use crate::{
cilassembly::{ChangeRefRc, CilAssembly},
metadata::{
tables::{FieldPtrRaw, TableDataOwned, TableId},
token::Token,
},
Error, Result,
};
#[derive(Debug, Clone)]
pub struct FieldPtrBuilder {
field: Option<u32>,
}
impl FieldPtrBuilder {
#[must_use]
pub fn new() -> Self {
Self { field: None }
}
#[must_use]
pub fn field(mut self, field: u32) -> Self {
self.field = Some(field);
self
}
pub fn build(self, assembly: &mut CilAssembly) -> Result<ChangeRefRc> {
let field = self.field.ok_or_else(|| {
Error::ModificationInvalid("Field RID is required for FieldPtr".to_string())
})?;
let field_ptr = FieldPtrRaw {
rid: 0,
token: Token::new(0),
offset: 0,
field,
};
assembly.table_row_add(TableId::FieldPtr, TableDataOwned::FieldPtr(field_ptr))
}
}
impl Default for FieldPtrBuilder {
fn default() -> Self {
Self::new()
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{
cilassembly::ChangeRefKind, test::factories::table::assemblyref::get_test_assembly,
};
#[test]
fn test_fieldptr_builder_new() {
let builder = FieldPtrBuilder::new();
assert!(builder.field.is_none());
}
#[test]
fn test_fieldptr_builder_default() {
let builder = FieldPtrBuilder::default();
assert!(builder.field.is_none());
}
#[test]
fn test_fieldptr_builder_basic() -> Result<()> {
let mut assembly = get_test_assembly()?;
let ref_ = FieldPtrBuilder::new()
.field(1)
.build(&mut assembly)
.expect("Should build successfully");
assert_eq!(ref_.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
Ok(())
}
#[test]
fn test_fieldptr_builder_reordering() -> Result<()> {
let mut assembly = get_test_assembly()?;
let ref_ = FieldPtrBuilder::new()
.field(10) .build(&mut assembly)
.expect("Should build successfully");
assert_eq!(ref_.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
Ok(())
}
#[test]
fn test_fieldptr_builder_missing_field() -> Result<()> {
let mut assembly = get_test_assembly()?;
let result = FieldPtrBuilder::new().build(&mut assembly);
assert!(result.is_err());
match result.unwrap_err() {
Error::ModificationInvalid(details) => {
assert!(details.contains("Field RID is required"));
}
_ => panic!("Expected ModificationInvalid error"),
}
Ok(())
}
#[test]
fn test_fieldptr_builder_clone() {
let builder = FieldPtrBuilder::new().field(5);
let cloned = builder.clone();
assert_eq!(builder.field, cloned.field);
}
#[test]
fn test_fieldptr_builder_debug() {
let builder = FieldPtrBuilder::new().field(8);
let debug_str = format!("{builder:?}");
assert!(debug_str.contains("FieldPtrBuilder"));
assert!(debug_str.contains("field"));
}
#[test]
fn test_fieldptr_builder_fluent_interface() -> Result<()> {
let mut assembly = get_test_assembly()?;
let ref_ = FieldPtrBuilder::new()
.field(25)
.build(&mut assembly)
.expect("Should build successfully");
assert_eq!(ref_.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
Ok(())
}
#[test]
fn test_fieldptr_builder_multiple_builds() -> Result<()> {
let mut assembly = get_test_assembly()?;
let ref1 = FieldPtrBuilder::new()
.field(10)
.build(&mut assembly)
.expect("Should build first pointer");
let ref2 = FieldPtrBuilder::new()
.field(5)
.build(&mut assembly)
.expect("Should build second pointer");
let ref3 = FieldPtrBuilder::new()
.field(15)
.build(&mut assembly)
.expect("Should build third pointer");
assert!(!std::sync::Arc::ptr_eq(&ref1, &ref2));
assert!(!std::sync::Arc::ptr_eq(&ref2, &ref3));
assert_eq!(ref1.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
assert_eq!(ref2.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
assert_eq!(ref3.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
Ok(())
}
#[test]
fn test_fieldptr_builder_large_field_rid() -> Result<()> {
let mut assembly = get_test_assembly()?;
let ref_ = FieldPtrBuilder::new()
.field(0xFFFF) .build(&mut assembly)
.expect("Should handle large field RID");
assert_eq!(ref_.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
Ok(())
}
#[test]
fn test_fieldptr_builder_field_ordering_scenario() -> Result<()> {
let mut assembly = get_test_assembly()?;
let logical_to_physical = [(1, 3), (2, 1), (3, 2)];
let mut refs = Vec::new();
for (logical_idx, physical_field) in logical_to_physical {
let ref_ = FieldPtrBuilder::new()
.field(physical_field)
.build(&mut assembly)
.expect("Should build field pointer");
refs.push((logical_idx, ref_));
}
for (i, (logical_idx, ref_)) in refs.iter().enumerate() {
assert_eq!(*logical_idx, i + 1);
assert_eq!(ref_.kind(), ChangeRefKind::TableRow(TableId::FieldPtr));
}
Ok(())
}
#[test]
fn test_fieldptr_builder_zero_field() -> Result<()> {
let mut assembly = get_test_assembly()?;
let result = FieldPtrBuilder::new().field(0).build(&mut assembly);
assert!(result.is_ok());
Ok(())
}
}