feat: order: LineItem add and update cmd,events&service #59

Merged
realaravinth merged 8 commits from ordering-line-item into master 2024-07-23 18:13:29 +05:30
3 changed files with 167 additions and 2 deletions
Showing only changes of commit 15c0241598 - Show all commits

View file

@ -10,19 +10,25 @@ pub mod errors;
//services //services
pub mod add_line_item_service; pub mod add_line_item_service;
pub mod update_line_item_service;
#[automock] #[automock]
pub trait OrderingServicesInterface: Send + Sync { pub trait OrderingServicesInterface: Send + Sync {
fn add_line_item(&self) -> add_line_item_service::AddLineItemServiceObj; fn add_line_item(&self) -> add_line_item_service::AddLineItemServiceObj;
fn update_line_item(&self) -> update_line_item_service::UpdateLineItemServiceObj;
} }
#[derive(Clone, Builder)] #[derive(Clone, Builder)]
pub struct OrderingServices { pub struct OrderingServices {
add_line_item: add_line_item_service::AddLineItemServiceObj, add_line_item: add_line_item_service::AddLineItemServiceObj,
update_line_item: update_line_item_service::UpdateLineItemServiceObj,
} }
impl OrderingServicesInterface for OrderingServices { impl OrderingServicesInterface for OrderingServices {
fn add_line_item(&self) -> add_line_item_service::AddLineItemServiceObj { fn add_line_item(&self) -> add_line_item_service::AddLineItemServiceObj {
self.add_line_item.clone() self.add_line_item.clone()
} }
fn update_line_item(&self) -> update_line_item_service::UpdateLineItemServiceObj {
self.update_line_item.clone()
}
} }

View file

@ -0,0 +1,134 @@
// SPDX-FileCopyrightText: 2024 Aravinth Manivannan <realaravinth@batsense.net>
//
// SPDX-License-Identifier: AGPL-3.0-or-later
use std::sync::Arc;
use derive_builder::Builder;
use mockall::predicate::*;
use mockall::*;
use time::OffsetDateTime;
use super::errors::*;
use crate::ordering::{
application::port::output::db::line_item_id_exists::*,
domain::{line_item_aggregate::*, line_item_updated_event::*, update_line_item_command::*},
};
use crate::utils::uuid::*;
#[automock]
#[async_trait::async_trait]
pub trait UpdateLineItemUseCase: Send + Sync {
async fn update_line_item(
&self,
cmd: UpdateLineItemCommand,
) -> OrderingResult<LineItemUpdatedEvent>;
}
pub type UpdateLineItemServiceObj = Arc<dyn UpdateLineItemUseCase>;
#[derive(Clone, Builder)]
pub struct UpdateLineItemService {
db_line_item_id_exists: LineItemIDExistsDBPortObj,
}
#[async_trait::async_trait]
impl UpdateLineItemUseCase for UpdateLineItemService {
async fn update_line_item(
&self,
cmd: UpdateLineItemCommand,
) -> OrderingResult<LineItemUpdatedEvent> {
if !self
.db_line_item_id_exists
.line_item_id_exists(cmd.old_line_item().line_item_id())
.await?
{
return Err(OrderingError::LineItemIDNotFound);
}
let new_line_item = LineItemBuilder::default()
.sale_time(cmd.sale_time().clone())
.product_name(cmd.product_name().into())
.product_id(*cmd.product_id())
.line_item_id(*cmd.old_line_item().line_item_id())
.quantity(cmd.quantity().clone())
.deleted(false)
.build()
.unwrap();
Ok(LineItemUpdatedEventBuilder::default()
.added_by_user(*cmd.adding_by())
.new_line_item(new_line_item)
.old_line_item(cmd.old_line_item().clone())
.build()
.unwrap())
}
}
#[cfg(test)]
pub mod tests {
use super::*;
use crate::ordering::domain::line_item_updated_event::tests::get_updated_line_item_event_from_command;
use crate::utils::uuid::tests::UUID;
use crate::{tests::bdd::*, utils::uuid::tests::mock_get_uuid};
pub fn mock_update_line_item_service(
times: Option<usize>,
cmd: UpdateLineItemCommand,
) -> UpdateLineItemServiceObj {
let mut m = MockUpdateLineItemUseCase::new();
let res = get_updated_line_item_event_from_command(&cmd);
if let Some(times) = times {
m.expect_update_line_item()
.times(times)
.returning(move |_| Ok(res.clone()));
} else {
m.expect_update_line_item()
.returning(move |_| Ok(res.clone()));
}
Arc::new(m)
}
#[actix_rt::test]
async fn test_service() {
let cmd = UpdateLineItemCommand::get_cmd();
let s = UpdateLineItemServiceBuilder::default()
.db_line_item_id_exists(mock_line_item_id_exists_db_port_true(IS_CALLED_ONLY_ONCE))
.build()
.unwrap();
let res = s.update_line_item(cmd.clone()).await.unwrap();
assert_eq!(res.new_line_item().product_name(), cmd.product_name());
assert_eq!(res.new_line_item().product_id(), cmd.product_id());
assert_eq!(res.new_line_item().quantity(), cmd.quantity());
assert_eq!(res.new_line_item().quantity(), cmd.quantity());
assert_eq!(
res.new_line_item().line_item_id(),
cmd.old_line_item().line_item_id()
);
assert!(!res.new_line_item().deleted());
assert_eq!(res.old_line_item(), cmd.old_line_item());
assert_eq!(res.added_by_user(), cmd.adding_by());
}
#[actix_rt::test]
async fn test_service_line_item_id_doesnt_exist() {
let cmd = UpdateLineItemCommand::get_cmd();
let s = UpdateLineItemServiceBuilder::default()
.db_line_item_id_exists(mock_line_item_id_exists_db_port_false(IS_CALLED_ONLY_ONCE))
.build()
.unwrap();
assert_eq!(
s.update_line_item(cmd.clone()).await,
Err(OrderingError::LineItemIDNotFound)
);
}
}

