1use std::collections::{BTreeMap, BTreeSet};
2use std::sync::{Arc, Mutex};
3
4use teaql_core::{Record, Value};
5
6#[derive(Debug, Clone)]
7pub struct EntityKey {
8 pub entity: String,
9 pub id: Value,
10 id_key: String,
11}
12
13impl EntityKey {
14 pub fn new(entity: impl Into<String>, id: impl Into<Value>) -> Self {
15 let id = id.into();
16 Self {
17 entity: entity.into(),
18 id_key: value_key(&id),
19 id,
20 }
21 }
22}
23
24impl PartialEq for EntityKey {
25 fn eq(&self, other: &Self) -> bool {
26 self.entity == other.entity && self.id_key == other.id_key
27 }
28}
29
30impl Eq for EntityKey {}
31
32impl PartialOrd for EntityKey {
33 fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
34 Some(self.cmp(other))
35 }
36}
37
38impl Ord for EntityKey {
39 fn cmp(&self, other: &Self) -> std::cmp::Ordering {
40 self.entity
41 .cmp(&other.entity)
42 .then_with(|| self.id_key.cmp(&other.id_key))
43 }
44}
45
46fn value_key(value: &Value) -> String {
47 match value {
48 Value::Null => "null".to_owned(),
49 Value::Bool(value) => format!("bool:{value}"),
50 Value::I64(value) => format!("i64:{value}"),
51 Value::U64(value) => format!("u64:{value}"),
52 Value::F64(value) => format!("f64:{value}"),
53 Value::Decimal(value) => format!("decimal:{value}"),
54 Value::Text(value) => format!("text:{value}"),
55 Value::Json(value) => format!("json:{value}"),
56 Value::Date(value) => format!("date:{value}"),
57 Value::Timestamp(value) => format!("timestamp:{}", value.to_rfc3339()),
58 Value::Object(_) => "object".to_owned(),
59 Value::List(_) => "list".to_owned(),
60 }
61}
62
63#[derive(Debug, Clone, Default, PartialEq)]
64pub struct EntityChangeSet {
65 changes: BTreeMap<EntityKey, Record>,
66}
67
68impl EntityChangeSet {
69 pub fn is_empty(&self) -> bool {
70 self.changes.is_empty()
71 }
72
73 pub fn set(&mut self, key: EntityKey, field: impl Into<String>, value: Value) {
74 self.changes
75 .entry(key)
76 .or_default()
77 .insert(field.into(), value);
78 }
79
80 pub fn get(&self, key: &EntityKey, field: &str) -> Option<&Value> {
81 self.changes.get(key).and_then(|changes| changes.get(field))
82 }
83
84 pub fn changes(&self) -> &BTreeMap<EntityKey, Record> {
85 &self.changes
86 }
87
88 pub fn clear_entity(&mut self, key: &EntityKey) {
90 self.changes.remove(key);
91 }
92
93 pub fn field_names(&self, key: &EntityKey) -> BTreeSet<String> {
95 self.changes
96 .get(key)
97 .map(|record| record.keys().cloned().collect())
98 .unwrap_or_default()
99 }
100}
101
102#[derive(Debug, Clone, Default, PartialEq)]
103pub struct ChangeSetStack {
104 stack: Vec<EntityChangeSet>,
105}
106
107impl ChangeSetStack {
108 pub fn current_mut(&mut self) -> &mut EntityChangeSet {
109 if self.stack.is_empty() {
110 self.stack.push(EntityChangeSet::default());
111 }
112 self.stack.last_mut().expect("change set stack has current")
113 }
114
115 pub fn current(&self) -> Option<&EntityChangeSet> {
116 self.stack.last()
117 }
118
119 pub fn push(&mut self) {
120 self.stack.push(EntityChangeSet::default());
121 }
122
123 pub fn pop(&mut self) -> Option<EntityChangeSet> {
124 self.stack.pop()
125 }
126
127 pub fn get(&self, key: &EntityKey, field: &str) -> Option<Value> {
128 self.stack
129 .iter()
130 .rev()
131 .find_map(|change_set| change_set.get(key, field).cloned())
132 }
133
134 pub fn set(&mut self, key: EntityKey, field: impl Into<String>, value: Value) {
135 self.current_mut().set(key, field, value);
136 }
137
138 pub fn clear_current(&mut self) {
139 if let Some(current) = self.stack.last_mut() {
140 *current = EntityChangeSet::default();
141 }
142 }
143
144 pub fn clear_entity(&mut self, key: &EntityKey) {
146 for change_set in &mut self.stack {
147 change_set.clear_entity(key);
148 }
149 }
150
151 pub fn changed_field_names(&self, key: &EntityKey) -> BTreeSet<String> {
154 let mut fields = BTreeSet::new();
155 for change_set in &self.stack {
156 fields.extend(change_set.field_names(key));
157 }
158 fields
159 }
160}
161
162#[derive(Debug, Default)]
163pub struct RootContext {
164 change_sets: ChangeSetStack,
165 comment: Option<String>,
167 deleted_keys: std::collections::BTreeSet<EntityKey>,
170 new_keys: std::collections::BTreeSet<EntityKey>,
172 original_record: Option<Record>,
174 trace_chains: std::collections::BTreeMap<EntityKey, Vec<teaql_core::TraceNode>>,
176 original_versions: std::collections::BTreeMap<EntityKey, i64>,
178 is_new: bool,
180}
181
182#[derive(Debug, Clone, Default)]
183pub struct EntityRoot {
184 inner: Arc<Mutex<RootContext>>,
185}
186
187impl PartialEq for EntityRoot {
188 fn eq(&self, other: &Self) -> bool {
189 Arc::ptr_eq(&self.inner, &other.inner)
190 }
191}
192
193impl EntityRoot {
194 pub fn push_change_set(&self) {
195 self.inner
196 .lock()
197 .unwrap_or_else(|e| e.into_inner())
198 .change_sets
199 .push();
200 }
201
202 pub fn pop_change_set(&self) -> Option<EntityChangeSet> {
203 self.inner
204 .lock()
205 .unwrap_or_else(|e| e.into_inner())
206 .change_sets
207 .pop()
208 }
209
210 pub fn clear_current_change_set(&self) {
211 self.inner
212 .lock()
213 .unwrap_or_else(|e| e.into_inner())
214 .change_sets
215 .clear_current();
216 }
217
218 pub fn set(&self, key: EntityKey, field: impl Into<String>, value: impl Into<Value>) {
219 self.inner
220 .lock()
221 .unwrap_or_else(|e| e.into_inner())
222 .change_sets
223 .set(key, field, value.into());
224 }
225
226 pub fn get(&self, key: &EntityKey, field: &str) -> Option<Value> {
227 self.inner
228 .lock()
229 .unwrap_or_else(|e| e.into_inner())
230 .change_sets
231 .get(key, field)
232 }
233
234 pub fn current_change_set(&self) -> EntityChangeSet {
235 self.inner
236 .lock()
237 .unwrap_or_else(|e| e.into_inner())
238 .change_sets
239 .current()
240 .cloned()
241 .unwrap_or_default()
242 }
243
244 pub fn set_comment(&self, comment: impl Into<String>) {
247 self.inner
248 .lock()
249 .unwrap_or_else(|e| e.into_inner())
250 .comment = Some(comment.into());
251 }
252
253 pub fn get_comment(&self) -> Option<String> {
255 self.inner
256 .lock()
257 .unwrap_or_else(|e| e.into_inner())
258 .comment
259 .clone()
260 }
261
262 pub fn mark_as_new(&self, key: EntityKey) {
264 self.inner
265 .lock()
266 .unwrap_or_else(|e| e.into_inner())
267 .new_keys
268 .insert(key);
269 }
270
271 pub fn is_new(&self, key: &EntityKey) -> bool {
273 self.inner
274 .lock()
275 .unwrap_or_else(|e| e.into_inner())
276 .new_keys
277 .contains(key)
278 }
279
280 pub fn set_original_record(&self, record: Record) {
282 self.inner
283 .lock()
284 .unwrap_or_else(|e| e.into_inner())
285 .original_record = Some(record);
286 }
287
288 pub fn original_record(&self) -> Option<Record> {
290 self.inner
291 .lock()
292 .unwrap_or_else(|e| e.into_inner())
293 .original_record
294 .clone()
295 }
296
297 pub fn mark_as_delete(&self, key: EntityKey) {
302 let mut ctx = self.inner.lock().unwrap_or_else(|e| e.into_inner());
303 ctx.change_sets.clear_entity(&key);
304 ctx.deleted_keys.insert(key);
305 }
306
307 pub fn is_marked_as_delete(&self, key: &EntityKey) -> bool {
309 self.inner
310 .lock()
311 .unwrap_or_else(|e| e.into_inner())
312 .deleted_keys
313 .contains(key)
314 }
315
316 pub fn changed_field_names(&self, key: &EntityKey) -> BTreeSet<String> {
319 self.inner
320 .lock()
321 .unwrap_or_else(|e| e.into_inner())
322 .change_sets
323 .changed_field_names(key)
324 }
325 pub fn deleted_keys(&self) -> std::collections::BTreeSet<EntityKey> {
326 self.inner.lock().unwrap_or_else(|e| e.into_inner()).deleted_keys.clone()
327 }
328
329 pub fn new_keys(&self) -> std::collections::BTreeSet<EntityKey> {
330 self.inner.lock().unwrap_or_else(|e| e.into_inner()).new_keys.clone()
331 }
332
333 pub fn get_original_version(&self, key: &EntityKey) -> Option<i64> {
334 self.inner.lock().unwrap_or_else(|e| e.into_inner()).original_versions.get(key).cloned()
335 }
336
337 pub fn get_trace_chain(&self, key: &EntityKey) -> Vec<teaql_core::TraceNode> {
338 self.inner.lock().unwrap_or_else(|e| e.into_inner()).trace_chains.get(key).cloned().unwrap_or_default()
339 }
340
341
342
343 pub fn set_original_version(&self, key: EntityKey, version: i64) {
344 self.inner.lock().unwrap_or_else(|e| e.into_inner()).original_versions.insert(key, version);
345 }
346}
347
348pub trait LedgerEntity: teaql_core::Entity {
349 fn entity_root(&self) -> Option<EntityRoot>;
350}