mirror of
https://github.com/scottlamb/moonfire-nvr.git
synced 2025-02-22 02:52:30 -05:00
* fully stop using ancient `failure` crate in favor of own error type * set an `ErrorKind` on everything
127 lines
3.6 KiB
Rust
127 lines
3.6 KiB
Rust
// This file is part of Moonfire NVR, a security camera network video recorder.
|
|
// Copyright (C) 2021 The Moonfire NVR Authors; see AUTHORS and LICENSE.txt.
|
|
// SPDX-License-Identifier: GPL-v3.0-or-later WITH GPL-3.0-linking-exception
|
|
|
|
//! HTTP body implementation using `ARefss<'static, [u8]>` chunks.
|
|
//!
|
|
//! Moonfire NVR uses this custom chunk type rather than [bytes::Bytes]. This
|
|
//! is mostly for historical reasons: we used to use `mmap`-backed chunks.
|
|
//! The custom chunk type also helps minimize reference-counting in `mp4::File`
|
|
//! as described [here](https://github.com/tokio-rs/bytes/issues/359#issuecomment-640812016),
|
|
//! although this is a pretty small optimization.
|
|
//!
|
|
//! Some day I expect [bytes::Bytes] will expose its vtable (see link above),
|
|
//! allowing us to minimize reference-counting while using the standard
|
|
//! [hyper::Body].
|
|
|
|
use base::Error;
|
|
use futures::{stream, Stream};
|
|
use reffers::ARefss;
|
|
use std::error::Error as StdError;
|
|
use std::pin::Pin;
|
|
use sync_wrapper::SyncWrapper;
|
|
|
|
pub struct Chunk(ARefss<'static, [u8]>);
|
|
|
|
pub type BoxedError = Box<dyn StdError + Send + Sync>;
|
|
pub type BodyStream = Box<dyn Stream<Item = Result<Chunk, BoxedError>> + Send>;
|
|
|
|
pub fn wrap_error(e: Error) -> BoxedError {
|
|
Box::new(e)
|
|
}
|
|
|
|
impl From<ARefss<'static, [u8]>> for Chunk {
|
|
fn from(r: ARefss<'static, [u8]>) -> Self {
|
|
Chunk(r)
|
|
}
|
|
}
|
|
|
|
impl From<&'static [u8]> for Chunk {
|
|
fn from(r: &'static [u8]) -> Self {
|
|
Chunk(ARefss::new(r))
|
|
}
|
|
}
|
|
|
|
impl From<&'static str> for Chunk {
|
|
fn from(r: &'static str) -> Self {
|
|
Chunk(ARefss::new(r.as_bytes()))
|
|
}
|
|
}
|
|
|
|
impl From<String> for Chunk {
|
|
fn from(r: String) -> Self {
|
|
Chunk(ARefss::new(r.into_bytes()))
|
|
}
|
|
}
|
|
|
|
impl From<Vec<u8>> for Chunk {
|
|
fn from(r: Vec<u8>) -> Self {
|
|
Chunk(ARefss::new(r))
|
|
}
|
|
}
|
|
|
|
impl hyper::body::Buf for Chunk {
|
|
fn remaining(&self) -> usize {
|
|
self.0.len()
|
|
}
|
|
fn chunk(&self) -> &[u8] {
|
|
&self.0
|
|
}
|
|
fn advance(&mut self, cnt: usize) {
|
|
self.0 = ::std::mem::replace(&mut self.0, ARefss::new(&[][..])).map(|b| &b[cnt..]);
|
|
}
|
|
}
|
|
|
|
// This SyncWrapper stuff is blindly copied from hyper's body type.
|
|
// See <https://github.com/hyperium/hyper/pull/2187>, matched by
|
|
// <https://github.com/scottlamb/http-serve/pull/18>.
|
|
pub struct Body(SyncWrapper<Pin<BodyStream>>);
|
|
|
|
impl hyper::body::HttpBody for Body {
|
|
type Data = Chunk;
|
|
type Error = BoxedError;
|
|
|
|
fn poll_data(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut std::task::Context,
|
|
) -> std::task::Poll<Option<Result<Self::Data, Self::Error>>> {
|
|
// This is safe because the pin is not structural.
|
|
// https://doc.rust-lang.org/std/pin/#pinning-is-not-structural-for-field
|
|
// (The field _holds_ a pin, but isn't itself pinned.)
|
|
unsafe { self.get_unchecked_mut() }
|
|
.0
|
|
.get_mut()
|
|
.as_mut()
|
|
.poll_next(cx)
|
|
}
|
|
|
|
fn poll_trailers(
|
|
self: Pin<&mut Self>,
|
|
_cx: &mut std::task::Context,
|
|
) -> std::task::Poll<Result<Option<http::header::HeaderMap>, Self::Error>> {
|
|
std::task::Poll::Ready(Ok(None))
|
|
}
|
|
}
|
|
|
|
impl From<BodyStream> for Body {
|
|
fn from(b: BodyStream) -> Self {
|
|
Body(SyncWrapper::new(Pin::from(b)))
|
|
}
|
|
}
|
|
|
|
impl<C: Into<Chunk>> From<C> for Body {
|
|
fn from(c: C) -> Self {
|
|
Body(SyncWrapper::new(Box::pin(stream::once(
|
|
futures::future::ok(c.into()),
|
|
))))
|
|
}
|
|
}
|
|
|
|
impl From<Error> for Body {
|
|
fn from(e: Error) -> Self {
|
|
Body(SyncWrapper::new(Box::pin(stream::once(
|
|
futures::future::err(wrap_error(e)),
|
|
))))
|
|
}
|
|
}
|