2
0
Fork 0
mirror of https://git.asonix.dog/asonix/pict-rs synced 2025-01-08 18:51:24 +00:00

ranged support for process endpoint, bugfixes

This commit is contained in:
John Doe 2021-01-14 08:52:11 -05:00
parent 61061f0451
commit c663337bcd
2 changed files with 75 additions and 38 deletions

View file

@ -74,6 +74,9 @@ pub(crate) enum UploadError {
#[error("{0}")] #[error("{0}")]
Json(#[from] serde_json::Error), Json(#[from] serde_json::Error),
#[error("Range header not satisfiable")]
Range,
} }
impl From<actix_web::client::SendRequestError> for UploadError { impl From<actix_web::client::SendRequestError> for UploadError {
@ -119,6 +122,7 @@ impl ResponseError for UploadError {
| UploadError::ParseReq(_) => StatusCode::BAD_REQUEST, | UploadError::ParseReq(_) => StatusCode::BAD_REQUEST,
UploadError::MissingAlias | UploadError::MissingFilename => StatusCode::NOT_FOUND, UploadError::MissingAlias | UploadError::MissingFilename => StatusCode::NOT_FOUND,
UploadError::InvalidToken => StatusCode::FORBIDDEN, UploadError::InvalidToken => StatusCode::FORBIDDEN,
UploadError::Range => StatusCode::RANGE_NOT_SATISFIABLE,
_ => StatusCode::INTERNAL_SERVER_ERROR, _ => StatusCode::INTERNAL_SERVER_ERROR,
} }
} }

View file

