summaryrefslogtreecommitdiff
path: root/src/lossless/lsb.rs
diff options
context:
space:
mode:
authorLibravatar Silas Bartha <silas@exvacuum.dev>2024-06-02 18:38:25 -0400
committerLibravatar Silas Bartha <silas@exvacuum.dev>2024-06-02 18:38:25 -0400
commited31c4201834719b2b75dca7cad1637abf1a80bb (patch)
tree3650833ed6d8c458bef9efff18f3585f4ccf17d9 /src/lossless/lsb.rs
parentccb19be9d0e918070029f31a86c7eb546121bd87 (diff)
Made codecs object-safe and deal with bytes only
Diffstat (limited to 'src/lossless/lsb.rs')
-rw-r--r--src/lossless/lsb.rs65
1 files changed, 24 insertions, 41 deletions
diff --git a/src/lossless/lsb.rs b/src/lossless/lsb.rs
index 59dcc0b..f8f1113 100644
--- a/src/lossless/lsb.rs
+++ b/src/lossless/lsb.rs
@@ -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
- },
-}