|
|
@ -10,12 +10,13 @@ |
|
|
|
)] |
|
|
|
)] |
|
|
|
|
|
|
|
|
|
|
|
use clap::{Parser, Subcommand}; |
|
|
|
use clap::{Parser, Subcommand}; |
|
|
|
|
|
|
|
use flate2::read::MultiGzDecoder; |
|
|
|
use oxhttp::model::{Body, HeaderName, HeaderValue, Request, Response, Status}; |
|
|
|
use oxhttp::model::{Body, HeaderName, HeaderValue, Request, Response, Status}; |
|
|
|
use oxhttp::Server; |
|
|
|
use oxhttp::Server; |
|
|
|
use oxigraph::io::{DatasetFormat, DatasetSerializer, GraphFormat, GraphSerializer}; |
|
|
|
use oxigraph::io::{DatasetFormat, DatasetSerializer, GraphFormat, GraphSerializer}; |
|
|
|
use oxigraph::model::{GraphName, GraphNameRef, IriParseError, NamedNode, NamedOrBlankNode}; |
|
|
|
use oxigraph::model::{GraphName, GraphNameRef, IriParseError, NamedNode, NamedOrBlankNode}; |
|
|
|
use oxigraph::sparql::{Query, QueryResults, Update}; |
|
|
|
use oxigraph::sparql::{Query, QueryResults, Update}; |
|
|
|
use oxigraph::store::Store; |
|
|
|
use oxigraph::store::{BulkLoader, Store}; |
|
|
|
use oxiri::Iri; |
|
|
|
use oxiri::Iri; |
|
|
|
use rand::random; |
|
|
|
use rand::random; |
|
|
|
use sparesults::{QueryResultsFormat, QueryResultsSerializer}; |
|
|
|
use sparesults::{QueryResultsFormat, QueryResultsSerializer}; |
|
|
@ -76,35 +77,36 @@ pub fn main() -> std::io::Result<()> { |
|
|
|
|
|
|
|
|
|
|
|
match matches.command { |
|
|
|
match matches.command { |
|
|
|
Command::Load { file } => { |
|
|
|
Command::Load { file } => { |
|
|
|
let handles = file.iter().map(|file| { |
|
|
|
let handles = file |
|
|
|
let store = store.clone(); |
|
|
|
.iter() |
|
|
|
let file = file.to_string(); |
|
|
|
.map(|file| { |
|
|
|
spawn(move || { |
|
|
|
let store = store.clone(); |
|
|
|
let f = file.clone(); |
|
|
|
let file = file.to_string(); |
|
|
|
let start = Instant::now(); |
|
|
|
spawn(move || { |
|
|
|
let loader = store.bulk_loader().on_progress(move |size| { |
|
|
|
let f = file.clone(); |
|
|
|
let elapsed = start.elapsed(); |
|
|
|
let start = Instant::now(); |
|
|
|
println!("{} triples loaded in {}s ({} t/s) from {}", size, elapsed.as_secs(), size / elapsed.as_secs(), f) |
|
|
|
let loader = store.bulk_loader().on_progress(move |size| { |
|
|
|
}); |
|
|
|
let elapsed = start.elapsed(); |
|
|
|
let reader = BufReader::new(File::open(&file)?); |
|
|
|
println!( |
|
|
|
if let Some(format) = file |
|
|
|
"{} triples loaded in {}s ({} t/s) from {}", |
|
|
|
.rsplit_once('.') |
|
|
|
size, |
|
|
|
.and_then(|(_, extension)| DatasetFormat::from_extension(extension)) { |
|
|
|
elapsed.as_secs(), |
|
|
|
loader.load_dataset(reader, format, None)?; |
|
|
|
size / elapsed.as_secs(), |
|
|
|
Ok(()) |
|
|
|
f |
|
|
|
} else if let Some(format) = file |
|
|
|
) |
|
|
|
.rsplit_once('.') |
|
|
|
}); |
|
|
|
.and_then(|(_, extension)| GraphFormat::from_extension(extension)) { |
|
|
|
if file.ends_with(".gz") { |
|
|
|
loader.load_graph(reader, format, GraphNameRef::DefaultGraph, None)?; |
|
|
|
bulk_load( |
|
|
|
Ok(()) |
|
|
|
loader, |
|
|
|
} else { |
|
|
|
&file[..file.len() - 3], |
|
|
|
Err(io::Error::new( |
|
|
|
MultiGzDecoder::new(File::open(&file)?), |
|
|
|
ErrorKind::InvalidInput, |
|
|
|
) |
|
|
|
"The server is not able to guess the file format of {} from its extension", |
|
|
|
} else { |
|
|
|
)) |
|
|
|
bulk_load(loader, &file, File::open(&file)?) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
}) |
|
|
|
}) |
|
|
|
}) |
|
|
|
}).collect::<Vec<JoinHandle<io::Result<()>>>>(); |
|
|
|
.collect::<Vec<JoinHandle<io::Result<()>>>>(); |
|
|
|
for handle in handles { |
|
|
|
for handle in handles { |
|
|
|
handle.join().unwrap()?; |
|
|
|
handle.join().unwrap()?; |
|
|
|
} |
|
|
|
} |
|
|
@ -124,6 +126,28 @@ pub fn main() -> std::io::Result<()> { |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fn bulk_load(loader: BulkLoader, file: &str, reader: impl Read) -> io::Result<()> { |
|
|
|
|
|
|
|
let (_, extension) = file.rsplit_once('.').ok_or_else(|| io::Error::new( |
|
|
|
|
|
|
|
ErrorKind::InvalidInput, |
|
|
|
|
|
|
|
format!("The server is not able to guess the file format of {} because the file name as no extension", file)))?; |
|
|
|
|
|
|
|
let reader = BufReader::new(reader); |
|
|
|
|
|
|
|
if let Some(format) = DatasetFormat::from_extension(extension) { |
|
|
|
|
|
|
|
loader.load_dataset(reader, format, None)?; |
|
|
|
|
|
|
|
Ok(()) |
|
|
|
|
|
|
|
} else if let Some(format) = GraphFormat::from_extension(extension) { |
|
|
|
|
|
|
|
loader.load_graph(reader, format, GraphNameRef::DefaultGraph, None)?; |
|
|
|
|
|
|
|
Ok(()) |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
Err(io::Error::new( |
|
|
|
|
|
|
|
ErrorKind::InvalidInput, |
|
|
|
|
|
|
|
format!( |
|
|
|
|
|
|
|
"The server is not able to guess the file format from the extension {}", |
|
|
|
|
|
|
|
extension |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
)) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
fn handle_request(request: &mut Request, store: Store) -> Response { |
|
|
|
fn handle_request(request: &mut Request, store: Store) -> Response { |
|
|
|
match (request.url().path(), request.method().as_ref()) { |
|
|
|
match (request.url().path(), request.method().as_ref()) { |
|
|
|
("/", "HEAD") => Response::builder(Status::OK) |
|
|
|
("/", "HEAD") => Response::builder(Status::OK) |
|
|
|