domain: add Media Catalog ports and MetadataResolver service

This commit is contained in:
2026-05-31 03:27:48 +02:00
parent 147206d8a5
commit ccb61b72d7
9 changed files with 139 additions and 1 deletions

View File

@@ -0,0 +1,10 @@
use async_trait::async_trait;
use crate::{entities::{AssetMetadata, MetadataSource}, errors::DomainError, value_objects::SystemId};
#[async_trait]
pub trait AssetMetadataRepository: Send + Sync {
async fn find_by_asset(&self, asset_id: &SystemId) -> Result<Vec<AssetMetadata>, DomainError>;
async fn find_by_asset_and_source(&self, asset_id: &SystemId, source: MetadataSource) -> Result<Option<AssetMetadata>, DomainError>;
async fn save(&self, metadata: &AssetMetadata) -> Result<(), DomainError>;
async fn delete_by_asset_and_source(&self, asset_id: &SystemId, source: MetadataSource) -> Result<(), DomainError>;
}

View File

@@ -0,0 +1,11 @@
use async_trait::async_trait;
use crate::{entities::Asset, errors::DomainError, value_objects::{Checksum, SystemId}};
#[async_trait]
pub trait AssetRepository: Send + Sync {
async fn find_by_id(&self, id: &SystemId) -> Result<Option<Asset>, DomainError>;
async fn find_by_checksum(&self, checksum: &Checksum) -> Result<Vec<Asset>, DomainError>;
async fn find_by_owner(&self, owner_id: &SystemId, limit: u32, offset: u32) -> Result<Vec<Asset>, DomainError>;
async fn save(&self, asset: &Asset) -> Result<(), DomainError>;
async fn delete(&self, id: &SystemId) -> Result<(), DomainError>;
}

View File

@@ -0,0 +1,10 @@
use async_trait::async_trait;
use crate::{entities::AssetStack, errors::DomainError, value_objects::SystemId};
#[async_trait]
pub trait AssetStackRepository: Send + Sync {
async fn find_by_id(&self, id: &SystemId) -> Result<Option<AssetStack>, DomainError>;
async fn find_by_asset(&self, asset_id: &SystemId) -> Result<Vec<AssetStack>, DomainError>;
async fn save(&self, stack: &AssetStack) -> Result<(), DomainError>;
async fn delete(&self, id: &SystemId) -> Result<(), DomainError>;
}

View File

@@ -0,0 +1,10 @@
use async_trait::async_trait;
use crate::{entities::{DerivativeAsset, DerivativeProfile}, errors::DomainError, value_objects::SystemId};
#[async_trait]
pub trait DerivativeRepository: Send + Sync {
async fn find_by_asset(&self, asset_id: &SystemId) -> Result<Vec<DerivativeAsset>, DomainError>;
async fn find_by_asset_and_profile(&self, asset_id: &SystemId, profile: DerivativeProfile) -> Result<Option<DerivativeAsset>, DomainError>;
async fn save(&self, derivative: &DerivativeAsset) -> Result<(), DomainError>;
async fn delete(&self, id: &SystemId) -> Result<(), DomainError>;
}

View File

@@ -0,0 +1,10 @@
use async_trait::async_trait;
use crate::{entities::DuplicateGroup, errors::DomainError, value_objects::SystemId};
#[async_trait]
pub trait DuplicateRepository: Send + Sync {
async fn find_by_id(&self, id: &SystemId) -> Result<Option<DuplicateGroup>, DomainError>;
async fn find_unresolved(&self) -> Result<Vec<DuplicateGroup>, DomainError>;
async fn find_by_asset(&self, asset_id: &SystemId) -> Result<Vec<DuplicateGroup>, DomainError>;
async fn save(&self, group: &DuplicateGroup) -> Result<(), DomainError>;
}

View File

@@ -9,6 +9,11 @@ mod library_path_repo;
mod ingest_session_repo;
mod quota_repo;
mod file_storage;
mod asset_repo;
mod asset_metadata_repo;
mod asset_stack_repo;
mod derivative_repo;
mod duplicate_repo;
pub use auth::{PasswordHasher, TokenIssuer};
pub use event_publisher::EventPublisher;
@@ -21,3 +26,8 @@ pub use library_path_repo::LibraryPathRepository;
pub use ingest_session_repo::IngestSessionRepository;
pub use quota_repo::{QuotaRepository, UsageLedgerRepository};
pub use file_storage::{FileEntry, FileStoragePort};
pub use asset_repo::AssetRepository;
pub use asset_metadata_repo::AssetMetadataRepository;
pub use asset_stack_repo::AssetStackRepository;
pub use derivative_repo::DerivativeRepository;
pub use duplicate_repo::DuplicateRepository;

View File

@@ -1 +1,32 @@
// Metadata resolver — will be implemented in Task 9
use crate::entities::{AssetMetadata, MetadataSource};
use crate::value_objects::{MetadataValue, StructuredData};
/// Merge metadata layers by priority: ExifExtracted < AiGenerated < UserEdited.
/// Later (higher-priority) layers overwrite earlier ones.
pub fn resolve_metadata(layers: &[AssetMetadata]) -> StructuredData {
let mut sorted = layers.to_vec();
sorted.sort_by_key(|l| l.metadata_source);
let mut result = StructuredData::new();
for layer in &sorted {
result.merge_from(layer.data.clone());
}
result
}
/// Return the source and value for a field from the highest-priority layer that contains it.
pub fn get_field_source<'a>(
layers: &'a [AssetMetadata],
field: &str,
) -> Option<(MetadataSource, &'a MetadataValue)> {
// Build index list sorted by metadata_source descending (highest priority first)
let mut indices: Vec<usize> = (0..layers.len()).collect();
indices.sort_by(|&a, &b| layers[b].metadata_source.cmp(&layers[a].metadata_source));
for i in indices {
if let Some(val) = layers[i].data.get(field) {
return Some((layers[i].metadata_source, val));
}
}
None
}