1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
//! Content-Encoding support on top of `http` crate

#![forbid(unsafe_code)]

pub mod error;

#[macro_use]
mod coder;
mod coding;
mod decode;
mod encode;

#[cfg(any(feature = "br", feature = "gz", feature = "de"))]
mod writer;

#[cfg(feature = "br")]
mod brotli {
    use std::io::{self, Write};

    use brotli2::write::{BrotliDecoder, BrotliEncoder};
    use bytes::Bytes;

    use super::{coder::Code, writer::BytesMutWriter};

    pub type Decoder = BrotliDecoder<BytesMutWriter>;
    pub struct Encoder(Option<BrotliEncoder<BytesMutWriter>>);

    impl Encoder {
        pub(crate) fn new(level: u32) -> Self {
            Self(Some(BrotliEncoder::new(BytesMutWriter::new(), level)))
        }
    }

    impl<T> Code<T> for BrotliDecoder<BytesMutWriter>
    where
        T: AsRef<[u8]>,
    {
        type Item = Bytes;

        fn code(&mut self, item: T) -> io::Result<Option<Self::Item>> {
            self.write_all(item.as_ref())?;
            self.flush()?;
            let b = self.get_mut().take();
            if !b.is_empty() {
                Ok(Some(b))
            } else {
                Ok(None)
            }
        }

        fn code_eof(&mut self) -> io::Result<Option<Self::Item>> {
            let b = self.finish()?.take_owned();
            if !b.is_empty() {
                Ok(Some(b))
            } else {
                Ok(None)
            }
        }
    }

    impl<T> Code<T> for Encoder
    where
        T: AsRef<[u8]>,
    {
        type Item = Bytes;

        fn code(&mut self, item: T) -> io::Result<Option<Self::Item>> {
            let encoder = self.0.as_mut().unwrap();
            encoder.write_all(item.as_ref())?;
            encoder.flush()?;
            let b = encoder.get_mut().take();
            if !b.is_empty() {
                Ok(Some(b))
            } else {
                Ok(None)
            }
        }

        fn code_eof(&mut self) -> io::Result<Option<Self::Item>> {
            match self.0.take() {
                Some(encoder) => {
                    let b = encoder.finish()?.take_owned();
                    assert!(!b.is_empty());
                    Ok(Some(b))
                }
                None => Ok(None),
            }
        }
    }
}

#[cfg(feature = "gz")]
mod gzip {
    use super::writer::BytesMutWriter;

    use flate2::write::{GzDecoder, GzEncoder};

    pub type Decoder = GzDecoder<BytesMutWriter>;
    pub type Encoder = GzEncoder<BytesMutWriter>;

    code_impl!(GzDecoder);
    code_impl!(GzEncoder);
}
#[cfg(feature = "de")]
mod deflate {
    use super::writer::BytesMutWriter;

    use flate2::write::{DeflateDecoder, DeflateEncoder};

    pub type Decoder = DeflateDecoder<BytesMutWriter>;
    pub type Encoder = DeflateEncoder<BytesMutWriter>;

    code_impl!(DeflateDecoder);
    code_impl!(DeflateEncoder);
}

pub use self::coder::{Code, Coder, FeaturedCode};
pub use self::coding::ContentEncoding;
pub use self::decode::try_decoder;
pub use self::encode::encoder;