mirror of
https://git.asonix.dog/asonix/pict-rs
synced 2025-01-08 18:51:24 +00:00
Reduce required Unpin bounds with pin-project-lite
This commit is contained in:
parent
4c5482b3a8
commit
2fb8d3e39a
6 changed files with 207 additions and 121 deletions
42
Cargo.lock
generated
42
Cargo.lock
generated
|
@ -77,9 +77,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "actix-macros"
|
name = "actix-macros"
|
||||||
version = "0.2.1"
|
version = "0.2.2"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c2f86cd6857c135e6e9fe57b1619a88d1f94a7df34c00e11fe13e64fd3438837"
|
checksum = "c8c999eaf5f8414142b5bb60e43dcb27ccd12c2c15e4133e470f3c5060472532"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"quote",
|
"quote",
|
||||||
"syn",
|
"syn",
|
||||||
|
@ -240,9 +240,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ahash"
|
name = "ahash"
|
||||||
version = "0.7.5"
|
version = "0.7.6"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "991984e3fd003e7ba02eb724f87a0f997b78677c46c0e91f8424ad7394c9886a"
|
checksum = "fcb51a0695d8f838b1ee009b3fbf66bda078cd64590202a864a8f3e8c4315c47"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"getrandom",
|
"getrandom",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
|
@ -560,9 +560,9 @@ checksum = "e78d4f1cc4ae33bbfc157ed5d5a5ef3bc29227303d595861deb238fcec4e9457"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "encoding_rs"
|
name = "encoding_rs"
|
||||||
version = "0.8.28"
|
version = "0.8.29"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "80df024fbc5ac80f87dfef0d9f5209a252f2a497f7f42944cff24d8253cac065"
|
checksum = "a74ea89a0a1b98f6332de42c95baff457ada66d1cb4030f9ff151b2041a1c746"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
]
|
]
|
||||||
|
@ -865,9 +865,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "instant"
|
name = "instant"
|
||||||
version = "0.1.11"
|
version = "0.1.12"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "716d3d89f35ac6a34fd0eed635395f4c3b76fa889338a4632e5231a8684216bd"
|
checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
]
|
]
|
||||||
|
@ -920,9 +920,9 @@ checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "libc"
|
name = "libc"
|
||||||
version = "0.2.103"
|
version = "0.2.104"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "dd8f7255a17a627354f321ef0055d63b898c6fb27eff628af4d1b66b7331edf6"
|
checksum = "7b2f96d100e1cf1929e7719b7edb3b90ab5298072638fccd77be9ce942ecdfce"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "local-channel"
|
name = "local-channel"
|
||||||
|
@ -998,9 +998,9 @@ checksum = "2a60c7ce501c71e03a9c9c0d35b861413ae925bd979cc7a4e30d060069aaac8d"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "mio"
|
name = "mio"
|
||||||
version = "0.7.13"
|
version = "0.7.14"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8c2bdb6314ec10835cd3293dd268473a835c02b7b352e788be788b3c6ca6bb16"
|
checksum = "8067b404fe97c70829f082dec8bcf4f71225d7eaea1d8645349cb76fa06205cc"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"libc",
|
"libc",
|
||||||
"log",
|
"log",
|
||||||
|
@ -1184,6 +1184,7 @@ dependencies = [
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"opentelemetry",
|
"opentelemetry",
|
||||||
"opentelemetry-otlp",
|
"opentelemetry-otlp",
|
||||||
|
"pin-project-lite",
|
||||||
"rand",
|
"rand",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
|
@ -1240,9 +1241,9 @@ checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ppv-lite86"
|
name = "ppv-lite86"
|
||||||
version = "0.2.10"
|
version = "0.2.14"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ac74c624d6b2d21f425f752262f42188365d7b8ff1aff74c82e45136510a4857"
|
checksum = "c3ca011bd0129ff4ae15cd04c4eef202cadf6c51c21e47aba319b4e0501db741"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro-error"
|
name = "proc-macro-error"
|
||||||
|
@ -1753,9 +1754,9 @@ checksum = "8ea5119cdb4c55b55d432abb513a0429384878c15dde60cc77b1c99de1a95a6a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "structopt"
|
name = "structopt"
|
||||||
version = "0.3.23"
|
version = "0.3.25"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "bf9d950ef167e25e0bdb073cf1d68e9ad2795ac826f2f3f59647817cf23c0bfa"
|
checksum = "40b9788f4202aa75c240ecc9c15c65185e6a39ccdeb0fd5d008b98825464c87c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"clap",
|
"clap",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
|
@ -1764,9 +1765,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "structopt-derive"
|
name = "structopt-derive"
|
||||||
version = "0.4.16"
|
version = "0.4.18"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "134d838a2c9943ac3125cf6df165eda53493451b719f3255b2a26b85f772d0ba"
|
checksum = "dcb5ae327f9cc13b68763b5749770cb9e048a99bd9dfdfa58d0cf05d5f64afe0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"heck",
|
"heck",
|
||||||
"proc-macro-error",
|
"proc-macro-error",
|
||||||
|
@ -2036,14 +2037,15 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "tower"
|
name = "tower"
|
||||||
version = "0.4.8"
|
version = "0.4.9"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f60422bc7fefa2f3ec70359b8ff1caff59d785877eb70595904605bcc412470f"
|
checksum = "d15a6b60cdff0cb039d81d3b37f8bc3d7e53dca09069aae3ef2502ca4834fe30"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"futures-core",
|
"futures-core",
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"indexmap",
|
"indexmap",
|
||||||
"pin-project",
|
"pin-project",
|
||||||
|
"pin-project-lite",
|
||||||
"rand",
|
"rand",
|
||||||
"slab",
|
"slab",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
|
|
@ -28,6 +28,7 @@ num_cpus = "1.13"
|
||||||
once_cell = "1.4.0"
|
once_cell = "1.4.0"
|
||||||
opentelemetry = { version = "0.16", features = ["rt-tokio"] }
|
opentelemetry = { version = "0.16", features = ["rt-tokio"] }
|
||||||
opentelemetry-otlp = "0.9"
|
opentelemetry-otlp = "0.9"
|
||||||
|
pin-project-lite = "0.2.7"
|
||||||
rand = "0.8.0"
|
rand = "0.8.0"
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
|
|
|
@ -5,39 +5,63 @@ use std::{
|
||||||
};
|
};
|
||||||
use tokio::io::{AsyncRead, ReadBuf};
|
use tokio::io::{AsyncRead, ReadBuf};
|
||||||
|
|
||||||
pub(crate) enum Either<Left, Right> {
|
pin_project_lite::pin_project! {
|
||||||
Left(Left),
|
#[project = EitherProj]
|
||||||
Right(Right),
|
#[project_replace = EitherProjReplace]
|
||||||
|
pub(crate) enum Either<Left, Right> {
|
||||||
|
Left {
|
||||||
|
#[pin]
|
||||||
|
left: Left,
|
||||||
|
},
|
||||||
|
Right {
|
||||||
|
#[pin]
|
||||||
|
right: Right,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<L, R> Either<L, R> {
|
||||||
|
pub(crate) fn left(left: L) -> Self {
|
||||||
|
Either::Left { left }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn right(right: R) -> Self {
|
||||||
|
Either::Right { right }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<Left, Right> AsyncRead for Either<Left, Right>
|
impl<Left, Right> AsyncRead for Either<Left, Right>
|
||||||
where
|
where
|
||||||
Left: AsyncRead + Unpin,
|
Left: AsyncRead,
|
||||||
Right: AsyncRead + Unpin,
|
Right: AsyncRead,
|
||||||
{
|
{
|
||||||
fn poll_read(
|
fn poll_read(
|
||||||
mut self: Pin<&mut Self>,
|
mut self: Pin<&mut Self>,
|
||||||
cx: &mut Context<'_>,
|
cx: &mut Context<'_>,
|
||||||
buf: &mut ReadBuf<'_>,
|
buf: &mut ReadBuf<'_>,
|
||||||
) -> Poll<std::io::Result<()>> {
|
) -> Poll<std::io::Result<()>> {
|
||||||
match *self {
|
let this = self.as_mut().project();
|
||||||
Self::Left(ref mut left) => Pin::new(left).poll_read(cx, buf),
|
|
||||||
Self::Right(ref mut right) => Pin::new(right).poll_read(cx, buf),
|
match this {
|
||||||
|
EitherProj::Left { left } => left.poll_read(cx, buf),
|
||||||
|
EitherProj::Right { right } => right.poll_read(cx, buf),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<Left, Right> Stream for Either<Left, Right>
|
impl<Left, Right> Stream for Either<Left, Right>
|
||||||
where
|
where
|
||||||
Left: Stream<Item = <Right as Stream>::Item> + Unpin,
|
Left: Stream<Item = <Right as Stream>::Item>,
|
||||||
Right: Stream + Unpin,
|
Right: Stream,
|
||||||
{
|
{
|
||||||
type Item = <Left as Stream>::Item;
|
type Item = <Left as Stream>::Item;
|
||||||
|
|
||||||
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
||||||
match *self {
|
let this = self.as_mut().project();
|
||||||
Self::Left(ref mut left) => Pin::new(left).poll_next(cx),
|
|
||||||
Self::Right(ref mut right) => Pin::new(right).poll_next(cx),
|
match this {
|
||||||
|
EitherProj::Left { left } => left.poll_next(cx),
|
||||||
|
EitherProj::Right { right } => right.poll_next(cx),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
157
src/file.rs
157
src/file.rs
|
@ -10,17 +10,24 @@ pub(crate) use io_uring::File;
|
||||||
#[cfg(not(feature = "io-uring"))]
|
#[cfg(not(feature = "io-uring"))]
|
||||||
pub(crate) use tokio_file::File;
|
pub(crate) use tokio_file::File;
|
||||||
|
|
||||||
struct CrateError<S>(S);
|
pin_project_lite::pin_project! {
|
||||||
|
struct CrateError<S> {
|
||||||
|
#[pin]
|
||||||
|
inner: S
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<T, E, S> Stream for CrateError<S>
|
impl<T, E, S> Stream for CrateError<S>
|
||||||
where
|
where
|
||||||
S: Stream<Item = Result<T, E>> + Unpin,
|
S: Stream<Item = Result<T, E>>,
|
||||||
crate::error::Error: From<E>,
|
crate::error::Error: From<E>,
|
||||||
{
|
{
|
||||||
type Item = Result<T, crate::error::Error>;
|
type Item = Result<T, crate::error::Error>;
|
||||||
|
|
||||||
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
||||||
Pin::new(&mut self.0)
|
let this = self.as_mut().project();
|
||||||
|
|
||||||
|
this.inner
|
||||||
.poll_next(cx)
|
.poll_next(cx)
|
||||||
.map(|opt| opt.map(|res| res.map_err(Into::into)))
|
.map(|opt| opt.map(|res| res.map_err(Into::into)))
|
||||||
}
|
}
|
||||||
|
@ -90,31 +97,35 @@ mod tokio_file {
|
||||||
mut self,
|
mut self,
|
||||||
from_start: Option<u64>,
|
from_start: Option<u64>,
|
||||||
len: Option<u64>,
|
len: Option<u64>,
|
||||||
) -> Result<
|
) -> Result<impl Stream<Item = Result<Bytes, crate::error::Error>>, crate::error::Error>
|
||||||
impl Stream<Item = Result<Bytes, crate::error::Error>> + Unpin,
|
{
|
||||||
crate::error::Error,
|
|
||||||
> {
|
|
||||||
let obj = match (from_start, len) {
|
let obj = match (from_start, len) {
|
||||||
(Some(lower), Some(upper)) => {
|
(Some(lower), Some(upper)) => {
|
||||||
self.inner.seek(SeekFrom::Start(lower)).await?;
|
self.inner.seek(SeekFrom::Start(lower)).await?;
|
||||||
Either::Left(self.inner.take(upper))
|
Either::left(self.inner.take(upper))
|
||||||
}
|
}
|
||||||
(None, Some(upper)) => Either::Left(self.inner.take(upper)),
|
(None, Some(upper)) => Either::left(self.inner.take(upper)),
|
||||||
(Some(lower), None) => {
|
(Some(lower), None) => {
|
||||||
self.inner.seek(SeekFrom::Start(lower)).await?;
|
self.inner.seek(SeekFrom::Start(lower)).await?;
|
||||||
Either::Right(self.inner)
|
Either::right(self.inner)
|
||||||
}
|
}
|
||||||
(None, None) => Either::Right(self.inner),
|
(None, None) => Either::right(self.inner),
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(super::CrateError(BytesFreezer(FramedRead::new(
|
Ok(super::CrateError {
|
||||||
obj,
|
inner: BytesFreezer {
|
||||||
BytesCodec::new(),
|
inner: FramedRead::new(obj, BytesCodec::new()),
|
||||||
))))
|
},
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct BytesFreezer<S>(S);
|
pin_project_lite::pin_project! {
|
||||||
|
struct BytesFreezer<S> {
|
||||||
|
#[pin]
|
||||||
|
inner: S,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<S, E> Stream for BytesFreezer<S>
|
impl<S, E> Stream for BytesFreezer<S>
|
||||||
where
|
where
|
||||||
|
@ -126,7 +137,9 @@ mod tokio_file {
|
||||||
mut self: std::pin::Pin<&mut Self>,
|
mut self: std::pin::Pin<&mut Self>,
|
||||||
cx: &mut std::task::Context<'_>,
|
cx: &mut std::task::Context<'_>,
|
||||||
) -> std::task::Poll<Option<Self::Item>> {
|
) -> std::task::Poll<Option<Self::Item>> {
|
||||||
std::pin::Pin::new(&mut self.0)
|
let this = self.as_mut().project();
|
||||||
|
|
||||||
|
this.inner
|
||||||
.poll_next(cx)
|
.poll_next(cx)
|
||||||
.map(|opt| opt.map(|res| res.map(BytesMut::freeze)))
|
.map(|opt| opt.map(|res| res.map(BytesMut::freeze)))
|
||||||
}
|
}
|
||||||
|
@ -299,21 +312,21 @@ mod io_uring {
|
||||||
self,
|
self,
|
||||||
from_start: Option<u64>,
|
from_start: Option<u64>,
|
||||||
len: Option<u64>,
|
len: Option<u64>,
|
||||||
) -> Result<
|
) -> Result<impl Stream<Item = Result<Bytes, crate::error::Error>>, crate::error::Error>
|
||||||
impl Stream<Item = Result<Bytes, crate::error::Error>> + Unpin,
|
{
|
||||||
crate::error::Error,
|
|
||||||
> {
|
|
||||||
let size = self.metadata().await?.len();
|
let size = self.metadata().await?.len();
|
||||||
|
|
||||||
let cursor = from_start.unwrap_or(0);
|
let cursor = from_start.unwrap_or(0);
|
||||||
let size = len.unwrap_or(size - cursor) + cursor;
|
let size = len.unwrap_or(size - cursor) + cursor;
|
||||||
|
|
||||||
Ok(super::CrateError(BytesStream {
|
Ok(super::CrateError {
|
||||||
file: Some(self),
|
inner: BytesStream {
|
||||||
|
state: ReadFileState::File { file: Some(self) },
|
||||||
size,
|
size,
|
||||||
cursor,
|
cursor,
|
||||||
fut: None,
|
callback: read_file,
|
||||||
}))
|
},
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn read_at<T: IoBufMut>(&self, buf: T, pos: u64) -> BufResult<usize, T> {
|
async fn read_at<T: IoBufMut>(&self, buf: T, pos: u64) -> BufResult<usize, T> {
|
||||||
|
@ -325,55 +338,89 @@ mod io_uring {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct BytesStream {
|
pin_project_lite::pin_project! {
|
||||||
file: Option<File>,
|
struct BytesStream<F, Fut> {
|
||||||
|
#[pin]
|
||||||
|
state: ReadFileState<Fut>,
|
||||||
size: u64,
|
size: u64,
|
||||||
cursor: u64,
|
cursor: u64,
|
||||||
fut: Option<Pin<Box<dyn Future<Output = (File, BufResult<usize, Vec<u8>>)>>>>,
|
#[pin]
|
||||||
|
callback: F,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Stream for BytesStream {
|
pin_project_lite::pin_project! {
|
||||||
type Item = std::io::Result<Bytes>;
|
#[project = ReadFileStateProj]
|
||||||
|
#[project_replace = ReadFileStateProjReplace]
|
||||||
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
enum ReadFileState<Fut> {
|
||||||
let mut fut = if let Some(fut) = self.fut.take() {
|
File {
|
||||||
fut
|
file: Option<File>,
|
||||||
} else {
|
},
|
||||||
let file = self.file.take().unwrap();
|
Future {
|
||||||
|
#[pin]
|
||||||
if self.cursor == self.size {
|
fut: Fut,
|
||||||
return Poll::Ready(None);
|
},
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let cursor = self.cursor;
|
async fn read_file(
|
||||||
let max_size = self.size - self.cursor;
|
file: File,
|
||||||
|
capacity: usize,
|
||||||
Box::pin(async move {
|
cursor: u64,
|
||||||
let buf = Vec::with_capacity(max_size.try_into().unwrap());
|
) -> (File, BufResult<usize, Vec<u8>>) {
|
||||||
|
let buf = Vec::with_capacity(capacity);
|
||||||
|
|
||||||
let buf_res = file.read_at(buf, cursor).await;
|
let buf_res = file.read_at(buf, cursor).await;
|
||||||
|
|
||||||
(file, buf_res)
|
(file, buf_res)
|
||||||
})
|
|
||||||
};
|
|
||||||
|
|
||||||
match Pin::new(&mut fut).poll(cx) {
|
|
||||||
Poll::Pending => {
|
|
||||||
self.fut = Some(fut);
|
|
||||||
Poll::Pending
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<F, Fut> Stream for BytesStream<F, Fut>
|
||||||
|
where
|
||||||
|
F: Fn(File, usize, u64) -> Fut,
|
||||||
|
Fut: Future<Output = (File, BufResult<usize, Vec<u8>>)> + 'static,
|
||||||
|
{
|
||||||
|
type Item = std::io::Result<Bytes>;
|
||||||
|
|
||||||
|
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
||||||
|
let mut this = self.as_mut().project();
|
||||||
|
|
||||||
|
match this.state.as_mut().project() {
|
||||||
|
ReadFileStateProj::File { file } => {
|
||||||
|
let cursor = *this.cursor;
|
||||||
|
let max_size = *this.size - *this.cursor;
|
||||||
|
|
||||||
|
if max_size == 0 {
|
||||||
|
return Poll::Ready(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
let capacity = max_size.min(65_356) as usize;
|
||||||
|
let file = file.take().unwrap();
|
||||||
|
|
||||||
|
let fut = (this.callback)(file, capacity, cursor);
|
||||||
|
|
||||||
|
this.state.project_replace(ReadFileState::Future { fut });
|
||||||
|
self.poll_next(cx)
|
||||||
|
}
|
||||||
|
ReadFileStateProj::Future { fut } => match fut.poll(cx) {
|
||||||
|
Poll::Pending => Poll::Pending,
|
||||||
Poll::Ready((file, (Ok(n), mut buf))) => {
|
Poll::Ready((file, (Ok(n), mut buf))) => {
|
||||||
self.file = Some(file);
|
this.state
|
||||||
|
.project_replace(ReadFileState::File { file: Some(file) });
|
||||||
|
|
||||||
let _ = buf.split_off(n);
|
let _ = buf.split_off(n);
|
||||||
let n: u64 = match n.try_into() {
|
let n: u64 = match n.try_into() {
|
||||||
Ok(n) => n,
|
Ok(n) => n,
|
||||||
Err(_) => return Poll::Ready(Some(Err(std::io::ErrorKind::Other.into()))),
|
Err(_) => {
|
||||||
|
return Poll::Ready(Some(Err(std::io::ErrorKind::Other.into())))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
self.cursor += n;
|
*this.cursor += n;
|
||||||
|
|
||||||
Poll::Ready(Some(Ok(Bytes::from(buf))))
|
Poll::Ready(Some(Ok(buf.into())))
|
||||||
}
|
}
|
||||||
Poll::Ready((_, (Err(e), _))) => Poll::Ready(Some(Err(e))),
|
Poll::Ready((_, (Err(e), _))) => Poll::Ready(Some(Err(e))),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
39
src/main.rs
39
src/main.rs
|
@ -91,16 +91,24 @@ static PROCESS_MAP: Lazy<ProcessMap> = Lazy::new(DashMap::new);
|
||||||
type OutcomeSender = Sender<(Details, web::Bytes)>;
|
type OutcomeSender = Sender<(Details, web::Bytes)>;
|
||||||
type ProcessMap = DashMap<PathBuf, Vec<OutcomeSender>>;
|
type ProcessMap = DashMap<PathBuf, Vec<OutcomeSender>>;
|
||||||
|
|
||||||
struct CancelSafeProcessor<F> {
|
struct CancelToken {
|
||||||
span: Span,
|
span: Span,
|
||||||
path: PathBuf,
|
path: PathBuf,
|
||||||
receiver: Option<Receiver<(Details, web::Bytes)>>,
|
receiver: Option<Receiver<(Details, web::Bytes)>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pin_project_lite::pin_project! {
|
||||||
|
struct CancelSafeProcessor<F> {
|
||||||
|
cancel_token: CancelToken,
|
||||||
|
|
||||||
|
#[pin]
|
||||||
fut: F,
|
fut: F,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<F> CancelSafeProcessor<F>
|
impl<F> CancelSafeProcessor<F>
|
||||||
where
|
where
|
||||||
F: Future<Output = Result<(Details, web::Bytes), Error>> + Unpin,
|
F: Future<Output = Result<(Details, web::Bytes), Error>>,
|
||||||
{
|
{
|
||||||
pub(crate) fn new(path: PathBuf, fut: F) -> Self {
|
pub(crate) fn new(path: PathBuf, fut: F) -> Self {
|
||||||
let entry = PROCESS_MAP.entry(path.clone());
|
let entry = PROCESS_MAP.entry(path.clone());
|
||||||
|
@ -127,9 +135,11 @@ where
|
||||||
};
|
};
|
||||||
|
|
||||||
CancelSafeProcessor {
|
CancelSafeProcessor {
|
||||||
|
cancel_token: CancelToken {
|
||||||
span,
|
span,
|
||||||
path,
|
path,
|
||||||
receiver,
|
receiver,
|
||||||
|
},
|
||||||
fut,
|
fut,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -137,21 +147,26 @@ where
|
||||||
|
|
||||||
impl<F> Future for CancelSafeProcessor<F>
|
impl<F> Future for CancelSafeProcessor<F>
|
||||||
where
|
where
|
||||||
F: Future<Output = Result<(Details, web::Bytes), Error>> + Unpin,
|
F: Future<Output = Result<(Details, web::Bytes), Error>>,
|
||||||
{
|
{
|
||||||
type Output = Result<(Details, web::Bytes), Error>;
|
type Output = Result<(Details, web::Bytes), Error>;
|
||||||
|
|
||||||
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
||||||
let span = self.span.clone();
|
let this = self.as_mut().project();
|
||||||
|
|
||||||
|
let span = &this.cancel_token.span;
|
||||||
|
let receiver = &mut this.cancel_token.receiver;
|
||||||
|
let path = &this.cancel_token.path;
|
||||||
|
let fut = this.fut;
|
||||||
|
|
||||||
span.in_scope(|| {
|
span.in_scope(|| {
|
||||||
if let Some(ref mut rx) = self.receiver {
|
if let Some(ref mut rx) = receiver {
|
||||||
Pin::new(rx)
|
Pin::new(rx)
|
||||||
.poll(cx)
|
.poll(cx)
|
||||||
.map(|res| res.map_err(|_| UploadError::Canceled.into()))
|
.map(|res| res.map_err(|_| UploadError::Canceled.into()))
|
||||||
} else {
|
} else {
|
||||||
Pin::new(&mut self.fut).poll(cx).map(|res| {
|
fut.poll(cx).map(|res| {
|
||||||
let opt = PROCESS_MAP.remove(&self.path);
|
let opt = PROCESS_MAP.remove(path);
|
||||||
res.map(|tup| {
|
res.map(|tup| {
|
||||||
if let Some((_, vec)) = opt {
|
if let Some((_, vec)) = opt {
|
||||||
for sender in vec {
|
for sender in vec {
|
||||||
|
@ -166,7 +181,7 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<F> Drop for CancelSafeProcessor<F> {
|
impl Drop for CancelToken {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
if self.receiver.is_none() {
|
if self.receiver.is_none() {
|
||||||
let completed = PROCESS_MAP.remove(&self.path).is_none();
|
let completed = PROCESS_MAP.remove(&self.path).is_none();
|
||||||
|
@ -591,7 +606,7 @@ async fn process(
|
||||||
};
|
};
|
||||||
|
|
||||||
let (details, bytes) =
|
let (details, bytes) =
|
||||||
CancelSafeProcessor::new(thumbnail_path.clone(), Box::pin(process_fut)).await?;
|
CancelSafeProcessor::new(thumbnail_path.clone(), process_fut).await?;
|
||||||
|
|
||||||
return match range {
|
return match range {
|
||||||
Some(range_header) => {
|
Some(range_header) => {
|
||||||
|
@ -717,7 +732,7 @@ async fn ranged_file_resp(
|
||||||
let mut builder = HttpResponse::PartialContent();
|
let mut builder = HttpResponse::PartialContent();
|
||||||
builder.insert_header(range.to_content_range(meta.len()));
|
builder.insert_header(range.to_content_range(meta.len()));
|
||||||
|
|
||||||
(builder, Either::Left(range.chop_file(file).await?))
|
(builder, Either::left(range.chop_file(file).await?))
|
||||||
} else {
|
} else {
|
||||||
return Err(UploadError::Range.into());
|
return Err(UploadError::Range.into());
|
||||||
}
|
}
|
||||||
|
@ -726,13 +741,13 @@ async fn ranged_file_resp(
|
||||||
None => {
|
None => {
|
||||||
let file = crate::file::File::open(path).await?;
|
let file = crate::file::File::open(path).await?;
|
||||||
let stream = file.read_to_stream(None, None).await?;
|
let stream = file.read_to_stream(None, None).await?;
|
||||||
(HttpResponse::Ok(), Either::Right(stream))
|
(HttpResponse::Ok(), Either::right(stream))
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(srv_response(
|
Ok(srv_response(
|
||||||
builder,
|
builder,
|
||||||
stream,
|
Box::pin(stream),
|
||||||
details.content_type(),
|
details.content_type(),
|
||||||
7 * DAYS,
|
7 * DAYS,
|
||||||
details.system_time(),
|
details.system_time(),
|
||||||
|
|
|
@ -42,10 +42,7 @@ impl Range {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn chop_bytes(
|
pub(crate) fn chop_bytes(&self, bytes: Bytes) -> impl Stream<Item = Result<Bytes, Error>> {
|
||||||
&self,
|
|
||||||
bytes: Bytes,
|
|
||||||
) -> impl Stream<Item = Result<Bytes, Error>> + Unpin {
|
|
||||||
match self {
|
match self {
|
||||||
Range::Start(start) => once(ready(Ok(bytes.slice(*start as usize..)))),
|
Range::Start(start) => once(ready(Ok(bytes.slice(*start as usize..)))),
|
||||||
Range::SuffixLength(from_start) => once(ready(Ok(bytes.slice(..*from_start as usize)))),
|
Range::SuffixLength(from_start) => once(ready(Ok(bytes.slice(..*from_start as usize)))),
|
||||||
|
@ -58,7 +55,7 @@ impl Range {
|
||||||
pub(crate) async fn chop_file(
|
pub(crate) async fn chop_file(
|
||||||
&self,
|
&self,
|
||||||
file: crate::file::File,
|
file: crate::file::File,
|
||||||
) -> Result<impl Stream<Item = Result<Bytes, Error>> + Unpin, Error> {
|
) -> Result<impl Stream<Item = Result<Bytes, Error>>, Error> {
|
||||||
match self {
|
match self {
|
||||||
Range::Start(start) => file.read_to_stream(Some(*start), None).await,
|
Range::Start(start) => file.read_to_stream(Some(*start), None).await,
|
||||||
Range::SuffixLength(from_start) => file.read_to_stream(None, Some(*from_start)).await,
|
Range::SuffixLength(from_start) => file.read_to_stream(None, Some(*from_start)).await,
|
||||||
|
|
Loading…
Reference in a new issue