mirror of
https://github.com/denoland/deno.git
synced 2025-01-18 03:44:05 -05:00
140 lines
3.9 KiB
Rust
140 lines
3.9 KiB
Rust
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use deno_core::error::type_error;
|
|
use deno_core::error::AnyError;
|
|
use deno_core::op2;
|
|
use flate2::write::DeflateDecoder;
|
|
use flate2::write::DeflateEncoder;
|
|
use flate2::write::GzDecoder;
|
|
use flate2::write::GzEncoder;
|
|
use flate2::write::ZlibDecoder;
|
|
use flate2::write::ZlibEncoder;
|
|
use flate2::Compression;
|
|
use std::cell::RefCell;
|
|
use std::io::Write;
|
|
|
|
#[derive(Debug)]
|
|
struct CompressionResource(RefCell<Option<Inner>>);
|
|
|
|
impl deno_core::GarbageCollected for CompressionResource {}
|
|
|
|
/// https://wicg.github.io/compression/#supported-formats
|
|
#[derive(Debug)]
|
|
enum Inner {
|
|
DeflateDecoder(ZlibDecoder<Vec<u8>>),
|
|
DeflateEncoder(ZlibEncoder<Vec<u8>>),
|
|
DeflateRawDecoder(DeflateDecoder<Vec<u8>>),
|
|
DeflateRawEncoder(DeflateEncoder<Vec<u8>>),
|
|
GzDecoder(GzDecoder<Vec<u8>>),
|
|
GzEncoder(GzEncoder<Vec<u8>>),
|
|
}
|
|
|
|
#[op2]
|
|
#[cppgc]
|
|
pub fn op_compression_new(
|
|
#[string] format: &str,
|
|
is_decoder: bool,
|
|
) -> Result<CompressionResource, AnyError> {
|
|
let w = Vec::new();
|
|
let inner = match (format, is_decoder) {
|
|
("deflate", true) => Inner::DeflateDecoder(ZlibDecoder::new(w)),
|
|
("deflate", false) => {
|
|
Inner::DeflateEncoder(ZlibEncoder::new(w, Compression::default()))
|
|
}
|
|
("deflate-raw", true) => Inner::DeflateRawDecoder(DeflateDecoder::new(w)),
|
|
("deflate-raw", false) => {
|
|
Inner::DeflateRawEncoder(DeflateEncoder::new(w, Compression::default()))
|
|
}
|
|
("gzip", true) => Inner::GzDecoder(GzDecoder::new(w)),
|
|
("gzip", false) => {
|
|
Inner::GzEncoder(GzEncoder::new(w, Compression::default()))
|
|
}
|
|
_ => return Err(type_error("Unsupported format")),
|
|
};
|
|
Ok(CompressionResource(RefCell::new(Some(inner))))
|
|
}
|
|
|
|
#[op2]
|
|
#[buffer]
|
|
pub fn op_compression_write(
|
|
#[cppgc] resource: &CompressionResource,
|
|
#[anybuffer] input: &[u8],
|
|
) -> Result<Vec<u8>, AnyError> {
|
|
let mut inner = resource.0.borrow_mut();
|
|
let inner = inner
|
|
.as_mut()
|
|
.ok_or_else(|| type_error("resource is closed"))?;
|
|
let out: Vec<u8> = match &mut *inner {
|
|
Inner::DeflateDecoder(d) => {
|
|
d.write_all(input).map_err(|e| type_error(e.to_string()))?;
|
|
d.flush()?;
|
|
d.get_mut().drain(..)
|
|
}
|
|
Inner::DeflateEncoder(d) => {
|
|
d.write_all(input).map_err(|e| type_error(e.to_string()))?;
|
|
d.flush()?;
|
|
d.get_mut().drain(..)
|
|
}
|
|
Inner::DeflateRawDecoder(d) => {
|
|
d.write_all(input).map_err(|e| type_error(e.to_string()))?;
|
|
d.flush()?;
|
|
d.get_mut().drain(..)
|
|
}
|
|
Inner::DeflateRawEncoder(d) => {
|
|
d.write_all(input).map_err(|e| type_error(e.to_string()))?;
|
|
d.flush()?;
|
|
d.get_mut().drain(..)
|
|
}
|
|
Inner::GzDecoder(d) => {
|
|
d.write_all(input).map_err(|e| type_error(e.to_string()))?;
|
|
d.flush()?;
|
|
d.get_mut().drain(..)
|
|
}
|
|
Inner::GzEncoder(d) => {
|
|
d.write_all(input).map_err(|e| type_error(e.to_string()))?;
|
|
d.flush()?;
|
|
d.get_mut().drain(..)
|
|
}
|
|
}
|
|
.collect();
|
|
Ok(out)
|
|
}
|
|
|
|
#[op2]
|
|
#[buffer]
|
|
pub fn op_compression_finish(
|
|
#[cppgc] resource: &CompressionResource,
|
|
report_errors: bool,
|
|
) -> Result<Vec<u8>, AnyError> {
|
|
let inner = resource
|
|
.0
|
|
.borrow_mut()
|
|
.take()
|
|
.ok_or_else(|| type_error("resource is closed"))?;
|
|
let out = match inner {
|
|
Inner::DeflateDecoder(d) => {
|
|
d.finish().map_err(|e| type_error(e.to_string()))
|
|
}
|
|
Inner::DeflateEncoder(d) => {
|
|
d.finish().map_err(|e| type_error(e.to_string()))
|
|
}
|
|
Inner::DeflateRawDecoder(d) => {
|
|
d.finish().map_err(|e| type_error(e.to_string()))
|
|
}
|
|
Inner::DeflateRawEncoder(d) => {
|
|
d.finish().map_err(|e| type_error(e.to_string()))
|
|
}
|
|
Inner::GzDecoder(d) => d.finish().map_err(|e| type_error(e.to_string())),
|
|
Inner::GzEncoder(d) => d.finish().map_err(|e| type_error(e.to_string())),
|
|
};
|
|
match out {
|
|
Err(err) => {
|
|
if report_errors {
|
|
Err(err)
|
|
} else {
|
|
Ok(Vec::with_capacity(0))
|
|
}
|
|
}
|
|
Ok(out) => Ok(out),
|
|
}
|
|
}
|