@ -1,6 +1,6 @@
use actix_form_data::{Field, Form, Value}; use actix_form_data::{Field, Form, Value};
use actix_fs::file; use actix_fs::file;
use actix_web::{App, HttpResponse, HttpServer, client::Client, guard, http::header::{ACCEPT_RANGES, CONTENT_LENGTH, CacheControl, CacheDirective, ContentRange, ContentRangeSpec, Header, LastModified}, middleware::{Compress, Logger}, web}; use actix_web::{App, HttpRequest, HttpResponse, HttpServer, client::Client, guard, http::{HeaderValue, header::{ACCEPT_RANGES, CONTENT_LENGTH, CacheControl, CacheDirective, ContentRange, ContentRangeSpec, Header, LastModified}}, middleware::{Compress, Logger}, web};
use bytes::Bytes; use bytes::Bytes;
use futures::{StreamExt, stream::{Stream, TryStreamExt}}; use futures::{StreamExt, stream::{Stream, TryStreamExt}};
use once_cell::sync::Lazy; use once_cell::sync::Lazy;
@ -333,6 +333,7 @@ async fn process_details(
/// Process files /// Process files
#[instrument(skip(manager, whitelist))] #[instrument(skip(manager, whitelist))]
async fn process( async fn process(
req: HttpRequest,
query: web::Query<ProcessQuery>, query: web::Query<ProcessQuery>,
ext: web::Path<String>, ext: web::Path<String>,
manager: web::Data<UploadManager>, manager: web::Data<UploadManager>,
@ -427,6 +428,26 @@ async fn process(
drop(entered); drop(entered);
}); });
match req.headers().get("Range") {
Some(range_head) => {
let range = parse_range_header(range_head)?;
let resp_bytes = img_bytes.slice(range[0] as usize..range[1] as usize);
let stream = Box::pin(futures::stream::once(async move {
Ok(resp_bytes) as Result<_, UploadError>
}));
return Ok(srv_ranged_response(
stream,
details.content_type(),
7 * DAYS,
details.system_time(),
Some((range[0], range[1])),
Some(img_bytes.len() as u64)));
}
None => {
return Ok(srv_response( return Ok(srv_response(
Box::pin(futures::stream::once(async { Box::pin(futures::stream::once(async {
Ok(img_bytes) as Result<_, UploadError> Ok(img_bytes) as Result<_, UploadError>
@ -436,6 +457,8 @@ async fn process(
details.system_time(), details.system_time(),
)); ));
} }
};
}
let details = if let Some(details) = details { let details = if let Some(details) = details {
details details
@ -447,14 +470,7 @@ async fn process(
details details
}; };
let stream = actix_fs::read_to_stream(thumbnail_path).await?; ranged_file_resp(thumbnail_path, req, details).await
Ok(srv_response(
stream,
details.content_type(),
7 * DAYS,
details.system_time(),
))
} }
/// Fetch file details /// Fetch file details
@ -504,14 +520,15 @@ async fn serve(
details details
}; };
match req.headers().get("Range") { ranged_file_resp(path, req, details).await
//Range header exists - return as ranged }
Some(range_head) => {
let range_hdr = range_head.to_str().map_err(|_| { fn parse_range_header(range_head: &HeaderValue) -> Result<Vec<u64>, UploadError> {
let range_head_str = range_head.to_str().map_err(|_| {
UploadError::ParseReq("Range header contains non-utf8 characters".to_string()) UploadError::ParseReq("Range header contains non-utf8 characters".to_string())
})?; })?;
let range_dashed = range_hdr let range_dashed = range_head_str
.split('=') .split('=')
.skip(1) .skip(1)
.next() .next()
@ -525,6 +542,18 @@ async fn serve(
UploadError::ParseReq("Cannot parse byte locations in range header".to_string()) UploadError::ParseReq("Cannot parse byte locations in range header".to_string())
})?; })?;
if range[0] > range[1] {
return Err(UploadError::Range);
}
Ok(range)
}
async fn ranged_file_resp(path: PathBuf, req: HttpRequest, details: Details) -> Result<HttpResponse, UploadError> {
match req.headers().get("Range") {
//Range header exists - return as ranged
Some(range_head) => {
let range = parse_range_header(range_head)?;
let (out_file, _) = file::seek( let (out_file, _) = file::seek(
file::open(path).await?, file::open(path).await?,
@ -538,10 +567,15 @@ async fn serve(
.await .await
.map_err(|_| UploadError::Upload("Error reading metadata".to_string()))?; .map_err(|_| UploadError::Upload("Error reading metadata".to_string()))?;
if meta.len() < range[0] {
return Err(UploadError::Range);
}
// file::read_to_stream() creates a stream in 65,356 byte chunks.
let whole_to = ((range[1] - range[0]) as f64 / 65_356.0).floor() as usize; let whole_to = ((range[1] - range[0]) as f64 / 65_356.0).floor() as usize;
let partial_len = ((range[1] - range[0]) % 65_536) as usize; let partial_len = ((range[1] - range[0]) % 65_356) as usize;
//debug!("Range of {}. Returning {} whole chunks, and {} bytes of the partial chunk", range[1]-range[0], whole_to, partial_len);
let stream = file::read_to_stream(out_file) let stream = file::read_to_stream(out_file)
.await? .await?
@ -550,7 +584,7 @@ async fn serve(
.map(move |bytes_res| { .map(move |bytes_res| {
match bytes_res.1 { match bytes_res.1 {
Ok(mut bytes) => { Ok(mut bytes) => {
if bytes_res.0 == whole_to { if bytes_res.0 == whole_to && partial_len <= bytes.len() {
return Ok(bytes.split_to(partial_len)); return Ok(bytes.split_to(partial_len));
} }
return Ok(bytes); return Ok(bytes);
@ -627,7 +661,6 @@ where
range, range,
instance_length, instance_length,
})) }))
.set_header(CONTENT_LENGTH, range.unwrap().1 - range.unwrap().0)
.set_header(ACCEPT_RANGES, "bytes") .set_header(ACCEPT_RANGES, "bytes")
.content_type(ext.to_string()) .content_type(ext.to_string())
.streaming(stream.err_into()) .streaming(stream.err_into())