hpx 2.4.10

High Performance HTTP Client
Documentation
use std::io::{self, Read, Result, Write};

use boring::ssl::{CertificateCompressionAlgorithm, CertificateCompressor};
use brotli::{CompressorWriter, Decompressor};
use flate2::{Compression, read::ZlibDecoder, write::ZlibEncoder};
// use zstd::stream::{Decoder as ZstdDecoder, Encoder as ZstdEncoder};

#[derive(Debug, Clone, Default)]
#[non_exhaustive]
pub struct BrotliCertificateCompressor;

impl CertificateCompressor for BrotliCertificateCompressor {
    const ALGORITHM: CertificateCompressionAlgorithm = CertificateCompressionAlgorithm::BROTLI;
    const CAN_COMPRESS: bool = true;
    const CAN_DECOMPRESS: bool = true;

    fn compress<W>(&self, input: &[u8], output: &mut W) -> Result<()>
    where
        W: Write,
    {
        let mut writer = CompressorWriter::new(output, input.len(), 11, 22);
        writer.write_all(input)?;
        writer.flush()?;
        Ok(())
    }

    fn decompress<W>(&self, input: &[u8], output: &mut W) -> Result<()>
    where
        W: Write,
    {
        let mut reader = Decompressor::new(input, 4096);
        let mut buf = [0u8; 4096];
        loop {
            match reader.read(&mut buf[..]) {
                Err(e) => {
                    if let io::ErrorKind::Interrupted = e.kind() {
                        continue;
                    }
                    return Err(e);
                }
                Ok(size) => {
                    if size == 0 {
                        break;
                    }
                    output.write_all(&buf[..size])?;
                }
            }
        }
        Ok(())
    }
}

#[derive(Debug, Clone, Default)]
#[non_exhaustive]
pub struct ZlibCertificateCompressor;

impl CertificateCompressor for ZlibCertificateCompressor {
    const ALGORITHM: CertificateCompressionAlgorithm = CertificateCompressionAlgorithm::ZLIB;
    const CAN_COMPRESS: bool = true;
    const CAN_DECOMPRESS: bool = true;

    fn compress<W>(&self, input: &[u8], output: &mut W) -> Result<()>
    where
        W: Write,
    {
        let mut encoder = ZlibEncoder::new(output, Compression::default());
        encoder.write_all(input)?;
        encoder.finish()?;
        Ok(())
    }

    fn decompress<W>(&self, input: &[u8], output: &mut W) -> Result<()>
    where
        W: Write,
    {
        let mut decoder = ZlibDecoder::new(input);
        io::copy(&mut decoder, output)?;
        Ok(())
    }
}

// #[derive(Debug, Clone, Default)]
// #[non_exhaustive]
// pub struct ZstdCertificateCompressor;

// impl CertificateCompressor for ZstdCertificateCompressor {
//     const ALGORITHM: CertificateCompressionAlgorithm = CertificateCompressionAlgorithm::ZSTD;
//     const CAN_COMPRESS: bool = true;
//     const CAN_DECOMPRESS: bool = true;
//
//     fn compress<W>(&self, input: &[u8], output: &mut W) -> Result<()>
//     where
//         W: Write,
//     {
//         let mut writer = ZstdEncoder::new(output, 0)?;
//         writer.write_all(input)?;
//         writer.flush()?;
//         Ok(())
//     }
//
//     fn decompress<W>(&self, input: &[u8], output: &mut W) -> Result<()>
//     where
//         W: Write,
//     {
//         let mut reader = ZstdDecoder::new(input)?;
//         let mut buf = [0u8; 4096];
//         loop {
//             match reader.read(&mut buf[..]) {
//                 Err(e) => {
//                     if let io::ErrorKind::Interrupted = e.kind() {
//                         continue;
//                     }
//                     return Err(e);
//                 }
//                 Ok(size) => {
//                     if size == 0 {
//                         break;
//                     }
//                     output.write_all(&buf[..size])?;
//                 }
//             }
//         }
//         Ok(())
//     }
// }