app: add organization + sharing commands/queries
This commit is contained in:
1
Cargo.lock
generated
1
Cargo.lock
generated
@@ -91,6 +91,7 @@ dependencies = [
|
||||
"anyhow",
|
||||
"async-trait",
|
||||
"bytes",
|
||||
"chrono",
|
||||
"domain",
|
||||
"serde",
|
||||
"thiserror",
|
||||
|
||||
@@ -12,3 +12,6 @@ uuid = { workspace = true }
|
||||
tokio = { workspace = true }
|
||||
bytes = { workspace = true }
|
||||
serde = { workspace = true }
|
||||
|
||||
[dev-dependencies]
|
||||
chrono = { workspace = true }
|
||||
|
||||
@@ -0,0 +1,43 @@
|
||||
use std::sync::Arc;
|
||||
use domain::{
|
||||
entities::Album,
|
||||
errors::DomainError,
|
||||
ports::AlbumRepository,
|
||||
value_objects::SystemId,
|
||||
};
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub enum AlbumAction {
|
||||
Add { asset_id: SystemId },
|
||||
Remove { asset_id: SystemId },
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub struct ManageAlbumEntriesCommand {
|
||||
pub album_id: SystemId,
|
||||
pub action: AlbumAction,
|
||||
pub user_id: SystemId,
|
||||
}
|
||||
|
||||
pub struct ManageAlbumEntriesHandler {
|
||||
album_repo: Arc<dyn AlbumRepository>,
|
||||
}
|
||||
|
||||
impl ManageAlbumEntriesHandler {
|
||||
pub fn new(album_repo: Arc<dyn AlbumRepository>) -> Self {
|
||||
Self { album_repo }
|
||||
}
|
||||
|
||||
pub async fn execute(&self, cmd: ManageAlbumEntriesCommand) -> Result<Album, DomainError> {
|
||||
let mut album = self.album_repo.find_by_id(&cmd.album_id).await?
|
||||
.ok_or_else(|| DomainError::NotFound(format!("Album {} not found", cmd.album_id)))?;
|
||||
|
||||
match cmd.action {
|
||||
AlbumAction::Add { asset_id } => album.add_asset(asset_id, cmd.user_id)?,
|
||||
AlbumAction::Remove { asset_id } => album.remove_asset(&asset_id)?,
|
||||
}
|
||||
|
||||
self.album_repo.save(&album).await?;
|
||||
Ok(album)
|
||||
}
|
||||
}
|
||||
@@ -1,3 +1,7 @@
|
||||
pub mod create_album;
|
||||
pub mod manage_album_entries;
|
||||
pub mod tag_asset;
|
||||
|
||||
pub use create_album::{CreateAlbumCommand, CreateAlbumHandler};
|
||||
pub use manage_album_entries::{AlbumAction, ManageAlbumEntriesCommand, ManageAlbumEntriesHandler};
|
||||
pub use tag_asset::{TagAssetCommand, TagAssetHandler};
|
||||
|
||||
44
crates/application/src/organization/commands/tag_asset.rs
Normal file
44
crates/application/src/organization/commands/tag_asset.rs
Normal file
@@ -0,0 +1,44 @@
|
||||
use std::sync::Arc;
|
||||
use domain::{
|
||||
entities::{AssetTag, Tag},
|
||||
errors::DomainError,
|
||||
ports::{AssetRepository, TagRepository},
|
||||
value_objects::SystemId,
|
||||
};
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub struct TagAssetCommand {
|
||||
pub asset_id: SystemId,
|
||||
pub tag_name: String,
|
||||
pub user_id: SystemId,
|
||||
}
|
||||
|
||||
pub struct TagAssetHandler {
|
||||
asset_repo: Arc<dyn AssetRepository>,
|
||||
tag_repo: Arc<dyn TagRepository>,
|
||||
}
|
||||
|
||||
impl TagAssetHandler {
|
||||
pub fn new(asset_repo: Arc<dyn AssetRepository>, tag_repo: Arc<dyn TagRepository>) -> Self {
|
||||
Self { asset_repo, tag_repo }
|
||||
}
|
||||
|
||||
pub async fn execute(&self, cmd: TagAssetCommand) -> Result<(Tag, AssetTag), DomainError> {
|
||||
self.asset_repo.find_by_id(&cmd.asset_id).await?
|
||||
.ok_or_else(|| DomainError::NotFound(format!("Asset {} not found", cmd.asset_id)))?;
|
||||
|
||||
let tag = match self.tag_repo.find_by_name(&cmd.tag_name).await? {
|
||||
Some(existing) => existing,
|
||||
None => {
|
||||
let new_tag = Tag::new_manual(&cmd.tag_name);
|
||||
self.tag_repo.save_tag(&new_tag).await?;
|
||||
new_tag
|
||||
}
|
||||
};
|
||||
|
||||
let asset_tag = AssetTag::new_manual(cmd.asset_id, tag.tag_id, cmd.user_id);
|
||||
self.tag_repo.save_asset_tag(&asset_tag).await?;
|
||||
|
||||
Ok((tag, asset_tag))
|
||||
}
|
||||
}
|
||||
@@ -1,3 +1,7 @@
|
||||
pub mod commands;
|
||||
pub mod queries;
|
||||
|
||||
pub use commands::{CreateAlbumCommand, CreateAlbumHandler};
|
||||
pub use commands::{AlbumAction, ManageAlbumEntriesCommand, ManageAlbumEntriesHandler};
|
||||
pub use commands::{TagAssetCommand, TagAssetHandler};
|
||||
pub use queries::get_album::{GetAlbumQuery, GetAlbumHandler};
|
||||
|
||||
27
crates/application/src/organization/queries/get_album.rs
Normal file
27
crates/application/src/organization/queries/get_album.rs
Normal file
@@ -0,0 +1,27 @@
|
||||
use std::sync::Arc;
|
||||
use domain::{
|
||||
entities::Album,
|
||||
errors::DomainError,
|
||||
ports::AlbumRepository,
|
||||
value_objects::SystemId,
|
||||
};
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub struct GetAlbumQuery {
|
||||
pub album_id: SystemId,
|
||||
}
|
||||
|
||||
pub struct GetAlbumHandler {
|
||||
album_repo: Arc<dyn AlbumRepository>,
|
||||
}
|
||||
|
||||
impl GetAlbumHandler {
|
||||
pub fn new(album_repo: Arc<dyn AlbumRepository>) -> Self {
|
||||
Self { album_repo }
|
||||
}
|
||||
|
||||
pub async fn execute(&self, query: GetAlbumQuery) -> Result<Album, DomainError> {
|
||||
self.album_repo.find_by_id(&query.album_id).await?
|
||||
.ok_or_else(|| DomainError::NotFound(format!("Album {} not found", query.album_id)))
|
||||
}
|
||||
}
|
||||
1
crates/application/src/organization/queries/mod.rs
Normal file
1
crates/application/src/organization/queries/mod.rs
Normal file
@@ -0,0 +1 @@
|
||||
pub mod get_album;
|
||||
@@ -0,0 +1,40 @@
|
||||
use std::sync::Arc;
|
||||
use domain::{
|
||||
entities::{LinkAccessLevel, ScopeType, ShareLink, ShareScope, ShareableType},
|
||||
errors::DomainError,
|
||||
ports::ShareRepository,
|
||||
value_objects::{DateTimeStamp, SystemId},
|
||||
};
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub struct GenerateShareLinkCommand {
|
||||
pub shareable_type: ShareableType,
|
||||
pub shareable_id: SystemId,
|
||||
pub access_level: LinkAccessLevel,
|
||||
pub created_by: SystemId,
|
||||
pub expires_at: Option<DateTimeStamp>,
|
||||
pub max_uses: Option<u32>,
|
||||
}
|
||||
|
||||
pub struct GenerateShareLinkHandler {
|
||||
share_repo: Arc<dyn ShareRepository>,
|
||||
}
|
||||
|
||||
impl GenerateShareLinkHandler {
|
||||
pub fn new(share_repo: Arc<dyn ShareRepository>) -> Self {
|
||||
Self { share_repo }
|
||||
}
|
||||
|
||||
pub async fn execute(&self, cmd: GenerateShareLinkCommand) -> Result<(ShareScope, ShareLink), DomainError> {
|
||||
let scope = ShareScope::new(ScopeType::Link, cmd.shareable_type, cmd.shareable_id, cmd.created_by);
|
||||
let token = uuid::Uuid::new_v4().to_string();
|
||||
let mut link = ShareLink::new(scope.scope_id, token, cmd.access_level);
|
||||
link.expires_at = cmd.expires_at;
|
||||
link.max_uses = cmd.max_uses;
|
||||
|
||||
self.share_repo.save_scope(&scope).await?;
|
||||
self.share_repo.save_link(&link).await?;
|
||||
|
||||
Ok((scope, link))
|
||||
}
|
||||
}
|
||||
7
crates/application/src/sharing/commands/mod.rs
Normal file
7
crates/application/src/sharing/commands/mod.rs
Normal file
@@ -0,0 +1,7 @@
|
||||
pub mod share_resource;
|
||||
pub mod generate_share_link;
|
||||
pub mod revoke_share;
|
||||
|
||||
pub use share_resource::{ShareResourceCommand, ShareResourceHandler};
|
||||
pub use generate_share_link::{GenerateShareLinkCommand, GenerateShareLinkHandler};
|
||||
pub use revoke_share::{RevokeShareCommand, RevokeShareHandler};
|
||||
39
crates/application/src/sharing/commands/revoke_share.rs
Normal file
39
crates/application/src/sharing/commands/revoke_share.rs
Normal file
@@ -0,0 +1,39 @@
|
||||
use std::sync::Arc;
|
||||
use domain::{
|
||||
errors::DomainError,
|
||||
events::DomainEvent,
|
||||
ports::{EventPublisher, ShareRepository},
|
||||
value_objects::{DateTimeStamp, SystemId},
|
||||
};
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub struct RevokeShareCommand {
|
||||
pub scope_id: SystemId,
|
||||
pub revoked_by: SystemId,
|
||||
}
|
||||
|
||||
pub struct RevokeShareHandler {
|
||||
share_repo: Arc<dyn ShareRepository>,
|
||||
event_pub: Arc<dyn EventPublisher>,
|
||||
}
|
||||
|
||||
impl RevokeShareHandler {
|
||||
pub fn new(share_repo: Arc<dyn ShareRepository>, event_pub: Arc<dyn EventPublisher>) -> Self {
|
||||
Self { share_repo, event_pub }
|
||||
}
|
||||
|
||||
pub async fn execute(&self, cmd: RevokeShareCommand) -> Result<(), DomainError> {
|
||||
self.share_repo.find_scope_by_id(&cmd.scope_id).await?
|
||||
.ok_or_else(|| DomainError::NotFound(format!("Share scope {} not found", cmd.scope_id)))?;
|
||||
|
||||
self.share_repo.delete_scope(&cmd.scope_id).await?;
|
||||
|
||||
self.event_pub.publish(DomainEvent::ShareRevoked {
|
||||
scope_id: cmd.scope_id,
|
||||
revoked_by: cmd.revoked_by,
|
||||
timestamp: DateTimeStamp::now(),
|
||||
}).await?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
51
crates/application/src/sharing/commands/share_resource.rs
Normal file
51
crates/application/src/sharing/commands/share_resource.rs
Normal file
@@ -0,0 +1,51 @@
|
||||
use std::sync::Arc;
|
||||
use domain::{
|
||||
entities::{ScopeType, ShareScope, ShareTarget, ShareableType, TargetType},
|
||||
errors::DomainError,
|
||||
events::DomainEvent,
|
||||
ports::{EventPublisher, ShareRepository},
|
||||
value_objects::{DateTimeStamp, SystemId},
|
||||
};
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub struct ShareResourceCommand {
|
||||
pub shareable_type: ShareableType,
|
||||
pub shareable_id: SystemId,
|
||||
pub target_type: TargetType,
|
||||
pub target_id: SystemId,
|
||||
pub role_id: SystemId,
|
||||
pub created_by: SystemId,
|
||||
}
|
||||
|
||||
pub struct ShareResourceHandler {
|
||||
share_repo: Arc<dyn ShareRepository>,
|
||||
event_pub: Arc<dyn EventPublisher>,
|
||||
}
|
||||
|
||||
impl ShareResourceHandler {
|
||||
pub fn new(share_repo: Arc<dyn ShareRepository>, event_pub: Arc<dyn EventPublisher>) -> Self {
|
||||
Self { share_repo, event_pub }
|
||||
}
|
||||
|
||||
pub async fn execute(&self, cmd: ShareResourceCommand) -> Result<(ShareScope, ShareTarget), DomainError> {
|
||||
let scope_type = match cmd.target_type {
|
||||
TargetType::User => ScopeType::User,
|
||||
TargetType::Group => ScopeType::Group,
|
||||
};
|
||||
|
||||
let scope = ShareScope::new(scope_type, cmd.shareable_type, cmd.shareable_id, cmd.created_by);
|
||||
let target = ShareTarget::new(scope.scope_id, cmd.target_type, cmd.target_id, cmd.role_id);
|
||||
|
||||
self.share_repo.save_scope(&scope).await?;
|
||||
self.share_repo.save_target(&target).await?;
|
||||
|
||||
self.event_pub.publish(DomainEvent::ShareCreated {
|
||||
scope_id: scope.scope_id,
|
||||
shareable_id: cmd.shareable_id,
|
||||
created_by: cmd.created_by,
|
||||
timestamp: DateTimeStamp::now(),
|
||||
}).await?;
|
||||
|
||||
Ok((scope, target))
|
||||
}
|
||||
}
|
||||
@@ -1 +1,7 @@
|
||||
// Sharing commands/queries (future: CreateShareLink, ManageAccess, etc.)
|
||||
pub mod commands;
|
||||
pub mod queries;
|
||||
|
||||
pub use commands::{ShareResourceCommand, ShareResourceHandler};
|
||||
pub use commands::{GenerateShareLinkCommand, GenerateShareLinkHandler};
|
||||
pub use commands::{RevokeShareCommand, RevokeShareHandler};
|
||||
pub use queries::access_shared_resource::{AccessSharedResourceQuery, AccessSharedResourceHandler};
|
||||
|
||||
@@ -0,0 +1,38 @@
|
||||
use std::sync::Arc;
|
||||
use domain::{
|
||||
entities::{LinkAccessLevel, ShareScope},
|
||||
errors::DomainError,
|
||||
ports::ShareRepository,
|
||||
};
|
||||
|
||||
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
|
||||
pub struct AccessSharedResourceQuery {
|
||||
pub token: String,
|
||||
}
|
||||
|
||||
pub struct AccessSharedResourceHandler {
|
||||
share_repo: Arc<dyn ShareRepository>,
|
||||
}
|
||||
|
||||
impl AccessSharedResourceHandler {
|
||||
pub fn new(share_repo: Arc<dyn ShareRepository>) -> Self {
|
||||
Self { share_repo }
|
||||
}
|
||||
|
||||
pub async fn execute(&self, query: AccessSharedResourceQuery) -> Result<(ShareScope, LinkAccessLevel), DomainError> {
|
||||
let mut link = self.share_repo.find_link_by_token(&query.token).await?
|
||||
.ok_or_else(|| DomainError::NotFound("Share link not found".to_string()))?;
|
||||
|
||||
if !link.is_valid() {
|
||||
return Err(DomainError::Forbidden("Link expired or exhausted".to_string()));
|
||||
}
|
||||
|
||||
link.record_use();
|
||||
self.share_repo.save_link(&link).await?;
|
||||
|
||||
let scope = self.share_repo.find_scope_by_id(&link.scope_id).await?
|
||||
.ok_or_else(|| DomainError::NotFound("Share scope not found".to_string()))?;
|
||||
|
||||
Ok((scope, link.access_level))
|
||||
}
|
||||
}
|
||||
1
crates/application/src/sharing/queries/mod.rs
Normal file
1
crates/application/src/sharing/queries/mod.rs
Normal file
@@ -0,0 +1 @@
|
||||
pub mod access_shared_resource;
|
||||
@@ -2,3 +2,4 @@ mod identity;
|
||||
mod organization;
|
||||
mod storage;
|
||||
mod catalog;
|
||||
mod sharing;
|
||||
|
||||
@@ -0,0 +1,92 @@
|
||||
use std::sync::Arc;
|
||||
use application::testing::InMemoryAlbumRepository;
|
||||
use application::organization::{
|
||||
AlbumAction, CreateAlbumCommand, CreateAlbumHandler,
|
||||
ManageAlbumEntriesCommand, ManageAlbumEntriesHandler,
|
||||
};
|
||||
use domain::errors::DomainError;
|
||||
use domain::value_objects::SystemId;
|
||||
|
||||
async fn setup_album(repo: &Arc<InMemoryAlbumRepository>, creator: SystemId) -> SystemId {
|
||||
let handler = CreateAlbumHandler::new(repo.clone());
|
||||
let album = handler.execute(CreateAlbumCommand {
|
||||
title: "Test Album".into(),
|
||||
creator_id: creator,
|
||||
}).await.unwrap();
|
||||
album.album_id
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn adds_asset_to_album() {
|
||||
let repo = Arc::new(InMemoryAlbumRepository::new());
|
||||
let user = SystemId::new();
|
||||
let album_id = setup_album(&repo, user).await;
|
||||
let asset_id = SystemId::new();
|
||||
|
||||
let handler = ManageAlbumEntriesHandler::new(repo.clone());
|
||||
let album = handler.execute(ManageAlbumEntriesCommand {
|
||||
album_id,
|
||||
action: AlbumAction::Add { asset_id },
|
||||
user_id: user,
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(album.asset_count(), 1);
|
||||
assert_eq!(album.entries[0].asset_id, asset_id);
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn removes_asset_from_album() {
|
||||
let repo = Arc::new(InMemoryAlbumRepository::new());
|
||||
let user = SystemId::new();
|
||||
let album_id = setup_album(&repo, user).await;
|
||||
let asset_id = SystemId::new();
|
||||
|
||||
let handler = ManageAlbumEntriesHandler::new(repo.clone());
|
||||
handler.execute(ManageAlbumEntriesCommand {
|
||||
album_id,
|
||||
action: AlbumAction::Add { asset_id },
|
||||
user_id: user,
|
||||
}).await.unwrap();
|
||||
|
||||
let album = handler.execute(ManageAlbumEntriesCommand {
|
||||
album_id,
|
||||
action: AlbumAction::Remove { asset_id },
|
||||
user_id: user,
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(album.asset_count(), 0);
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn rejects_nonexistent_album() {
|
||||
let repo = Arc::new(InMemoryAlbumRepository::new());
|
||||
let handler = ManageAlbumEntriesHandler::new(repo);
|
||||
let result = handler.execute(ManageAlbumEntriesCommand {
|
||||
album_id: SystemId::new(),
|
||||
action: AlbumAction::Add { asset_id: SystemId::new() },
|
||||
user_id: SystemId::new(),
|
||||
}).await;
|
||||
assert!(matches!(result, Err(DomainError::NotFound(_))));
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn rejects_duplicate_add() {
|
||||
let repo = Arc::new(InMemoryAlbumRepository::new());
|
||||
let user = SystemId::new();
|
||||
let album_id = setup_album(&repo, user).await;
|
||||
let asset_id = SystemId::new();
|
||||
|
||||
let handler = ManageAlbumEntriesHandler::new(repo.clone());
|
||||
handler.execute(ManageAlbumEntriesCommand {
|
||||
album_id,
|
||||
action: AlbumAction::Add { asset_id },
|
||||
user_id: user,
|
||||
}).await.unwrap();
|
||||
|
||||
let result = handler.execute(ManageAlbumEntriesCommand {
|
||||
album_id,
|
||||
action: AlbumAction::Add { asset_id },
|
||||
user_id: user,
|
||||
}).await;
|
||||
assert!(matches!(result, Err(DomainError::Conflict(_))));
|
||||
}
|
||||
@@ -1 +1,3 @@
|
||||
mod create_album;
|
||||
mod manage_album_entries;
|
||||
mod tag_asset;
|
||||
|
||||
81
crates/application/tests/organization/commands/tag_asset.rs
Normal file
81
crates/application/tests/organization/commands/tag_asset.rs
Normal file
@@ -0,0 +1,81 @@
|
||||
use std::sync::Arc;
|
||||
use application::testing::{InMemoryAssetRepository, InMemoryTagRepository};
|
||||
use application::organization::{TagAssetCommand, TagAssetHandler};
|
||||
use domain::entities::{Asset, AssetType, SourceReference};
|
||||
use domain::errors::DomainError;
|
||||
use domain::ports::{AssetRepository, TagRepository};
|
||||
use domain::value_objects::{Checksum, SystemId};
|
||||
|
||||
async fn seed_asset(repo: &Arc<InMemoryAssetRepository>) -> SystemId {
|
||||
let owner = SystemId::new();
|
||||
let asset = Asset::new(
|
||||
SourceReference {
|
||||
volume_id: SystemId::new(),
|
||||
relative_path: "photos/test.jpg".into(),
|
||||
checksum: Checksum::new("a".repeat(64)).unwrap(),
|
||||
},
|
||||
AssetType::Image,
|
||||
"image/jpeg",
|
||||
1024,
|
||||
owner,
|
||||
);
|
||||
let id = asset.asset_id;
|
||||
repo.save(&asset).await.unwrap();
|
||||
id
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn tags_asset_creates_new_tag() {
|
||||
let asset_repo = Arc::new(InMemoryAssetRepository::new());
|
||||
let tag_repo = Arc::new(InMemoryTagRepository::new());
|
||||
let asset_id = seed_asset(&asset_repo).await;
|
||||
let user = SystemId::new();
|
||||
|
||||
let handler = TagAssetHandler::new(asset_repo, tag_repo);
|
||||
let (tag, asset_tag) = handler.execute(TagAssetCommand {
|
||||
asset_id,
|
||||
tag_name: "sunset".into(),
|
||||
user_id: user,
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(tag.name, "sunset");
|
||||
assert_eq!(asset_tag.asset_id, asset_id);
|
||||
assert_eq!(asset_tag.tag_id, tag.tag_id);
|
||||
assert_eq!(asset_tag.tagged_by_user_id, Some(user));
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn reuses_existing_tag() {
|
||||
let asset_repo = Arc::new(InMemoryAssetRepository::new());
|
||||
let tag_repo = Arc::new(InMemoryTagRepository::new());
|
||||
let asset_id = seed_asset(&asset_repo).await;
|
||||
let user = SystemId::new();
|
||||
|
||||
// Pre-create tag
|
||||
let existing = domain::entities::Tag::new_manual("sunset");
|
||||
let existing_id = existing.tag_id;
|
||||
tag_repo.save_tag(&existing).await.unwrap();
|
||||
|
||||
let handler = TagAssetHandler::new(asset_repo, tag_repo);
|
||||
let (tag, _) = handler.execute(TagAssetCommand {
|
||||
asset_id,
|
||||
tag_name: "sunset".into(),
|
||||
user_id: user,
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(tag.tag_id, existing_id);
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn rejects_nonexistent_asset() {
|
||||
let asset_repo = Arc::new(InMemoryAssetRepository::new());
|
||||
let tag_repo = Arc::new(InMemoryTagRepository::new());
|
||||
|
||||
let handler = TagAssetHandler::new(asset_repo, tag_repo);
|
||||
let result = handler.execute(TagAssetCommand {
|
||||
asset_id: SystemId::new(),
|
||||
tag_name: "sunset".into(),
|
||||
user_id: SystemId::new(),
|
||||
}).await;
|
||||
assert!(matches!(result, Err(DomainError::NotFound(_))));
|
||||
}
|
||||
@@ -1 +1,2 @@
|
||||
mod commands;
|
||||
mod queries;
|
||||
|
||||
38
crates/application/tests/organization/queries/get_album.rs
Normal file
38
crates/application/tests/organization/queries/get_album.rs
Normal file
@@ -0,0 +1,38 @@
|
||||
use std::sync::Arc;
|
||||
use application::testing::InMemoryAlbumRepository;
|
||||
use application::organization::{
|
||||
CreateAlbumCommand, CreateAlbumHandler,
|
||||
GetAlbumQuery, GetAlbumHandler,
|
||||
};
|
||||
use domain::errors::DomainError;
|
||||
use domain::value_objects::SystemId;
|
||||
|
||||
#[tokio::test]
|
||||
async fn returns_album() {
|
||||
let repo = Arc::new(InMemoryAlbumRepository::new());
|
||||
let creator = SystemId::new();
|
||||
|
||||
let create_handler = CreateAlbumHandler::new(repo.clone());
|
||||
let album = create_handler.execute(CreateAlbumCommand {
|
||||
title: "My Album".into(),
|
||||
creator_id: creator,
|
||||
}).await.unwrap();
|
||||
|
||||
let query_handler = GetAlbumHandler::new(repo);
|
||||
let found = query_handler.execute(GetAlbumQuery {
|
||||
album_id: album.album_id,
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(found.album_id, album.album_id);
|
||||
assert_eq!(found.title, "My Album");
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn rejects_nonexistent() {
|
||||
let repo = Arc::new(InMemoryAlbumRepository::new());
|
||||
let handler = GetAlbumHandler::new(repo);
|
||||
let result = handler.execute(GetAlbumQuery {
|
||||
album_id: SystemId::new(),
|
||||
}).await;
|
||||
assert!(matches!(result, Err(DomainError::NotFound(_))));
|
||||
}
|
||||
1
crates/application/tests/organization/queries/mod.rs
Normal file
1
crates/application/tests/organization/queries/mod.rs
Normal file
@@ -0,0 +1 @@
|
||||
mod get_album;
|
||||
@@ -0,0 +1,46 @@
|
||||
use std::sync::Arc;
|
||||
use application::testing::InMemoryShareRepository;
|
||||
use application::sharing::{GenerateShareLinkCommand, GenerateShareLinkHandler};
|
||||
use domain::entities::{LinkAccessLevel, ScopeType, ShareableType};
|
||||
use domain::value_objects::{DateTimeStamp, SystemId};
|
||||
|
||||
#[tokio::test]
|
||||
async fn generates_link() {
|
||||
let share_repo = Arc::new(InMemoryShareRepository::new());
|
||||
let handler = GenerateShareLinkHandler::new(share_repo);
|
||||
|
||||
let (scope, link) = handler.execute(GenerateShareLinkCommand {
|
||||
shareable_type: ShareableType::Album,
|
||||
shareable_id: SystemId::new(),
|
||||
access_level: LinkAccessLevel::ViewOnly,
|
||||
created_by: SystemId::new(),
|
||||
expires_at: None,
|
||||
max_uses: None,
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(scope.scope_type, ScopeType::Link);
|
||||
assert!(!link.token.is_empty());
|
||||
assert_eq!(link.access_level, LinkAccessLevel::ViewOnly);
|
||||
assert!(link.expires_at.is_none());
|
||||
assert!(link.max_uses.is_none());
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn generates_link_with_expiry_and_max_uses() {
|
||||
let share_repo = Arc::new(InMemoryShareRepository::new());
|
||||
let handler = GenerateShareLinkHandler::new(share_repo);
|
||||
|
||||
let expiry = DateTimeStamp::now();
|
||||
let (_, link) = handler.execute(GenerateShareLinkCommand {
|
||||
shareable_type: ShareableType::Collection,
|
||||
shareable_id: SystemId::new(),
|
||||
access_level: LinkAccessLevel::LimitedSearch,
|
||||
created_by: SystemId::new(),
|
||||
expires_at: Some(expiry),
|
||||
max_uses: Some(10),
|
||||
}).await.unwrap();
|
||||
|
||||
assert!(link.expires_at.is_some());
|
||||
assert_eq!(link.max_uses, Some(10));
|
||||
assert_eq!(link.access_level, LinkAccessLevel::LimitedSearch);
|
||||
}
|
||||
3
crates/application/tests/sharing/commands/mod.rs
Normal file
3
crates/application/tests/sharing/commands/mod.rs
Normal file
@@ -0,0 +1,3 @@
|
||||
mod share_resource;
|
||||
mod generate_share_link;
|
||||
mod revoke_share;
|
||||
48
crates/application/tests/sharing/commands/revoke_share.rs
Normal file
48
crates/application/tests/sharing/commands/revoke_share.rs
Normal file
@@ -0,0 +1,48 @@
|
||||
use std::sync::Arc;
|
||||
use application::testing::{InMemoryShareRepository, StubEventPublisher};
|
||||
use application::sharing::{
|
||||
GenerateShareLinkCommand, GenerateShareLinkHandler,
|
||||
RevokeShareCommand, RevokeShareHandler,
|
||||
};
|
||||
use domain::entities::{LinkAccessLevel, ShareableType};
|
||||
use domain::errors::DomainError;
|
||||
use domain::value_objects::SystemId;
|
||||
|
||||
#[tokio::test]
|
||||
async fn revokes_share() {
|
||||
let share_repo = Arc::new(InMemoryShareRepository::new());
|
||||
let event_pub = Arc::new(StubEventPublisher::new());
|
||||
|
||||
// Create a scope first via generate_share_link
|
||||
let gen_handler = GenerateShareLinkHandler::new(share_repo.clone());
|
||||
let (scope, _) = gen_handler.execute(GenerateShareLinkCommand {
|
||||
shareable_type: ShareableType::Album,
|
||||
shareable_id: SystemId::new(),
|
||||
access_level: LinkAccessLevel::ViewOnly,
|
||||
created_by: SystemId::new(),
|
||||
expires_at: None,
|
||||
max_uses: None,
|
||||
}).await.unwrap();
|
||||
|
||||
let handler = RevokeShareHandler::new(share_repo, event_pub.clone());
|
||||
handler.execute(RevokeShareCommand {
|
||||
scope_id: scope.scope_id,
|
||||
revoked_by: SystemId::new(),
|
||||
}).await.unwrap();
|
||||
|
||||
let events = event_pub.published().await;
|
||||
assert_eq!(events.len(), 1);
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn rejects_nonexistent_scope() {
|
||||
let share_repo = Arc::new(InMemoryShareRepository::new());
|
||||
let event_pub = Arc::new(StubEventPublisher::new());
|
||||
let handler = RevokeShareHandler::new(share_repo, event_pub);
|
||||
|
||||
let result = handler.execute(RevokeShareCommand {
|
||||
scope_id: SystemId::new(),
|
||||
revoked_by: SystemId::new(),
|
||||
}).await;
|
||||
assert!(matches!(result, Err(DomainError::NotFound(_))));
|
||||
}
|
||||
47
crates/application/tests/sharing/commands/share_resource.rs
Normal file
47
crates/application/tests/sharing/commands/share_resource.rs
Normal file
@@ -0,0 +1,47 @@
|
||||
use std::sync::Arc;
|
||||
use application::testing::{InMemoryShareRepository, StubEventPublisher};
|
||||
use application::sharing::{ShareResourceCommand, ShareResourceHandler};
|
||||
use domain::entities::{ScopeType, ShareableType, TargetType};
|
||||
use domain::value_objects::SystemId;
|
||||
|
||||
#[tokio::test]
|
||||
async fn shares_with_user() {
|
||||
let share_repo = Arc::new(InMemoryShareRepository::new());
|
||||
let event_pub = Arc::new(StubEventPublisher::new());
|
||||
|
||||
let handler = ShareResourceHandler::new(share_repo, event_pub.clone());
|
||||
let (scope, target) = handler.execute(ShareResourceCommand {
|
||||
shareable_type: ShareableType::Album,
|
||||
shareable_id: SystemId::new(),
|
||||
target_type: TargetType::User,
|
||||
target_id: SystemId::new(),
|
||||
role_id: SystemId::new(),
|
||||
created_by: SystemId::new(),
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(scope.scope_type, ScopeType::User);
|
||||
assert_eq!(target.target_type, TargetType::User);
|
||||
assert_eq!(target.scope_id, scope.scope_id);
|
||||
|
||||
let events = event_pub.published().await;
|
||||
assert_eq!(events.len(), 1);
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn shares_with_group() {
|
||||
let share_repo = Arc::new(InMemoryShareRepository::new());
|
||||
let event_pub = Arc::new(StubEventPublisher::new());
|
||||
|
||||
let handler = ShareResourceHandler::new(share_repo, event_pub.clone());
|
||||
let (scope, target) = handler.execute(ShareResourceCommand {
|
||||
shareable_type: ShareableType::Asset,
|
||||
shareable_id: SystemId::new(),
|
||||
target_type: TargetType::Group,
|
||||
target_id: SystemId::new(),
|
||||
role_id: SystemId::new(),
|
||||
created_by: SystemId::new(),
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(scope.scope_type, ScopeType::Group);
|
||||
assert_eq!(target.target_type, TargetType::Group);
|
||||
}
|
||||
2
crates/application/tests/sharing/mod.rs
Normal file
2
crates/application/tests/sharing/mod.rs
Normal file
@@ -0,0 +1,2 @@
|
||||
mod commands;
|
||||
mod queries;
|
||||
@@ -0,0 +1,67 @@
|
||||
use std::sync::Arc;
|
||||
use application::testing::InMemoryShareRepository;
|
||||
use application::sharing::{
|
||||
AccessSharedResourceQuery, AccessSharedResourceHandler,
|
||||
GenerateShareLinkCommand, GenerateShareLinkHandler,
|
||||
};
|
||||
use chrono::{DateTime, Utc};
|
||||
use domain::entities::{LinkAccessLevel, ShareableType};
|
||||
use domain::errors::DomainError;
|
||||
use domain::value_objects::{DateTimeStamp, SystemId};
|
||||
|
||||
async fn create_link(
|
||||
repo: &Arc<InMemoryShareRepository>,
|
||||
expires_at: Option<DateTimeStamp>,
|
||||
max_uses: Option<u32>,
|
||||
) -> String {
|
||||
let handler = GenerateShareLinkHandler::new(repo.clone());
|
||||
let (_, link) = handler.execute(GenerateShareLinkCommand {
|
||||
shareable_type: ShareableType::Album,
|
||||
shareable_id: SystemId::new(),
|
||||
access_level: LinkAccessLevel::ViewOnly,
|
||||
created_by: SystemId::new(),
|
||||
expires_at,
|
||||
max_uses,
|
||||
}).await.unwrap();
|
||||
link.token
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn valid_link_returns_scope() {
|
||||
let repo = Arc::new(InMemoryShareRepository::new());
|
||||
let token = create_link(&repo, None, None).await;
|
||||
|
||||
let handler = AccessSharedResourceHandler::new(repo);
|
||||
let (scope, access_level) = handler.execute(AccessSharedResourceQuery {
|
||||
token,
|
||||
}).await.unwrap();
|
||||
|
||||
assert_eq!(access_level, LinkAccessLevel::ViewOnly);
|
||||
assert_eq!(scope.shareable_type, ShareableType::Album);
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn expired_link_rejected() {
|
||||
let repo = Arc::new(InMemoryShareRepository::new());
|
||||
// Create link with past expiry
|
||||
let past = DateTimeStamp::from_datetime(DateTime::<Utc>::from_timestamp(0, 0).unwrap());
|
||||
let token = create_link(&repo, Some(past), None).await;
|
||||
|
||||
let handler = AccessSharedResourceHandler::new(repo);
|
||||
let result = handler.execute(AccessSharedResourceQuery { token }).await;
|
||||
assert!(matches!(result, Err(DomainError::Forbidden(_))));
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn exhausted_link_rejected() {
|
||||
let repo = Arc::new(InMemoryShareRepository::new());
|
||||
let token = create_link(&repo, None, Some(1)).await;
|
||||
|
||||
// Use it once
|
||||
let handler = AccessSharedResourceHandler::new(repo.clone());
|
||||
handler.execute(AccessSharedResourceQuery { token: token.clone() }).await.unwrap();
|
||||
|
||||
// Second use should fail
|
||||
let result = handler.execute(AccessSharedResourceQuery { token }).await;
|
||||
assert!(matches!(result, Err(DomainError::Forbidden(_))));
|
||||
}
|
||||
1
crates/application/tests/sharing/queries/mod.rs
Normal file
1
crates/application/tests/sharing/queries/mod.rs
Normal file
@@ -0,0 +1 @@
|
||||
mod access_shared_resource;
|
||||
Reference in New Issue
Block a user