Initial Commit

This commit is contained in:
Silas Bartha 2024-05-27 11:18:36 -04:00
commit 16d1838e5b
Signed by: soaos
GPG Key ID: 9BD3DCC0D56A09B2
8 changed files with 247 additions and 0 deletions

1
.gitignore vendored Normal file
View File

@ -0,0 +1 @@
/target

16
Cargo.toml Normal file
View File

@ -0,0 +1,16 @@
[package]
name = "occule"
version = "0.1.0"
edition = "2021"
[features]
default = ["jpg", "png", "lossless"]
jpg = []
png = []
lossless = []
[dependencies]
colored = "2.1.0"
image = "0.24"
img-parts = "0.3.0"
thiserror = "1.0.61"

14
src/codec.rs Normal file
View File

@ -0,0 +1,14 @@
pub trait Codec {
type Carrier;
type Payload;
type Output;
type Error;
fn encode(
&self,
carrier: impl Into<Self::Carrier>,
payload: impl Into<Self::Payload>,
) -> Result<Self::Output, Self::Error>;
fn decode(&self, encoded: impl Into<Self::Output>) -> Result<(Self::Carrier, Self::Payload), Self::Error>;
}

2
src/jpg/mod.rs Normal file
View File

@ -0,0 +1,2 @@
mod segment;
pub use segment::*;

66
src/jpg/segment.rs Normal file
View File

@ -0,0 +1,66 @@
use std::{mem::size_of, usize};
use img_parts::jpeg::{markers, Jpeg, JpegSegment};
use thiserror::Error;
use crate::codec::Codec;
#[derive(Debug, PartialEq, Eq)]
pub struct JpegSegmentCodec {
pub start_index: usize,
}
impl Codec for JpegSegmentCodec {
type Carrier = Vec<u8>;
type Payload = Vec<u8>;
type Output = Self::Carrier;
type Error = JpegSegmentError;
fn encode(&self, carrier: impl Into<Self::Carrier>, payload: impl Into<Self::Payload>) -> Result<Self::Output, Self::Error> {
let mut jpeg = match Jpeg::from_bytes(carrier.into().into()) {
Ok(image) => image,
Err(err) => return Err(JpegSegmentError::ParseFailed { inner: err })
};
let mut payload_bytes: Self::Carrier = payload.into();
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() {
let segment = JpegSegment::new_with_contents(markers::COM, payload_chunk.to_vec().into());
jpeg.segments_mut().insert(self.start_index + index, segment);
}
Ok(jpeg.encoder().bytes().to_vec())
}
fn decode(&self, encoded: impl Into<Self::Output>) -> Result<(Self::Carrier, Self::Payload), Self::Error> {
let mut jpeg = match Jpeg::from_bytes(encoded.into().into()) {
Ok(image) => image,
Err(err) => return Err(JpegSegmentError::ParseFailed { inner: err })
};
let segment = jpeg.segments_mut().remove(self.start_index);
let segment_bytes = segment.contents();
let segment_count = u64::from_le_bytes(segment_bytes[0..size_of::<u64>()].try_into().unwrap()) as usize;
let mut payload_vec: Vec<u8> = Vec::with_capacity((u16::MAX as usize - size_of::<u16>()) * segment_count);
payload_vec.extend(segment_bytes[size_of::<u64>()..].to_vec());
for _ in 0..segment_count-1 {
let segment = jpeg.segments_mut().remove(self.start_index);
payload_vec.extend(segment.contents());
}
Ok((jpeg.encoder().bytes().to_vec(), payload_vec))
}
}
impl Default for JpegSegmentCodec {
fn default() -> Self {
Self {
start_index: 3,
}
}
}
#[derive(Error, Debug)]
pub enum JpegSegmentError {
#[error("Failed to parse JPEG data: {inner:?}")]
ParseFailed { inner: img_parts::Error }
}

7
src/lib.rs Normal file
View File

@ -0,0 +1,7 @@
pub mod codec;
#[cfg(feature = "jpg")]
pub mod jpg;
#[cfg(feature = "lossless")]
pub mod lossless;

139
src/lossless/champleve.rs Normal file
View File

