#![allow(dead_code)]
use derive_builder::Builder;
use serde::{Deserialize, Serialize};
use es_entity::*;
es_entity::entity_id! { UserId }
#[derive(EsEvent, Debug, Serialize, Deserialize)]
#[serde(tag = "type", rename_all = "snake_case")]
#[es_event(id = "UserId", event_context)]
pub enum UserEvent {
Initialized { id: UserId, name: String },
NameUpdated { name: String },
}
#[derive(EsEntity, Builder)]
#[builder(pattern = "owned", build_fn(error = "EntityHydrationError"))]
pub struct User {
pub id: UserId,
pub name: String,
events: EntityEvents<UserEvent>,
}
impl User {
pub fn update_name(&mut self, new_name: impl Into<String>) -> Idempotent<()> {
let new_name = new_name.into();
idempotency_guard!(
self.events.iter_all().rev(),
already_applied: UserEvent::NameUpdated { name } if name == &new_name,
resets_on: UserEvent::NameUpdated { .. }
);
self.name = new_name.clone();
self.events.push(UserEvent::NameUpdated { name: new_name });
Idempotent::Executed(())
}
}
impl TryFromEvents<UserEvent> for User {
fn try_from_events(events: EntityEvents<UserEvent>) -> Result<Self, EntityHydrationError> {
let mut builder = UserBuilder::default();
for event in events.iter_all() {
match event {
UserEvent::Initialized { id, name } => {
builder = builder.id(*id).name(name.clone());
}
UserEvent::NameUpdated { name } => {
builder = builder.name(name.clone());
}
}
}
builder.events(events).build()
}
}
#[derive(Debug, Builder)]
pub struct NewUser {
#[builder(setter(into))]
pub id: UserId,
#[builder(setter(into))]
pub name: String,
}
impl NewUser {
pub fn builder() -> NewUserBuilder {
NewUserBuilder::default()
}
}
impl IntoEvents<UserEvent> for NewUser {
fn into_events(self) -> EntityEvents<UserEvent> {
EntityEvents::init(
self.id,
[UserEvent::Initialized {
id: self.id,
name: self.name,
}],
)
}
}