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

View File

@ -1,6 +1,6 @@
[package]
name = "occule"
version = "0.1.0"
version = "0.2.0"
edition = "2021"
[features]
@ -9,7 +9,7 @@ jpeg = ["dep:img-parts"]
lossless = ["dep:image"]
[dependencies]
thiserror = "1.0.61"
thiserror = "^1.0"
[dependencies.img-parts]
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.
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.
fn encode<C, P>(&self, carrier: C, payload: P) -> Result<Self::Output, Self::Error>
where
C: Into<Self::Carrier>,
P: Into<Self::Payload>;
fn encode(&self, carrier: &[u8], payload: &[u8]) -> Result<Vec<u8>, CodecError>;
/// Extracts payload data from an encoded carrier, returning the carrier with data removed and the
/// payload data.
fn decode<E>(&self, encoded: E) -> Result<(Self::Carrier, Self::Payload), Self::Error>
where
E: Into<Self::Output>;
fn decode(&self, encoded: &[u8]) -> Result<(Vec<u8>, Vec<u8>), CodecError>;
}
/// 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 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
/// 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 {
type Carrier = Vec<u8>;
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>,
fn encode(&self, carrier: &[u8], payload: &[u8]) -> Result<Vec<u8>, CodecError>
{
let mut jpeg = match Jpeg::from_bytes(carrier.into().into()) {
Ok(image) => image,
Err(err) => return Err(JpegSegmentError::ParseFailed { inner: err })
let mut jpeg = match Jpeg::from_bytes(carrier.to_vec().into()) {
Ok(v) => v,
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);
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() {
@ -38,13 +29,11 @@ impl Codec for JpegSegmentCodec {
Ok(jpeg.encoder().bytes().to_vec())
}
fn decode<E>(&self, encoded: E) -> Result<(Self::Carrier, Self::Payload), Self::Error>
where
E: Into<Self::Output>,
fn decode(&self, encoded: &[u8]) -> Result<(Vec<u8>, Vec<u8>), CodecError>
{
let mut jpeg = match Jpeg::from_bytes(encoded.into().into()) {
Ok(image) => image,
Err(err) => return Err(JpegSegmentError::ParseFailed { inner: err })
let mut jpeg = match Jpeg::from_bytes(encoded.to_vec().into()) {
Ok(v) => v,
Err(e) => return Err(CodecError::DependencyError(e.to_string()))
};
let segment = jpeg.segments_mut().remove(self.start_index);
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 thiserror::Error;
use image::{DynamicImage, GenericImageView, Pixel};
use crate::codec::Codec;
use crate::{codec::Codec, CodecError};
/// 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
/// 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.
#[derive(Debug)]
#[derive(Debug, Default)]
pub struct LsbCodec;
impl Codec for LsbCodec {
type Carrier = DynamicImage;
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>,
fn encode(&self, carrier: &[u8], payload: &[u8]) -> Result<Vec<u8>, CodecError>
{
let mut image: DynamicImage = carrier.into();
let payload: Vec<u8> = payload.into();
let image_format = image::guess_format(carrier.into()).unwrap();
let mut image: DynamicImage = image::load_from_memory(carrier.into()).unwrap();
let payload: &[u8] = payload.into();
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();
@ -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>
where
E: Into<Self::Output>,
fn decode(&self, carrier: &[u8]) -> Result<(Vec<u8>, Vec<u8>), CodecError>
{
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();
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)
}
/// 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
},
}