use crate::{
metadata::tables::{
assembly::AssemblyRaw,
types::{RowWritable, TableInfoRef},
},
utils::{write_le_at, write_le_at_dyn},
Result,
};
impl RowWritable for AssemblyRaw {
fn row_write(
&self,
data: &mut [u8],
offset: &mut usize,
_rid: u32,
sizes: &TableInfoRef,
) -> Result<()> {
write_le_at(data, offset, self.hash_alg_id)?;
write_le_at(
data,
offset,
u16::try_from(self.major_version).map_err(|_| {
malformed_error!(
"Assembly major version out of range: {}",
self.major_version
)
})?,
)?;
write_le_at(
data,
offset,
u16::try_from(self.minor_version).map_err(|_| {
malformed_error!(
"Assembly minor version out of range: {}",
self.minor_version
)
})?,
)?;
write_le_at(
data,
offset,
u16::try_from(self.build_number).map_err(|_| {
malformed_error!("Assembly build number out of range: {}", self.build_number)
})?,
)?;
write_le_at(
data,
offset,
u16::try_from(self.revision_number).map_err(|_| {
malformed_error!(
"Assembly revision number out of range: {}",
self.revision_number
)
})?,
)?;
write_le_at(data, offset, self.flags)?;
write_le_at_dyn(data, offset, self.public_key, sizes.is_large_blob())?;
write_le_at_dyn(data, offset, self.name, sizes.is_large_str())?;
write_le_at_dyn(data, offset, self.culture, sizes.is_large_str())?;
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::metadata::{
tables::types::{RowReadable, TableInfo, TableRow},
token::Token,
};
#[test]
fn test_round_trip_serialization_small_heaps() {
let original_row = AssemblyRaw {
rid: 1,
token: Token::new(0x20000001),
offset: 0,
hash_alg_id: 0x01010101,
major_version: 0x0202,
minor_version: 0x0303,
build_number: 0x0404,
revision_number: 0x0505,
flags: 0x06060606,
public_key: 0x0707,
name: 0x0808,
culture: 0x0909,
};
let table_info = std::sync::Arc::new(TableInfo::new_test(&[], false, false, false));
let row_size = <AssemblyRaw as TableRow>::row_size(&table_info) as usize;
let mut buffer = vec![0u8; row_size];
let mut offset = 0;
original_row
.row_write(&mut buffer, &mut offset, 1, &table_info)
.expect("Serialization should succeed");
let mut read_offset = 0;
let deserialized_row = AssemblyRaw::row_read(&buffer, &mut read_offset, 1, &table_info)
.expect("Deserialization should succeed");
assert_eq!(original_row.hash_alg_id, deserialized_row.hash_alg_id);
assert_eq!(original_row.major_version, deserialized_row.major_version);
assert_eq!(original_row.minor_version, deserialized_row.minor_version);
assert_eq!(original_row.build_number, deserialized_row.build_number);
assert_eq!(
original_row.revision_number,
deserialized_row.revision_number
);
assert_eq!(original_row.flags, deserialized_row.flags);
assert_eq!(original_row.public_key, deserialized_row.public_key);
assert_eq!(original_row.name, deserialized_row.name);
assert_eq!(original_row.culture, deserialized_row.culture);
assert_eq!(offset, row_size, "Offset should match expected row size");
}
#[test]
fn test_round_trip_serialization_large_heaps() {
let original_row = AssemblyRaw {
rid: 1,
token: Token::new(0x20000001),
offset: 0,
hash_alg_id: 0x01010101,
major_version: 0x0202,
minor_version: 0x0303,
build_number: 0x0404,
revision_number: 0x0505,
flags: 0x06060606,
public_key: 0x07070707,
name: 0x08080808,
culture: 0x09090909,
};
let table_info = std::sync::Arc::new(TableInfo::new_test(&[], true, true, true));
let row_size = <AssemblyRaw as TableRow>::row_size(&table_info) as usize;
let mut buffer = vec![0u8; row_size];
let mut offset = 0;
original_row
.row_write(&mut buffer, &mut offset, 1, &table_info)
.expect("Serialization should succeed");
let mut read_offset = 0;
let deserialized_row = AssemblyRaw::row_read(&buffer, &mut read_offset, 1, &table_info)
.expect("Deserialization should succeed");
assert_eq!(original_row.hash_alg_id, deserialized_row.hash_alg_id);
assert_eq!(original_row.major_version, deserialized_row.major_version);
assert_eq!(original_row.minor_version, deserialized_row.minor_version);
assert_eq!(original_row.build_number, deserialized_row.build_number);
assert_eq!(
original_row.revision_number,
deserialized_row.revision_number
);
assert_eq!(original_row.flags, deserialized_row.flags);
assert_eq!(original_row.public_key, deserialized_row.public_key);
assert_eq!(original_row.name, deserialized_row.name);
assert_eq!(original_row.culture, deserialized_row.culture);
assert_eq!(offset, row_size, "Offset should match expected row size");
}
#[test]
fn test_known_binary_format_small_heaps() {
let assembly_row = AssemblyRaw {
rid: 1,
token: Token::new(0x20000001),
offset: 0,
hash_alg_id: 0x01010101,
major_version: 0x0202,
minor_version: 0x0303,
build_number: 0x0404,
revision_number: 0x0505,
flags: 0x06060606,
public_key: 0x0707,
name: 0x0808,
culture: 0x0909,
};
let table_info = std::sync::Arc::new(TableInfo::new_test(&[], false, false, false));
let mut buffer = vec![0u8; <AssemblyRaw as TableRow>::row_size(&table_info) as usize];
let mut offset = 0;
assembly_row
.row_write(&mut buffer, &mut offset, 1, &table_info)
.expect("Serialization should succeed");
let expected = vec![
0x01, 0x01, 0x01, 0x01, 0x02, 0x02, 0x03, 0x03, 0x04, 0x04, 0x05, 0x05, 0x06, 0x06, 0x06, 0x06, 0x07, 0x07, 0x08, 0x08, 0x09, 0x09, ];
assert_eq!(
buffer, expected,
"Binary output should match expected format"
);
}
#[test]
fn test_known_binary_format_large_heaps() {
let assembly_row = AssemblyRaw {
rid: 1,
token: Token::new(0x20000001),
offset: 0,
hash_alg_id: 0x01010101,
major_version: 0x0202,
minor_version: 0x0303,
build_number: 0x0404,
revision_number: 0x0505,
flags: 0x06060606,
public_key: 0x07070707,
name: 0x08080808,
culture: 0x09090909,
};
let table_info = std::sync::Arc::new(TableInfo::new_test(&[], true, true, true));
let mut buffer = vec![0u8; <AssemblyRaw as TableRow>::row_size(&table_info) as usize];
let mut offset = 0;
assembly_row
.row_write(&mut buffer, &mut offset, 1, &table_info)
.expect("Serialization should succeed");
let expected = vec![
0x01, 0x01, 0x01, 0x01, 0x02, 0x02, 0x03, 0x03, 0x04, 0x04, 0x05, 0x05, 0x06, 0x06, 0x06, 0x06, 0x07, 0x07, 0x07, 0x07, 0x08, 0x08, 0x08, 0x08, 0x09, 0x09, 0x09, 0x09, ];
assert_eq!(
buffer, expected,
"Binary output should match expected format"
);
}
#[test]
fn test_row_size_calculation() {
let table_info_small = std::sync::Arc::new(TableInfo::new_test(&[], false, false, false));
let small_size = <AssemblyRaw as TableRow>::row_size(&table_info_small);
assert_eq!(small_size, 4 + 2 + 2 + 2 + 2 + 4 + 2 + 2 + 2);
let table_info_large = std::sync::Arc::new(TableInfo::new_test(&[], true, true, true));
let large_size = <AssemblyRaw as TableRow>::row_size(&table_info_large);
assert_eq!(large_size, 4 + 2 + 2 + 2 + 2 + 4 + 4 + 4 + 4); }
}