@x12i/memorix-writer 1.0.0

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 (204) hide show
  1. package/README.md +145 -0
  2. package/catalox-seeds/memorix-write-descriptors.manifest.json +164 -0
  3. package/dist/client/create-client.d.ts +3 -0
  4. package/dist/client/create-client.d.ts.map +1 -0
  5. package/dist/client/create-client.js +81 -0
  6. package/dist/client/create-client.js.map +1 -0
  7. package/dist/client/create-from-env.d.ts +3 -0
  8. package/dist/client/create-from-env.d.ts.map +1 -0
  9. package/dist/client/create-from-env.js +21 -0
  10. package/dist/client/create-from-env.js.map +1 -0
  11. package/dist/client/ensure-tier.d.ts +3 -0
  12. package/dist/client/ensure-tier.d.ts.map +1 -0
  13. package/dist/client/ensure-tier.js +4 -0
  14. package/dist/client/ensure-tier.js.map +1 -0
  15. package/dist/client/types.d.ts +165 -0
  16. package/dist/client/types.d.ts.map +1 -0
  17. package/dist/client/types.js +2 -0
  18. package/dist/client/types.js.map +1 -0
  19. package/dist/constants.d.ts +13 -0
  20. package/dist/constants.d.ts.map +1 -0
  21. package/dist/constants.js +13 -0
  22. package/dist/constants.js.map +1 -0
  23. package/dist/content/compute-content-metadata.d.ts +16 -0
  24. package/dist/content/compute-content-metadata.d.ts.map +1 -0
  25. package/dist/content/compute-content-metadata.js +13 -0
  26. package/dist/content/compute-content-metadata.js.map +1 -0
  27. package/dist/content/generate-content-key.d.ts +13 -0
  28. package/dist/content/generate-content-key.d.ts.map +1 -0
  29. package/dist/content/generate-content-key.js +22 -0
  30. package/dist/content/generate-content-key.js.map +1 -0
  31. package/dist/content/generate-preview.d.ts +2 -0
  32. package/dist/content/generate-preview.d.ts.map +1 -0
  33. package/dist/content/generate-preview.js +7 -0
  34. package/dist/content/generate-preview.js.map +1 -0
  35. package/dist/content/map-content-write-errors.d.ts +3 -0
  36. package/dist/content/map-content-write-errors.d.ts.map +1 -0
  37. package/dist/content/map-content-write-errors.js +9 -0
  38. package/dist/content/map-content-write-errors.js.map +1 -0
  39. package/dist/content/resolve-content-write-config.d.ts +25 -0
  40. package/dist/content/resolve-content-write-config.d.ts.map +1 -0
  41. package/dist/content/resolve-content-write-config.js +35 -0
  42. package/dist/content/resolve-content-write-config.js.map +1 -0
  43. package/dist/content/write-content-object.d.ts +40 -0
  44. package/dist/content/write-content-object.d.ts.map +1 -0
  45. package/dist/content/write-content-object.js +117 -0
  46. package/dist/content/write-content-object.js.map +1 -0
  47. package/dist/data/collection-name.d.ts +30 -0
  48. package/dist/data/collection-name.d.ts.map +1 -0
  49. package/dist/data/collection-name.js +86 -0
  50. package/dist/data/collection-name.js.map +1 -0
  51. package/dist/data/identity.d.ts +13 -0
  52. package/dist/data/identity.d.ts.map +1 -0
  53. package/dist/data/identity.js +29 -0
  54. package/dist/data/identity.js.map +1 -0
  55. package/dist/data/memorix-write-adapter.d.ts +33 -0
  56. package/dist/data/memorix-write-adapter.d.ts.map +1 -0
  57. package/dist/data/memorix-write-adapter.js +45 -0
  58. package/dist/data/memorix-write-adapter.js.map +1 -0
  59. package/dist/descriptors/entity-descriptor-types.d.ts +59 -0
  60. package/dist/descriptors/entity-descriptor-types.d.ts.map +1 -0
  61. package/dist/descriptors/entity-descriptor-types.js +2 -0
  62. package/dist/descriptors/entity-descriptor-types.js.map +1 -0
  63. package/dist/descriptors/load-entity-descriptor.d.ts +7 -0
  64. package/dist/descriptors/load-entity-descriptor.d.ts.map +1 -0
  65. package/dist/descriptors/load-entity-descriptor.js +23 -0
  66. package/dist/descriptors/load-entity-descriptor.js.map +1 -0
  67. package/dist/descriptors/load-write-descriptor.d.ts +5 -0
  68. package/dist/descriptors/load-write-descriptor.d.ts.map +1 -0
  69. package/dist/descriptors/load-write-descriptor.js +19 -0
  70. package/dist/descriptors/load-write-descriptor.js.map +1 -0
  71. package/dist/descriptors/secret-scan.d.ts +2 -0
  72. package/dist/descriptors/secret-scan.d.ts.map +1 -0
  73. package/dist/descriptors/secret-scan.js +39 -0
  74. package/dist/descriptors/secret-scan.js.map +1 -0
  75. package/dist/descriptors/validate-write-descriptor.d.ts +4 -0
  76. package/dist/descriptors/validate-write-descriptor.d.ts.map +1 -0
  77. package/dist/descriptors/validate-write-descriptor.js +45 -0
  78. package/dist/descriptors/validate-write-descriptor.js.map +1 -0
  79. package/dist/descriptors/write-descriptor-types.d.ts +112 -0
  80. package/dist/descriptors/write-descriptor-types.d.ts.map +1 -0
  81. package/dist/descriptors/write-descriptor-types.js +2 -0
  82. package/dist/descriptors/write-descriptor-types.js.map +1 -0
  83. package/dist/errors/errors.d.ts +6 -0
  84. package/dist/errors/errors.d.ts.map +1 -0
  85. package/dist/errors/errors.js +12 -0
  86. package/dist/errors/errors.js.map +1 -0
  87. package/dist/errors/issues.d.ts +13 -0
  88. package/dist/errors/issues.d.ts.map +1 -0
  89. package/dist/errors/issues.js +9 -0
  90. package/dist/errors/issues.js.map +1 -0
  91. package/dist/index.d.ts +20 -0
  92. package/dist/index.d.ts.map +1 -0
  93. package/dist/index.js +15 -0
  94. package/dist/index.js.map +1 -0
  95. package/dist/mongo/collection-documents.d.ts +54 -0
  96. package/dist/mongo/collection-documents.d.ts.map +1 -0
  97. package/dist/mongo/collection-documents.js +100 -0
  98. package/dist/mongo/collection-documents.js.map +1 -0
  99. package/dist/mongo/collection-name-validate.d.ts +2 -0
  100. package/dist/mongo/collection-name-validate.d.ts.map +1 -0
  101. package/dist/mongo/collection-name-validate.js +19 -0
  102. package/dist/mongo/collection-name-validate.js.map +1 -0
  103. package/dist/mongo/connection.d.ts +3 -0
  104. package/dist/mongo/connection.d.ts.map +1 -0
  105. package/dist/mongo/connection.js +8 -0
  106. package/dist/mongo/connection.js.map +1 -0
  107. package/dist/mongo/default-collections.d.ts +3 -0
  108. package/dist/mongo/default-collections.d.ts.map +1 -0
  109. package/dist/mongo/default-collections.js +14 -0
  110. package/dist/mongo/default-collections.js.map +1 -0
  111. package/dist/mongo/env.d.ts +6 -0
  112. package/dist/mongo/env.d.ts.map +1 -0
  113. package/dist/mongo/env.js +33 -0
  114. package/dist/mongo/env.js.map +1 -0
  115. package/dist/mongo/id-normalize.d.ts +3 -0
  116. package/dist/mongo/id-normalize.d.ts.map +1 -0
  117. package/dist/mongo/id-normalize.js +23 -0
  118. package/dist/mongo/id-normalize.js.map +1 -0
  119. package/dist/mongo/index.d.ts +5 -0
  120. package/dist/mongo/index.d.ts.map +1 -0
  121. package/dist/mongo/index.js +5 -0
  122. package/dist/mongo/index.js.map +1 -0
  123. package/dist/mongo/target.d.ts +4 -0
  124. package/dist/mongo/target.d.ts.map +1 -0
  125. package/dist/mongo/target.js +9 -0
  126. package/dist/mongo/target.js.map +1 -0
  127. package/dist/mongo/tier-env.d.ts +9 -0
  128. package/dist/mongo/tier-env.d.ts.map +1 -0
  129. package/dist/mongo/tier-env.js +11 -0
  130. package/dist/mongo/tier-env.js.map +1 -0
  131. package/dist/mongo/tier.d.ts +22 -0
  132. package/dist/mongo/tier.d.ts.map +1 -0
  133. package/dist/mongo/tier.js +50 -0
  134. package/dist/mongo/tier.js.map +1 -0
  135. package/dist/utils/path.d.ts +4 -0
  136. package/dist/utils/path.d.ts.map +1 -0
  137. package/dist/utils/path.js +42 -0
  138. package/dist/utils/path.js.map +1 -0
  139. package/dist/validation/validate-content.d.ts +17 -0
  140. package/dist/validation/validate-content.d.ts.map +1 -0
  141. package/dist/validation/validate-content.js +47 -0
  142. package/dist/validation/validate-content.js.map +1 -0
  143. package/dist/validation/validate-fields.d.ts +8 -0
  144. package/dist/validation/validate-fields.d.ts.map +1 -0
  145. package/dist/validation/validate-fields.js +49 -0
  146. package/dist/validation/validate-fields.js.map +1 -0
  147. package/dist/validation/validate-identity.d.ts +3 -0
  148. package/dist/validation/validate-identity.d.ts.map +1 -0
  149. package/dist/validation/validate-identity.js +2 -0
  150. package/dist/validation/validate-identity.js.map +1 -0
  151. package/dist/validation/validate-immutability.d.ts +9 -0
  152. package/dist/validation/validate-immutability.d.ts.map +1 -0
  153. package/dist/validation/validate-immutability.js +35 -0
  154. package/dist/validation/validate-immutability.js.map +1 -0
  155. package/dist/validation/validate-values.d.ts +4 -0
  156. package/dist/validation/validate-values.d.ts.map +1 -0
  157. package/dist/validation/validate-values.js +55 -0
  158. package/dist/validation/validate-values.js.map +1 -0
  159. package/dist/writer/apply-audit.d.ts +10 -0
  160. package/dist/writer/apply-audit.d.ts.map +1 -0
  161. package/dist/writer/apply-audit.js +39 -0
  162. package/dist/writer/apply-audit.js.map +1 -0
  163. package/dist/writer/apply-field-mapping.d.ts +9 -0
  164. package/dist/writer/apply-field-mapping.d.ts.map +1 -0
  165. package/dist/writer/apply-field-mapping.js +32 -0
  166. package/dist/writer/apply-field-mapping.js.map +1 -0
  167. package/dist/writer/apply-timestamps.d.ts +8 -0
  168. package/dist/writer/apply-timestamps.d.ts.map +1 -0
  169. package/dist/writer/apply-timestamps.js +12 -0
  170. package/dist/writer/apply-timestamps.js.map +1 -0
  171. package/dist/writer/build-record.d.ts +28 -0
  172. package/dist/writer/build-record.d.ts.map +1 -0
  173. package/dist/writer/build-record.js +36 -0
  174. package/dist/writer/build-record.js.map +1 -0
  175. package/dist/writer/resolve-conflict.d.ts +12 -0
  176. package/dist/writer/resolve-conflict.d.ts.map +1 -0
  177. package/dist/writer/resolve-conflict.js +32 -0
  178. package/dist/writer/resolve-conflict.js.map +1 -0
  179. package/dist/writer/resolve-idempotency.d.ts +23 -0
  180. package/dist/writer/resolve-idempotency.d.ts.map +1 -0
  181. package/dist/writer/resolve-idempotency.js +69 -0
  182. package/dist/writer/resolve-idempotency.js.map +1 -0
  183. package/dist/writer/resolve-operation.d.ts +4 -0
  184. package/dist/writer/resolve-operation.d.ts.map +1 -0
  185. package/dist/writer/resolve-operation.js +9 -0
  186. package/dist/writer/resolve-operation.js.map +1 -0
  187. package/dist/writer/resolve-write-field-target.d.ts +4 -0
  188. package/dist/writer/resolve-write-field-target.d.ts.map +1 -0
  189. package/dist/writer/resolve-write-field-target.js +7 -0
  190. package/dist/writer/resolve-write-field-target.js.map +1 -0
  191. package/dist/writer/validate-write.d.ts +3 -0
  192. package/dist/writer/validate-write.d.ts.map +1 -0
  193. package/dist/writer/validate-write.js +72 -0
  194. package/dist/writer/validate-write.js.map +1 -0
  195. package/dist/writer/write-record.d.ts +3 -0
  196. package/dist/writer/write-record.d.ts.map +1 -0
  197. package/dist/writer/write-record.js +472 -0
  198. package/dist/writer/write-record.js.map +1 -0
  199. package/dist/writer/write-records.d.ts +3 -0
  200. package/dist/writer/write-records.d.ts.map +1 -0
  201. package/dist/writer/write-records.js +79 -0
  202. package/dist/writer/write-records.js.map +1 -0
  203. package/docs/MEMORIX-DATABASE-CONVENTIONS.md +324 -0
  204. package/package.json +51 -0
