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

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 (205) hide show
  1. package/SKILL.md +304 -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/generated/builtin-schemas.d.ts +87 -162
  7. package/dist/generated/builtin-schemas.js +139 -236
  8. package/dist/open-lix.d.ts +103 -14
  9. package/dist/open-lix.js +3 -0
  10. package/dist/sqlite/index.js +99 -22
  11. package/dist-engine-src/README.md +18 -0
  12. package/dist-engine-src/src/backend/kv.rs +358 -0
  13. package/dist-engine-src/src/backend/mod.rs +12 -0
  14. package/dist-engine-src/src/backend/testing.rs +658 -0
  15. package/dist-engine-src/src/backend/types.rs +96 -0
  16. package/dist-engine-src/src/binary_cas/chunking.rs +31 -0
  17. package/dist-engine-src/src/binary_cas/codec.rs +346 -0
  18. package/dist-engine-src/src/binary_cas/context.rs +139 -0
  19. package/dist-engine-src/src/binary_cas/kv.rs +1063 -0
  20. package/dist-engine-src/src/binary_cas/mod.rs +11 -0
  21. package/dist-engine-src/src/binary_cas/types.rs +121 -0
  22. package/dist-engine-src/src/catalog/context.rs +412 -0
  23. package/dist-engine-src/src/catalog/mod.rs +10 -0
  24. package/dist-engine-src/src/catalog/schema.rs +4 -0
  25. package/dist-engine-src/src/catalog/snapshot.rs +1114 -0
  26. package/dist-engine-src/src/cel/context.rs +86 -0
  27. package/dist-engine-src/src/cel/error.rs +19 -0
  28. package/dist-engine-src/src/cel/mod.rs +8 -0
  29. package/dist-engine-src/src/cel/provider.rs +9 -0
  30. package/dist-engine-src/src/cel/runtime.rs +167 -0
  31. package/dist-engine-src/src/cel/value.rs +50 -0
  32. package/dist-engine-src/src/commit_graph/context.rs +901 -0
  33. package/dist-engine-src/src/commit_graph/mod.rs +11 -0
  34. package/dist-engine-src/src/commit_graph/types.rs +109 -0
  35. package/dist-engine-src/src/commit_graph/walker.rs +756 -0
  36. package/dist-engine-src/src/commit_store/codec.rs +887 -0
  37. package/dist-engine-src/src/commit_store/context.rs +944 -0
  38. package/dist-engine-src/src/commit_store/materialization.rs +84 -0
  39. package/dist-engine-src/src/commit_store/mod.rs +16 -0
  40. package/dist-engine-src/src/commit_store/storage.rs +600 -0
  41. package/dist-engine-src/src/commit_store/types.rs +215 -0
  42. package/dist-engine-src/src/common/error.rs +313 -0
  43. package/dist-engine-src/src/common/fingerprint.rs +3 -0
  44. package/dist-engine-src/src/common/fs_path.rs +1336 -0
  45. package/dist-engine-src/src/common/identity.rs +145 -0
  46. package/dist-engine-src/src/common/json_pointer.rs +67 -0
  47. package/dist-engine-src/src/common/metadata.rs +40 -0
  48. package/dist-engine-src/src/common/mod.rs +23 -0
  49. package/dist-engine-src/src/common/types.rs +105 -0
  50. package/dist-engine-src/src/common/wire.rs +222 -0
  51. package/dist-engine-src/src/domain.rs +324 -0
  52. package/dist-engine-src/src/engine.rs +225 -0
  53. package/dist-engine-src/src/entity_identity.rs +405 -0
  54. package/dist-engine-src/src/functions/context.rs +292 -0
  55. package/dist-engine-src/src/functions/deterministic.rs +113 -0
  56. package/dist-engine-src/src/functions/mod.rs +18 -0
  57. package/dist-engine-src/src/functions/provider.rs +130 -0
  58. package/dist-engine-src/src/functions/state.rs +336 -0
  59. package/dist-engine-src/src/functions/types.rs +37 -0
  60. package/dist-engine-src/src/init.rs +558 -0
  61. package/dist-engine-src/src/json_store/compression.rs +77 -0
  62. package/dist-engine-src/src/json_store/context.rs +423 -0
  63. package/dist-engine-src/src/json_store/encoded.rs +15 -0
  64. package/dist-engine-src/src/json_store/mod.rs +12 -0
  65. package/dist-engine-src/src/json_store/store.rs +1109 -0
  66. package/dist-engine-src/src/json_store/types.rs +217 -0
  67. package/dist-engine-src/src/lib.rs +62 -0
  68. package/dist-engine-src/src/live_state/context.rs +2019 -0
  69. package/dist-engine-src/src/live_state/mod.rs +15 -0
  70. package/dist-engine-src/src/live_state/overlay.rs +75 -0
  71. package/dist-engine-src/src/live_state/reader.rs +23 -0
  72. package/dist-engine-src/src/live_state/types.rs +222 -0
  73. package/dist-engine-src/src/live_state/visibility.rs +223 -0
  74. package/dist-engine-src/src/plugin/archive.rs +438 -0
  75. package/dist-engine-src/src/plugin/component.rs +183 -0
  76. package/dist-engine-src/src/plugin/install.rs +619 -0
  77. package/dist-engine-src/src/plugin/manifest.rs +516 -0
  78. package/dist-engine-src/src/plugin/materializer.rs +477 -0
  79. package/dist-engine-src/src/plugin/mod.rs +33 -0
  80. package/dist-engine-src/src/plugin/plugin_manifest.json +118 -0
  81. package/dist-engine-src/src/plugin/storage.rs +74 -0
  82. package/dist-engine-src/src/schema/annotations/defaults.rs +275 -0
  83. package/dist-engine-src/src/schema/annotations/mod.rs +1 -0
  84. package/dist-engine-src/src/schema/builtin/lix_account.json +21 -0
  85. package/dist-engine-src/src/schema/builtin/lix_active_account.json +29 -0
  86. package/dist-engine-src/src/schema/builtin/lix_binary_blob_ref.json +29 -0
  87. package/dist-engine-src/src/schema/builtin/lix_change.json +63 -0
  88. package/dist-engine-src/src/schema/builtin/lix_change_author.json +45 -0
  89. package/dist-engine-src/src/schema/builtin/lix_commit.json +24 -0
  90. package/dist-engine-src/src/schema/builtin/lix_commit_edge.json +53 -0
  91. package/dist-engine-src/src/schema/builtin/lix_directory_descriptor.json +52 -0
  92. package/dist-engine-src/src/schema/builtin/lix_file_descriptor.json +52 -0
  93. package/dist-engine-src/src/schema/builtin/lix_key_value.json +40 -0
  94. package/dist-engine-src/src/schema/builtin/lix_label.json +29 -0
  95. package/dist-engine-src/src/schema/builtin/lix_label_assignment.json +74 -0
  96. package/dist-engine-src/src/schema/builtin/lix_registered_schema.json +25 -0
  97. package/dist-engine-src/src/schema/builtin/lix_version_descriptor.json +34 -0
  98. package/dist-engine-src/src/schema/builtin/lix_version_ref.json +48 -0
  99. package/dist-engine-src/src/schema/builtin/mod.rs +222 -0
  100. package/dist-engine-src/src/schema/compatibility.rs +787 -0
  101. package/dist-engine-src/src/schema/definition.json +187 -0
  102. package/dist-engine-src/src/schema/definition.rs +742 -0
  103. package/dist-engine-src/src/schema/key.rs +138 -0
  104. package/dist-engine-src/src/schema/mod.rs +20 -0
  105. package/dist-engine-src/src/schema/seed.rs +14 -0
  106. package/dist-engine-src/src/schema/tests.rs +780 -0
  107. package/dist-engine-src/src/session/context.rs +364 -0
  108. package/dist-engine-src/src/session/create_version.rs +88 -0
  109. package/dist-engine-src/src/session/execute.rs +478 -0
  110. package/dist-engine-src/src/session/merge/analysis.rs +102 -0
  111. package/dist-engine-src/src/session/merge/apply.rs +23 -0
  112. package/dist-engine-src/src/session/merge/conflicts.rs +63 -0
  113. package/dist-engine-src/src/session/merge/mod.rs +11 -0
  114. package/dist-engine-src/src/session/merge/stats.rs +65 -0
  115. package/dist-engine-src/src/session/merge/version.rs +427 -0
  116. package/dist-engine-src/src/session/mod.rs +27 -0
  117. package/dist-engine-src/src/session/optimization9_sql2_bench.rs +100 -0
  118. package/dist-engine-src/src/session/switch_version.rs +109 -0
  119. package/dist-engine-src/src/sql2/change_provider.rs +331 -0
  120. package/dist-engine-src/src/sql2/classify.rs +182 -0
  121. package/dist-engine-src/src/sql2/context.rs +311 -0
  122. package/dist-engine-src/src/sql2/directory_history_provider.rs +631 -0
  123. package/dist-engine-src/src/sql2/directory_provider.rs +2453 -0
  124. package/dist-engine-src/src/sql2/dml.rs +148 -0
  125. package/dist-engine-src/src/sql2/entity_history_provider.rs +440 -0
  126. package/dist-engine-src/src/sql2/entity_provider.rs +3211 -0
  127. package/dist-engine-src/src/sql2/error.rs +216 -0
  128. package/dist-engine-src/src/sql2/execute.rs +3440 -0
  129. package/dist-engine-src/src/sql2/file_history_provider.rs +910 -0
  130. package/dist-engine-src/src/sql2/file_provider.rs +3679 -0
  131. package/dist-engine-src/src/sql2/filesystem_planner.rs +1490 -0
  132. package/dist-engine-src/src/sql2/filesystem_predicates.rs +159 -0
  133. package/dist-engine-src/src/sql2/filesystem_visibility.rs +383 -0
  134. package/dist-engine-src/src/sql2/history_projection.rs +56 -0
  135. package/dist-engine-src/src/sql2/history_provider.rs +412 -0
  136. package/dist-engine-src/src/sql2/history_route.rs +657 -0
  137. package/dist-engine-src/src/sql2/lix_state_provider.rs +2512 -0
  138. package/dist-engine-src/src/sql2/mod.rs +46 -0
  139. package/dist-engine-src/src/sql2/predicate_typecheck.rs +246 -0
  140. package/dist-engine-src/src/sql2/public_bind/assignment.rs +46 -0
  141. package/dist-engine-src/src/sql2/public_bind/capability.rs +41 -0
  142. package/dist-engine-src/src/sql2/public_bind/dml.rs +166 -0
  143. package/dist-engine-src/src/sql2/public_bind/mod.rs +25 -0
  144. package/dist-engine-src/src/sql2/public_bind/table.rs +168 -0
  145. package/dist-engine-src/src/sql2/read_only.rs +63 -0
  146. package/dist-engine-src/src/sql2/record_batch.rs +17 -0
  147. package/dist-engine-src/src/sql2/result_metadata.rs +29 -0
  148. package/dist-engine-src/src/sql2/runtime.rs +60 -0
  149. package/dist-engine-src/src/sql2/session.rs +132 -0
  150. package/dist-engine-src/src/sql2/udfs/common.rs +295 -0
  151. package/dist-engine-src/src/sql2/udfs/lix_active_version_commit_id.rs +53 -0
  152. package/dist-engine-src/src/sql2/udfs/lix_empty_blob.rs +47 -0
  153. package/dist-engine-src/src/sql2/udfs/lix_json.rs +100 -0
  154. package/dist-engine-src/src/sql2/udfs/lix_json_get.rs +99 -0
  155. package/dist-engine-src/src/sql2/udfs/lix_json_get_text.rs +99 -0
  156. package/dist-engine-src/src/sql2/udfs/lix_text_decode.rs +82 -0
  157. package/dist-engine-src/src/sql2/udfs/lix_text_encode.rs +85 -0
  158. package/dist-engine-src/src/sql2/udfs/lix_timestamp.rs +76 -0
  159. package/dist-engine-src/src/sql2/udfs/lix_uuid_v7.rs +76 -0
  160. package/dist-engine-src/src/sql2/udfs/mod.rs +89 -0
  161. package/dist-engine-src/src/sql2/udfs/public_call.rs +211 -0
  162. package/dist-engine-src/src/sql2/version_provider.rs +1202 -0
  163. package/dist-engine-src/src/sql2/version_scope.rs +394 -0
  164. package/dist-engine-src/src/sql2/write_normalization.rs +345 -0
  165. package/dist-engine-src/src/storage/context.rs +356 -0
  166. package/dist-engine-src/src/storage/mod.rs +14 -0
  167. package/dist-engine-src/src/storage/read_scope.rs +88 -0
  168. package/dist-engine-src/src/storage/types.rs +501 -0
  169. package/dist-engine-src/src/storage_bench.rs +4863 -0
  170. package/dist-engine-src/src/test_support.rs +228 -0
  171. package/dist-engine-src/src/tracked_state/by_file_index.rs +98 -0
  172. package/dist-engine-src/src/tracked_state/codec.rs +2085 -0
  173. package/dist-engine-src/src/tracked_state/context.rs +1867 -0
  174. package/dist-engine-src/src/tracked_state/diff.rs +686 -0
  175. package/dist-engine-src/src/tracked_state/materialization.rs +403 -0
  176. package/dist-engine-src/src/tracked_state/materializer.rs +488 -0
  177. package/dist-engine-src/src/tracked_state/merge.rs +492 -0
  178. package/dist-engine-src/src/tracked_state/mod.rs +32 -0
  179. package/dist-engine-src/src/tracked_state/storage.rs +375 -0
  180. package/dist-engine-src/src/tracked_state/tree.rs +3187 -0
  181. package/dist-engine-src/src/tracked_state/types.rs +231 -0
  182. package/dist-engine-src/src/transaction/commit.rs +1484 -0
  183. package/dist-engine-src/src/transaction/context.rs +1548 -0
  184. package/dist-engine-src/src/transaction/live_state_overlay.rs +35 -0
  185. package/dist-engine-src/src/transaction/mod.rs +13 -0
  186. package/dist-engine-src/src/transaction/normalization.rs +890 -0
  187. package/dist-engine-src/src/transaction/prep.rs +37 -0
  188. package/dist-engine-src/src/transaction/schema_resolver.rs +149 -0
  189. package/dist-engine-src/src/transaction/staging.rs +1731 -0
  190. package/dist-engine-src/src/transaction/types.rs +460 -0
  191. package/dist-engine-src/src/transaction/validation.rs +5830 -0
  192. package/dist-engine-src/src/untracked_state/codec.rs +307 -0
  193. package/dist-engine-src/src/untracked_state/context.rs +98 -0
  194. package/dist-engine-src/src/untracked_state/materialization.rs +63 -0
  195. package/dist-engine-src/src/untracked_state/mod.rs +15 -0
  196. package/dist-engine-src/src/untracked_state/storage.rs +396 -0
  197. package/dist-engine-src/src/untracked_state/types.rs +146 -0
  198. package/dist-engine-src/src/version/context.rs +40 -0
  199. package/dist-engine-src/src/version/lifecycle.rs +221 -0
  200. package/dist-engine-src/src/version/mod.rs +13 -0
  201. package/dist-engine-src/src/version/refs.rs +330 -0
  202. package/dist-engine-src/src/version/stage_rows.rs +67 -0
  203. package/dist-engine-src/src/version/types.rs +21 -0
  204. package/dist-engine-src/src/wasm/mod.rs +60 -0
  205. package/package.json +68 -64