View file

@ -92,7 +92,8 @@ impl Aggregate for LineItem {
Ok(vec![OrderingEvent::LineItemAdded(res)]) Ok(vec![OrderingEvent::LineItemAdded(res)])
} }
OrderingCommand::UpdateLineItem(cmd) => { OrderingCommand::UpdateLineItem(cmd) => {
unimplemented!() let res = services.update_line_item().update_line_item(cmd).await?;
Ok(vec![OrderingEvent::LineItemUpdated(res)])
} }
OrderingCommand::DeleteLineItem(cmd) => { OrderingCommand::DeleteLineItem(cmd) => {
unimplemented!() unimplemented!()
@ -116,9 +117,12 @@ mod aggregate_tests {
use add_line_item_service::tests::mock_add_line_item_service; use add_line_item_service::tests::mock_add_line_item_service;
use cqrs_es::test::TestFramework; use cqrs_es::test::TestFramework;
use update_line_item_service::tests::mock_update_line_item_service;
use super::*; use super::*;
use crate::ordering::domain::line_item_updated_event::tests::get_updated_line_item_event_from_command;
use crate::ordering::domain::update_line_item_command::UpdateLineItemCommand;
use crate::tests::bdd::*; use crate::tests::bdd::*;
use crate::ordering::domain::{ use crate::ordering::domain::{
@ -129,7 +133,7 @@ mod aggregate_tests {
type LineItemTestFramework = TestFramework<LineItem>; type LineItemTestFramework = TestFramework<LineItem>;
#[test] #[test]
fn test_create_product() { fn test_add_line_item() {
let cmd = AddLineItemCommand::get_cmd(); let cmd = AddLineItemCommand::get_cmd();
let expected = get_added_line_item_event_from_command(&cmd); let expected = get_added_line_item_event_from_command(&cmd);
let expected = OrderingEvent::LineItemAdded(expected); let expected = OrderingEvent::LineItemAdded(expected);
@ -145,4 +149,25 @@ mod aggregate_tests {
.when(OrderingCommand::AddLineItem(cmd)) .when(OrderingCommand::AddLineItem(cmd))
.then_expect_events(vec![expected]); .then_expect_events(vec![expected]);
} }
#[test]
fn test_update_line_item() {
let cmd = UpdateLineItemCommand::get_cmd();
let expected = get_updated_line_item_event_from_command(&cmd);
let expected = OrderingEvent::LineItemUpdated(expected);
let mut services = MockOrderingServicesInterface::new();
services
.expect_update_line_item()
.times(IS_CALLED_ONLY_ONCE.unwrap())
.return_const(mock_update_line_item_service(
IS_CALLED_ONLY_ONCE,
cmd.clone(),
));
LineItemTestFramework::with(Arc::new(services))
.given_no_previous_events()
.when(OrderingCommand::UpdateLineItem(cmd))
.then_expect_events(vec![expected]);
}
} }