@@ -0,0 +1,49 @@
1
+ import { writeIssue } from "../errors/issues.js";
2
+ import { resolveInputPath } from "../writer/resolve-write-field-target.js";
3
+ export function validateInputFields(args) {
4
+ const issues = [];
5
+ const { writeDescriptor, input, contentFieldNames = [] } = args;
6
+ const rejectUnknown = writeDescriptor.validation?.rejectUnknownFields ?? true;
7
+ const fieldEntries = Object.entries(writeDescriptor.fields);
8
+ const allowedInputPaths = new Set();
9
+ for (const [name, field] of fieldEntries) {
10
+ if (field.writable === false)
11
+ continue;
12
+ allowedInputPaths.add(resolveInputPath(name, field));
13
+ allowedInputPaths.add(name);
14
+ }
15
+ for (const cf of contentFieldNames) {
16
+ allowedInputPaths.add(cf);
17
+ }
18
+ if (rejectUnknown) {
19
+ for (const key of Object.keys(input)) {
20
+ if (!allowedInputPaths.has(key)) {
21
+ issues.push(writeIssue("UNKNOWN_FIELD", `unknown input field "${key}"`, {
22
+ path: key,
23
+ severity: "error",
24
+ }));
25
+ }
26
+ }
27
+ }
28
+ let writableProvided = 0;
29
+ for (const [name, field] of fieldEntries) {
30
+ const inputPath = resolveInputPath(name, field);
31
+ const value = input[inputPath] ?? input[name];
32
+ if (field.required && (value === undefined || value === null)) {
33
+ issues.push(writeIssue("REQUIRED_FIELD_MISSING", `required field "${name}" is missing`, {
34
+ path: inputPath,
35
+ severity: "error",
36
+ }));
37
+ }
38
+ if (field.writable !== false && value !== undefined) {
39
+ writableProvided++;
40
+ }
41
+ }
42
+ if (writeDescriptor.validation?.requireAtLeastOneWritableField &&
43
+ writableProvided === 0 &&
44
+ contentFieldNames.length === 0) {
45
+ issues.push(writeIssue("VALIDATION_FAILED", "at least one writable field must be provided", { severity: "error" }));
46
+ }
47
+ return issues;
48
+ }
49
+ //# sourceMappingURL=validate-fields.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-fields.js","sourceRoot":"","sources":["../../src/validation/validate-fields.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AACjD,OAAO,EAAE,gBAAgB,EAAE,MAAM,yCAAyC,CAAC;AAE3E,MAAM,UAAU,mBAAmB,CAAC,IAInC;IACC,MAAM,MAAM,GAAwB,EAAE,CAAC;IACvC,MAAM,EAAE,eAAe,EAAE,KAAK,EAAE,iBAAiB,GAAG,EAAE,EAAE,GAAG,IAAI,CAAC;IAChE,MAAM,aAAa,GAAG,eAAe,CAAC,UAAU,EAAE,mBAAmB,IAAI,IAAI,CAAC;IAC9E,MAAM,YAAY,GAAG,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC;IAE5D,MAAM,iBAAiB,GAAG,IAAI,GAAG,EAAU,CAAC;IAC5C,KAAK,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,YAAY,EAAE,CAAC;QACzC,IAAI,KAAK,CAAC,QAAQ,KAAK,KAAK;YAAE,SAAS;QACvC,iBAAiB,CAAC,GAAG,CAAC,gBAAgB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;QACrD,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC9B,CAAC;IACD,KAAK,MAAM,EAAE,IAAI,iBAAiB,EAAE,CAAC;QACnC,iBAAiB,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAC5B,CAAC;IAED,IAAI,aAAa,EAAE,CAAC;QAClB,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YACrC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;gBAChC,MAAM,CAAC,IAAI,CACT,UAAU,CAAC,eAAe,EAAE,wBAAwB,GAAG,GAAG,EAAE;oBAC1D,IAAI,EAAE,GAAG;oBACT,QAAQ,EAAE,OAAO;iBAClB,CAAC,CACH,CAAC;YACJ,CAAC;QACH,CAAC;IACH,CAAC;IAED,IAAI,gBAAgB,GAAG,CAAC,CAAC;IACzB,KAAK,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,YAAY,EAAE,CAAC;QACzC,MAAM,SAAS,GAAG,gBAAgB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAChD,MAAM,KAAK,GAAG,KAAK,CAAC,SAAS,CAAC,IAAI,KAAK,CAAC,IAAI,CAAC,CAAC;QAE9C,IAAI,KAAK,CAAC,QAAQ,IAAI,CAAC,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,CAAC,EAAE,CAAC;YAC9D,MAAM,CAAC,IAAI,CACT,UAAU,CAAC,wBAAwB,EAAE,mBAAmB,IAAI,cAAc,EAAE;gBAC1E,IAAI,EAAE,SAAS;gBACf,QAAQ,EAAE,OAAO;aAClB,CAAC,CACH,CAAC;QACJ,CAAC;QAED,IAAI,KAAK,CAAC,QAAQ,KAAK,KAAK,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YACpD,gBAAgB,EAAE,CAAC;QACrB,CAAC;IACH,CAAC;IAED,IACE,eAAe,CAAC,UAAU,EAAE,8BAA8B;QAC1D,gBAAgB,KAAK,CAAC;QACtB,iBAAiB,CAAC,MAAM,KAAK,CAAC,EAC9B,CAAC;QACD,MAAM,CAAC,IAAI,CACT,UAAU,CACR,mBAAmB,EACnB,8CAA8C,EAC9C,EAAE,QAAQ,EAAE,OAAO,EAAE,CACtB,CACF,CAAC;IACJ,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC"}
@@ -0,0 +1,3 @@
1
+ export { resolveRequestIdentity, assertDescriptorIdField } from "../data/identity.js";
2
+ export type { ResolvedIdentity } from "../data/identity.js";
3
+ //# sourceMappingURL=validate-identity.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-identity.d.ts","sourceRoot":"","sources":["../../src/validation/validate-identity.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAAE,uBAAuB,EAAE,MAAM,qBAAqB,CAAC;AACtF,YAAY,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { resolveRequestIdentity, assertDescriptorIdField } from "../data/identity.js";
2
+ //# sourceMappingURL=validate-identity.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-identity.js","sourceRoot":"","sources":["../../src/validation/validate-identity.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAAE,uBAAuB,EAAE,MAAM,qBAAqB,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { MemorixWriteFieldDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ import type { MemorixWriteIssue } from "../errors/issues.js";
3
+ export declare function validateImmutableFieldsOnPatch(args: {
4
+ existing: Record<string, unknown>;
5
+ fields: Record<string, MemorixWriteFieldDescriptor>;
6
+ input: Record<string, unknown>;
7
+ inputPathResolver: (name: string, field: MemorixWriteFieldDescriptor) => string;
8
+ }): MemorixWriteIssue[];
9
+ //# sourceMappingURL=validate-immutability.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-immutability.d.ts","sourceRoot":"","sources":["../../src/validation/validate-immutability.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0CAA0C,CAAC;AAC5F,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAY7D,wBAAgB,8BAA8B,CAAC,IAAI,EAAE;IACnD,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAClC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,2BAA2B,CAAC,CAAC;IACpD,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC/B,iBAAiB,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,2BAA2B,KAAK,MAAM,CAAC;CACjF,GAAG,iBAAiB,EAAE,CA0BtB"}
@@ -0,0 +1,35 @@
1
+ import { writeIssue } from "../errors/issues.js";
2
+ import { deepGet } from "../utils/path.js";
3
+ const PROTECTED_PATHS = new Set([
4
+ "entityId",
5
+ "eventId",
6
+ "recordId",
7
+ "createdAt",
8
+ "_memorix.writeDescriptorId",
9
+ ]);
10
+ export function validateImmutableFieldsOnPatch(args) {
11
+ const issues = [];
12
+ for (const [name, field] of Object.entries(args.fields)) {
13
+ if (!field.immutable)
14
+ continue;
15
+ const inputPath = args.inputPathResolver(name, field);
16
+ if (!(inputPath in args.input) && !(name in args.input))
17
+ continue;
18
+ const existingVal = deepGet(args.existing, field.target.path);
19
+ const newVal = args.input[inputPath] ?? args.input[name];
20
+ if (newVal !== undefined && newVal !== existingVal) {
21
+ issues.push(writeIssue("IMMUTABLE_FIELD_UPDATE", `field "${name}" is immutable`, {
22
+ path: field.target.path,
23
+ }));
24
+ }
25
+ }
26
+ for (const path of PROTECTED_PATHS) {
27
+ if (path in args.input) {
28
+ issues.push(writeIssue("IMMUTABLE_FIELD_UPDATE", `system field "${path}" cannot be patched`, {
29
+ path,
30
+ }));
31
+ }
32
+ }
33
+ return issues;
34
+ }
35
+ //# sourceMappingURL=validate-immutability.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-immutability.js","sourceRoot":"","sources":["../../src/validation/validate-immutability.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AACjD,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAE3C,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC;IAC9B,UAAU;IACV,SAAS;IACT,UAAU;IACV,WAAW;IACX,4BAA4B;CAC7B,CAAC,CAAC;AAEH,MAAM,UAAU,8BAA8B,CAAC,IAK9C;IACC,MAAM,MAAM,GAAwB,EAAE,CAAC;IACvC,KAAK,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;QACxD,IAAI,CAAC,KAAK,CAAC,SAAS;YAAE,SAAS;QAC/B,MAAM,SAAS,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QACtD,IAAI,CAAC,CAAC,SAAS,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,IAAI,IAAI,CAAC,KAAK,CAAC;YAAE,SAAS;QAClE,MAAM,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAC9D,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACzD,IAAI,MAAM,KAAK,SAAS,IAAI,MAAM,KAAK,WAAW,EAAE,CAAC;YACnD,MAAM,CAAC,IAAI,CACT,UAAU,CAAC,wBAAwB,EAAE,UAAU,IAAI,gBAAgB,EAAE;gBACnE,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI;aACxB,CAAC,CACH,CAAC;QACJ,CAAC;IACH,CAAC;IACD,KAAK,MAAM,IAAI,IAAI,eAAe,EAAE,CAAC;QACnC,IAAI,IAAI,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACvB,MAAM,CAAC,IAAI,CACT,UAAU,CAAC,wBAAwB,EAAE,iBAAiB,IAAI,qBAAqB,EAAE;gBAC/E,IAAI;aACL,CAAC,CACH,CAAC;QACJ,CAAC;IACH,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { MemorixWriteFieldDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ import type { MemorixWriteIssue } from "../errors/issues.js";
3
+ export declare function validateFieldValue(fieldName: string, field: MemorixWriteFieldDescriptor, value: unknown): MemorixWriteIssue[];
4
+ //# sourceMappingURL=validate-values.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-values.d.ts","sourceRoot":"","sources":["../../src/validation/validate-values.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0CAA0C,CAAC;AAC5F,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAY7D,wBAAgB,kBAAkB,CAChC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,2BAA2B,EAClC,KAAK,EAAE,OAAO,GACb,iBAAiB,EAAE,CAuDrB"}
@@ -0,0 +1,55 @@
1
+ import { writeIssue } from "../errors/issues.js";
2
+ function isContentPointer(value) {
3
+ return (value != null &&
4
+ typeof value === "object" &&
5
+ "contentKey" in value &&
6
+ typeof value.contentKey === "string");
7
+ }
8
+ export function validateFieldValue(fieldName, field, value) {
9
+ const issues = [];
10
+ if (value === undefined || value === null) {
11
+ if (value === null && field.nullBehavior === "error") {
12
+ issues.push(writeIssue("VALIDATION_FAILED", `null not allowed for "${fieldName}"`, {
13
+ path: fieldName,
14
+ }));
15
+ }
16
+ return issues;
17
+ }
18
+ const vt = field.valueType;
19
+ if (!vt)
20
+ return issues;
21
+ if (vt === "content") {
22
+ if (!isContentPointer(value) && typeof value !== "object") {
23
+ issues.push(writeIssue("INVALID_VALUE_TYPE", `"${fieldName}" must be content object or provided via content.fields`, {
24
+ path: fieldName,
25
+ }));
26
+ }
27
+ return issues;
28
+ }
29
+ if (vt === "enum") {
30
+ if (typeof value !== "string" || !field.enumValues?.includes(value)) {
31
+ issues.push(writeIssue("INVALID_ENUM_VALUE", `"${fieldName}" must be one of: ${field.enumValues?.join(", ") ?? ""}`, { path: fieldName }));
32
+ }
33
+ return issues;
34
+ }
35
+ if (vt === "string" && typeof value !== "string") {
36
+ issues.push(writeIssue("INVALID_VALUE_TYPE", `"${fieldName}" must be a string`, { path: fieldName }));
37
+ }
38
+ else if (vt === "number" && typeof value !== "number") {
39
+ issues.push(writeIssue("INVALID_VALUE_TYPE", `"${fieldName}" must be a number`, { path: fieldName }));
40
+ }
41
+ else if (vt === "boolean" && typeof value !== "boolean") {
42
+ issues.push(writeIssue("INVALID_VALUE_TYPE", `"${fieldName}" must be a boolean`, { path: fieldName }));
43
+ }
44
+ else if ((vt === "date" || vt === "datetime") && typeof value !== "string") {
45
+ issues.push(writeIssue("INVALID_VALUE_TYPE", `"${fieldName}" must be an ISO date string`, { path: fieldName }));
46
+ }
47
+ else if (vt === "array" && !Array.isArray(value)) {
48
+ issues.push(writeIssue("INVALID_VALUE_TYPE", `"${fieldName}" must be an array`, { path: fieldName }));
49
+ }
50
+ else if (vt === "object" && (typeof value !== "object" || Array.isArray(value))) {
51
+ issues.push(writeIssue("INVALID_VALUE_TYPE", `"${fieldName}" must be an object`, { path: fieldName }));
52
+ }
53
+ return issues;
54
+ }
55
+ //# sourceMappingURL=validate-values.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-values.js","sourceRoot":"","sources":["../../src/validation/validate-values.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import type { MemorixWriteDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ import type { WriteMemorixRecordRequest } from "../client/types.js";
3
+ export declare function applyAuditMetadata(args: {
4
+ record: Record<string, unknown>;
5
+ writeDescriptor: MemorixWriteDescriptor;
6
+ request: Pick<WriteMemorixRecordRequest, "source" | "actor" | "options">;
7
+ operation: string;
8
+ metadataRootPath: string;
9
+ }): void;
10
+ //# sourceMappingURL=apply-audit.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apply-audit.d.ts","sourceRoot":"","sources":["../../src/writer/apply-audit.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,0CAA0C,CAAC;AACvF,OAAO,KAAK,EAAE,yBAAyB,EAAE,MAAM,oBAAoB,CAAC;AAGpE,wBAAgB,kBAAkB,CAAC,IAAI,EAAE;IACvC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAChC,eAAe,EAAE,sBAAsB,CAAC;IACxC,OAAO,EAAE,IAAI,CAAC,yBAAyB,EAAE,QAAQ,GAAG,OAAO,GAAG,SAAS,CAAC,CAAC;IACzE,SAAS,EAAE,MAAM,CAAC;IAClB,gBAAgB,EAAE,MAAM,CAAC;CAC1B,GAAG,IAAI,CA4CP"}
@@ -0,0 +1,39 @@
1
+ import { deepSet } from "../utils/path.js";
2
+ export function applyAuditMetadata(args) {
3
+ const audit = args.writeDescriptor.audit;
4
+ if (audit?.enabled === false)
5
+ return;
6
+ const auditPath = audit?.path ?? "_memorix.audit";
7
+ const block = {
8
+ at: new Date().toISOString(),
9
+ operation: args.operation,
10
+ };
11
+ if (audit?.includeDescriptorId !== false) {
12
+ block.writeDescriptorId = args.writeDescriptor.id;
13
+ }
14
+ if (audit?.includeSource !== false && args.request.source) {
15
+ block.source = args.request.source;
16
+ }
17
+ if (audit?.includeActor !== false && args.request.actor) {
18
+ block.actor = args.request.actor;
19
+ }
20
+ if (audit?.includeCorrelationId !== false && args.request.options?.correlationId) {
21
+ block.correlationId = args.request.options.correlationId;
22
+ }
23
+ const relative = auditPath.startsWith(`${args.metadataRootPath}.`)
24
+ ? auditPath.slice(args.metadataRootPath.length + 1)
25
+ : auditPath === args.metadataRootPath
26
+ ? ""
27
+ : auditPath.replace(/^_memorix\.?/, "");
28
+ if (!relative) {
29
+ const meta = args.record[args.metadataRootPath] ?? {};
30
+ Object.assign(meta, block);
31
+ args.record[args.metadataRootPath] = meta;
32
+ return;
33
+ }
34
+ if (!args.record[args.metadataRootPath]) {
35
+ args.record[args.metadataRootPath] = {};
36
+ }
37
+ deepSet(args.record[args.metadataRootPath], relative, block);
38
+ }
39
+ //# sourceMappingURL=apply-audit.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apply-audit.js","sourceRoot":"","sources":["../../src/writer/apply-audit.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAE3C,MAAM,UAAU,kBAAkB,CAAC,IAMlC;IACC,MAAM,KAAK,GAAG,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC;IACzC,IAAI,KAAK,EAAE,OAAO,KAAK,KAAK;QAAE,OAAO;IAErC,MAAM,SAAS,GAAG,KAAK,EAAE,IAAI,IAAI,gBAAgB,CAAC;IAClD,MAAM,KAAK,GAA4B;QACrC,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QAC5B,SAAS,EAAE,IAAI,CAAC,SAAS;KAC1B,CAAC;IACF,IAAI,KAAK,EAAE,mBAAmB,KAAK,KAAK,EAAE,CAAC;QACzC,KAAK,CAAC,iBAAiB,GAAG,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC;IACpD,CAAC;IACD,IAAI,KAAK,EAAE,aAAa,KAAK,KAAK,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;QAC1D,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC;IACrC,CAAC;IACD,IAAI,KAAK,EAAE,YAAY,KAAK,KAAK,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QACxD,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC;IACnC,CAAC;IACD,IAAI,KAAK,EAAE,oBAAoB,KAAK,KAAK,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,aAAa,EAAE,CAAC;QACjF,KAAK,CAAC,aAAa,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,aAAa,CAAC;IAC3D,CAAC;IAED,MAAM,QAAQ,GAAG,SAAS,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC,gBAAgB,GAAG,CAAC;QAChE,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC;QACnD,CAAC,CAAC,SAAS,KAAK,IAAI,CAAC,gBAAgB;YACnC,CAAC,CAAC,EAAE;YACJ,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAE5C,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,MAAM,IAAI,GACP,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAA6B,IAAI,EAAE,CAAC;QACxE,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAC3B,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC;QAC1C,OAAO;IACT,CAAC;IAED,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC;QACxC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,GAAG,EAAE,CAAC;IAC1C,CAAC;IACD,OAAO,CACL,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAA4B,EAC7D,QAAQ,EACR,KAAK,CACN,CAAC;AACJ,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { MemorixWriteFieldDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ import type { MemorixContentObject } from "../descriptors/entity-descriptor-types.js";
3
+ export declare function applyFieldMapping(args: {
4
+ record: Record<string, unknown>;
5
+ fields: Record<string, MemorixWriteFieldDescriptor>;
6
+ input: Record<string, unknown>;
7
+ contentPointers?: Record<string, MemorixContentObject>;
8
+ }): void;
9
+ //# sourceMappingURL=apply-field-mapping.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apply-field-mapping.d.ts","sourceRoot":"","sources":["../../src/writer/apply-field-mapping.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0CAA0C,CAAC;AAC5F,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,2CAA2C,CAAC;AAItF,wBAAgB,iBAAiB,CAAC,IAAI,EAAE;IACtC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAChC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,2BAA2B,CAAC,CAAC;IACpD,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC/B,eAAe,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,oBAAoB,CAAC,CAAC;CACxD,GAAG,IAAI,CA6BP"}
@@ -0,0 +1,32 @@
1
+ import { deepSet } from "../utils/path.js";
2
+ import { resolveInputPath } from "./resolve-write-field-target.js";
3
+ export function applyFieldMapping(args) {
4
+ for (const [name, field] of Object.entries(args.fields)) {
5
+ if (field.writable === false)
6
+ continue;
7
+ const inputPath = resolveInputPath(name, field);
8
+ let value = args.contentPointers?.[name] ??
9
+ args.input[inputPath] ??
10
+ args.input[name];
11
+ if (value === undefined) {
12
+ if (field.defaultValue !== undefined) {
13
+ value = field.defaultValue;
14
+ }
15
+ else if (field.nullBehavior === "ignore") {
16
+ continue;
17
+ }
18
+ else {
19
+ continue;
20
+ }
21
+ }
22
+ if (value === null && field.nullBehavior === "ignore")
23
+ continue;
24
+ if (value === null && field.nullBehavior === "clear") {
25
+ deepSet(args.record, field.target.path, null);
26
+ continue;
27
+ }
28
+ const targetRoot = args.record;
29
+ deepSet(targetRoot, field.target.path, value);
30
+ }
31
+ }
32
+ //# sourceMappingURL=apply-field-mapping.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apply-field-mapping.js","sourceRoot":"","sources":["../../src/writer/apply-field-mapping.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC3C,OAAO,EAAE,gBAAgB,EAAE,MAAM,iCAAiC,CAAC;AAEnE,MAAM,UAAU,iBAAiB,CAAC,IAKjC;IACC,KAAK,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;QACxD,IAAI,KAAK,CAAC,QAAQ,KAAK,KAAK;YAAE,SAAS;QAEvC,MAAM,SAAS,GAAG,gBAAgB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAChD,IAAI,KAAK,GACP,IAAI,CAAC,eAAe,EAAE,CAAC,IAAI,CAAC;YAC5B,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC;YACrB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAEnB,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YACxB,IAAI,KAAK,CAAC,YAAY,KAAK,SAAS,EAAE,CAAC;gBACrC,KAAK,GAAG,KAAK,CAAC,YAAY,CAAC;YAC7B,CAAC;iBAAM,IAAI,KAAK,CAAC,YAAY,KAAK,QAAQ,EAAE,CAAC;gBAC3C,SAAS;YACX,CAAC;iBAAM,CAAC;gBACN,SAAS;YACX,CAAC;QACH,CAAC;QAED,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,CAAC,YAAY,KAAK,QAAQ;YAAE,SAAS;QAChE,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,CAAC,YAAY,KAAK,OAAO,EAAE,CAAC;YACrD,OAAO,CAAC,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAC9C,SAAS;QACX,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC;QAC/B,OAAO,CAAC,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IAChD,CAAC;AACH,CAAC"}
@@ -0,0 +1,8 @@
1
+ import type { MemorixWriteDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ export declare function applyTimestamps(args: {
3
+ record: Record<string, unknown>;
4
+ writeDescriptor: MemorixWriteDescriptor;
5
+ now: string;
6
+ isNew: boolean;
7
+ }): void;
8
+ //# sourceMappingURL=apply-timestamps.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apply-timestamps.d.ts","sourceRoot":"","sources":["../../src/writer/apply-timestamps.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,0CAA0C,CAAC;AAEvF,wBAAgB,eAAe,CAAC,IAAI,EAAE;IACpC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAChC,eAAe,EAAE,sBAAsB,CAAC;IACxC,GAAG,EAAE,MAAM,CAAC;IACZ,KAAK,EAAE,OAAO,CAAC;CAChB,GAAG,IAAI,CAWP"}
@@ -0,0 +1,12 @@
1
+ export function applyTimestamps(args) {
2
+ const createdPath = args.writeDescriptor.timestamps?.createdAtPath ?? "createdAt";
3
+ const updatedPath = args.writeDescriptor.timestamps?.updatedAtPath ?? "modifiedAt";
4
+ const source = args.writeDescriptor.timestamps?.writeTimeSource ?? "server";
5
+ if (source === "server") {
6
+ if (args.isNew) {
7
+ args.record[createdPath] = args.now;
8
+ }
9
+ args.record[updatedPath] = args.now;
10
+ }
11
+ }
12
+ //# sourceMappingURL=apply-timestamps.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apply-timestamps.js","sourceRoot":"","sources":["../../src/writer/apply-timestamps.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,eAAe,CAAC,IAK/B;IACC,MAAM,WAAW,GAAG,IAAI,CAAC,eAAe,CAAC,UAAU,EAAE,aAAa,IAAI,WAAW,CAAC;IAClF,MAAM,WAAW,GAAG,IAAI,CAAC,eAAe,CAAC,UAAU,EAAE,aAAa,IAAI,YAAY,CAAC;IACnF,MAAM,MAAM,GAAG,IAAI,CAAC,eAAe,CAAC,UAAU,EAAE,eAAe,IAAI,QAAQ,CAAC;IAE5E,IAAI,MAAM,KAAK,QAAQ,EAAE,CAAC;QACxB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC;QACtC,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC;IACtC,CAAC;AACH,CAAC"}
@@ -0,0 +1,28 @@
1
+ import type { MemorixEntityDescriptor } from "../descriptors/entity-descriptor-types.js";
2
+ import type { MemorixWriteDescriptor } from "../descriptors/write-descriptor-types.js";
3
+ import type { ResolvedIdentity } from "../data/identity.js";
4
+ import type { MemorixWriterDefaults } from "../client/types.js";
5
+ export type MemorixWrittenRecordBase = {
6
+ entityId?: string;
7
+ eventId?: string;
8
+ recordId: string;
9
+ createdAt: string;
10
+ modifiedAt: string;
11
+ data: Record<string, unknown>;
12
+ _memorix?: Record<string, unknown>;
13
+ };
14
+ export declare function buildBaseRecord(args: {
15
+ identity: ResolvedIdentity;
16
+ writeDescriptor: MemorixWriteDescriptor;
17
+ entity: MemorixEntityDescriptor;
18
+ defaults: MemorixWriterDefaults;
19
+ recordId?: string;
20
+ now: string;
21
+ idempotencyKey?: string;
22
+ correlationId?: string;
23
+ source?: Record<string, unknown>;
24
+ actor?: Record<string, unknown>;
25
+ operation: string;
26
+ }): Record<string, unknown>;
27
+ export declare function setPayloadHash(record: Record<string, unknown>, hash: string, path?: string): void;
28
+ //# sourceMappingURL=build-record.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"build-record.d.ts","sourceRoot":"","sources":["../../src/writer/build-record.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,uBAAuB,EAAE,MAAM,2CAA2C,CAAC;AACzF,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,0CAA0C,CAAC;AACvF,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;AAC5D,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAGhE,MAAM,MAAM,wBAAwB,GAAG;IACrC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC9B,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACpC,CAAC;AAEF,wBAAgB,eAAe,CAAC,IAAI,EAAE;IACpC,QAAQ,EAAE,gBAAgB,CAAC;IAC3B,eAAe,EAAE,sBAAsB,CAAC;IACxC,MAAM,EAAE,uBAAuB,CAAC;IAChC,QAAQ,EAAE,qBAAqB,CAAC;IAChC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,GAAG,EAAE,MAAM,CAAC;IACZ,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACjC,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAChC,SAAS,EAAE,MAAM,CAAC;CACnB,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CA2B1B;AAED,wBAAgB,cAAc,CAC5B,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,IAAI,EAAE,MAAM,EACZ,IAAI,SAAyB,GAC5B,IAAI,CAEN"}
@@ -0,0 +1,36 @@
1
+ import { randomUUID } from "node:crypto";
2
+ import { deepSet } from "../utils/path.js";
3
+ export function buildBaseRecord(args) {
4
+ const recordId = args.recordId ?? `rec_${randomUUID().replace(/-/g, "")}`;
5
+ const metaPath = args.defaults.memorixMetadataPath;
6
+ const record = {
7
+ recordId,
8
+ data: {},
9
+ [args.writeDescriptor.timestamps?.createdAtPath ?? "createdAt"]: args.now,
10
+ [args.writeDescriptor.timestamps?.updatedAtPath ?? "modifiedAt"]: args.now,
11
+ };
12
+ if (args.identity.entityId)
13
+ record.entityId = args.identity.entityId;
14
+ if (args.identity.eventId)
15
+ record.eventId = args.identity.eventId;
16
+ const meta = {
17
+ entity: args.entity.entityName,
18
+ contentType: args.writeDescriptor.target.contentType,
19
+ writeDescriptorId: args.writeDescriptor.id,
20
+ operation: args.operation,
21
+ };
22
+ if (args.idempotencyKey)
23
+ meta.idempotencyKey = args.idempotencyKey;
24
+ if (args.correlationId)
25
+ meta.correlationId = args.correlationId;
26
+ if (args.source)
27
+ meta.source = args.source;
28
+ if (args.actor)
29
+ meta.actor = args.actor;
30
+ record[metaPath] = meta;
31
+ return record;
32
+ }
33
+ export function setPayloadHash(record, hash, path = "_memorix.payloadHash") {
34
+ deepSet(record, path, hash);
35
+ }
36
+ //# sourceMappingURL=build-record.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"build-record.js","sourceRoot":"","sources":["../../src/writer/build-record.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAKzC,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAY3C,MAAM,UAAU,eAAe,CAAC,IAY/B;IACC,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,IAAI,OAAO,UAAU,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,EAAE,CAAC;IAC1E,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,mBAAmB,CAAC;IAEnD,MAAM,MAAM,GAA4B;QACtC,QAAQ;QACR,IAAI,EAAE,EAAE;QACR,CAAC,IAAI,CAAC,eAAe,CAAC,UAAU,EAAE,aAAa,IAAI,WAAW,CAAC,EAAE,IAAI,CAAC,GAAG;QACzE,CAAC,IAAI,CAAC,eAAe,CAAC,UAAU,EAAE,aAAa,IAAI,YAAY,CAAC,EAAE,IAAI,CAAC,GAAG;KAC3E,CAAC;IAEF,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ;QAAE,MAAM,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC;IACrE,IAAI,IAAI,CAAC,QAAQ,CAAC,OAAO;QAAE,MAAM,CAAC,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC;IAElE,MAAM,IAAI,GAA4B;QACpC,MAAM,EAAE,IAAI,CAAC,MAAM,CAAC,UAAU;QAC9B,WAAW,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,WAAW;QACpD,iBAAiB,EAAE,IAAI,CAAC,eAAe,CAAC,EAAE;QAC1C,SAAS,EAAE,IAAI,CAAC,SAAS;KAC1B,CAAC;IACF,IAAI,IAAI,CAAC,cAAc;QAAE,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;IACnE,IAAI,IAAI,CAAC,aAAa;QAAE,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC;IAChE,IAAI,IAAI,CAAC,MAAM;QAAE,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;IAC3C,IAAI,IAAI,CAAC,KAAK;QAAE,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;IAExC,MAAM,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC;IACxB,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,UAAU,cAAc,CAC5B,MAA+B,EAC/B,IAAY,EACZ,IAAI,GAAG,sBAAsB;IAE7B,OAAO,CAAC,MAAM,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;AAC9B,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { MemorixWriteDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ import type { MemorixWriteAdapter } from "../data/memorix-write-adapter.js";
3
+ export declare function buildMatchFilter(matchBy: string[], identity: Record<string, unknown>, input: Record<string, unknown>): Record<string, unknown>;
4
+ export declare function findExistingForUpsert(args: {
5
+ adapter: MemorixWriteAdapter;
6
+ collection: string;
7
+ writeDescriptor: MemorixWriteDescriptor;
8
+ identityFilter: Record<string, unknown>;
9
+ input: Record<string, unknown>;
10
+ }): Promise<Record<string, unknown>[]>;
11
+ export declare function resolveConflictAction(writeDescriptor: MemorixWriteDescriptor, matchCount: number): "insert" | "update" | "skip" | "fail" | "append";
12
+ //# sourceMappingURL=resolve-conflict.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-conflict.d.ts","sourceRoot":"","sources":["../../src/writer/resolve-conflict.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,0CAA0C,CAAC;AACvF,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AAG5E,wBAAgB,gBAAgB,CAC9B,OAAO,EAAE,MAAM,EAAE,EACjB,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EACjC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAC7B,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAMzB;AAED,wBAAsB,qBAAqB,CAAC,IAAI,EAAE;IAChD,OAAO,EAAE,mBAAmB,CAAC;IAC7B,UAAU,EAAE,MAAM,CAAC;IACnB,eAAe,EAAE,sBAAsB,CAAC;IACxC,cAAc,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACxC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CAChC,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAAC,CAMrC;AAED,wBAAgB,qBAAqB,CACnC,eAAe,EAAE,sBAAsB,EACvC,UAAU,EAAE,MAAM,GACjB,QAAQ,GAAG,QAAQ,GAAG,MAAM,GAAG,MAAM,GAAG,QAAQ,CAalD"}
@@ -0,0 +1,32 @@
1
+ import { throwWriterError } from "../errors/errors.js";
2
+ export function buildMatchFilter(matchBy, identity, input) {
3
+ const filter = { ...identity };
4
+ for (const key of matchBy) {
5
+ if (key in input)
6
+ filter[key] = input[key];
7
+ }
8
+ return filter;
9
+ }
10
+ export async function findExistingForUpsert(args) {
11
+ const matchBy = args.writeDescriptor.conflict?.matchBy ?? [
12
+ args.writeDescriptor.identity.idField,
13
+ ];
14
+ const filter = buildMatchFilter(matchBy, args.identityFilter, args.input);
15
+ return args.adapter.findMany(args.collection, filter, { limit: 25 });
16
+ }
17
+ export function resolveConflictAction(writeDescriptor, matchCount) {
18
+ if (matchCount === 0)
19
+ return "insert";
20
+ const onConflict = writeDescriptor.conflict?.onConflict ?? "update";
21
+ if (matchCount > 1 && onConflict === "fail") {
22
+ throwWriterError("CONFLICT", `multiple records match upsert criteria (${matchCount})`);
23
+ }
24
+ if (matchCount > 1 && onConflict === "append")
25
+ return "insert";
26
+ if (onConflict === "skip")
27
+ return "skip";
28
+ if (onConflict === "replace")
29
+ return "update";
30
+ return onConflict === "append" ? "insert" : "update";
31
+ }
32
+ //# sourceMappingURL=resolve-conflict.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-conflict.js","sourceRoot":"","sources":["../../src/writer/resolve-conflict.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;AAEvD,MAAM,UAAU,gBAAgB,CAC9B,OAAiB,EACjB,QAAiC,EACjC,KAA8B;IAE9B,MAAM,MAAM,GAA4B,EAAE,GAAG,QAAQ,EAAE,CAAC;IACxD,KAAK,MAAM,GAAG,IAAI,OAAO,EAAE,CAAC;QAC1B,IAAI,GAAG,IAAI,KAAK;YAAE,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;IAC7C,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,qBAAqB,CAAC,IAM3C;IACC,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,OAAO,IAAI;QACxD,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,OAAO;KACtC,CAAC;IACF,MAAM,MAAM,GAAG,gBAAgB,CAAC,OAAO,EAAE,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;IAC1E,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,EAAE,MAAM,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC;AACvE,CAAC;AAED,MAAM,UAAU,qBAAqB,CACnC,eAAuC,EACvC,UAAkB;IAElB,IAAI,UAAU,KAAK,CAAC;QAAE,OAAO,QAAQ,CAAC;IACtC,MAAM,UAAU,GAAG,eAAe,CAAC,QAAQ,EAAE,UAAU,IAAI,QAAQ,CAAC;IACpE,IAAI,UAAU,GAAG,CAAC,IAAI,UAAU,KAAK,MAAM,EAAE,CAAC;QAC5C,gBAAgB,CACd,UAAU,EACV,2CAA2C,UAAU,GAAG,CACzD,CAAC;IACJ,CAAC;IACD,IAAI,UAAU,GAAG,CAAC,IAAI,UAAU,KAAK,QAAQ;QAAE,OAAO,QAAQ,CAAC;IAC/D,IAAI,UAAU,KAAK,MAAM;QAAE,OAAO,MAAM,CAAC;IACzC,IAAI,UAAU,KAAK,SAAS;QAAE,OAAO,QAAQ,CAAC;IAC9C,OAAO,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC;AACvD,CAAC"}
@@ -0,0 +1,23 @@
1
+ import type { MemorixWriteDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ import type { MemorixWriteAdapter } from "../data/memorix-write-adapter.js";
3
+ import type { MemorixWriteIssue } from "../errors/issues.js";
4
+ export declare function hashPayload(payload: unknown): string;
5
+ export type IdempotencyCheckResult = {
6
+ action: "proceed";
7
+ } | {
8
+ action: "deduplicated";
9
+ existing: Record<string, unknown>;
10
+ } | {
11
+ action: "conflict";
12
+ issues: MemorixWriteIssue[];
13
+ };
14
+ export declare function checkIdempotency(args: {
15
+ adapter: MemorixWriteAdapter;
16
+ collection: string;
17
+ writeDescriptor: MemorixWriteDescriptor;
18
+ idempotencyKey: string;
19
+ payloadHash: string;
20
+ identityFilter: Record<string, unknown>;
21
+ }): Promise<IdempotencyCheckResult>;
22
+ export declare function applyIdempotencyToRecord(record: Record<string, unknown>, idempotencyKey: string, payloadHash: string, writeDescriptor: MemorixWriteDescriptor): void;
23
+ //# sourceMappingURL=resolve-idempotency.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-idempotency.d.ts","sourceRoot":"","sources":["../../src/writer/resolve-idempotency.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,0CAA0C,CAAC;AACvF,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AAC5E,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAO7D,wBAAgB,WAAW,CAAC,OAAO,EAAE,OAAO,GAAG,MAAM,CAIpD;AAED,MAAM,MAAM,sBAAsB,GAC9B;IAAE,MAAM,EAAE,SAAS,CAAA;CAAE,GACrB;IAAE,MAAM,EAAE,cAAc,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,GAC7D;IAAE,MAAM,EAAE,UAAU,CAAC;IAAC,MAAM,EAAE,iBAAiB,EAAE,CAAA;CAAE,CAAC;AAExD,wBAAsB,gBAAgB,CAAC,IAAI,EAAE;IAC3C,OAAO,EAAE,mBAAmB,CAAC;IAC7B,UAAU,EAAE,MAAM,CAAC;IACnB,eAAe,EAAE,sBAAsB,CAAC;IACxC,cAAc,EAAE,MAAM,CAAC;IACvB,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACzC,GAAG,OAAO,CAAC,sBAAsB,CAAC,CAkClC;AASD,wBAAgB,wBAAwB,CACtC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,cAAc,EAAE,MAAM,EACtB,WAAW,EAAE,MAAM,EACnB,eAAe,EAAE,sBAAsB,GACtC,IAAI,CAsBN"}
@@ -0,0 +1,69 @@
1
+ import { createHash } from "node:crypto";
2
+ import { writeIssue } from "../errors/issues.js";
3
+ import { DEFAULT_IDEMPOTENCY_KEY_PATH, DEFAULT_PAYLOAD_HASH_PATH, } from "../constants.js";
4
+ export function hashPayload(payload) {
5
+ return createHash("sha256")
6
+ .update(JSON.stringify(payload))
7
+ .digest("hex");
8
+ }
9
+ export async function checkIdempotency(args) {
10
+ const cfg = args.writeDescriptor.conflict?.idempotency;
11
+ if (!cfg?.enabled)
12
+ return { action: "proceed" };
13
+ const keyPath = cfg.keyPath ?? DEFAULT_IDEMPOTENCY_KEY_PATH;
14
+ const hashPath = cfg.payloadHashPath ?? DEFAULT_PAYLOAD_HASH_PATH;
15
+ const filter = {
16
+ ...args.identityFilter,
17
+ [keyPath]: args.idempotencyKey,
18
+ };
19
+ const existing = await args.adapter.findOne(args.collection, filter);
20
+ if (!existing)
21
+ return { action: "proceed" };
22
+ const existingHash = getByPath(existing, hashPath);
23
+ if (existingHash === args.payloadHash) {
24
+ const behavior = cfg.behavior ?? "deduplicate";
25
+ if (behavior === "failOnDifferentPayload") {
26
+ return { action: "deduplicated", existing };
27
+ }
28
+ return { action: "deduplicated", existing };
29
+ }
30
+ return {
31
+ action: "conflict",
32
+ issues: [
33
+ writeIssue("IDEMPOTENCY_CONFLICT", `idempotency key "${args.idempotencyKey}" already used with different payload`, { severity: "error", details: { idempotencyKey: args.idempotencyKey } }),
34
+ ],
35
+ };
36
+ }
37
+ function getByPath(obj, path) {
38
+ return path.split(".").reduce((acc, p) => {
39
+ if (acc == null || typeof acc !== "object")
40
+ return undefined;
41
+ return acc[p];
42
+ }, obj);
43
+ }
44
+ export function applyIdempotencyToRecord(record, idempotencyKey, payloadHash, writeDescriptor) {
45
+ const cfg = writeDescriptor.conflict?.idempotency;
46
+ if (!cfg?.enabled)
47
+ return;
48
+ const keyPath = cfg.storePath ?? cfg.keyPath ?? DEFAULT_IDEMPOTENCY_KEY_PATH;
49
+ const hashPath = cfg.payloadHashPath ?? DEFAULT_PAYLOAD_HASH_PATH;
50
+ const parts = keyPath.split(".");
51
+ let cur = record;
52
+ for (let i = 0; i < parts.length - 1; i++) {
53
+ const p = parts[i];
54
+ if (!cur[p] || typeof cur[p] !== "object")
55
+ cur[p] = {};
56
+ cur = cur[p];
57
+ }
58
+ cur[parts[parts.length - 1]] = idempotencyKey;
59
+ const hashParts = hashPath.split(".");
60
+ cur = record;
61
+ for (let i = 0; i < hashParts.length - 1; i++) {
62
+ const p = hashParts[i];
63
+ if (!cur[p] || typeof cur[p] !== "object")
64
+ cur[p] = {};
65
+ cur = cur[p];
66
+ }
67
+ cur[hashParts[hashParts.length - 1]] = payloadHash;
68
+ }
69
+ //# sourceMappingURL=resolve-idempotency.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-idempotency.js","sourceRoot":"","sources":["../../src/writer/resolve-idempotency.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { MemorixWriteDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ export type WriteOperation = "add" | "upsert" | "patch" | "replace";
3
+ export declare function resolveWriteOperation(writeDescriptor: MemorixWriteDescriptor, requested?: WriteOperation): WriteOperation;
4
+ //# sourceMappingURL=resolve-operation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-operation.d.ts","sourceRoot":"","sources":["../../src/writer/resolve-operation.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,0CAA0C,CAAC;AAGvF,MAAM,MAAM,cAAc,GAAG,KAAK,GAAG,QAAQ,GAAG,OAAO,GAAG,SAAS,CAAC;AAEpE,wBAAgB,qBAAqB,CACnC,eAAe,EAAE,sBAAsB,EACvC,SAAS,CAAC,EAAE,cAAc,GACzB,cAAc,CAShB"}
@@ -0,0 +1,9 @@
1
+ import { throwWriterError } from "../errors/errors.js";
2
+ export function resolveWriteOperation(writeDescriptor, requested) {
3
+ const op = requested ?? writeDescriptor.operations.default;
4
+ if (!writeDescriptor.operations.allowed.includes(op)) {
5
+ throwWriterError("OPERATION_NOT_ALLOWED", `operation "${op}" is not allowed by write descriptor "${writeDescriptor.id}"`);
6
+ }
7
+ return op;
8
+ }
9
+ //# sourceMappingURL=resolve-operation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-operation.js","sourceRoot":"","sources":["../../src/writer/resolve-operation.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;AAIvD,MAAM,UAAU,qBAAqB,CACnC,eAAuC,EACvC,SAA0B;IAE1B,MAAM,EAAE,GAAG,SAAS,IAAI,eAAe,CAAC,UAAU,CAAC,OAAO,CAAC;IAC3D,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC,EAAE,CAAC;QACrD,gBAAgB,CACd,uBAAuB,EACvB,cAAc,EAAE,yCAAyC,eAAe,CAAC,EAAE,GAAG,CAC/E,CAAC;IACJ,CAAC;IACD,OAAO,EAAE,CAAC;AACZ,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { MemorixWriteFieldDescriptor } from "../descriptors/write-descriptor-types.js";
2
+ export declare function resolveMemorixWriteFieldTarget(fieldName: string, field: MemorixWriteFieldDescriptor): string;
3
+ export declare function resolveInputPath(fieldName: string, field: MemorixWriteFieldDescriptor): string;
4
+ //# sourceMappingURL=resolve-write-field-target.d.ts.map