Made codecs object-safe and deal with bytes only
This commit is contained in:
parent
ccb19be9d0
commit
ed31c42018
7
Cargo.lock
generated
7
Cargo.lock
generated
@ -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",
|
||||||
|
@ -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"
|
||||||
|
40
src/codec.rs
40
src/codec.rs
@ -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),
|
||||||
}
|
}
|
||||||
|
@ -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,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -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
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user