1use chrono::{DateTime, Utc};
4
5use super::{error::EsEntityError, traits::*};
6
7pub type LastPersisted<'a, E> = std::slice::Iter<'a, PersistedEvent<E>>;
9
10pub struct GenericEvent<Id> {
16 pub entity_id: Id,
17 pub sequence: i32,
18 pub event: serde_json::Value,
19 pub recorded_at: DateTime<Utc>,
20}
21
22pub struct PersistedEvent<E: EsEvent> {
28 pub entity_id: <E as EsEvent>::EntityId,
30 pub recorded_at: DateTime<Utc>,
32 pub sequence: usize,
34 pub event: E,
36}
37
38impl<E: Clone + EsEvent> Clone for PersistedEvent<E> {
39 fn clone(&self) -> Self {
40 PersistedEvent {
41 entity_id: self.entity_id.clone(),
42 recorded_at: self.recorded_at,
43 sequence: self.sequence,
44 event: self.event.clone(),
45 }
46 }
47}
48
49pub struct EntityEvents<T: EsEvent> {
54 pub entity_id: <T as EsEvent>::EntityId,
56 persisted_events: Vec<PersistedEvent<T>>,
58 new_events: Vec<T>,
60}
61
62impl<T: Clone + EsEvent> Clone for EntityEvents<T> {
63 fn clone(&self) -> Self {
64 Self {
65 entity_id: self.entity_id.clone(),
66 persisted_events: self.persisted_events.clone(),
67 new_events: self.new_events.clone(),
68 }
69 }
70}
71
72impl<T> EntityEvents<T>
73where
74 T: EsEvent,
75{
76 pub fn init(id: <T as EsEvent>::EntityId, initial_events: impl IntoIterator<Item = T>) -> Self {
78 Self {
79 entity_id: id,
80 persisted_events: Vec::new(),
81 new_events: initial_events.into_iter().collect(),
82 }
83 }
84
85 pub fn id(&self) -> &<T as EsEvent>::EntityId {
87 &self.entity_id
88 }
89
90 pub fn entity_first_persisted_at(&self) -> Option<DateTime<Utc>> {
92 self.persisted_events.first().map(|e| e.recorded_at)
93 }
94
95 pub fn entity_last_modified_at(&self) -> Option<DateTime<Utc>> {
97 self.persisted_events.last().map(|e| e.recorded_at)
98 }
99
100 pub fn push(&mut self, event: T) {
102 self.new_events.push(event);
103 }
104
105 pub fn extend(&mut self, events: impl IntoIterator<Item = T>) {
107 self.new_events.extend(events);
108 }
109
110 pub fn any_new(&self) -> bool {
112 !self.new_events.is_empty()
113 }
114
115 pub fn len_persisted(&self) -> usize {
117 self.persisted_events.len()
118 }
119
120 pub fn iter_persisted(&self) -> impl DoubleEndedIterator<Item = &PersistedEvent<T>> {
122 self.persisted_events.iter()
123 }
124
125 pub fn last_persisted(&self, n: usize) -> LastPersisted<'_, T> {
127 let start = self.persisted_events.len() - n;
128 self.persisted_events[start..].iter()
129 }
130
131 pub fn iter_all(&self) -> impl DoubleEndedIterator<Item = &T> {
133 self.persisted_events
134 .iter()
135 .map(|e| &e.event)
136 .chain(self.new_events.iter())
137 }
138
139 pub fn load_first<E: EsEntity<Event = T>>(
141 events: impl IntoIterator<Item = GenericEvent<<T as EsEvent>::EntityId>>,
142 ) -> Result<E, EsEntityError> {
143 let mut current_id = None;
144 let mut current = None;
145 for e in events {
146 if current_id.is_none() {
147 current_id = Some(e.entity_id.clone());
148 current = Some(Self {
149 entity_id: e.entity_id.clone(),
150 persisted_events: Vec::new(),
151 new_events: Vec::new(),
152 });
153 }
154 if current_id.as_ref() != Some(&e.entity_id) {
155 break;
156 }
157 let cur = current.as_mut().expect("Could not get current");
158 cur.persisted_events.push(PersistedEvent {
159 entity_id: e.entity_id,
160 recorded_at: e.recorded_at,
161 sequence: e.sequence as usize,
162 event: serde_json::from_value(e.event)?,
163 });
164 }
165 if let Some(current) = current {
166 E::try_from_events(current)
167 } else {
168 Err(EsEntityError::NotFound)
169 }
170 }
171
172 pub fn load_n<E: EsEntity<Event = T>>(
175 events: impl IntoIterator<Item = GenericEvent<<T as EsEvent>::EntityId>>,
176 n: usize,
177 ) -> Result<(Vec<E>, bool), EsEntityError> {
178 let mut ret: Vec<E> = Vec::new();
179 let mut current_id = None;
180 let mut current = None;
181 for e in events {
182 if current_id.as_ref() != Some(&e.entity_id) {
183 if let Some(current) = current.take() {
184 ret.push(E::try_from_events(current)?);
185 if ret.len() == n {
186 return Ok((ret, true));
187 }
188 }
189
190 current_id = Some(e.entity_id.clone());
191 current = Some(Self {
192 entity_id: e.entity_id.clone(),
193 persisted_events: Vec::new(),
194 new_events: Vec::new(),
195 });
196 }
197 let cur = current.as_mut().expect("Could not get current");
198 cur.persisted_events.push(PersistedEvent {
199 entity_id: e.entity_id,
200 recorded_at: e.recorded_at,
201 sequence: e.sequence as usize,
202 event: serde_json::from_value(e.event)?,
203 });
204 }
205 if let Some(current) = current.take() {
206 ret.push(E::try_from_events(current)?);
207 }
208 Ok((ret, false))
209 }
210
211 #[doc(hidden)]
212 pub fn mark_new_events_persisted_at(
213 &mut self,
214 recorded_at: chrono::DateTime<chrono::Utc>,
215 ) -> usize {
216 let n = self.new_events.len();
217 let offset = self.persisted_events.len() + 1;
218 self.persisted_events
219 .extend(
220 self.new_events
221 .drain(..)
222 .enumerate()
223 .map(|(i, event)| PersistedEvent {
224 entity_id: self.entity_id.clone(),
225 recorded_at,
226 sequence: i + offset,
227 event,
228 }),
229 );
230 n
231 }
232
233 #[doc(hidden)]
234 pub fn serialize_new_events(&self) -> Vec<serde_json::Value> {
235 self.new_events
236 .iter()
237 .map(|event| serde_json::to_value(event).expect("Failed to serialize event"))
238 .collect()
239 }
240}
241
242#[cfg(test)]
243mod tests {
244 use super::*;
245 use uuid::Uuid;
246
247 #[derive(Debug, serde::Serialize, serde::Deserialize)]
248 enum DummyEntityEvent {
249 Created(String),
250 }
251
252 impl EsEvent for DummyEntityEvent {
253 type EntityId = Uuid;
254 }
255
256 struct DummyEntity {
257 name: String,
258
259 events: EntityEvents<DummyEntityEvent>,
260 }
261
262 impl EsEntity for DummyEntity {
263 type Event = DummyEntityEvent;
264 type New = NewDummyEntity;
265
266 fn events_mut(&mut self) -> &mut EntityEvents<DummyEntityEvent> {
267 &mut self.events
268 }
269 fn events(&self) -> &EntityEvents<DummyEntityEvent> {
270 &self.events
271 }
272 }
273
274 impl TryFromEvents<DummyEntityEvent> for DummyEntity {
275 fn try_from_events(events: EntityEvents<DummyEntityEvent>) -> Result<Self, EsEntityError> {
276 let name = events
277 .iter_persisted()
278 .map(|e| match &e.event {
279 DummyEntityEvent::Created(name) => name.clone(),
280 })
281 .next()
282 .expect("Could not find name");
283 Ok(Self { name, events })
284 }
285 }
286
287 struct NewDummyEntity {}
288
289 impl IntoEvents<DummyEntityEvent> for NewDummyEntity {
290 fn into_events(self) -> EntityEvents<DummyEntityEvent> {
291 EntityEvents::init(
292 Uuid::now_v7(),
293 vec![DummyEntityEvent::Created("".to_owned())],
294 )
295 }
296 }
297
298 #[test]
299 fn load_zero_events() {
300 let generic_events = vec![];
301 let res = EntityEvents::load_first::<DummyEntity>(generic_events);
302 assert!(matches!(res, Err(EsEntityError::NotFound)));
303 }
304
305 #[test]
306 fn load_first() {
307 let generic_events = vec![GenericEvent {
308 entity_id: uuid::Uuid::now_v7(),
309 sequence: 1,
310 event: serde_json::to_value(DummyEntityEvent::Created("dummy-name".to_owned()))
311 .expect("Could not serialize"),
312 recorded_at: chrono::Utc::now(),
313 }];
314 let entity: DummyEntity = EntityEvents::load_first(generic_events).expect("Could not load");
315 assert!(entity.name == "dummy-name");
316 }
317
318 #[test]
319 fn load_n() {
320 let generic_events = vec![
321 GenericEvent {
322 entity_id: uuid::Uuid::now_v7(),
323 sequence: 1,
324 event: serde_json::to_value(DummyEntityEvent::Created("dummy-name".to_owned()))
325 .expect("Could not serialize"),
326 recorded_at: chrono::Utc::now(),
327 },
328 GenericEvent {
329 entity_id: uuid::Uuid::now_v7(),
330 sequence: 1,
331 event: serde_json::to_value(DummyEntityEvent::Created("other-name".to_owned()))
332 .expect("Could not serialize"),
333 recorded_at: chrono::Utc::now(),
334 },
335 ];
336 let (entity, more): (Vec<DummyEntity>, _) =
337 EntityEvents::load_n(generic_events, 2).expect("Could not load");
338 assert!(!more);
339 assert_eq!(entity.len(), 2);
340 }
341}