0.1.20 upd

This commit is contained in:
Michael Wain 2025-02-15 19:23:52 +03:00
parent e1e8d0a12c
commit b7bc68933e
3 changed files with 216 additions and 2 deletions

View File

@ -1,6 +1,6 @@
[package]
name = "itunesdb"
version = "0.1.19"
version = "0.1.20"
edition = "2021"
authors = ["alterwain"]

214
src/artworkdb.rs Normal file
View File

@ -0,0 +1,214 @@
pub mod deserializer {
use log::info;
use crate::artworkdb::aobjects::{ADataSet, ADatabase, AImageItem, ASomeList};
use crate::artworkdb::objects::{ChunkHeader, ChunkType, DataSet, Database, ImageItem, ImageName};
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<ChunkHeader> = 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: match ds.data_type {
_ => ASomeList::Images(Vec::new()), // 1
}});
},
ChunkType::ImageList => {
u = usize::try_from(header.end_of_chunk).unwrap() - 12;
},
ChunkType::ImageItem => {
u = usize::try_from(header.end_of_chunk).unwrap() - 12;
let ai: ImageItem = bincode::deserialize(&data[i..i+u]).unwrap();
if let ASomeList::Images(images) = &mut adb.find_dataset(1).child {
images.push(AImageItem { data: ai, name: None });
}
},
ChunkType::LocationTag => {
u = usize::try_from(header.end_of_chunk).unwrap() - 12;
},
ChunkType::ImageName => {
u = usize::try_from(header.end_of_chunk).unwrap() - 12;
let ds: ImageName = bincode::deserialize(&data[i..i + u]).unwrap();
if let ASomeList::Images(images) = &mut adb.find_dataset(1).child {
images.last_mut().unwrap().name = Some(ds);
}
},
_ => { u = 1; info!("Unknown stuff happened"); }
}
i += u;
chunk_header = None;
ChunkState::Header
}
}
}
adb
}
}
pub mod aobjects {
use crate::artworkdb::objects::{ChunkHeader, DataSet, Database, ImageItem, ImageName};
#[derive(Debug, serde::Serialize)]
pub struct ADatabase {
pub header: Option<ChunkHeader>,
pub data: Option<Database>,
pub children: Vec<ADataSet>
}
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: ASomeList
}
#[derive(Debug, serde::Serialize)]
pub enum ASomeList {
Images(Vec<AImageItem>), // 1
Albums, // 2
Files, // 3
}
#[derive(Debug, serde::Serialize)]
pub struct AImageItem {
pub data: ImageItem,
pub name: Option<ImageName>,
}
}
pub mod objects {
use serde::{Deserialize, Serialize};
pub enum ChunkType {
ArtworkDB,
DataSet,
ImageList,
ImageItem,
LocationTag,
ImageName,
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,
_ => ChunkType::Unknown,
}
}
}
impl From<ChunkType> 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],
}
}
}
#[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 {
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 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,
}
}

View File

@ -3,7 +3,7 @@ pub mod objects;
pub mod xobjects;
pub mod deserializer;
pub mod serializer;
mod artworkdb;
/*
[lib]
crate-type = ["staticlib", "cdylib", "lib"]