@lix-js/sdk 0.6.0-preview.1 → 0.6.0-preview.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (191) hide show
  1. package/SKILL.md +305 -320
  2. package/dist/engine-wasm/wasm/lix_engine.d.ts +5 -0
  3. package/dist/engine-wasm/wasm/lix_engine.js +9 -13
  4. package/dist/engine-wasm/wasm/lix_engine.wasm +0 -0
  5. package/dist/engine-wasm/wasm/lix_engine.wasm.d.ts +1 -0
  6. package/dist/open-lix.d.ts +103 -14
  7. package/dist/open-lix.js +3 -0
  8. package/dist/sqlite/index.js +99 -22
  9. package/dist-engine-src/README.md +18 -0
  10. package/dist-engine-src/src/backend/kv.rs +358 -0
  11. package/dist-engine-src/src/backend/mod.rs +12 -0
  12. package/dist-engine-src/src/backend/testing.rs +658 -0
  13. package/dist-engine-src/src/backend/types.rs +96 -0
  14. package/dist-engine-src/src/binary_cas/chunking.rs +31 -0
  15. package/dist-engine-src/src/binary_cas/codec.rs +346 -0
  16. package/dist-engine-src/src/binary_cas/context.rs +139 -0
  17. package/dist-engine-src/src/binary_cas/kv.rs +1063 -0
  18. package/dist-engine-src/src/binary_cas/mod.rs +11 -0
  19. package/dist-engine-src/src/binary_cas/types.rs +127 -0
  20. package/dist-engine-src/src/cel/context.rs +86 -0
  21. package/dist-engine-src/src/cel/error.rs +19 -0
  22. package/dist-engine-src/src/cel/mod.rs +8 -0
  23. package/dist-engine-src/src/cel/provider.rs +9 -0
  24. package/dist-engine-src/src/cel/runtime.rs +167 -0
  25. package/dist-engine-src/src/cel/value.rs +50 -0
  26. package/dist-engine-src/src/changelog/codec.rs +321 -0
  27. package/dist-engine-src/src/changelog/context.rs +92 -0
  28. package/dist-engine-src/src/changelog/materialization.rs +121 -0
  29. package/dist-engine-src/src/changelog/mod.rs +13 -0
  30. package/dist-engine-src/src/changelog/reader.rs +20 -0
  31. package/dist-engine-src/src/changelog/storage.rs +220 -0
  32. package/dist-engine-src/src/changelog/types.rs +38 -0
  33. package/dist-engine-src/src/commit_graph/context.rs +1588 -0
  34. package/dist-engine-src/src/commit_graph/mod.rs +12 -0
  35. package/dist-engine-src/src/commit_graph/types.rs +145 -0
  36. package/dist-engine-src/src/commit_graph/walker.rs +780 -0
  37. package/dist-engine-src/src/common/error.rs +313 -0
  38. package/dist-engine-src/src/common/fingerprint.rs +3 -0
  39. package/dist-engine-src/src/common/fs_path.rs +1336 -0
  40. package/dist-engine-src/src/common/identity.rs +135 -0
  41. package/dist-engine-src/src/common/metadata.rs +35 -0
  42. package/dist-engine-src/src/common/mod.rs +23 -0
  43. package/dist-engine-src/src/common/types.rs +105 -0
  44. package/dist-engine-src/src/common/wire.rs +222 -0
  45. package/dist-engine-src/src/engine.rs +239 -0
  46. package/dist-engine-src/src/entity_identity.rs +285 -0
  47. package/dist-engine-src/src/functions/context.rs +327 -0
  48. package/dist-engine-src/src/functions/deterministic.rs +113 -0
  49. package/dist-engine-src/src/functions/mod.rs +18 -0
  50. package/dist-engine-src/src/functions/provider.rs +130 -0
  51. package/dist-engine-src/src/functions/state.rs +363 -0
  52. package/dist-engine-src/src/functions/types.rs +37 -0
  53. package/dist-engine-src/src/init.rs +505 -0
  54. package/dist-engine-src/src/json_store/compression.rs +77 -0
  55. package/dist-engine-src/src/json_store/context.rs +129 -0
  56. package/dist-engine-src/src/json_store/encoded.rs +15 -0
  57. package/dist-engine-src/src/json_store/mod.rs +9 -0
  58. package/dist-engine-src/src/json_store/store.rs +236 -0
  59. package/dist-engine-src/src/json_store/types.rs +52 -0
  60. package/dist-engine-src/src/lib.rs +61 -0
  61. package/dist-engine-src/src/live_state/context.rs +2241 -0
  62. package/dist-engine-src/src/live_state/mod.rs +15 -0
  63. package/dist-engine-src/src/live_state/overlay.rs +75 -0
  64. package/dist-engine-src/src/live_state/reader.rs +23 -0
  65. package/dist-engine-src/src/live_state/types.rs +239 -0
  66. package/dist-engine-src/src/live_state/visibility.rs +218 -0
  67. package/dist-engine-src/src/plugin/archive.rs +441 -0
  68. package/dist-engine-src/src/plugin/component.rs +183 -0
  69. package/dist-engine-src/src/plugin/install.rs +637 -0
  70. package/dist-engine-src/src/plugin/manifest.rs +516 -0
  71. package/dist-engine-src/src/plugin/materializer.rs +477 -0
  72. package/dist-engine-src/src/plugin/mod.rs +33 -0
  73. package/dist-engine-src/src/plugin/plugin_manifest.json +119 -0
  74. package/dist-engine-src/src/plugin/storage.rs +74 -0
  75. package/dist-engine-src/src/schema/annotations/defaults.rs +280 -0
  76. package/dist-engine-src/src/schema/annotations/mod.rs +1 -0
  77. package/dist-engine-src/src/schema/builtin/lix_account.json +22 -0
  78. package/dist-engine-src/src/schema/builtin/lix_active_account.json +30 -0
  79. package/dist-engine-src/src/schema/builtin/lix_binary_blob_ref.json +30 -0
  80. package/dist-engine-src/src/schema/builtin/lix_change.json +62 -0
  81. package/dist-engine-src/src/schema/builtin/lix_change_author.json +46 -0
  82. package/dist-engine-src/src/schema/builtin/lix_change_set.json +18 -0
  83. package/dist-engine-src/src/schema/builtin/lix_change_set_element.json +75 -0
  84. package/dist-engine-src/src/schema/builtin/lix_commit.json +62 -0
  85. package/dist-engine-src/src/schema/builtin/lix_commit_edge.json +46 -0
  86. package/dist-engine-src/src/schema/builtin/lix_directory_descriptor.json +53 -0
  87. package/dist-engine-src/src/schema/builtin/lix_entity_label.json +63 -0
  88. package/dist-engine-src/src/schema/builtin/lix_file_descriptor.json +53 -0
  89. package/dist-engine-src/src/schema/builtin/lix_key_value.json +41 -0
  90. package/dist-engine-src/src/schema/builtin/lix_label.json +22 -0
  91. package/dist-engine-src/src/schema/builtin/lix_registered_schema.json +31 -0
  92. package/dist-engine-src/src/schema/builtin/lix_version_descriptor.json +35 -0
  93. package/dist-engine-src/src/schema/builtin/lix_version_ref.json +49 -0
  94. package/dist-engine-src/src/schema/builtin/mod.rs +271 -0
  95. package/dist-engine-src/src/schema/definition.json +157 -0
  96. package/dist-engine-src/src/schema/definition.rs +636 -0
  97. package/dist-engine-src/src/schema/key.rs +206 -0
  98. package/dist-engine-src/src/schema/mod.rs +20 -0
  99. package/dist-engine-src/src/schema/seed.rs +14 -0
  100. package/dist-engine-src/src/schema/tests.rs +739 -0
  101. package/dist-engine-src/src/schema_registry.rs +294 -0
  102. package/dist-engine-src/src/session/context.rs +366 -0
  103. package/dist-engine-src/src/session/create_version.rs +80 -0
  104. package/dist-engine-src/src/session/execute.rs +447 -0
  105. package/dist-engine-src/src/session/merge/analysis.rs +102 -0
  106. package/dist-engine-src/src/session/merge/apply.rs +23 -0
  107. package/dist-engine-src/src/session/merge/conflicts.rs +62 -0
  108. package/dist-engine-src/src/session/merge/mod.rs +11 -0
  109. package/dist-engine-src/src/session/merge/stats.rs +65 -0
  110. package/dist-engine-src/src/session/merge/version.rs +437 -0
  111. package/dist-engine-src/src/session/mod.rs +25 -0
  112. package/dist-engine-src/src/session/switch_version.rs +121 -0
  113. package/dist-engine-src/src/sql2/change_provider.rs +337 -0
  114. package/dist-engine-src/src/sql2/classify.rs +147 -0
  115. package/dist-engine-src/src/sql2/commit_derived_provider.rs +591 -0
  116. package/dist-engine-src/src/sql2/context.rs +307 -0
  117. package/dist-engine-src/src/sql2/directory_history_provider.rs +623 -0
  118. package/dist-engine-src/src/sql2/directory_provider.rs +2405 -0
  119. package/dist-engine-src/src/sql2/dml.rs +148 -0
  120. package/dist-engine-src/src/sql2/entity_history_provider.rs +444 -0
  121. package/dist-engine-src/src/sql2/entity_provider.rs +2700 -0
  122. package/dist-engine-src/src/sql2/error.rs +196 -0
  123. package/dist-engine-src/src/sql2/execute.rs +3379 -0
  124. package/dist-engine-src/src/sql2/file_history_provider.rs +902 -0
  125. package/dist-engine-src/src/sql2/file_provider.rs +3254 -0
  126. package/dist-engine-src/src/sql2/filesystem_planner.rs +1526 -0
  127. package/dist-engine-src/src/sql2/filesystem_predicates.rs +159 -0
  128. package/dist-engine-src/src/sql2/filesystem_visibility.rs +369 -0
  129. package/dist-engine-src/src/sql2/history_projection.rs +80 -0
  130. package/dist-engine-src/src/sql2/history_provider.rs +418 -0
  131. package/dist-engine-src/src/sql2/history_route.rs +643 -0
  132. package/dist-engine-src/src/sql2/lix_state_provider.rs +2430 -0
  133. package/dist-engine-src/src/sql2/mod.rs +43 -0
  134. package/dist-engine-src/src/sql2/read_only.rs +65 -0
  135. package/dist-engine-src/src/sql2/record_batch.rs +17 -0
  136. package/dist-engine-src/src/sql2/result_metadata.rs +29 -0
  137. package/dist-engine-src/src/sql2/runtime.rs +60 -0
  138. package/dist-engine-src/src/sql2/session.rs +135 -0
  139. package/dist-engine-src/src/sql2/udfs/common.rs +295 -0
  140. package/dist-engine-src/src/sql2/udfs/lix_active_version_commit_id.rs +53 -0
  141. package/dist-engine-src/src/sql2/udfs/lix_empty_blob.rs +47 -0
  142. package/dist-engine-src/src/sql2/udfs/lix_json.rs +100 -0
  143. package/dist-engine-src/src/sql2/udfs/lix_json_get.rs +99 -0
  144. package/dist-engine-src/src/sql2/udfs/lix_json_get_text.rs +99 -0
  145. package/dist-engine-src/src/sql2/udfs/lix_text_decode.rs +82 -0
  146. package/dist-engine-src/src/sql2/udfs/lix_text_encode.rs +85 -0
  147. package/dist-engine-src/src/sql2/udfs/lix_uuid_v7.rs +76 -0
  148. package/dist-engine-src/src/sql2/udfs/mod.rs +82 -0
  149. package/dist-engine-src/src/sql2/version_provider.rs +1187 -0
  150. package/dist-engine-src/src/sql2/version_scope.rs +394 -0
  151. package/dist-engine-src/src/sql2/write_normalization.rs +345 -0
  152. package/dist-engine-src/src/storage/context.rs +356 -0
  153. package/dist-engine-src/src/storage/mod.rs +14 -0
  154. package/dist-engine-src/src/storage/read_scope.rs +88 -0
  155. package/dist-engine-src/src/storage/types.rs +501 -0
  156. package/dist-engine-src/src/storage_bench.rs +3406 -0
  157. package/dist-engine-src/src/test_support.rs +81 -0
  158. package/dist-engine-src/src/tracked_state/by_file_index.rs +102 -0
  159. package/dist-engine-src/src/tracked_state/codec.rs +747 -0
  160. package/dist-engine-src/src/tracked_state/context.rs +983 -0
  161. package/dist-engine-src/src/tracked_state/diff.rs +494 -0
  162. package/dist-engine-src/src/tracked_state/materialization.rs +141 -0
  163. package/dist-engine-src/src/tracked_state/merge.rs +474 -0
  164. package/dist-engine-src/src/tracked_state/mod.rs +31 -0
  165. package/dist-engine-src/src/tracked_state/rebuild.rs +771 -0
  166. package/dist-engine-src/src/tracked_state/storage.rs +243 -0
  167. package/dist-engine-src/src/tracked_state/tree.rs +2744 -0
  168. package/dist-engine-src/src/tracked_state/tree_types.rs +176 -0
  169. package/dist-engine-src/src/tracked_state/types.rs +61 -0
  170. package/dist-engine-src/src/transaction/commit.rs +1224 -0
  171. package/dist-engine-src/src/transaction/context.rs +1307 -0
  172. package/dist-engine-src/src/transaction/live_state_overlay.rs +34 -0
  173. package/dist-engine-src/src/transaction/mod.rs +11 -0
  174. package/dist-engine-src/src/transaction/normalization.rs +1026 -0
  175. package/dist-engine-src/src/transaction/schema_resolver.rs +127 -0
  176. package/dist-engine-src/src/transaction/staging.rs +1436 -0
  177. package/dist-engine-src/src/transaction/types.rs +351 -0
  178. package/dist-engine-src/src/transaction/validation.rs +4811 -0
  179. package/dist-engine-src/src/untracked_state/codec.rs +363 -0
  180. package/dist-engine-src/src/untracked_state/context.rs +82 -0
  181. package/dist-engine-src/src/untracked_state/materialization.rs +157 -0
  182. package/dist-engine-src/src/untracked_state/mod.rs +17 -0
  183. package/dist-engine-src/src/untracked_state/storage.rs +348 -0
  184. package/dist-engine-src/src/untracked_state/types.rs +96 -0
  185. package/dist-engine-src/src/version/context.rs +52 -0
  186. package/dist-engine-src/src/version/mod.rs +12 -0
  187. package/dist-engine-src/src/version/refs.rs +421 -0
  188. package/dist-engine-src/src/version/stage_rows.rs +71 -0
  189. package/dist-engine-src/src/version/types.rs +21 -0
  190. package/dist-engine-src/src/wasm/mod.rs +60 -0
  191. package/package.json +68 -64
