@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.
- package/README.md +145 -0
- package/catalox-seeds/memorix-write-descriptors.manifest.json +164 -0
- package/dist/client/create-client.d.ts +3 -0
- package/dist/client/create-client.d.ts.map +1 -0
- package/dist/client/create-client.js +81 -0
- package/dist/client/create-client.js.map +1 -0
- package/dist/client/create-from-env.d.ts +3 -0
- package/dist/client/create-from-env.d.ts.map +1 -0
- package/dist/client/create-from-env.js +21 -0
- package/dist/client/create-from-env.js.map +1 -0
- package/dist/client/ensure-tier.d.ts +3 -0
- package/dist/client/ensure-tier.d.ts.map +1 -0
- package/dist/client/ensure-tier.js +4 -0
- package/dist/client/ensure-tier.js.map +1 -0
- package/dist/client/types.d.ts +165 -0
- package/dist/client/types.d.ts.map +1 -0
- package/dist/client/types.js +2 -0
- package/dist/client/types.js.map +1 -0
- package/dist/constants.d.ts +13 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +13 -0
- package/dist/constants.js.map +1 -0
- package/dist/content/compute-content-metadata.d.ts +16 -0
- package/dist/content/compute-content-metadata.d.ts.map +1 -0
- package/dist/content/compute-content-metadata.js +13 -0
- package/dist/content/compute-content-metadata.js.map +1 -0
- package/dist/content/generate-content-key.d.ts +13 -0
- package/dist/content/generate-content-key.d.ts.map +1 -0
- package/dist/content/generate-content-key.js +22 -0
- package/dist/content/generate-content-key.js.map +1 -0
- package/dist/content/generate-preview.d.ts +2 -0
- package/dist/content/generate-preview.d.ts.map +1 -0
- package/dist/content/generate-preview.js +7 -0
- package/dist/content/generate-preview.js.map +1 -0
- package/dist/content/map-content-write-errors.d.ts +3 -0
- package/dist/content/map-content-write-errors.d.ts.map +1 -0
- package/dist/content/map-content-write-errors.js +9 -0
- package/dist/content/map-content-write-errors.js.map +1 -0
- package/dist/content/resolve-content-write-config.d.ts +25 -0
- package/dist/content/resolve-content-write-config.d.ts.map +1 -0
- package/dist/content/resolve-content-write-config.js +35 -0
- package/dist/content/resolve-content-write-config.js.map +1 -0
- package/dist/content/write-content-object.d.ts +40 -0
- package/dist/content/write-content-object.d.ts.map +1 -0
- package/dist/content/write-content-object.js +117 -0
- package/dist/content/write-content-object.js.map +1 -0
- package/dist/data/collection-name.d.ts +30 -0
- package/dist/data/collection-name.d.ts.map +1 -0
- package/dist/data/collection-name.js +86 -0
- package/dist/data/collection-name.js.map +1 -0
- package/dist/data/identity.d.ts +13 -0
- package/dist/data/identity.d.ts.map +1 -0
- package/dist/data/identity.js +29 -0
- package/dist/data/identity.js.map +1 -0
- package/dist/data/memorix-write-adapter.d.ts +33 -0
- package/dist/data/memorix-write-adapter.d.ts.map +1 -0
- package/dist/data/memorix-write-adapter.js +45 -0
- package/dist/data/memorix-write-adapter.js.map +1 -0
- package/dist/descriptors/entity-descriptor-types.d.ts +59 -0
- package/dist/descriptors/entity-descriptor-types.d.ts.map +1 -0
- package/dist/descriptors/entity-descriptor-types.js +2 -0
- package/dist/descriptors/entity-descriptor-types.js.map +1 -0
- package/dist/descriptors/load-entity-descriptor.d.ts +7 -0
- package/dist/descriptors/load-entity-descriptor.d.ts.map +1 -0
- package/dist/descriptors/load-entity-descriptor.js +23 -0
- package/dist/descriptors/load-entity-descriptor.js.map +1 -0
- package/dist/descriptors/load-write-descriptor.d.ts +5 -0
- package/dist/descriptors/load-write-descriptor.d.ts.map +1 -0
- package/dist/descriptors/load-write-descriptor.js +19 -0
- package/dist/descriptors/load-write-descriptor.js.map +1 -0
- package/dist/descriptors/secret-scan.d.ts +2 -0
- package/dist/descriptors/secret-scan.d.ts.map +1 -0
- package/dist/descriptors/secret-scan.js +39 -0
- package/dist/descriptors/secret-scan.js.map +1 -0
- package/dist/descriptors/validate-write-descriptor.d.ts +4 -0
- package/dist/descriptors/validate-write-descriptor.d.ts.map +1 -0
- package/dist/descriptors/validate-write-descriptor.js +45 -0
- package/dist/descriptors/validate-write-descriptor.js.map +1 -0
- package/dist/descriptors/write-descriptor-types.d.ts +112 -0
- package/dist/descriptors/write-descriptor-types.d.ts.map +1 -0
- package/dist/descriptors/write-descriptor-types.js +2 -0
- package/dist/descriptors/write-descriptor-types.js.map +1 -0
- package/dist/errors/errors.d.ts +6 -0
- package/dist/errors/errors.d.ts.map +1 -0
- package/dist/errors/errors.js +12 -0
- package/dist/errors/errors.js.map +1 -0
- package/dist/errors/issues.d.ts +13 -0
- package/dist/errors/issues.d.ts.map +1 -0
- package/dist/errors/issues.js +9 -0
- package/dist/errors/issues.js.map +1 -0
- package/dist/index.d.ts +20 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +15 -0
- package/dist/index.js.map +1 -0
- package/dist/mongo/collection-documents.d.ts +54 -0
- package/dist/mongo/collection-documents.d.ts.map +1 -0
- package/dist/mongo/collection-documents.js +100 -0
- package/dist/mongo/collection-documents.js.map +1 -0
- package/dist/mongo/collection-name-validate.d.ts +2 -0
- package/dist/mongo/collection-name-validate.d.ts.map +1 -0
- package/dist/mongo/collection-name-validate.js +19 -0
- package/dist/mongo/collection-name-validate.js.map +1 -0
- package/dist/mongo/connection.d.ts +3 -0
- package/dist/mongo/connection.d.ts.map +1 -0
- package/dist/mongo/connection.js +8 -0
- package/dist/mongo/connection.js.map +1 -0
- package/dist/mongo/default-collections.d.ts +3 -0
- package/dist/mongo/default-collections.d.ts.map +1 -0
- package/dist/mongo/default-collections.js +14 -0
- package/dist/mongo/default-collections.js.map +1 -0
- package/dist/mongo/env.d.ts +6 -0
- package/dist/mongo/env.d.ts.map +1 -0
- package/dist/mongo/env.js +33 -0
- package/dist/mongo/env.js.map +1 -0
- package/dist/mongo/id-normalize.d.ts +3 -0
- package/dist/mongo/id-normalize.d.ts.map +1 -0
- package/dist/mongo/id-normalize.js +23 -0
- package/dist/mongo/id-normalize.js.map +1 -0
- package/dist/mongo/index.d.ts +5 -0
- package/dist/mongo/index.d.ts.map +1 -0
- package/dist/mongo/index.js +5 -0
- package/dist/mongo/index.js.map +1 -0
- package/dist/mongo/target.d.ts +4 -0
- package/dist/mongo/target.d.ts.map +1 -0
- package/dist/mongo/target.js +9 -0
- package/dist/mongo/target.js.map +1 -0
- package/dist/mongo/tier-env.d.ts +9 -0
- package/dist/mongo/tier-env.d.ts.map +1 -0
- package/dist/mongo/tier-env.js +11 -0
- package/dist/mongo/tier-env.js.map +1 -0
- package/dist/mongo/tier.d.ts +22 -0
- package/dist/mongo/tier.d.ts.map +1 -0
- package/dist/mongo/tier.js +50 -0
- package/dist/mongo/tier.js.map +1 -0
- package/dist/utils/path.d.ts +4 -0
- package/dist/utils/path.d.ts.map +1 -0
- package/dist/utils/path.js +42 -0
- package/dist/utils/path.js.map +1 -0
- package/dist/validation/validate-content.d.ts +17 -0
- package/dist/validation/validate-content.d.ts.map +1 -0
- package/dist/validation/validate-content.js +47 -0
- package/dist/validation/validate-content.js.map +1 -0
- package/dist/validation/validate-fields.d.ts +8 -0
- package/dist/validation/validate-fields.d.ts.map +1 -0
- package/dist/validation/validate-fields.js +49 -0
- package/dist/validation/validate-fields.js.map +1 -0
- package/dist/validation/validate-identity.d.ts +3 -0
- package/dist/validation/validate-identity.d.ts.map +1 -0
- package/dist/validation/validate-identity.js +2 -0
- package/dist/validation/validate-identity.js.map +1 -0
- package/dist/validation/validate-immutability.d.ts +9 -0
- package/dist/validation/validate-immutability.d.ts.map +1 -0
- package/dist/validation/validate-immutability.js +35 -0
- package/dist/validation/validate-immutability.js.map +1 -0
- package/dist/validation/validate-values.d.ts +4 -0
- package/dist/validation/validate-values.d.ts.map +1 -0
- package/dist/validation/validate-values.js +55 -0
- package/dist/validation/validate-values.js.map +1 -0
- package/dist/writer/apply-audit.d.ts +10 -0
- package/dist/writer/apply-audit.d.ts.map +1 -0
- package/dist/writer/apply-audit.js +39 -0
- package/dist/writer/apply-audit.js.map +1 -0
- package/dist/writer/apply-field-mapping.d.ts +9 -0
- package/dist/writer/apply-field-mapping.d.ts.map +1 -0
- package/dist/writer/apply-field-mapping.js +32 -0
- package/dist/writer/apply-field-mapping.js.map +1 -0
- package/dist/writer/apply-timestamps.d.ts +8 -0
- package/dist/writer/apply-timestamps.d.ts.map +1 -0
- package/dist/writer/apply-timestamps.js +12 -0
- package/dist/writer/apply-timestamps.js.map +1 -0
- package/dist/writer/build-record.d.ts +28 -0
- package/dist/writer/build-record.d.ts.map +1 -0
- package/dist/writer/build-record.js +36 -0
- package/dist/writer/build-record.js.map +1 -0
- package/dist/writer/resolve-conflict.d.ts +12 -0
- package/dist/writer/resolve-conflict.d.ts.map +1 -0
- package/dist/writer/resolve-conflict.js +32 -0
- package/dist/writer/resolve-conflict.js.map +1 -0
- package/dist/writer/resolve-idempotency.d.ts +23 -0
- package/dist/writer/resolve-idempotency.d.ts.map +1 -0
- package/dist/writer/resolve-idempotency.js +69 -0
- package/dist/writer/resolve-idempotency.js.map +1 -0
- package/dist/writer/resolve-operation.d.ts +4 -0
- package/dist/writer/resolve-operation.d.ts.map +1 -0
- package/dist/writer/resolve-operation.js +9 -0
- package/dist/writer/resolve-operation.js.map +1 -0
- package/dist/writer/resolve-write-field-target.d.ts +4 -0
- package/dist/writer/resolve-write-field-target.d.ts.map +1 -0
- package/dist/writer/resolve-write-field-target.js +7 -0
- package/dist/writer/resolve-write-field-target.js.map +1 -0
- package/dist/writer/validate-write.d.ts +3 -0
- package/dist/writer/validate-write.d.ts.map +1 -0
- package/dist/writer/validate-write.js +72 -0
- package/dist/writer/validate-write.js.map +1 -0
- package/dist/writer/write-record.d.ts +3 -0
- package/dist/writer/write-record.d.ts.map +1 -0
- package/dist/writer/write-record.js +472 -0
- package/dist/writer/write-record.js.map +1 -0
- package/dist/writer/write-records.d.ts +3 -0
- package/dist/writer/write-records.d.ts.map +1 -0
- package/dist/writer/write-records.js +79 -0
- package/dist/writer/write-records.js.map +1 -0
- package/docs/MEMORIX-DATABASE-CONVENTIONS.md +324 -0
- 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 @@
|
|
|
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 @@
|
|
|
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":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAIzC,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AACjD,OAAO,EACL,4BAA4B,EAC5B,yBAAyB,GAC1B,MAAM,iBAAiB,CAAC;AAEzB,MAAM,UAAU,WAAW,CAAC,OAAgB;IAC1C,OAAO,UAAU,CAAC,QAAQ,CAAC;SACxB,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;SAC/B,MAAM,CAAC,KAAK,CAAC,CAAC;AACnB,CAAC;AAOD,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,IAOtC;IACC,MAAM,GAAG,GAAG,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,WAAW,CAAC;IACvD,IAAI,CAAC,GAAG,EAAE,OAAO;QAAE,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC;IAEhD,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,IAAI,4BAA4B,CAAC;IAC5D,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,IAAI,yBAAyB,CAAC;IAElE,MAAM,MAAM,GAAG;QACb,GAAG,IAAI,CAAC,cAAc;QACtB,CAAC,OAAO,CAAC,EAAE,IAAI,CAAC,cAAc;KAC/B,CAAC;IAEF,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;IACrE,IAAI,CAAC,QAAQ;QAAE,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC;IAE5C,MAAM,YAAY,GAAG,SAAS,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IACnD,IAAI,YAAY,KAAK,IAAI,CAAC,WAAW,EAAE,CAAC;QACtC,MAAM,QAAQ,GAAG,GAAG,CAAC,QAAQ,IAAI,aAAa,CAAC;QAC/C,IAAI,QAAQ,KAAK,wBAAwB,EAAE,CAAC;YAC1C,OAAO,EAAE,MAAM,EAAE,cAAc,EAAE,QAAQ,EAAE,CAAC;QAC9C,CAAC;QACD,OAAO,EAAE,MAAM,EAAE,cAAc,EAAE,QAAQ,EAAE,CAAC;IAC9C,CAAC;IAED,OAAO;QACL,MAAM,EAAE,UAAU;QAClB,MAAM,EAAE;YACN,UAAU,CACR,sBAAsB,EACtB,oBAAoB,IAAI,CAAC,cAAc,uCAAuC,EAC9E,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,EAAE,cAAc,EAAE,IAAI,CAAC,cAAc,EAAE,EAAE,CACxE;SACF;KACF,CAAC;AACJ,CAAC;AAED,SAAS,SAAS,CAAC,GAA4B,EAAE,IAAY;IAC3D,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAU,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE;QAChD,IAAI,GAAG,IAAI,IAAI,IAAI,OAAO,GAAG,KAAK,QAAQ;YAAE,OAAO,SAAS,CAAC;QAC7D,OAAQ,GAA+B,CAAC,CAAC,CAAC,CAAC;IAC7C,CAAC,EAAE,GAAG,CAAC,CAAC;AACV,CAAC;AAED,MAAM,UAAU,wBAAwB,CACtC,MAA+B,EAC/B,cAAsB,EACtB,WAAmB,EACnB,eAAuC;IAEvC,MAAM,GAAG,GAAG,eAAe,CAAC,QAAQ,EAAE,WAAW,CAAC;IAClD,IAAI,CAAC,GAAG,EAAE,OAAO;QAAE,OAAO;IAC1B,MAAM,OAAO,GAAG,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,OAAO,IAAI,4BAA4B,CAAC;IAC7E,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,IAAI,yBAAyB,CAAC;IAClE,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACjC,IAAI,GAAG,GAA4B,MAAM,CAAC;IAC1C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QAC1C,MAAM,CAAC,GAAG,KAAK,CAAC,CAAC,CAAE,CAAC;QACpB,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,OAAO,GAAG,CAAC,CAAC,CAAC,KAAK,QAAQ;YAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC;QACvD,GAAG,GAAG,GAAG,CAAC,CAAC,CAA4B,CAAC;IAC1C,CAAC;IACD,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAE,CAAC,GAAG,cAAc,CAAC;IAE/C,MAAM,SAAS,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACtC,GAAG,GAAG,MAAM,CAAC;IACb,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QAC9C,MAAM,CAAC,GAAG,SAAS,CAAC,CAAC,CAAE,CAAC;QACxB,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,OAAO,GAAG,CAAC,CAAC,CAAC,KAAK,QAAQ;YAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC;QACvD,GAAG,GAAG,GAAG,CAAC,CAAC,CAA4B,CAAC;IAC1C,CAAC;IACD,GAAG,CAAC,SAAS,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,CAAE,CAAC,GAAG,WAAW,CAAC;AACtD,CAAC"}
|
|
@@ -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
|