@ -0,0 +1,139 @@
use std::cmp::Ordering;
use image::{ColorType, DynamicImage, GenericImageView, Pixel};
use thiserror::Error;
use crate::codec::Codec;
#[derive(Debug)]
pub struct ChampleveCodec;
impl Codec for ChampleveCodec {
type Carrier = DynamicImage;
type Payload = Vec<u8>;
type Output = Self::Carrier;
type Error = ChampleveError;
fn encode(&self, carrier: impl Into<Self::Carrier>, payload: impl Into<Self::Payload>) -> Result<Self::Output, Self::Error> {
let mut image: DynamicImage = carrier.into();
let payload: Vec<u8> = payload.into();
if image.pixels().count() < payload.len() {
return Err(ChampleveError::PayloadTooBig);
}
let mut payload_iter = payload.iter();
match image {
DynamicImage::ImageRgba8(ref mut image) => {
for pixel in image.pixels_mut() {
if let Some(payload_byte) = payload_iter.next() {
encode_pixel(pixel, *payload_byte, false);
} else {
encode_pixel(pixel, 0, true);
}
}
},
DynamicImage::ImageRgb8(ref mut image) => {
for pixel in image.pixels_mut() {
if let Some(payload_byte) = payload_iter.next() {
encode_pixel(pixel, *payload_byte, false);
} else {
encode_pixel(pixel, 0, true);
}
}
},
_ => return Err(ChampleveError::UnsupportedFormat { format: image.color() })
}
Ok(image)
}
fn decode(&self, carrier: impl Into<Self::Output>) -> Result<(Self::Carrier, Self::Payload), ChampleveError> {
let mut image: DynamicImage = carrier.into();
let mut payload: Vec<u8> = Vec::new();
match image {
DynamicImage::ImageRgba8(ref mut image) => {
for pixel in image.pixels_mut() {
if let Some(payload_byte) = decode_pixel(pixel) {
payload.push(payload_byte);
} else {
break;
}
}
},
DynamicImage::ImageRgb8(ref mut image) => {
for pixel in image.pixels_mut() {
if let Some(payload_byte) = decode_pixel(pixel) {
payload.push(payload_byte);
} else {
break;
}
}
},
_ => return Err(ChampleveError::UnsupportedFormat { format: image.color() })
}
Ok((image, payload))
}
}
fn encode_pixel<P: Pixel<Subpixel = u8>>(pixel: &mut P, payload_byte: u8, end_of_data: bool) {
let mut bits_remaining: i32 = 8;
for channel in pixel.channels_mut() {
*channel &= 0b11111000;
bits_remaining -= 3;
if bits_remaining <= -3 {
break;
}
let mask = match bits_remaining.cmp(&0) {
Ordering::Less => payload_byte << -bits_remaining,
_ => payload_byte >> bits_remaining,
} & 0b00000111;
*channel |= mask;
}
// Add end-of-data marker to final bit if necessary
if end_of_data {
*pixel.channels_mut().last_mut().unwrap() |= 1;
}
}
fn decode_pixel<P: Pixel<Subpixel = u8>>(pixel: &mut P) -> Option<u8> {
// Final bit as end-of-data marker
if pixel.channels().last().unwrap() & 1 == 1 {
return None;
}
let mut bits_remaining: i32 = 8;
let mut payload_byte: u8 = 0;
for channel in pixel.channels_mut() {
bits_remaining -= 3;
if bits_remaining <= -3 {
break;
}
let channel_bits = *channel & 0b00000111;
*channel &= 0b11111000;
let mask = match bits_remaining.cmp(&0) {
Ordering::Less => channel_bits >> -bits_remaining,
_ => channel_bits << bits_remaining,
};
payload_byte |= mask;
}
Some(payload_byte)
}
#[derive(Error, Debug)]
pub enum ChampleveError {
#[error("Payload is too big for the carrier. Choose a smaller payload or an image with greater pixel dimensions.")]
PayloadTooBig,
#[error("Specified image format ({format:?}) is unsupported.")]
UnsupportedFormat {
format: ColorType
},
}

2
src/lossless/mod.rs Normal file
View File

@ -0,0 +1,2 @@
mod champleve;
pub use champleve::*;