@@ -0,0 +1,492 @@
1
+ use std::collections::{BTreeMap, BTreeSet};
2
+
3
+ use crate::tracked_state::{
4
+ MaterializedTrackedStateRow, TrackedStateDiff, TrackedStateDiffEntry, TrackedStateDiffIdentity,
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: MaterializedTrackedStateRow,
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) -> &MaterializedTrackedStateRow {
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_json_array_text()?
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_json_array_text()?
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: &MaterializedTrackedStateRow) -> bool {
172
+ row.snapshot_content.is_some()
173
+ }
174
+
175
+ fn tracked_row_payload_eq(
176
+ left: &MaterializedTrackedStateRow,
177
+ right: &MaterializedTrackedStateRow,
178
+ ) -> bool {
179
+ left.snapshot_content == right.snapshot_content && left.metadata == right.metadata
180
+ }
181
+
182
+ #[cfg(test)]
183
+ mod tests {
184
+ use super::*;
185
+ use crate::entity_identity::EntityIdentity;
186
+ use crate::tracked_state::TrackedStateDiffKind;
187
+
188
+ #[test]
189
+ fn source_add_applies() {
190
+ let plan = plan_merge(
191
+ &TrackedStateDiff::default(),
192
+ &diff(vec![entry(
193
+ "entity-a",
194
+ TrackedStateDiffKind::Added,
195
+ None,
196
+ Some(row("entity-a", "source")),
197
+ )]),
198
+ )
199
+ .expect("merge should plan");
200
+
201
+ assert_eq!(patch_ids(&plan), vec!["entity-a"]);
202
+ assert!(plan.conflicts.is_empty());
203
+ }
204
+
205
+ #[test]
206
+ fn source_modify_applies() {
207
+ let plan = plan_merge(
208
+ &TrackedStateDiff::default(),
209
+ &diff(vec![entry(
210
+ "entity-a",
211
+ TrackedStateDiffKind::Modified,
212
+ Some(row_with_value("entity-a", "base", "base")),
213
+ Some(row_with_value("entity-a", "source", "source")),
214
+ )]),
215
+ )
216
+ .expect("merge should plan");
217
+
218
+ assert_eq!(patch_ids(&plan), vec!["entity-a"]);
219
+ assert_eq!(
220
+ plan.patches[0].projected_row().snapshot_content.as_deref(),
221
+ Some("{\"value\":\"source\"}")
222
+ );
223
+ assert_eq!(plan.patches[0].change_id(), "source");
224
+ }
225
+
226
+ #[test]
227
+ fn source_delete_applies_tombstone() {
228
+ let plan = plan_merge(
229
+ &TrackedStateDiff::default(),
230
+ &diff(vec![entry(
231
+ "entity-a",
232
+ TrackedStateDiffKind::Removed,
233
+ Some(row("entity-a", "base")),
234
+ Some(tombstone("entity-a", "source-delete")),
235
+ )]),
236
+ )
237
+ .expect("merge should plan");
238
+
239
+ assert_eq!(patch_ids(&plan), vec!["entity-a"]);
240
+ assert_eq!(plan.patches[0].projected_row().snapshot_content, None);
241
+ assert_eq!(plan.patches[0].change_id(), "source-delete");
242
+ }
243
+
244
+ #[test]
245
+ fn target_only_change_is_noop() {
246
+ let plan = plan_merge(
247
+ &diff(vec![entry(
248
+ "entity-a",
249
+ TrackedStateDiffKind::Modified,
250
+ Some(row("entity-a", "base")),
251
+ Some(row("entity-a", "target")),
252
+ )]),
253
+ &TrackedStateDiff::default(),
254
+ )
255
+ .expect("merge should plan");
256
+
257
+ assert!(plan.patches.is_empty());
258
+ assert!(plan.conflicts.is_empty());
259
+ }
260
+
261
+ #[test]
262
+ fn both_sides_same_final_value_is_convergent_noop() {
263
+ let target = entry(
264
+ "entity-a",
265
+ TrackedStateDiffKind::Modified,
266
+ Some(row_with_value("entity-a", "base", "base")),
267
+ Some(row_with_value("entity-a", "target", "same")),
268
+ );
269
+ let source = entry(
270
+ "entity-a",
271
+ TrackedStateDiffKind::Modified,
272
+ Some(row_with_value("entity-a", "base", "base")),
273
+ Some(row_with_value("entity-a", "source", "same")),
274
+ );
275
+
276
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
277
+
278
+ assert!(plan.patches.is_empty());
279
+ assert!(plan.conflicts.is_empty());
280
+ }
281
+
282
+ #[test]
283
+ fn both_sides_delete_is_convergent_noop() {
284
+ let target = entry(
285
+ "entity-a",
286
+ TrackedStateDiffKind::Removed,
287
+ Some(row("entity-a", "base")),
288
+ Some(tombstone("entity-a", "target-delete")),
289
+ );
290
+ let source = entry(
291
+ "entity-a",
292
+ TrackedStateDiffKind::Removed,
293
+ Some(row("entity-a", "base")),
294
+ Some(tombstone("entity-a", "source-delete")),
295
+ );
296
+
297
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
298
+
299
+ assert!(plan.patches.is_empty());
300
+ assert!(plan.conflicts.is_empty());
301
+ }
302
+
303
+ #[test]
304
+ fn different_modifications_conflict() {
305
+ let target = entry(
306
+ "entity-a",
307
+ TrackedStateDiffKind::Modified,
308
+ Some(row_with_value("entity-a", "base", "base")),
309
+ Some(row_with_value("entity-a", "target", "target")),
310
+ );
311
+ let source = entry(
312
+ "entity-a",
313
+ TrackedStateDiffKind::Modified,
314
+ Some(row_with_value("entity-a", "base", "base")),
315
+ Some(row_with_value("entity-a", "source", "source")),
316
+ );
317
+
318
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
319
+
320
+ assert!(plan.patches.is_empty());
321
+ assert_eq!(conflict_ids(&plan), vec!["entity-a"]);
322
+ }
323
+
324
+ #[test]
325
+ fn delete_modify_conflicts() {
326
+ let target = entry(
327
+ "entity-a",
328
+ TrackedStateDiffKind::Removed,
329
+ Some(row("entity-a", "base")),
330
+ Some(tombstone("entity-a", "target-delete")),
331
+ );
332
+ let source = entry(
333
+ "entity-a",
334
+ TrackedStateDiffKind::Modified,
335
+ Some(row("entity-a", "base")),
336
+ Some(row_with_value("entity-a", "source", "source")),
337
+ );
338
+
339
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
340
+
341
+ assert_eq!(conflict_ids(&plan), vec!["entity-a"]);
342
+ }
343
+
344
+ #[test]
345
+ fn modify_delete_conflicts() {
346
+ let target = entry(
347
+ "entity-a",
348
+ TrackedStateDiffKind::Modified,
349
+ Some(row("entity-a", "base")),
350
+ Some(row_with_value("entity-a", "target", "target")),
351
+ );
352
+ let source = entry(
353
+ "entity-a",
354
+ TrackedStateDiffKind::Removed,
355
+ Some(row("entity-a", "base")),
356
+ Some(tombstone("entity-a", "source-delete")),
357
+ );
358
+
359
+ let plan = plan_merge(&diff(vec![target]), &diff(vec![source])).expect("merge should plan");
360
+
361
+ assert_eq!(conflict_ids(&plan), vec!["entity-a"]);
362
+ }
363
+
364
+ #[test]
365
+ fn source_removal_without_tombstone_errors() {
366
+ let error = plan_merge(
367
+ &TrackedStateDiff::default(),
368
+ &diff(vec![entry(
369
+ "entity-a",
370
+ TrackedStateDiffKind::Removed,
371
+ Some(row("entity-a", "base")),
372
+ None,
373
+ )]),
374
+ )
375
+ .expect_err("merge should reject impossible source removal");
376
+
377
+ assert!(error.message.contains("without a tombstone row"));
378
+ }
379
+
380
+ #[test]
381
+ fn patch_and_conflict_order_is_deterministic_by_identity() {
382
+ let target = diff(vec![entry(
383
+ "entity-b",
384
+ TrackedStateDiffKind::Modified,
385
+ Some(row_with_value("entity-b", "base", "base")),
386
+ Some(row_with_value("entity-b", "target", "target")),
387
+ )]);
388
+ let source = diff(vec![
389
+ entry(
390
+ "entity-c",
391
+ TrackedStateDiffKind::Added,
392
+ None,
393
+ Some(row("entity-c", "source-c")),
394
+ ),
395
+ entry(
396
+ "entity-a",
397
+ TrackedStateDiffKind::Added,
398
+ None,
399
+ Some(row("entity-a", "source-a")),
400
+ ),
401
+ entry(
402
+ "entity-b",
403
+ TrackedStateDiffKind::Modified,
404
+ Some(row_with_value("entity-b", "base", "base")),
405
+ Some(row_with_value("entity-b", "source", "source")),
406
+ ),
407
+ ]);
408
+
409
+ let plan = plan_merge(&target, &source).expect("merge should plan");
410
+
411
+ assert_eq!(patch_ids(&plan), vec!["entity-a", "entity-c"]);
412
+ assert_eq!(conflict_ids(&plan), vec!["entity-b"]);
413
+ }
414
+
415
+ fn diff(entries: Vec<TrackedStateDiffEntry>) -> TrackedStateDiff {
416
+ TrackedStateDiff { entries }
417
+ }
418
+
419
+ fn entry(
420
+ entity_id: &str,
421
+ kind: TrackedStateDiffKind,
422
+ before: Option<MaterializedTrackedStateRow>,
423
+ after: Option<MaterializedTrackedStateRow>,
424
+ ) -> TrackedStateDiffEntry {
425
+ TrackedStateDiffEntry {
426
+ identity: TrackedStateDiffIdentity {
427
+ schema_key: "test_schema".to_string(),
428
+ entity_id: EntityIdentity::single(entity_id),
429
+ file_id: None,
430
+ },
431
+ kind,
432
+ before,
433
+ after,
434
+ }
435
+ }
436
+
437
+ fn patch_ids(plan: &TrackedStateMergePlan) -> Vec<String> {
438
+ plan.patches
439
+ .iter()
440
+ .map(|entry| {
441
+ entry
442
+ .identity()
443
+ .entity_id
444
+ .as_single_string_owned()
445
+ .expect("identity")
446
+ })
447
+ .collect()
448
+ }
449
+
450
+ fn conflict_ids(plan: &TrackedStateMergePlan) -> Vec<String> {
451
+ plan.conflicts
452
+ .iter()
453
+ .map(|entry| {
454
+ entry
455
+ .identity
456
+ .entity_id
457
+ .as_single_string_owned()
458
+ .expect("identity")
459
+ })
460
+ .collect()
461
+ }
462
+
463
+ fn tombstone(entity_id: &str, change_id: &str) -> MaterializedTrackedStateRow {
464
+ let mut row = row(entity_id, change_id);
465
+ row.snapshot_content = None;
466
+ row.deleted = true;
467
+ row
468
+ }
469
+
470
+ fn row(entity_id: &str, change_id: &str) -> MaterializedTrackedStateRow {
471
+ row_with_value(entity_id, change_id, "value")
472
+ }
473
+
474
+ fn row_with_value(
475
+ entity_id: &str,
476
+ change_id: &str,
477
+ value: &str,
478
+ ) -> MaterializedTrackedStateRow {
479
+ MaterializedTrackedStateRow {
480
+ entity_id: EntityIdentity::single(entity_id),
481
+ schema_key: "test_schema".to_string(),
482
+ file_id: None,
483
+ snapshot_content: Some(format!("{{\"value\":\"{value}\"}}")),
484
+ metadata: None,
485
+ deleted: false,
486
+ created_at: "2026-01-01T00:00:00Z".to_string(),
487
+ updated_at: "2026-01-01T00:00:00Z".to_string(),
488
+ change_id: change_id.to_string(),
489
+ commit_id: change_id.replace("change", "commit"),
490
+ }
491
+ }
492
+ }
@@ -0,0 +1,32 @@
1
+ mod by_file_index;
2
+ mod codec;
3
+ mod context;
4
+ mod diff;
5
+ mod materialization;
6
+ mod materializer;
7
+ mod merge;
8
+ mod storage;
9
+ mod tree;
10
+ mod types;
11
+
12
+ #[allow(unused_imports)]
13
+ pub(crate) use context::{
14
+ TrackedStateContext, TrackedStateMaterializer, TrackedStateStoreReader, TrackedStateWriter,
15
+ };
16
+ #[allow(unused_imports)]
17
+ pub(crate) use diff::{
18
+ TrackedStateDiff, TrackedStateDiffEntry, TrackedStateDiffIdentity, TrackedStateDiffKind,
19
+ TrackedStateDiffRequest,
20
+ };
21
+ pub(crate) use materialization::{materialize_index_entries, TrackedMaterializationProjection};
22
+ #[allow(unused_imports)]
23
+ pub(crate) use merge::{
24
+ plan_merge, TrackedStateMergeConflict, TrackedStateMergePatch, TrackedStateMergePlan,
25
+ };
26
+ pub(crate) use storage::{load_delta_pack, DeltaJsonPackIndexesRef};
27
+ #[allow(unused_imports)]
28
+ pub(crate) use types::{
29
+ MaterializedTrackedStateRow, TrackedStateDeltaRef, TrackedStateFilter,
30
+ TrackedStateIndexValueRef, TrackedStateKeyRef, TrackedStateProjection, TrackedStateRowRequest,
31
+ TrackedStateScanRequest,
32
+ };