diff --git a/src/lib.rs b/src/lib.rs index 462f476..359637e 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -475,7 +475,7 @@ where } let meta_size = - ObjectHeader::byte_count::() + 4 + (blocks + 1) * M::object_location_bytes(); + ObjectHeader::byte_count::() + 8 + (blocks + 1) * M::object_location_bytes(); match self .blocks @@ -770,6 +770,12 @@ impl ObjectMover for DataObject { &old_object_reader.path_hash.to_le_bytes(), ) .await?; + meta_writer + .write( + &mut storage.medium, + &old_object_reader.directory_hash.to_le_bytes(), + ) + .await?; let filename_location = if old_object_reader.filename_location == object.location() { destination diff --git a/src/ll/objects.rs b/src/ll/objects.rs index c7bd642..a7293af 100644 --- a/src/ll/objects.rs +++ b/src/ll/objects.rs @@ -12,6 +12,7 @@ use crate::{ pub enum ObjectType { FileMetadata = 0x8F, FileData = 0x8E, + DirectoryMetadata = 0x8D, } impl ObjectType { @@ -19,6 +20,7 @@ impl ObjectType { match byte { v if v == Self::FileMetadata as u8 => Some(Self::FileMetadata), v if v == Self::FileData as u8 => Some(Self::FileData), + v if v == Self::DirectoryMetadata as u8 => Some(Self::DirectoryMetadata), _ => None, } } @@ -431,9 +433,9 @@ impl ObjectHeader { pub struct MetadataObjectHeader { pub object: ObjectHeader, pub path_hash: u32, + pub directory_hash: u32, pub filename_location: ObjectLocation, data_object_cursor: usize, // Used to iterate through the list of object locations. - _parent: Option, _medium: PhantomData, } @@ -444,7 +446,7 @@ impl MetadataObjectHeader { ) -> Result, StorageError> { log::trace!("MetadataObjectHeader::next_object_location()"); - let data_offset = 4 + M::object_location_bytes(); // path hash + filename location + let data_offset = 4 + 4 + M::object_location_bytes(); // path hash + folder hash + filename location if self.data_object_cursor >= self .object @@ -774,13 +776,23 @@ impl ObjectInfo { ) .await?; + let mut directory_hash_bytes = [0; 4]; + let directory_hash_offset = path_hash_offset + 4; + medium + .read( + self.location().block, + directory_hash_offset, + &mut directory_hash_bytes, + ) + .await?; + let mut filename_location_bytes = [0; 8]; let filename_location_bytes = &mut filename_location_bytes[0..M::object_location_bytes()]; medium .read( self.location().block, - path_hash_offset + 4, + directory_hash_offset + 4, filename_location_bytes, ) .await?; @@ -788,9 +800,9 @@ impl ObjectInfo { Ok(MetadataObjectHeader { object: self.header, path_hash: u32::from_le_bytes(path_hash_bytes), + directory_hash: u32::from_le_bytes(directory_hash_bytes), filename_location: ObjectLocation::from_bytes::(filename_location_bytes), data_object_cursor: 0, - _parent: None, _medium: PhantomData, }) } diff --git a/src/writer.rs b/src/writer.rs index fe963d6..a27e4d6 100644 --- a/src/writer.rs +++ b/src/writer.rs @@ -164,12 +164,14 @@ where ) -> Result<(), StorageError> { log::debug!("Writer::create(path = {:?})", path); + // TODO: lift this? we need to look up the directory if path.contains(&['/', '\\'][..]) { log::warn!("Path contains invalid characters"); return Err(StorageError::InvalidOperation); } let path_hash = hash_path(path); + let directory_hash = hash_path(""); // TODO let est_page_count = 1 + storage.estimate_data_chunks(op.estimate_length())?; // this is mutable because we can fail mid-writing. 4 bytes to store the path hash @@ -199,6 +201,9 @@ where meta_writer .write(&mut storage.medium, &path_hash.to_le_bytes()) .await?; + meta_writer + .write(&mut storage.medium, &directory_hash.to_le_bytes()) + .await?; storage .write_location(&mut meta_writer, filename_location)