/
gift_card_event_repository.rs
118 lines (109 loc) · 3.78 KB
/
gift_card_event_repository.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
use async_trait::async_trait;
use derive_more::Display;
use fmodel_rust::aggregate::EventRepository;
use serde_derive::{Deserialize, Serialize};
use synapse_client::apis::aggregate_api::read_aggregate_events;
use synapse_client::apis::configuration;
use synapse_client::apis::events_api::publish_event_message;
use synapse_client::models::{EventMessage, PublishableEventMessage};
use crate::gift_card_api::{GiftCardCommand, GiftCardEvent};
/// Error type for the application/aggregate
#[derive(Debug, Display, Serialize, Deserialize)]
#[allow(dead_code)]
pub enum AggregateError {
FetchEvents(String),
SaveEvents(String),
FetchState(String),
SaveState(String),
}
/// Map to domain events of type GiftCardEvent
pub trait ToGiftCardEvent {
fn to_gift_card_event(&self) -> Option<GiftCardEvent>;
}
/// Map from Axon EventMessage to domain events of type GiftCardEvent
impl ToGiftCardEvent for EventMessage {
fn to_gift_card_event(&self) -> Option<GiftCardEvent> {
let value = self.payload.clone().unwrap().unwrap();
let event = serde_json::from_value(value);
match event {
Ok(event) => Some(event),
Err(_err) => None,
}
}
}
/// Map to Axon EventMessage
trait ToEventMessage {
fn to_event_message(&self, version: i64) -> PublishableEventMessage;
}
/// Map from domain events of type GiftCardEvent to Axon EventMessage
impl ToEventMessage for GiftCardEvent {
fn to_event_message(&self, version: i64) -> PublishableEventMessage {
let payload = serde_json::to_value(self).unwrap();
PublishableEventMessage {
payload_type: Some(self.payload_type()),
name: self.payload_type(),
aggregate_id: Some(self.id()),
aggregate_type: Some(self.aggregate_type()),
sequence_number: Some(version),
date_time: None,
index: None,
id: None,
meta_data: None,
payload: Some(Some(payload)),
payload_revision: None,
}
}
}
/// Axon Server event repository
pub struct AxonServerEventRepository {
pub configuration: configuration::Configuration,
pub context: String,
}
/// Event repository implementation for Axon Server
#[async_trait]
impl EventRepository<GiftCardCommand, GiftCardEvent, i64, AggregateError>
for AxonServerEventRepository
{
async fn fetch_events(
&self,
command: &GiftCardCommand,
) -> Result<Vec<(GiftCardEvent, i64)>, AggregateError> {
let result = read_aggregate_events(&self.configuration, &self.context, &command.id()).await;
match result {
Ok(events) => Ok(events
.items
.unwrap_or_default()
.into_iter()
.map(|event| {
(
event.to_gift_card_event().unwrap(),
event.sequence_number.unwrap(),
)
})
.collect()),
Err(err) => Err(AggregateError::FetchEvents(err.to_string())),
}
}
async fn save(
&self,
events: &[GiftCardEvent],
version: &Option<i64>,
) -> Result<Vec<(GiftCardEvent, i64)>, AggregateError> {
let mut saved_events: Vec<(GiftCardEvent, i64)> = vec![];
let mut version = version.unwrap_or(-1);
for evt in events {
version += 1;
let result = publish_event_message(
&self.configuration,
&self.context,
Some(evt.to_event_message(version)),
)
.await;
match result {
Ok(_) => saved_events.push((evt.to_owned(), version)),
Err(err) => return Err(AggregateError::SaveEvents(err.to_string())),
};
}
Ok(saved_events)
}
}