pub mod deserializer { use log::info; use crate::artworkdb::aobjects::{ADataSet, ADatabase, AImageItem, AImageName, SecondTypeTag, ThirdTypeTag}; use crate::artworkdb::objects::{ChunkHeader, ChunkType, DataSet, Database, ImageFile, ImageItem, ImageName, LocationTag}; enum ChunkState { Header, Data } pub fn parse_bytes(data: &[u8]) -> ADatabase { let mut adb = ADatabase {data: None, header: None, children: Vec::new() }; let mut state = ChunkState::Header; let mut chunk_header: Option = None; let mut last_type: u32 = 0; let mut i = 0; while i < data.len() { state = match state { ChunkState::Header => { if i + 12 >= data.len() { break; } chunk_header = Some(bincode::deserialize(&data[i..i + 12]).unwrap()); i += 12; ChunkState::Data }, ChunkState::Data => { let mut u = 0; let header = chunk_header.unwrap(); match ChunkType::from(header.chunk_type) { ChunkType::ArtworkDB => { u = usize::try_from(header.end_of_chunk).unwrap() - 12; let db: Database = bincode::deserialize(&data[i..i+u]).unwrap(); info!("val: {:?}", db); adb.data = Some(db); adb.header = Some(header); }, ChunkType::DataSet => { u = usize::try_from(header.end_of_chunk).unwrap() - 12; let ds: DataSet = bincode::deserialize(&data[i..i + u]).unwrap(); last_type = ds.data_type; adb.children.push(ADataSet { header, data: ds, child: Vec::new()}); }, ChunkType::ImageList | ChunkType::AlbumList | ChunkType::FileList => { u = usize::try_from(header.end_of_chunk).unwrap() - 12; }, ChunkType::ImageItem => { info!("ImageItem tag"); u = usize::try_from(header.end_of_chunk).unwrap() - 12; let ai: ImageItem = bincode::deserialize(&data[i..i+u]).unwrap(); let images = &mut adb.find_dataset(last_type).child; images.push(AImageItem { data: Some(ai), tag: Vec::new(), file: None }); }, ChunkType::FileImage => { u = usize::try_from(header.end_of_chunk).unwrap() - 12; let ai: ImageFile = bincode::deserialize(&data[i..i+u]).unwrap(); let images = &mut adb.find_dataset(last_type).child; images.push(AImageItem { data: None, tag: Vec::new(), file: Some(ai) }); }, ChunkType::LocationTag => { u = usize::try_from(header.end_of_chunk).unwrap() - 12; let ds: LocationTag = bincode::deserialize(&data[i..i + u]).unwrap(); let images = &mut adb.find_dataset(last_type).child; let mut str = None; match ds.tag_type { 3 => { let mut bytes = Vec::new(); let str_end = u32::from_le_bytes(data[i+12..i+16].try_into().unwrap()) as usize; let mut h = i+24; u += 12 + str_end; while h < i+24+str_end { bytes.push(u16::from_le_bytes(data[h..h+2].try_into().unwrap())); h+=2; } let g = String::from_utf16(&bytes).unwrap(); info!("str: {}", g); str = Some(g); images.last_mut().unwrap().tag.last_mut().unwrap().child.as_mut().unwrap().tag = Some(ThirdTypeTag { data: ds, str }); }, 2 => { images.last_mut().unwrap().tag.push(SecondTypeTag { data: ds, str, child: None }); }, _ => { u = usize::try_from(header.children_count).unwrap() - 12; } } }, ChunkType::ImageName => { info!("ImageName tag"); u = usize::try_from(header.end_of_chunk).unwrap() - 12; let ds: ImageName = bincode::deserialize(&data[i..i + u]).unwrap(); let images = &mut adb.find_dataset(last_type).child; images.last_mut().unwrap().tag.last_mut().unwrap().child = Some(AImageName { iname: ds, tag: None }); }, _ => { u = 1; info!("Unknown stuff happened {:X?}", header.chunk_type.to_vec()); } } i += u; chunk_header = None; ChunkState::Header } } } adb } } pub mod serializer { use log::info; use crate::artworkdb::aobjects::ADatabase; use crate::artworkdb::objects::{ChunkHeader, ChunkType}; pub fn to_bytes(adb: ADatabase) -> Vec { let mut bytes: Vec = Vec::new(); for i in 0..(adb.children.len()) { let data_set = adb.children.get(i).unwrap(); let mut entry_bytes = Vec::new(); match data_set.data.data_type { 1 => { entry_bytes.append(&mut generate_header_raw(ChunkType::ImageList, 80, data_set.child.len())); entry_bytes.append(&mut [0; 80].to_vec()); } // Image List _ => { info!("Unknown data_set type!"); } } for img in data_set.child.iter() { let mut args = Vec::new(); if img.data.is_some() { let mut item = Vec::new(); item.append(&mut generate_header(ChunkType::ImageItem, 140, args.len())); item.append(&mut bincode::serialize(img.data.as_ref().unwrap()).unwrap()); item.append(&mut [0; 100].to_vec()); entry_bytes.append(&mut item); for o in &img.tag { let mut data = [2u32.to_le_bytes(), 0u32.to_le_bytes(), 0u32.to_le_bytes()].concat(); entry_bytes.append(&mut generate_header(ChunkType::LocationTag, 12, 0)); entry_bytes.append(&mut data); if let Some(name) = &o.child { let mut nb = Vec::new(); let arg = name.tag.as_ref().unwrap(); let mut str_b = string_to_ipod16(arg.str.as_ref().unwrap()); str_b = [3u32.to_le_bytes().to_vec(), 0u32.to_le_bytes().to_vec(), 0u32.to_le_bytes().to_vec(), (str_b.len() as u32).to_le_bytes().to_vec(), 2u32.to_le_bytes().to_vec(), 0u32.to_le_bytes().to_vec(), str_b ].concat(); nb.append(&mut generate_header(ChunkType::LocationTag, 12, str_b.len())); nb.append(&mut str_b); entry_bytes.append(&mut generate_header(ChunkType::ImageName, 76, nb.len())); entry_bytes.append(&mut bincode::serialize(&name.iname).unwrap()); entry_bytes.append(&mut nb); } } } if img.file.is_some() { entry_bytes.append(&mut generate_header(ChunkType::FileImage, 12, 0)); entry_bytes.append(&mut bincode::serialize(img.file.as_ref().unwrap()).unwrap()); } entry_bytes.append(&mut args); } bytes.append(&mut generate_header(ChunkType::DataSet, 84, entry_bytes.len())); bytes.append(&mut bincode::serialize(&data_set.data).unwrap()); bytes.append(&mut [0; 80].to_vec()); bytes.append(&mut entry_bytes); } bytes } fn string_to_ipod16(str: &str) -> Vec { str.encode_utf16().flat_map(|f| [f as u8, (f >> 8) as u8]).collect() } fn generate_header(ct: ChunkType, header_size: usize, data_len: usize) -> Vec { let header_size = 12 + header_size as u32; let header = ChunkHeader{ chunk_type: ct.into(), end_of_chunk: header_size, children_count: header_size + data_len as u32}; bincode::serialize(&header).unwrap() } fn generate_header_raw(ct: ChunkType, header_size: usize, child_cnt: usize) -> Vec { let header_size = 12 + header_size as u32; let header = ChunkHeader{ chunk_type: ct.into(), end_of_chunk: header_size, children_count: child_cnt as u32}; bincode::serialize(&header).unwrap() } } pub mod aobjects { use crate::artworkdb::objects::{ChunkHeader, DataSet, Database, ImageFile, ImageItem, ImageName, LocationTag}; #[derive(Debug, serde::Serialize)] pub struct ADatabase { pub header: Option, pub data: Option, pub children: Vec } impl ADatabase { pub(crate) fn find_dataset(&mut self, p0: u32) -> &mut ADataSet { self.children.iter_mut().find(|d| d.data.data_type == p0).unwrap() } } #[derive(Debug, serde::Serialize)] pub struct ADataSet { pub header: ChunkHeader, pub data: DataSet, pub child: Vec } #[derive(Debug, serde::Serialize)] pub struct AImageItem { pub tag: Vec, pub data: Option, pub file: Option, } #[derive(Debug, serde::Serialize)] pub struct AImageName { pub iname: ImageName, pub tag: Option, } #[derive(Debug, serde::Serialize)] pub struct SecondTypeTag { pub data: LocationTag, pub str: Option, pub child: Option } #[derive(Debug, serde::Serialize)] pub struct ThirdTypeTag { pub data: LocationTag, pub str: Option } } pub mod objects { use serde::{Deserialize, Serialize}; pub enum ChunkType { ArtworkDB, DataSet, ImageList, AlbumList, FileList, ImageItem, LocationTag, ImageName, FileImage, Unknown, } impl From<[u8; 4]> for ChunkType { fn from(value: [u8; 4]) -> Self { match value { [0x6D, 0x68, 0x66, 0x64] => ChunkType::ArtworkDB, [0x6D, 0x68, 0x73, 0x64] => ChunkType::DataSet, [0x6D, 0x68, 0x6C, 0x69] => ChunkType::ImageList, [0x6D, 0x68, 0x69, 0x69] => ChunkType::ImageItem, [0x6D, 0x68, 0x6F, 0x64] => ChunkType::LocationTag, [0x6D, 0x68, 0x6E, 0x69] => ChunkType::ImageName, [0x6D, 0x68, 0x6C, 0x61] => ChunkType::AlbumList, [0x6D, 0x68, 0x6C, 0x66] => ChunkType::FileList, [0x6D, 0x68, 0x69, 0x66] => ChunkType::FileImage, _ => ChunkType::Unknown, } } } impl From for [u8; 4] { fn from(value: ChunkType) -> Self { match value { ChunkType::ArtworkDB => [0x6D, 0x68, 0x66, 0x64], ChunkType::DataSet => [0x6D, 0x68, 0x73, 0x64], ChunkType::Unknown => [0x00, 0x00, 0x00, 0x00], ChunkType::ImageList => [0x6D, 0x68, 0x6C, 0x69], ChunkType::ImageItem => [0x6D, 0x68, 0x69, 0x69], ChunkType::LocationTag => [0x6D, 0x68, 0x6F, 0x64], ChunkType::ImageName => [0x6D, 0x68, 0x6E, 0x69], ChunkType::AlbumList => [0x6D, 0x68, 0x6C, 0x61], ChunkType::FileList => [0x6D, 0x68, 0x6C, 0x66], ChunkType::FileImage => [0x6D, 0x68, 0x69, 0x66], } } } #[derive(Serialize, Deserialize, PartialEq, Debug, Clone, Copy)] pub struct ChunkHeader { pub chunk_type: [u8; 4], pub end_of_chunk: u32, pub children_count: u32, } #[derive(Serialize, Deserialize, PartialEq, Debug)] pub struct Database { unknown1: u32, unknown2: u32, number_of_children: u32, unknown3: u32, next_id_for_mhii: u32, unknown5: u64, unknown6: u64, unknown7: u32, unknown8: u32, unknown9: u32, unknown10: u64 } #[derive(Serialize, Deserialize, PartialEq, Debug, Clone, Copy)] pub struct DataSet { pub data_type: u32, } #[derive(Serialize, Deserialize, PartialEq, Debug, Clone, Copy)] pub struct LocationTag { pub tag_type: u32, unk1: u32, unk2: u32, } #[derive(Serialize, Deserialize, PartialEq, Debug)] pub struct ImageItem { number_of_children: u32, id: u32, song_dbid: u64, unknown4: u32, rating: u32, unknown6: u32, original_date: u32, digitized_date: u32, source_image_size: u32 } #[derive(Serialize, Deserialize, PartialEq, Debug)] pub struct ImageFile { unknown1: u32, correlation_id: u32, image_size: u32 } #[derive(Serialize, Deserialize, PartialEq, Debug)] pub struct ImageName { number_of_children: u32, correlation_id: u32, ithmb_offset: u32, image_size: u32, vertical_padding: u16, horizontal_padding: u16, image_height: u16, image_width: u16, unknown: u32, image_size_n: u32, } }