Made codecs object-safe and deal with bytes only

This commit is contained in:
Silas Bartha 2024-06-02 18:38:25 -04:00
parent ccb19be9d0
commit ed31c42018
Signed by: soaos
GPG Key ID: 9BD3DCC0D56A09B2
5 changed files with 64 additions and 94 deletions

7
Cargo.lock generated
View File

@ -8,6 +8,12 @@ version = "1.0.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe"
[[package]]
name = "anyhow"
version = "1.0.86"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b3d1d046238990b9cf5bcde22a3fb3584ee5cf65fb2765f454ed428c7a0063da"
[[package]] [[package]]
name = "autocfg" name = "autocfg"
version = "1.3.0" version = "1.3.0"
@ -252,6 +258,7 @@ dependencies = [
name = "occule" name = "occule"
version = "0.1.0" version = "0.1.0"
dependencies = [ dependencies = [
"anyhow",
"image", "image",
"img-parts", "img-parts",
"thiserror", "thiserror",

View File

@ -1,6 +1,6 @@
[package] [package]
name = "occule" name = "occule"
version = "0.1.0" version = "0.2.0"
edition = "2021" edition = "2021"
[features] [features]
@ -9,7 +9,7 @@ jpeg = ["dep:img-parts"]
lossless = ["dep:image"] lossless = ["dep:image"]
[dependencies] [dependencies]
thiserror = "1.0.61" thiserror = "^1.0"
[dependencies.img-parts] [dependencies.img-parts]
version = "0.3.0" version = "0.3.0"

View File

@ -1,26 +1,28 @@
use thiserror::Error;
/// Codecs enable the concealment of payload data inside the data of a carrier. /// Codecs enable the concealment of payload data inside the data of a carrier.
pub trait Codec { pub trait Codec {
/// Data type representing the carrier.
type Carrier;
/// Data type representing the payload.
type Payload;
/// Data type representing encoder output/decoder input (usually the same as the carrier).
type Output;
/// Type of errors produced by this codec.
type Error;
/// Embeds payload data inside carrier, returning the result. /// Embeds payload data inside carrier, returning the result.
fn encode<C, P>(&self, carrier: C, payload: P) -> Result<Self::Output, Self::Error> fn encode(&self, carrier: &[u8], payload: &[u8]) -> Result<Vec<u8>, CodecError>;
where
C: Into<Self::Carrier>,
P: Into<Self::Payload>;
/// Extracts payload data from an encoded carrier, returning the carrier with data removed and the /// Extracts payload data from an encoded carrier, returning the carrier with data removed and the
/// payload data. /// payload data.
fn decode<E>(&self, encoded: E) -> Result<(Self::Carrier, Self::Payload), Self::Error> fn decode(&self, encoded: &[u8]) -> Result<(Vec<u8>, Vec<u8>), CodecError>;
where }
E: Into<Self::Output>;
/// Errors produced by a codec
#[derive(Debug, Error)]
pub enum CodecError {
/// Variant used when data is determined not to be encoded. Note that a codec may have no way
/// of knowing this, so this may not be returned even if the data was not encoded
#[error("Data was not encoded with this codec")]
DataNotEncoded,
/// Variant used when data is invalid in some way. Allows a message string for further context
#[error("Provided data invalid: {0}")]
DataInvalid(String),
/// Variant used when some dependency, such as a file load, fails
#[error("Error occured in dependency: {0}")]
DependencyError(String),
} }

View File

@ -1,9 +1,8 @@
use std::{mem::size_of, usize}; use std::{mem::size_of, usize};
use img_parts::jpeg::{markers, Jpeg, JpegSegment}; use img_parts::jpeg::{markers, Jpeg, JpegSegment};
use thiserror::Error;
use crate::codec::Codec; use crate::{codec::Codec, CodecError};
/// Codec for storing payload data in JPEG comment (COM) segments. Can store an arbitrary amount of /// Codec for storing payload data in JPEG comment (COM) segments. Can store an arbitrary amount of
/// data, as long as the number of comment segments does not exceed u64::MAX. /// data, as long as the number of comment segments does not exceed u64::MAX.
@ -14,21 +13,13 @@ pub struct JpegSegmentCodec {
} }
impl Codec for JpegSegmentCodec { impl Codec for JpegSegmentCodec {
type Carrier = Vec<u8>; fn encode(&self, carrier: &[u8], payload: &[u8]) -> Result<Vec<u8>, CodecError>
type Payload = Vec<u8>;
type Output = Self::Carrier;
type Error = JpegSegmentError;
fn encode<C, P>(&self, carrier: C, payload: P) -> Result<Self::Output, Self::Error>
where
C: Into<Self::Carrier>,
P: Into<Self::Payload>,
{ {
let mut jpeg = match Jpeg::from_bytes(carrier.into().into()) { let mut jpeg = match Jpeg::from_bytes(carrier.to_vec().into()) {
Ok(image) => image, Ok(v) => v,
Err(err) => return Err(JpegSegmentError::ParseFailed { inner: err }) Err(e) => return Err(CodecError::DependencyError(e.to_string()))
}; };
let mut payload_bytes: Self::Carrier = payload.into(); let mut payload_bytes = payload.to_vec();
let segment_count = ((payload_bytes.len() + size_of::<u64>()) as u64).div_ceil((u16::MAX as usize - size_of::<u16>()) as u64); let segment_count = ((payload_bytes.len() + size_of::<u64>()) as u64).div_ceil((u16::MAX as usize - size_of::<u16>()) as u64);
payload_bytes.splice(0..0, segment_count.to_le_bytes()); payload_bytes.splice(0..0, segment_count.to_le_bytes());
for (index, payload_chunk) in payload_bytes.chunks(u16::MAX as usize - size_of::<u16>()).enumerate() { for (index, payload_chunk) in payload_bytes.chunks(u16::MAX as usize - size_of::<u16>()).enumerate() {
@ -38,13 +29,11 @@ impl Codec for JpegSegmentCodec {
Ok(jpeg.encoder().bytes().to_vec()) Ok(jpeg.encoder().bytes().to_vec())
} }
fn decode<E>(&self, encoded: E) -> Result<(Self::Carrier, Self::Payload), Self::Error> fn decode(&self, encoded: &[u8]) -> Result<(Vec<u8>, Vec<u8>), CodecError>
where
E: Into<Self::Output>,
{ {
let mut jpeg = match Jpeg::from_bytes(encoded.into().into()) { let mut jpeg = match Jpeg::from_bytes(encoded.to_vec().into()) {
Ok(image) => image, Ok(v) => v,
Err(err) => return Err(JpegSegmentError::ParseFailed { inner: err }) Err(e) => return Err(CodecError::DependencyError(e.to_string()))
}; };
let segment = jpeg.segments_mut().remove(self.start_index); let segment = jpeg.segments_mut().remove(self.start_index);
let segment_bytes = segment.contents(); let segment_bytes = segment.contents();
@ -68,14 +57,3 @@ impl Default for JpegSegmentCodec {
} }
} }
} }
/// Errors thrown by the JPEG segment codec.
#[derive(Error, Debug)]
pub enum JpegSegmentError {
/// Parsing JPEG data failed.
#[error("Failed to parse JPEG data: {inner:?}")]
ParseFailed {
/// Error thrown by parser.
inner: img_parts::Error,
}
}

View File

@ -1,33 +1,25 @@
use std::cmp::Ordering; use std::{cmp::Ordering, io::{BufWriter, Cursor}};
use image::{ColorType, DynamicImage, GenericImageView, Pixel}; use image::{DynamicImage, GenericImageView, Pixel};
use thiserror::Error;
use crate::codec::Codec; use crate::{codec::Codec, CodecError};
/// Least-significant bit (LSB) steganography encodes data in the least-significant bits of colors /// Least-significant bit (LSB) steganography encodes data in the least-significant bits of colors
/// in an image. This implementation reduces the colors in the carrier (irreversibly) in order to /// in an image. This implementation reduces the colors in the carrier (irreversibly) in order to
/// allow a byte of data to fit in each pixel of the image. 3 bits of data are encoded per pixel, /// allow a byte of data to fit in each pixel of the image. 3 bits of data are encoded per pixel,
/// and the 9th bit is used to signal the end of data. /// and the 9th bit is used to signal the end of data.
#[derive(Debug)] #[derive(Debug, Default)]
pub struct LsbCodec; pub struct LsbCodec;
impl Codec for LsbCodec { impl Codec for LsbCodec {
type Carrier = DynamicImage; fn encode(&self, carrier: &[u8], payload: &[u8]) -> Result<Vec<u8>, CodecError>
type Payload = Vec<u8>;
type Output = Self::Carrier;
type Error = LsbError;
fn encode<C, P>(&self, carrier: C, payload: P) -> Result<Self::Output, Self::Error>
where
C: Into<Self::Carrier>,
P: Into<Self::Payload>,
{ {
let mut image: DynamicImage = carrier.into(); let image_format = image::guess_format(carrier.into()).unwrap();
let payload: Vec<u8> = payload.into(); let mut image: DynamicImage = image::load_from_memory(carrier.into()).unwrap();
let payload: &[u8] = payload.into();
if image.pixels().count() < payload.len() { if image.pixels().count() < payload.len() {
return Err(LsbError::PayloadTooBig); return Err(CodecError::DataInvalid("Payload Too Big for Carrier".into()));
} }
let mut payload_iter = payload.iter(); let mut payload_iter = payload.iter();
@ -51,17 +43,20 @@ impl Codec for LsbCodec {
} }
} }
}, },
_ => return Err(LsbError::UnsupportedFormat { format: image.color() }) _ => return Err(CodecError::DataInvalid("Unsupported Image Color Format".into()))
} }
Ok(image) let mut buf = BufWriter::new(Cursor::new(Vec::<u8>::new()));
if let Err(e) = image.write_to(&mut buf, image_format) {
return Err(CodecError::DependencyError(e.to_string()))
}
Ok(buf.into_inner().unwrap().into_inner())
} }
fn decode<E>(&self, carrier: E) -> Result<(Self::Carrier, Self::Payload), LsbError> fn decode(&self, carrier: &[u8]) -> Result<(Vec<u8>, Vec<u8>), CodecError>
where
E: Into<Self::Output>,
{ {
let mut image: DynamicImage = carrier.into(); let image_format = image::guess_format(carrier.into()).unwrap();
let mut image: DynamicImage = image::load_from_memory(carrier.into()).unwrap();
let mut payload: Vec<u8> = Vec::new(); let mut payload: Vec<u8> = Vec::new();
match image { match image {
@ -83,10 +78,14 @@ impl Codec for LsbCodec {
} }
} }
}, },
_ => return Err(LsbError::UnsupportedFormat { format: image.color() }) _ => return Err(CodecError::DataInvalid("Unsupported Image Color Format".into()))
} }
Ok((image, payload)) let mut buf = BufWriter::new(Cursor::new(Vec::<u8>::new()));
if let Err(e) = image.write_to(&mut buf, image_format) {
return Err(CodecError::DependencyError(e.to_string()))
}
Ok((buf.into_inner().unwrap().into_inner(), payload))
} }
} }
@ -138,19 +137,3 @@ fn decode_pixel<P: Pixel<Subpixel = u8>>(pixel: &mut P) -> Option<u8> {
} }
Some(payload_byte) Some(payload_byte)
} }
/// Errors thrown by the LSB Codec.
#[derive(Error, Debug)]
pub enum LsbError {
/// Error thrown when payload is too big for the carrier.
#[error("Payload is too big for the carrier. Choose a smaller payload or an image with greater pixel dimensions.")]
PayloadTooBig,
/// Error thrown when pixel format is unsupported.
#[error("Specified image format ({format:?}) is unsupported.")]
UnsupportedFormat {
/// Provided (invalid) format.
format: ColorType
},
}