@@ -0,0 +1,474 @@
1
+ use std::collections::{BTreeMap, BTreeSet};
2
+
3
+ use crate::tracked_state::{
4
+ TrackedStateDiff, TrackedStateDiffEntry, TrackedStateDiffIdentity, TrackedStateRow,
5
+ };
6
+ use crate::LixError;
7
+
8
+ /// Planned tracked-state merge result.
9
+ ///
10
+ /// This is intentionally a pure planner. It does not know about versions,
11
+ /// sessions, changelog writes, or live-state overlays. Callers provide two
12
+ /// diffs from the same merge base:
13
+ ///
14
+ /// - `base -> target`: what the destination version changed.
15
+ /// - `base -> source`: what the incoming version changed.
16
+ ///
17
+ /// The planner returns source-side patches that can be applied to the target
18
+ /// root plus first-class conflicts for identities changed differently on both
19
+ /// sides.
20
+ #[derive(Debug, Clone, PartialEq, Eq, Default)]
21
+ pub(crate) struct TrackedStateMergePlan {
22
+ pub(crate) patches: Vec<TrackedStateMergePatch>,
23
+ pub(crate) conflicts: Vec<TrackedStateMergeConflict>,
24
+ }
25
+
26
+ /// One source-side patch to apply to the target root.
27
+ ///
28
+ /// Merge patches are expressed as canonical change adoption, not as new row
29
+ /// writes. The projected row carries the target-root materialization shape,
30
+ /// including tombstones, while `change_id` preserves the source canonical
31
+ /// change identity.
32
+ #[derive(Debug, Clone, PartialEq, Eq)]
33
+ pub(crate) enum TrackedStateMergePatch {
34
+ Adopt {
35
+ identity: TrackedStateDiffIdentity,
36
+ change_id: String,
37
+ projected_row: TrackedStateRow,
38
+ },
39
+ }
40
+
41
+ impl TrackedStateMergePatch {
42
+ #[cfg(test)]
43
+ pub(crate) fn identity(&self) -> &TrackedStateDiffIdentity {
44
+ match self {
45
+ Self::Adopt { identity, .. } => identity,
46
+ }
47
+ }
48
+
49
+ pub(crate) fn change_id(&self) -> &str {
50
+ match self {
51
+ Self::Adopt { change_id, .. } => change_id,
52
+ }
53
+ }
54
+
55
+ pub(crate) fn projected_row(&self) -> &TrackedStateRow {
56
+ match self {
57
+ Self::Adopt { projected_row, .. } => projected_row,
58
+ }
59
+ }
60
+ }
61
+
62
+ /// One identity that both sides changed incompatibly.
63
+ #[derive(Debug, Clone, PartialEq, Eq)]
64
+ pub(crate) struct TrackedStateMergeConflict {
65
+ pub(crate) identity: TrackedStateDiffIdentity,
66
+ pub(crate) target: TrackedStateDiffEntry,
67
+ pub(crate) source: TrackedStateDiffEntry,
68
+ }
69
+
70
+ /// Plans a three-way tracked-state merge from two base-relative diffs.
71
+ ///
72
+ /// This follows the same shape as prolly-tree merge systems: compare
73
+ /// `base -> target` and `base -> source` by identity, emit source-only patches
74
+ /// for the target root, ignore target-only changes, collapse convergent
75
+ /// changes, and report divergent same-identity changes as conflicts.
76
+ pub(crate) fn plan_merge(
77
+ target_diff: &TrackedStateDiff,
78
+ source_diff: &TrackedStateDiff,
79
+ ) -> Result<TrackedStateMergePlan, LixError> {
80
+ let target_by_identity = diff_by_identity(target_diff)?;
81
+ let source_by_identity = diff_by_identity(source_diff)?;
82
+ let identities = target_by_identity
83
+ .keys()
84
+ .chain(source_by_identity.keys())
85
+ .cloned()
86
+ .collect::<BTreeSet<_>>();
87
+
88
+ let mut plan = TrackedStateMergePlan::default();
89
+ for identity in identities {
90
+ match (
91
+ target_by_identity.get(&identity),
92
+ source_by_identity.get(&identity),
93
+ ) {
94
+ (None, None) => {}
95
+ (Some(_target), None) => {
96
+ // Target already changed this identity. Source did not, so
97
+ // there is nothing to apply.
98
+ }
99
+ (None, Some(source)) => {
100
+ plan.patches
101
+ .push(adopt_source_change_patch(identity, source)?);
102
+ }
103
+ (Some(target), Some(source)) if same_final_state(target, source) => {
104
+ // Both sides reached the same visible state. Keep target to
105
+ // avoid writing duplicate source metadata.
106
+ }
107
+ (Some(target), Some(source)) => {
108
+ plan.conflicts.push(TrackedStateMergeConflict {
109
+ identity,
110
+ target: (*target).clone(),
111
+ source: (*source).clone(),
112
+ });
113
+ }
114
+ }
115
+ }
116
+
117
+ Ok(plan)
118
+ }
119
+
120
+ fn diff_by_identity(
121
+ diff: &TrackedStateDiff,
122
+ ) -> Result<BTreeMap<TrackedStateDiffIdentity, &TrackedStateDiffEntry>, LixError> {
123
+ let mut entries = BTreeMap::new();
124
+ for entry in &diff.entries {
125
+ if entries.insert(entry.identity.clone(), entry).is_some() {
126
+ return Err(LixError::new(
127
+ "LIX_ERROR_UNKNOWN",
128
+ format!(
129
+ "tracked-state merge received duplicate diff entry for schema '{}' entity '{}'",
130
+ entry.identity.schema_key,
131
+ entry.identity.entity_id.as_string()?
132
+ ),
133
+ ));
134
+ }
135
+ }
136
+ Ok(entries)
137
+ }
138
+
139
+ fn adopt_source_change_patch(
140
+ identity: TrackedStateDiffIdentity,
141
+ entry: &TrackedStateDiffEntry,
142
+ ) -> Result<TrackedStateMergePatch, LixError> {
143
+ let Some(row) = entry.after.clone() else {
144
+ return Err(LixError::new(
145
+ "LIX_ERROR_UNKNOWN",
146
+ format!(
147
+ "tracked-state merge cannot apply source removal for schema '{}' entity '{}' without a tombstone row",
148
+ entry.identity.schema_key,
149
+ entry.identity.entity_id.as_string()?
150
+ ),
151
+ ));
152
+ };
153
+ Ok(TrackedStateMergePatch::Adopt {
154
+ identity,
155
+ change_id: row.change_id.clone(),
156
+ projected_row: row,
157
+ })
158
+ }
159
+
160
+ fn same_final_state(target: &TrackedStateDiffEntry, source: &TrackedStateDiffEntry) -> bool {
161
+ match (target.after.as_ref(), source.after.as_ref()) {
162
+ (None, None) => true,
163
+ (Some(target), Some(source)) if !row_is_live(target) && !row_is_live(source) => true,
164
+ (Some(target), Some(source)) if row_is_live(target) && row_is_live(source) => {
165
+ tracked_row_payload_eq(target, source)
166
+ }
167
+ _ => false,
168
+ }
169
+ }
170
+
171
+ fn row_is_live(row: &TrackedStateRow) -> bool {
172
+ row.snapshot_content.is_some()
173
+ }
174
+
175
+ fn tracked_row_payload_eq(left: &TrackedStateRow, right: &TrackedStateRow) -> bool {
176
+ left.snapshot_content == right.snapshot_content
177
+ && left.metadata == right.metadata
178
+ && left.schema_version == right.schema_version
179
+ }
180
+
181
+ #[cfg(test)]
182
+ mod tests {
183
+ use super::*;
184
+ use crate::entity_identity::EntityIdentity;
185
+ use crate::tracked_state::TrackedStateDiffKind;
186
+
187
+ #[test]
188
+ fn source_add_applies() {
189
+ let plan = plan_merge(
190
+ &TrackedStateDiff::default(),
191
+ &diff(vec![entry(
192
+ "entity-a",
193
+ TrackedStateDiffKind::Added,
194
+ None,
195
+ Some(row("entity-a", "source")),
196
+ )]),
197
+ )
198
+ .expect("merge should plan");
199
+
200
+ assert_eq!(patch_ids(&plan), vec!["entity-a"]);
201
+ assert!(plan.conflicts.is_empty());
202
+ }
203
+
204
+ #[test]
205
+ fn source_modify_applies() {
206
+ let plan = plan_merge(
207
+ &TrackedStateDiff::default(),
208
+ &diff(vec![entry(
209
+ "entity-a",
210
+ TrackedStateDiffKind::Modified,
211
+ Some(row_with_value("entity-a", "base", "base")),
212
+ Some(row_with_value("entity-a", "source", "source")),
213
+ )]),
214
+ )
215
+ .expect("merge should plan");
216
+
217
+ assert_eq!(patch_ids(&plan), vec!["entity-a"]);
218
+ assert_eq!(
219
+ plan.patches[0].projected_row().snapshot_content.as_deref(),
220
+ Some("{\"value\":\"source\"}")
221
+ );
222
+ assert_eq!(plan.patches[0].change_id(), "source");
223
+ }
224
+
225
+ #[test]
226
+ fn source_delete_applies_tombstone() {
227
+ let plan = plan_merge(
228
+ &TrackedStateDiff::default(),
229
+ &diff(vec![entry(
230
+ "entity-a",
231
+ TrackedStateDiffKind::Removed,
232
+ Some(row("entity-a", "base")),
233
+ Some(tombstone("entity-a", "source-delete")),
234
+ )]),
235
+ )
236
+ .expect("merge should plan");
237
+
238
+ assert_eq!(patch_ids(&plan), vec!["entity-a"]);
239
+ assert_eq!(plan.patches[0].projected_row().snapshot_content, None);
240
+ assert_eq!(plan.patches[0].change_id(), "source-delete");
241
+ }
242
+
243
+ #[test]
244
+ fn target_only_change_is_noop() {
245
+ let plan = plan_merge(
246
+ &diff(vec![entry(
247
+ "entity-a",
248
+ TrackedStateDiffKind::Modified,
249
+ Some(row("entity-a", "base")),
250
+ Some(row("entity-a", "target")),
251
+ )]),
252
+ &TrackedStateDiff::default(),
253
+ )
254
+ .expect("merge should plan");
255
+
256
+ assert!(plan.patches.is_empty());
257
+ assert!(plan.conflicts.is_empty());
258
+ }
259
+
260
+ #[test]
261
+ fn both_sides_same_final_value_is_convergent_noop() {
262
+ let target = entry(
263
+ "entity-a",
264
+ TrackedStateDiffKind::Modified,
265
+ Some(row_with_value("entity-a", "base", "base")),
266
+ Some(row_with_value("entity-a", "target", "same")),
267
+ );
268
+ let source = entry(
269
+ "entity-a",
270
+ TrackedStateDiffKind::Modified,
271
+ Some(row_with_value("entity-a", "base", "base")),
272
+ Some(row_with_value("entity-a", "source", "same")),
273
+ );
274
+
275
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
276
+
277
+ assert!(plan.patches.is_empty());
278
+ assert!(plan.conflicts.is_empty());
279
+ }
280
+
281
+ #[test]
282
+ fn both_sides_delete_is_convergent_noop() {
283
+ let target = entry(
284
+ "entity-a",
285
+ TrackedStateDiffKind::Removed,
286
+ Some(row("entity-a", "base")),
287
+ Some(tombstone("entity-a", "target-delete")),
288
+ );
289
+ let source = entry(
290
+ "entity-a",
291
+ TrackedStateDiffKind::Removed,
292
+ Some(row("entity-a", "base")),
293
+ Some(tombstone("entity-a", "source-delete")),
294
+ );
295
+
296
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
297
+
298
+ assert!(plan.patches.is_empty());
299
+ assert!(plan.conflicts.is_empty());
300
+ }
301
+
302
+ #[test]
303
+ fn different_modifications_conflict() {
304
+ let target = entry(
305
+ "entity-a",
306
+ TrackedStateDiffKind::Modified,
307
+ Some(row_with_value("entity-a", "base", "base")),
308
+ Some(row_with_value("entity-a", "target", "target")),
309
+ );
310
+ let source = entry(
311
+ "entity-a",
312
+ TrackedStateDiffKind::Modified,
313
+ Some(row_with_value("entity-a", "base", "base")),
314
+ Some(row_with_value("entity-a", "source", "source")),
315
+ );
316
+
317
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
318
+
319
+ assert!(plan.patches.is_empty());
320
+ assert_eq!(conflict_ids(&plan), vec!["entity-a"]);
321
+ }
322
+
323
+ #[test]
324
+ fn delete_modify_conflicts() {
325
+ let target = entry(
326
+ "entity-a",
327
+ TrackedStateDiffKind::Removed,
328
+ Some(row("entity-a", "base")),
329
+ Some(tombstone("entity-a", "target-delete")),
330
+ );
331
+ let source = entry(
332
+ "entity-a",
333
+ TrackedStateDiffKind::Modified,
334
+ Some(row("entity-a", "base")),
335
+ Some(row_with_value("entity-a", "source", "source")),
336
+ );
337
+
338
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
339
+
340
+ assert_eq!(conflict_ids(&plan), vec!["entity-a"]);
341
+ }
342
+
343
+ #[test]
344
+ fn modify_delete_conflicts() {
345
+ let target = entry(
346
+ "entity-a",
347
+ TrackedStateDiffKind::Modified,
348
+ Some(row("entity-a", "base")),
349
+ Some(row_with_value("entity-a", "target", "target")),
350
+ );
351
+ let source = entry(
352
+ "entity-a",
353
+ TrackedStateDiffKind::Removed,
354
+ Some(row("entity-a", "base")),
355
+ Some(tombstone("entity-a", "source-delete")),
356
+ );
357
+
358
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
359
+
360
+ assert_eq!(conflict_ids(&plan), vec!["entity-a"]);
361
+ }
362
+
363
+ #[test]
364
+ fn source_removal_without_tombstone_errors() {
365
+ let error = plan_merge(
366
+ &TrackedStateDiff::default(),
367
+ &diff(vec![entry(
368
+ "entity-a",
369
+ TrackedStateDiffKind::Removed,
370
+ Some(row("entity-a", "base")),
371
+ None,
372
+ )]),
373
+ )
374
+ .expect_err("merge should reject impossible source removal");
375
+
376
+ assert!(error.message.contains("without a tombstone row"));
377
+ }
378
+
379
+ #[test]
380
+ fn patch_and_conflict_order_is_deterministic_by_identity() {
381
+ let target = diff(vec![entry(
382
+ "entity-b",
383
+ TrackedStateDiffKind::Modified,
384
+ Some(row_with_value("entity-b", "base", "base")),
385
+ Some(row_with_value("entity-b", "target", "target")),
386
+ )]);
387
+ let source = diff(vec![
388
+ entry(
389
+ "entity-c",
390
+ TrackedStateDiffKind::Added,
391
+ None,
392
+ Some(row("entity-c", "source-c")),
393
+ ),
394
+ entry(
395
+ "entity-a",
396
+ TrackedStateDiffKind::Added,
397
+ None,
398
+ Some(row("entity-a", "source-a")),
399
+ ),
400
+ entry(
401
+ "entity-b",
402
+ TrackedStateDiffKind::Modified,
403
+ Some(row_with_value("entity-b", "base", "base")),
404
+ Some(row_with_value("entity-b", "source", "source")),
405
+ ),
406
+ ]);
407
+
408
+ let plan = plan_merge(&target, &source).expect("merge should plan");
409
+
410
+ assert_eq!(patch_ids(&plan), vec!["entity-a", "entity-c"]);
411
+ assert_eq!(conflict_ids(&plan), vec!["entity-b"]);
412
+ }
413
+
414
+ fn diff(entries: Vec<TrackedStateDiffEntry>) -> TrackedStateDiff {
415
+ TrackedStateDiff { entries }
416
+ }
417
+
418
+ fn entry(
419
+ entity_id: &str,
420
+ kind: TrackedStateDiffKind,
421
+ before: Option<TrackedStateRow>,
422
+ after: Option<TrackedStateRow>,
423
+ ) -> TrackedStateDiffEntry {
424
+ TrackedStateDiffEntry {
425
+ identity: TrackedStateDiffIdentity {
426
+ schema_key: "test_schema".to_string(),
427
+ entity_id: EntityIdentity::single(entity_id),
428
+ file_id: None,
429
+ },
430
+ kind,
431
+ before,
432
+ after,
433
+ }
434
+ }
435
+
436
+ fn patch_ids(plan: &TrackedStateMergePlan) -> Vec<String> {
437
+ plan.patches
438
+ .iter()
439
+ .map(|entry| entry.identity().entity_id.as_string().expect("identity"))
440
+ .collect()
441
+ }
442
+
443
+ fn conflict_ids(plan: &TrackedStateMergePlan) -> Vec<String> {
444
+ plan.conflicts
445
+ .iter()
446
+ .map(|entry| entry.identity.entity_id.as_string().expect("identity"))
447
+ .collect()
448
+ }
449
+
450
+ fn tombstone(entity_id: &str, change_id: &str) -> TrackedStateRow {
451
+ let mut row = row(entity_id, change_id);
452
+ row.snapshot_content = None;
453
+ row
454
+ }
455
+
456
+ fn row(entity_id: &str, change_id: &str) -> TrackedStateRow {
457
+ row_with_value(entity_id, change_id, "value")
458
+ }
459
+
460
+ fn row_with_value(entity_id: &str, change_id: &str, value: &str) -> TrackedStateRow {
461
+ TrackedStateRow {
462
+ entity_id: EntityIdentity::single(entity_id),
463
+ schema_key: "test_schema".to_string(),
464
+ file_id: None,
465
+ snapshot_content: Some(format!("{{\"value\":\"{value}\"}}")),
466
+ metadata: None,
467
+ schema_version: "1".to_string(),
468
+ created_at: "2026-01-01T00:00:00Z".to_string(),
469
+ updated_at: "2026-01-01T00:00:00Z".to_string(),
470
+ change_id: change_id.to_string(),
471
+ commit_id: change_id.replace("change", "commit"),
472
+ }
473
+ }
474
+ }
@@ -0,0 +1,31 @@
1
+ mod by_file_index;
2
+ mod codec;
3
+ mod context;
4
+ mod diff;
5
+ mod materialization;
6
+ mod merge;
7
+ pub(crate) mod rebuild;
8
+ mod storage;
9
+ mod tree;
10
+ mod tree_types;
11
+ mod types;
12
+
13
+ #[allow(unused_imports)]
14
+ pub(crate) use context::{TrackedStateContext, TrackedStateStoreReader, TrackedStateWriter};
15
+ #[allow(unused_imports)]
16
+ pub(crate) use diff::{
17
+ TrackedStateDiff, TrackedStateDiffEntry, TrackedStateDiffIdentity, TrackedStateDiffKind,
18
+ TrackedStateDiffRequest,
19
+ };
20
+ pub(crate) use materialization::{
21
+ canonicalize_materialized_row, materialize_value, TrackedMaterializationProjection,
22
+ };
23
+ #[allow(unused_imports)]
24
+ pub(crate) use merge::{
25
+ plan_merge, TrackedStateMergeConflict, TrackedStateMergePatch, TrackedStateMergePlan,
26
+ };
27
+ #[allow(unused_imports)]
28
+ pub(crate) use types::{
29
+ TrackedStateFilter, TrackedStateProjection, TrackedStateRow, TrackedStateRowRequest,
30
+ TrackedStateScanRequest,
31
+ };