@xyo-network/payload-builder 3.5.2 → 3.6.0-rc.1
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/dist/browser/Builder.d.ts +13 -24
- package/dist/browser/Builder.d.ts.map +1 -1
- package/dist/browser/BuilderBase.d.ts +12 -13
- package/dist/browser/BuilderBase.d.ts.map +1 -1
- package/dist/browser/Options.d.ts +1 -1
- package/dist/browser/Options.d.ts.map +1 -1
- package/dist/browser/index.mjs +43 -142
- package/dist/browser/index.mjs.map +1 -1
- package/dist/neutral/Builder.d.ts +13 -24
- package/dist/neutral/Builder.d.ts.map +1 -1
- package/dist/neutral/BuilderBase.d.ts +12 -13
- package/dist/neutral/BuilderBase.d.ts.map +1 -1
- package/dist/neutral/Options.d.ts +1 -1
- package/dist/neutral/Options.d.ts.map +1 -1
- package/dist/neutral/index.mjs +43 -142
- package/dist/neutral/index.mjs.map +1 -1
- package/dist/node/Builder.d.ts +13 -24
- package/dist/node/Builder.d.ts.map +1 -1
- package/dist/node/BuilderBase.d.ts +12 -13
- package/dist/node/BuilderBase.d.ts.map +1 -1
- package/dist/node/Options.d.ts +1 -1
- package/dist/node/Options.d.ts.map +1 -1
- package/dist/node/index.mjs +43 -142
- package/dist/node/index.mjs.map +1 -1
- package/package.json +12 -11
- package/src/Builder.ts +39 -146
- package/src/BuilderBase.ts +29 -50
- package/src/Options.ts +1 -1
package/dist/node/index.mjs
CHANGED
|
@@ -1,7 +1,5 @@
|
|
|
1
1
|
// src/Builder.ts
|
|
2
|
-
import {
|
|
3
|
-
import { isJsonObject as isJsonObject2, omitBy as omitBy2 } from "@xylabs/object";
|
|
4
|
-
import { PayloadHasher } from "@xyo-network/hash";
|
|
2
|
+
import { ObjectHasher } from "@xyo-network/hash";
|
|
5
3
|
|
|
6
4
|
// src/BuilderBase.ts
|
|
7
5
|
import { assertEx } from "@xylabs/assert";
|
|
@@ -12,143 +10,85 @@ import {
|
|
|
12
10
|
} from "@xylabs/object";
|
|
13
11
|
import { removeEmptyFields } from "@xyo-network/hash";
|
|
14
12
|
var removeMetaAndSchema = (payload) => {
|
|
15
|
-
const { ...result } = payload;
|
|
16
|
-
delete result.$hash;
|
|
17
|
-
delete result.$meta;
|
|
13
|
+
const { ...result } = PayloadBuilder.omitMeta(payload);
|
|
18
14
|
delete result.schema;
|
|
19
15
|
return result;
|
|
20
16
|
};
|
|
21
|
-
var
|
|
17
|
+
var omitByPrefixPredicate = (prefix) => (_, key) => {
|
|
22
18
|
assertEx(typeof key === "string", () => `Invalid key type [${key}, ${typeof key}]`);
|
|
23
19
|
return key.startsWith(prefix);
|
|
24
20
|
};
|
|
25
21
|
var PayloadBuilderBase = class _PayloadBuilderBase {
|
|
26
22
|
constructor(options) {
|
|
27
23
|
this.options = options;
|
|
28
|
-
const {
|
|
29
|
-
schema,
|
|
30
|
-
fields,
|
|
31
|
-
meta
|
|
32
|
-
} = options;
|
|
24
|
+
const { schema, fields } = options;
|
|
33
25
|
this._schema = schema;
|
|
34
|
-
this._fields = removeEmptyFields(fields ?? {});
|
|
35
|
-
this._$meta = meta;
|
|
26
|
+
this._fields = removeMetaAndSchema(removeEmptyFields(structuredClone(fields ?? {})));
|
|
36
27
|
}
|
|
37
|
-
_$meta;
|
|
38
28
|
_fields;
|
|
39
29
|
_schema;
|
|
40
|
-
static dataHashableFields(schema,
|
|
41
|
-
const cleanFields =
|
|
30
|
+
static dataHashableFields(schema, payload) {
|
|
31
|
+
const cleanFields = removeEmptyFields({ ...payload, schema });
|
|
42
32
|
assertEx(
|
|
43
33
|
cleanFields === void 0 || isJsonObject(cleanFields),
|
|
44
34
|
() => `Fields must be JsonObject: ${JSON.stringify(toJson(cleanFields), null, 2)}`
|
|
45
35
|
);
|
|
46
|
-
return
|
|
36
|
+
return this.omitMeta(cleanFields);
|
|
47
37
|
}
|
|
48
|
-
static
|
|
49
|
-
|
|
50
|
-
if (!meta.timestamp && stamp) {
|
|
51
|
-
meta.timestamp = meta.timestamp ?? Date.now();
|
|
52
|
-
}
|
|
53
|
-
return meta;
|
|
38
|
+
static omitClientMeta(payload, maxDepth = 100) {
|
|
39
|
+
return omitBy(payload, omitByPrefixPredicate("$"), maxDepth);
|
|
54
40
|
}
|
|
55
|
-
|
|
56
|
-
this.
|
|
57
|
-
|
|
41
|
+
static omitMeta(payload, maxDepth = 100) {
|
|
42
|
+
return this.omitStorageMeta(this.omitClientMeta(payload, maxDepth), maxDepth);
|
|
43
|
+
}
|
|
44
|
+
static omitStorageMeta(payload, maxDepth = 100) {
|
|
45
|
+
return omitBy(payload, omitByPrefixPredicate("_"), maxDepth);
|
|
58
46
|
}
|
|
59
47
|
async dataHashableFields() {
|
|
60
48
|
return await _PayloadBuilderBase.dataHashableFields(
|
|
61
49
|
assertEx(this._schema, () => "Payload: Missing Schema"),
|
|
50
|
+
// TDOD: Add verification that required fields are present
|
|
62
51
|
this._fields
|
|
63
52
|
);
|
|
64
53
|
}
|
|
65
|
-
// we do not require sending in $hash since it will be generated anyway
|
|
66
54
|
fields(fields) {
|
|
67
55
|
if (fields) {
|
|
68
|
-
const
|
|
69
|
-
|
|
70
|
-
$meta,
|
|
71
|
-
$hash,
|
|
72
|
-
schema,
|
|
73
|
-
...fieldsOnly
|
|
74
|
-
} = fields;
|
|
75
|
-
if ($meta) {
|
|
76
|
-
this.$meta($meta);
|
|
77
|
-
}
|
|
56
|
+
const fieldsClone = structuredClone(fields);
|
|
57
|
+
const { schema } = fieldsClone;
|
|
78
58
|
if (schema) {
|
|
79
59
|
this.schema(schema);
|
|
80
60
|
}
|
|
81
|
-
this._fields = removeMetaAndSchema(
|
|
61
|
+
this._fields = removeEmptyFields(removeMetaAndSchema(fieldsClone));
|
|
82
62
|
}
|
|
83
63
|
return this;
|
|
84
64
|
}
|
|
85
65
|
schema(value) {
|
|
86
66
|
this._schema = value;
|
|
87
67
|
}
|
|
88
|
-
async metaFields(dataHash, stamp = true) {
|
|
89
|
-
return await _PayloadBuilderBase.metaFields(dataHash, this._$meta, stamp);
|
|
90
|
-
}
|
|
91
68
|
};
|
|
92
69
|
|
|
93
70
|
// src/Builder.ts
|
|
94
|
-
var omitByPredicate2 = (prefix) => (_, key) => {
|
|
95
|
-
assertEx2(typeof key === "string", () => `Invalid key type [${String(key)}, ${typeof key}]`);
|
|
96
|
-
return key.startsWith(prefix);
|
|
97
|
-
};
|
|
98
71
|
var PayloadBuilder = class _PayloadBuilder extends PayloadBuilderBase {
|
|
99
|
-
static async
|
|
100
|
-
|
|
101
|
-
return await Promise.all(payload.map((payload2) => this.build(payload2, options)));
|
|
102
|
-
} else {
|
|
103
|
-
const { stamp = false, validate = true } = options;
|
|
104
|
-
const {
|
|
105
|
-
schema,
|
|
106
|
-
$hash: incomingDataHash,
|
|
107
|
-
$meta: incomingMeta = {}
|
|
108
|
-
} = payload;
|
|
109
|
-
const { _signatures } = payload;
|
|
110
|
-
if (_signatures && !incomingMeta.signatures) {
|
|
111
|
-
incomingMeta.signatures = _signatures;
|
|
112
|
-
}
|
|
113
|
-
const fields = removeMetaAndSchema(payload);
|
|
114
|
-
const dataHashableFields = await _PayloadBuilder.dataHashableFields(schema, fields);
|
|
115
|
-
const $hash = validate || incomingDataHash === void 0 ? await PayloadHasher.hash(dataHashableFields) : incomingDataHash;
|
|
116
|
-
const $meta = { ...incomingMeta };
|
|
117
|
-
if ($meta.timestamp === void 0 && stamp) {
|
|
118
|
-
$meta.timestamp = Date.now();
|
|
119
|
-
}
|
|
120
|
-
const hashableFields = {
|
|
121
|
-
...dataHashableFields,
|
|
122
|
-
$hash,
|
|
123
|
-
schema
|
|
124
|
-
};
|
|
125
|
-
if (Object.keys($meta).length > 0) {
|
|
126
|
-
hashableFields.$meta = $meta;
|
|
127
|
-
}
|
|
128
|
-
return hashableFields;
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
static async dataHash(payload, options) {
|
|
132
|
-
return (await this.build(payload, options)).$hash;
|
|
72
|
+
static async dataHash(payload) {
|
|
73
|
+
return await ObjectHasher.hash(this.omitMeta(payload));
|
|
133
74
|
}
|
|
134
|
-
static async dataHashPairs(payloads
|
|
75
|
+
static async dataHashPairs(payloads) {
|
|
135
76
|
return await Promise.all(
|
|
136
77
|
payloads.map(async (payload) => {
|
|
137
|
-
const
|
|
138
|
-
return [
|
|
78
|
+
const dataHash = await this.dataHash(payload);
|
|
79
|
+
return [payload, dataHash];
|
|
139
80
|
})
|
|
140
81
|
);
|
|
141
82
|
}
|
|
142
|
-
static async dataHashes(payloads
|
|
83
|
+
static async dataHashes(payloads) {
|
|
143
84
|
return payloads ? await Promise.all(
|
|
144
85
|
payloads.map(async (payload) => {
|
|
145
|
-
|
|
146
|
-
return built.$hash;
|
|
86
|
+
return await _PayloadBuilder.dataHash(payload);
|
|
147
87
|
})
|
|
148
88
|
) : void 0;
|
|
149
89
|
}
|
|
150
90
|
static async filterExclude(payloads = [], hash) {
|
|
151
|
-
return await
|
|
91
|
+
return await ObjectHasher.filterExcludeByHash(await this.filterExcludeByDataHash(payloads, hash), hash);
|
|
152
92
|
}
|
|
153
93
|
static async filterExcludeByDataHash(payloads = [], hash) {
|
|
154
94
|
const hashes = Array.isArray(hash) ? hash : [hash];
|
|
@@ -161,53 +101,33 @@ var PayloadBuilder = class _PayloadBuilder extends PayloadBuilderBase {
|
|
|
161
101
|
static async findByDataHash(payloads = [], hash) {
|
|
162
102
|
return (await this.dataHashPairs(payloads)).find(([_, objHash]) => objHash === hash)?.[0];
|
|
163
103
|
}
|
|
164
|
-
static async hash(payload
|
|
165
|
-
return await
|
|
104
|
+
static async hash(payload) {
|
|
105
|
+
return await ObjectHasher.hash(this.omitStorageMeta(payload));
|
|
166
106
|
}
|
|
167
107
|
/**
|
|
168
108
|
* Creates an array of payload/hash tuples based on the payloads passed in
|
|
169
109
|
* @param objs Any array of payloads
|
|
170
110
|
* @returns An array of payload/hash tuples
|
|
171
111
|
*/
|
|
172
|
-
static async hashPairs(payloads
|
|
112
|
+
static async hashPairs(payloads) {
|
|
173
113
|
return await Promise.all(
|
|
174
114
|
payloads.map(async (payload) => {
|
|
175
|
-
|
|
176
|
-
return [built, await _PayloadBuilder.hash(built)];
|
|
115
|
+
return [payload, await _PayloadBuilder.hash(payload)];
|
|
177
116
|
})
|
|
178
117
|
);
|
|
179
118
|
}
|
|
180
|
-
static
|
|
181
|
-
|
|
182
|
-
assertEx2($meta === void 0 || isJsonObject2($meta), () => "$meta must be JsonObject");
|
|
183
|
-
const result = omitBy2(
|
|
184
|
-
{
|
|
185
|
-
...dataFields,
|
|
186
|
-
$hash: $hash ?? await _PayloadBuilder.dataHash(dataFields),
|
|
187
|
-
schema
|
|
188
|
-
},
|
|
189
|
-
omitByPredicate2("_")
|
|
190
|
-
);
|
|
191
|
-
const clonedMeta = { ...$meta };
|
|
192
|
-
if (timestamp) {
|
|
193
|
-
clonedMeta.timestamp = timestamp;
|
|
194
|
-
}
|
|
195
|
-
if (clonedMeta.timestamp === void 0 && stamp) {
|
|
196
|
-
clonedMeta.timestamp = Date.now();
|
|
197
|
-
}
|
|
198
|
-
if (Object.keys(clonedMeta).length > 0) {
|
|
199
|
-
result.$meta = clonedMeta;
|
|
200
|
-
}
|
|
201
|
-
return result;
|
|
119
|
+
static hashableFields(payload) {
|
|
120
|
+
return this.omitStorageMeta(payload);
|
|
202
121
|
}
|
|
203
122
|
static async hashes(payloads) {
|
|
204
|
-
return await
|
|
123
|
+
return await ObjectHasher.hashes(payloads);
|
|
205
124
|
}
|
|
206
|
-
static async toAllHashMap(
|
|
125
|
+
static async toAllHashMap(payloads) {
|
|
207
126
|
const result = {};
|
|
208
|
-
for (const pair of await this.hashPairs(
|
|
127
|
+
for (const pair of await this.hashPairs(payloads)) {
|
|
128
|
+
const dataHash = await this.dataHash(pair[0]);
|
|
209
129
|
result[pair[1]] = pair[0];
|
|
210
|
-
result[
|
|
130
|
+
result[dataHash] = pair[0];
|
|
211
131
|
}
|
|
212
132
|
return result;
|
|
213
133
|
}
|
|
@@ -230,30 +150,11 @@ var PayloadBuilder = class _PayloadBuilder extends PayloadBuilderBase {
|
|
|
230
150
|
}
|
|
231
151
|
return result;
|
|
232
152
|
}
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
const { $meta, ...result } = payloads;
|
|
239
|
-
return result;
|
|
240
|
-
}
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
async build(options) {
|
|
244
|
-
const dataHashableFields = await this.dataHashableFields();
|
|
245
|
-
return await _PayloadBuilder.build({
|
|
246
|
-
...dataHashableFields,
|
|
247
|
-
$meta: this._$meta,
|
|
248
|
-
schema: this._schema
|
|
249
|
-
}, options);
|
|
250
|
-
}
|
|
251
|
-
async hashableFields() {
|
|
252
|
-
return await _PayloadBuilder.hashableFields(
|
|
253
|
-
assertEx2(this._schema, () => "Payload: Missing Schema"),
|
|
254
|
-
this._fields,
|
|
255
|
-
this._$meta
|
|
256
|
-
);
|
|
153
|
+
build() {
|
|
154
|
+
return {
|
|
155
|
+
schema: this._schema,
|
|
156
|
+
...this._fields
|
|
157
|
+
};
|
|
257
158
|
}
|
|
258
159
|
};
|
|
259
160
|
export {
|
package/dist/node/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/Builder.ts","../../src/BuilderBase.ts"],"sourcesContent":["import { assertEx } from '@xylabs/assert'\nimport type { Hash } from '@xylabs/hex'\nimport type {\n AnyObject, JsonArray, JsonObject,\n} from '@xylabs/object'\nimport { isJsonObject, omitBy } from '@xylabs/object'\nimport { PayloadHasher } from '@xyo-network/hash'\nimport type {\n Payload, PayloadWithMeta, WithMeta,\n} from '@xyo-network/payload-model'\n\nimport type { WithoutMeta, WithoutSchema } from './BuilderBase.ts'\nimport { PayloadBuilderBase, removeMetaAndSchema } from './BuilderBase.ts'\nimport type { PayloadBuilderOptions } from './Options.ts'\n\nexport interface BuildOptions {\n stamp?: boolean\n validate?: boolean\n}\n\nconst omitByPredicate = <T extends object>(prefix: string) => (_: T[keyof T], key: keyof T) => {\n assertEx(typeof key === 'string', () => `Invalid key type [${String(key)}, ${typeof key}]`)\n return (key as string).startsWith(prefix)\n}\n\nexport class PayloadBuilder<\n T extends Payload = Payload<AnyObject>,\n O extends PayloadBuilderOptions<T> = PayloadBuilderOptions<T>,\n> extends PayloadBuilderBase<T, O> {\n static async build<T extends Payload = Payload<AnyObject>>(payload: T, options?: BuildOptions): Promise<WithMeta<T>>\n static async build<T extends Payload = Payload<AnyObject>>(payload: T[], options?: BuildOptions): Promise<WithMeta<T>[]>\n static async build<T extends Payload = Payload<AnyObject>>(payload: T | T[], options: BuildOptions = {}) {\n if (Array.isArray(payload)) {\n return await Promise.all(payload.map(payload => this.build(payload, options)))\n } else {\n const { stamp = false, validate = true } = options\n const {\n schema, $hash: incomingDataHash, $meta: incomingMeta = {},\n } = payload as WithMeta<T>\n\n // check for legacy signatures\n const { _signatures } = payload as { _signatures?: JsonArray }\n if (_signatures && !incomingMeta.signatures) {\n incomingMeta.signatures = _signatures\n }\n\n const fields = removeMetaAndSchema(payload)\n const dataHashableFields = await PayloadBuilder.dataHashableFields(schema, fields)\n const $hash = validate || incomingDataHash === undefined ? await PayloadHasher.hash(dataHashableFields) : incomingDataHash\n const $meta: JsonObject = { ...incomingMeta }\n if ($meta.timestamp === undefined && stamp) {\n $meta.timestamp = Date.now()\n }\n const hashableFields: WithMeta<Payload> = {\n ...dataHashableFields, $hash, schema,\n }\n\n if (Object.keys($meta).length > 0) {\n hashableFields.$meta = $meta\n }\n\n return hashableFields as WithMeta<T>\n }\n }\n\n static async dataHash<T extends Payload>(payload: T, options?: BuildOptions): Promise<Hash> {\n return (await this.build(payload, options)).$hash\n }\n\n static async dataHashPairs<T extends Payload>(payloads: T[], options?: BuildOptions): Promise<[WithMeta<T>, Hash][]> {\n return await Promise.all(\n payloads.map(async (payload) => {\n const built = await PayloadBuilder.build(payload, options)\n return [built, built.$hash]\n }),\n )\n }\n\n static async dataHashes(payloads: undefined, options?: BuildOptions): Promise<undefined>\n static async dataHashes<T extends Payload>(payloads: T[], options?: BuildOptions): Promise<Hash[]>\n static async dataHashes<T extends Payload>(payloads?: T[], options?: BuildOptions): Promise<Hash[] | undefined> {\n return payloads\n ? await Promise.all(\n payloads.map(async (payload) => {\n const built = await PayloadBuilder.build(payload, options)\n return built.$hash\n }),\n )\n : undefined\n }\n\n static async filterExclude<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {\n return await PayloadHasher.filterExcludeByHash(await this.filterExcludeByDataHash(payloads, hash), hash)\n }\n\n static async filterExcludeByDataHash<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {\n const hashes = Array.isArray(hash) ? hash : [hash]\n return (await this.dataHashPairs(payloads)).filter(([_, objHash]) => !hashes.includes(objHash))?.map(pair => pair[0])\n }\n\n static async filterIncludeByDataHash<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {\n const hashes = Array.isArray(hash) ? hash : [hash]\n return (await this.dataHashPairs(payloads)).filter(([_, objHash]) => hashes.includes(objHash))?.map(pair => pair[0])\n }\n\n static async findByDataHash<T extends Payload>(payloads: T[] = [], hash: Hash): Promise<T | undefined> {\n return (await this.dataHashPairs(payloads)).find(([_, objHash]) => objHash === hash)?.[0]\n }\n\n static async hash<T extends Payload>(payload: T, options?: BuildOptions): Promise<Hash> {\n return await PayloadHasher.hash(await PayloadBuilder.build(payload, options))\n }\n\n /**\n * Creates an array of payload/hash tuples based on the payloads passed in\n * @param objs Any array of payloads\n * @returns An array of payload/hash tuples\n */\n static async hashPairs<T extends Payload>(payloads: T[], options?: BuildOptions): Promise<[WithMeta<T>, Hash][]> {\n return await Promise.all(\n payloads.map<Promise<[WithMeta<T>, Hash]>>(async (payload) => {\n const built = await PayloadBuilder.build(payload, options)\n return [built, await PayloadBuilder.hash(built)]\n }),\n )\n }\n\n static async hashableFields<T extends Payload = Payload<AnyObject>>(\n schema: string,\n fields?: WithoutSchema<WithoutMeta<T>>,\n $meta?: JsonObject,\n $hash?: Hash,\n timestamp?: number,\n stamp = false,\n ): Promise<WithMeta<T>> {\n const dataFields = await this.dataHashableFields<T>(schema, fields)\n assertEx($meta === undefined || isJsonObject($meta), () => '$meta must be JsonObject')\n const result: WithMeta<T> = omitBy(\n {\n ...dataFields,\n $hash: $hash ?? (await PayloadBuilder.dataHash(dataFields)),\n schema,\n } as WithMeta<T>,\n omitByPredicate('_'),\n ) as WithMeta<T>\n\n const clonedMeta = { ...$meta }\n\n if (timestamp) {\n clonedMeta.timestamp = timestamp\n }\n\n if (clonedMeta.timestamp === undefined && stamp) {\n clonedMeta.timestamp = Date.now()\n }\n\n if (Object.keys(clonedMeta).length > 0) {\n result.$meta = clonedMeta\n }\n\n return result\n }\n\n static async hashes(payloads: undefined): Promise<undefined>\n static async hashes<T extends Payload>(payloads: T[]): Promise<Hash[]>\n static async hashes<T extends Payload>(payloads?: T[]): Promise<Hash[] | undefined> {\n return await PayloadHasher.hashes(payloads)\n }\n\n static async toAllHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, WithMeta<T>>> {\n const result: Record<Hash, WithMeta<T>> = {}\n for (const pair of await this.hashPairs(objs)) {\n result[pair[1]] = pair[0]\n result[pair[0].$hash] = pair[0]\n }\n return result\n }\n\n static async toDataHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, WithMeta<T>>> {\n const result: Record<Hash, WithMeta<T>> = {}\n for (const pair of await this.dataHashPairs(objs)) {\n result[pair[1]] = pair[0]\n }\n return result\n }\n\n /**\n * Creates an object map of payload hashes to payloads based on the payloads passed in\n * @param objs Any array of payloads\n * @returns A map of hashes to payloads\n */\n static async toHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, WithMeta<T>>> {\n const result: Record<Hash, WithMeta<T>> = {}\n for (const pair of await this.hashPairs(objs)) {\n result[pair[1]] = pair[0]\n }\n return result\n }\n\n static withoutMeta(payload: undefined): undefined\n static withoutMeta<T extends PayloadWithMeta>(payload: T): Omit<T, '$meta'>\n static withoutMeta<T extends PayloadWithMeta>(payloads: T[]): Omit<T, '$meta'>[]\n static withoutMeta<T extends PayloadWithMeta>(payloads: T | T[]) {\n if (Array.isArray(payloads)) {\n return payloads.map(payload => this.withoutMeta(payload))\n } else {\n if (payloads) {\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n const { $meta, ...result } = payloads\n return result as Omit<T, '$meta'>\n }\n }\n }\n\n async build(options?: BuildOptions): Promise<WithMeta<T>> {\n const dataHashableFields = await this.dataHashableFields()\n return await PayloadBuilder.build<T>({\n ...dataHashableFields, $meta: this._$meta, schema: this._schema,\n } as Payload as T, options)\n }\n\n async hashableFields() {\n return await PayloadBuilder.hashableFields(\n assertEx(this._schema, () => 'Payload: Missing Schema'),\n this._fields,\n this._$meta,\n )\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport type { Hash } from '@xylabs/hex'\nimport type { AnyObject, JsonObject } from '@xylabs/object'\nimport {\n isJsonObject, omitBy, toJson,\n} from '@xylabs/object'\nimport type { Promisable } from '@xylabs/promise'\nimport { removeEmptyFields } from '@xyo-network/hash'\nimport type {\n Payload, Schema, WithMeta, WithOptionalMeta,\n} from '@xyo-network/payload-model'\n\nimport type { PayloadBuilderOptions } from './Options.ts'\n\nexport type WithOptionalSchema<T extends Payload> = Omit<T, 'schema'> & Partial<T>\n\nexport type WithoutSchema<T extends WithOptionalSchema<Payload>> = Omit<T, 'schema'>\n\nexport type WithoutMeta<T extends WithOptionalMeta<Payload>> = Omit<T, '$hash' | '$meta'>\n\nexport const removeMetaAndSchema = <T extends Payload>(payload: WithOptionalSchema<WithOptionalMeta<T>>): WithoutSchema<WithoutMeta<T>> => {\n const { ...result } = payload\n delete result.$hash\n delete result.$meta\n delete result.schema\n return result as Omit<T, 'schema'>\n}\n\nconst omitByPredicate = (prefix: string) => (_: unknown, key: string) => {\n assertEx(typeof key === 'string', () => `Invalid key type [${key}, ${typeof key}]`)\n return key.startsWith(prefix)\n}\n\nexport class PayloadBuilderBase<T extends Payload = Payload<AnyObject>, O extends PayloadBuilderOptions<T> = PayloadBuilderOptions<T>> {\n protected _$meta?: JsonObject\n protected _fields?: WithoutSchema<WithoutMeta<T>>\n protected _schema: Schema\n\n constructor(readonly options: O) {\n const {\n schema, fields, meta,\n } = options\n this._schema = schema\n this._fields = removeEmptyFields(fields ?? {}) as WithoutSchema<WithoutMeta<T>>\n this._$meta = meta\n }\n\n static dataHashableFields<T extends Payload = Payload<AnyObject>>(\n schema: string,\n fields?: WithoutSchema<WithoutMeta<T>>,\n ): Promisable<Omit<T, '$hash' | '$meta'>> {\n const cleanFields = fields ? removeEmptyFields(fields) : undefined\n assertEx(\n cleanFields === undefined || isJsonObject(cleanFields),\n () => `Fields must be JsonObject: ${JSON.stringify(toJson(cleanFields), null, 2)}`,\n )\n return omitBy(omitBy({ schema, ...cleanFields }, omitByPredicate('$')), omitByPredicate('_')) as unknown as T\n }\n\n protected static metaFields(dataHash: Hash, otherMeta?: JsonObject, stamp = true): Promisable<JsonObject> {\n const meta: JsonObject = { ...otherMeta }\n\n if (!meta.timestamp && stamp) {\n meta.timestamp = meta.timestamp ?? Date.now()\n }\n\n return meta\n }\n\n $meta(meta?: JsonObject) {\n this._$meta = meta ?? (this._fields as WithMeta<T>).$meta\n return this\n }\n\n async dataHashableFields() {\n return await PayloadBuilderBase.dataHashableFields(\n assertEx(this._schema, () => 'Payload: Missing Schema'),\n this._fields,\n )\n }\n\n // we do not require sending in $hash since it will be generated anyway\n fields(fields: WithOptionalSchema<WithOptionalMeta<T>>) {\n if (fields) {\n const {\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n $meta, $hash, schema, ...fieldsOnly\n } = fields\n if ($meta) {\n this.$meta($meta)\n }\n if (schema) {\n this.schema(schema)\n }\n this._fields = removeMetaAndSchema<T>(fields)\n }\n return this\n }\n\n schema(value: Schema) {\n this._schema = value\n }\n\n protected async metaFields(dataHash: Hash, stamp = true): Promise<JsonObject> {\n return await PayloadBuilderBase.metaFields(dataHash, this._$meta, stamp)\n }\n}\n"],"mappings":"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","names":["assertEx","isJsonObject","omitBy","omitByPredicate","assertEx","payload","isJsonObject","omitBy"]}
|
|
1
|
+
{"version":3,"sources":["../../src/Builder.ts","../../src/BuilderBase.ts"],"sourcesContent":["import type { Hash } from '@xylabs/hex'\nimport type { AnyObject } from '@xylabs/object'\nimport { ObjectHasher } from '@xyo-network/hash'\nimport type { Payload } from '@xyo-network/payload-model'\n\nimport { PayloadBuilderBase } from './BuilderBase.ts'\nimport type { PayloadBuilderOptions } from './Options.ts'\n\nexport class PayloadBuilder<\n T extends Payload = Payload<AnyObject>,\n O extends PayloadBuilderOptions<T> = PayloadBuilderOptions<T>,\n> extends PayloadBuilderBase<T, O> {\n static async dataHash<T extends Payload>(payload: T): Promise<Hash> {\n return await ObjectHasher.hash(this.omitMeta(payload))\n }\n\n static async dataHashPairs<T extends Payload>(payloads: T[]): Promise<[T, Hash][]> {\n return await Promise.all(\n payloads.map(async (payload) => {\n const dataHash = await this.dataHash(payload)\n return [payload, dataHash]\n }),\n )\n }\n\n static async dataHashes(payloads: undefined): Promise<undefined>\n static async dataHashes<T extends Payload>(payloads: T[]): Promise<Hash[]>\n static async dataHashes<T extends Payload>(payloads?: T[]): Promise<Hash[] | undefined> {\n return payloads\n ? await Promise.all(\n payloads.map(async (payload) => {\n return await PayloadBuilder.dataHash(payload)\n }),\n )\n : undefined\n }\n\n static async filterExclude<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {\n return await ObjectHasher.filterExcludeByHash(await this.filterExcludeByDataHash(payloads, hash), hash)\n }\n\n static async filterExcludeByDataHash<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {\n const hashes = Array.isArray(hash) ? hash : [hash]\n return (await this.dataHashPairs(payloads)).filter(([_, objHash]) => !hashes.includes(objHash))?.map(pair => pair[0])\n }\n\n static async filterIncludeByDataHash<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {\n const hashes = Array.isArray(hash) ? hash : [hash]\n return (await this.dataHashPairs(payloads)).filter(([_, objHash]) => hashes.includes(objHash))?.map(pair => pair[0])\n }\n\n static async findByDataHash<T extends Payload>(payloads: T[] = [], hash: Hash): Promise<T | undefined> {\n return (await this.dataHashPairs(payloads)).find(([_, objHash]) => objHash === hash)?.[0]\n }\n\n static async hash<T extends Payload>(payload: T): Promise<Hash> {\n return await ObjectHasher.hash(this.omitStorageMeta(payload))\n }\n\n /**\n * Creates an array of payload/hash tuples based on the payloads passed in\n * @param objs Any array of payloads\n * @returns An array of payload/hash tuples\n */\n static async hashPairs<T extends Payload>(payloads: T[]): Promise<[T, Hash][]> {\n return await Promise.all(\n payloads.map<Promise<[T, Hash]>>(async (payload) => {\n return [payload, await PayloadBuilder.hash(payload)]\n }),\n )\n }\n\n static hashableFields<T extends Payload>(\n payload: T,\n ): T {\n return this.omitStorageMeta(payload)\n }\n\n static async hashes(payloads: undefined): Promise<undefined>\n static async hashes<T extends Payload>(payloads: T[]): Promise<Hash[]>\n static async hashes<T extends Payload>(payloads?: T[]): Promise<Hash[] | undefined> {\n return await ObjectHasher.hashes(payloads)\n }\n\n static async toAllHashMap<T extends Payload>(payloads: T[]): Promise<Record<Hash, T>> {\n const result: Record<Hash, T> = {}\n for (const pair of await this.hashPairs(payloads)) {\n const dataHash = await this.dataHash(pair[0])\n result[pair[1]] = pair[0]\n result[dataHash] = pair[0]\n }\n return result\n }\n\n static async toDataHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, T>> {\n const result: Record<Hash, T> = {}\n for (const pair of await this.dataHashPairs(objs)) {\n result[pair[1]] = pair[0]\n }\n return result\n }\n\n /**\n * Creates an object map of payload hashes to payloads based on the payloads passed in\n * @param objs Any array of payloads\n * @returns A map of hashes to payloads\n */\n static async toHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, T>> {\n const result: Record<Hash, T> = {}\n for (const pair of await this.hashPairs(objs)) {\n result[pair[1]] = pair[0]\n }\n return result\n }\n\n build(): T {\n return {\n schema: this._schema,\n ...this._fields,\n } as T\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport type { AnyObject } from '@xylabs/object'\nimport {\n isJsonObject, omitBy, toJson,\n} from '@xylabs/object'\nimport type { Promisable } from '@xylabs/promise'\nimport { removeEmptyFields } from '@xyo-network/hash'\nimport type { Payload, Schema } from '@xyo-network/payload-model'\n\nimport { PayloadBuilder } from './Builder.ts'\nimport type { PayloadBuilderOptions } from './Options.ts'\n\nexport type WithOptionalSchema<T extends Payload> = Omit<T, 'schema'> & Partial<T>\n\nexport type WithoutSchema<T extends WithOptionalSchema<Payload>> = Omit<T, 'schema'>\n\nexport const removeMetaAndSchema = <T extends Payload>(payload: Partial<WithOptionalSchema<T>>): WithoutSchema<T> => {\n const { ...result } = PayloadBuilder.omitMeta(payload as T) as WithOptionalSchema<T>\n delete result.schema\n return result as Omit<T, 'schema'>\n}\n\nconst omitByPrefixPredicate = (prefix: string) => (_: unknown, key: string) => {\n assertEx(typeof key === 'string', () => `Invalid key type [${key}, ${typeof key}]`)\n return key.startsWith(prefix)\n}\n\nexport class PayloadBuilderBase<T extends Payload = Payload<AnyObject>, O extends PayloadBuilderOptions<T> = PayloadBuilderOptions<T>> {\n protected _fields?: Partial<WithoutSchema<T>>\n protected _schema: Schema\n\n constructor(readonly options: O) {\n const { schema, fields } = options\n this._schema = schema\n this._fields = removeMetaAndSchema(removeEmptyFields(structuredClone(fields ?? {})))\n }\n\n static dataHashableFields<T extends Payload>(\n schema: Schema,\n payload: WithoutSchema<T>,\n\n ): Promisable<Payload> {\n const cleanFields = removeEmptyFields({ ...payload, schema })\n assertEx(\n cleanFields === undefined || isJsonObject(cleanFields),\n () => `Fields must be JsonObject: ${JSON.stringify(toJson(cleanFields), null, 2)}`,\n )\n return this.omitMeta(cleanFields) as T\n }\n\n static omitClientMeta<T extends Payload>(payload: T, maxDepth = 100): T {\n return omitBy(payload, omitByPrefixPredicate('$'), maxDepth) as T\n }\n\n static omitMeta<T extends Payload>(payload: T, maxDepth = 100): T {\n return this.omitStorageMeta(this.omitClientMeta(payload, maxDepth), maxDepth)\n }\n\n static omitStorageMeta<T extends Payload>(payload: T, maxDepth = 100): T {\n return omitBy(payload, omitByPrefixPredicate('_'), maxDepth) as T\n }\n\n async dataHashableFields() {\n return await PayloadBuilderBase.dataHashableFields(\n assertEx(this._schema, () => 'Payload: Missing Schema'),\n // TDOD: Add verification that required fields are present\n this._fields as T,\n )\n }\n\n fields(fields: WithOptionalSchema<T>) {\n if (fields) {\n const fieldsClone = structuredClone(fields)\n const { schema } = fieldsClone\n if (schema) {\n this.schema(schema)\n }\n this._fields = removeEmptyFields(removeMetaAndSchema<T>(fieldsClone))\n }\n return this\n }\n\n schema(value: Schema) {\n this._schema = value\n }\n}\n"],"mappings":"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","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@xyo-network/payload-builder",
|
|
3
|
-
"version": "3.
|
|
3
|
+
"version": "3.6.0-rc.1",
|
|
4
4
|
"description": "Primary SDK for using XYO Protocol 2.0",
|
|
5
5
|
"homepage": "https://xyo.network",
|
|
6
6
|
"bugs": {
|
|
@@ -29,22 +29,23 @@
|
|
|
29
29
|
"module": "dist/neutral/index.mjs",
|
|
30
30
|
"types": "dist/neutral/index.d.ts",
|
|
31
31
|
"dependencies": {
|
|
32
|
-
"@xylabs/assert": "^4.4.
|
|
33
|
-
"@xylabs/hex": "^4.4.
|
|
34
|
-
"@xylabs/logger": "^4.4.
|
|
35
|
-
"@xylabs/object": "^4.4.
|
|
36
|
-
"@xylabs/promise": "^4.4.
|
|
37
|
-
"@xyo-network/hash": "^3.
|
|
38
|
-
"@xyo-network/payload-model": "^3.
|
|
32
|
+
"@xylabs/assert": "^4.4.12",
|
|
33
|
+
"@xylabs/hex": "^4.4.12",
|
|
34
|
+
"@xylabs/logger": "^4.4.12",
|
|
35
|
+
"@xylabs/object": "^4.4.12",
|
|
36
|
+
"@xylabs/promise": "^4.4.12",
|
|
37
|
+
"@xyo-network/hash": "^3.6.0-rc.1",
|
|
38
|
+
"@xyo-network/payload-model": "^3.6.0-rc.1"
|
|
39
39
|
},
|
|
40
40
|
"devDependencies": {
|
|
41
41
|
"@xylabs/ts-scripts-yarn3": "^4.2.4",
|
|
42
42
|
"@xylabs/tsconfig": "^4.2.4",
|
|
43
|
-
"@xylabs/vitest-extended": "^4.4.
|
|
43
|
+
"@xylabs/vitest-extended": "^4.4.12",
|
|
44
44
|
"typescript": "^5.7.2",
|
|
45
|
-
"vitest": "^2.1.
|
|
45
|
+
"vitest": "^2.1.8"
|
|
46
46
|
},
|
|
47
47
|
"publishConfig": {
|
|
48
48
|
"access": "public"
|
|
49
|
-
}
|
|
49
|
+
},
|
|
50
|
+
"stableVersion": "3.5.2"
|
|
50
51
|
}
|
package/src/Builder.ts
CHANGED
|
@@ -1,96 +1,42 @@
|
|
|
1
|
-
import { assertEx } from '@xylabs/assert'
|
|
2
1
|
import type { Hash } from '@xylabs/hex'
|
|
3
|
-
import type {
|
|
4
|
-
|
|
5
|
-
} from '@
|
|
6
|
-
import { isJsonObject, omitBy } from '@xylabs/object'
|
|
7
|
-
import { PayloadHasher } from '@xyo-network/hash'
|
|
8
|
-
import type {
|
|
9
|
-
Payload, PayloadWithMeta, WithMeta,
|
|
10
|
-
} from '@xyo-network/payload-model'
|
|
11
|
-
|
|
12
|
-
import type { WithoutMeta, WithoutSchema } from './BuilderBase.ts'
|
|
13
|
-
import { PayloadBuilderBase, removeMetaAndSchema } from './BuilderBase.ts'
|
|
14
|
-
import type { PayloadBuilderOptions } from './Options.ts'
|
|
15
|
-
|
|
16
|
-
export interface BuildOptions {
|
|
17
|
-
stamp?: boolean
|
|
18
|
-
validate?: boolean
|
|
19
|
-
}
|
|
2
|
+
import type { AnyObject } from '@xylabs/object'
|
|
3
|
+
import { ObjectHasher } from '@xyo-network/hash'
|
|
4
|
+
import type { Payload } from '@xyo-network/payload-model'
|
|
20
5
|
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
return (key as string).startsWith(prefix)
|
|
24
|
-
}
|
|
6
|
+
import { PayloadBuilderBase } from './BuilderBase.ts'
|
|
7
|
+
import type { PayloadBuilderOptions } from './Options.ts'
|
|
25
8
|
|
|
26
9
|
export class PayloadBuilder<
|
|
27
10
|
T extends Payload = Payload<AnyObject>,
|
|
28
11
|
O extends PayloadBuilderOptions<T> = PayloadBuilderOptions<T>,
|
|
29
12
|
> extends PayloadBuilderBase<T, O> {
|
|
30
|
-
static async
|
|
31
|
-
|
|
32
|
-
static async build<T extends Payload = Payload<AnyObject>>(payload: T | T[], options: BuildOptions = {}) {
|
|
33
|
-
if (Array.isArray(payload)) {
|
|
34
|
-
return await Promise.all(payload.map(payload => this.build(payload, options)))
|
|
35
|
-
} else {
|
|
36
|
-
const { stamp = false, validate = true } = options
|
|
37
|
-
const {
|
|
38
|
-
schema, $hash: incomingDataHash, $meta: incomingMeta = {},
|
|
39
|
-
} = payload as WithMeta<T>
|
|
40
|
-
|
|
41
|
-
// check for legacy signatures
|
|
42
|
-
const { _signatures } = payload as { _signatures?: JsonArray }
|
|
43
|
-
if (_signatures && !incomingMeta.signatures) {
|
|
44
|
-
incomingMeta.signatures = _signatures
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
const fields = removeMetaAndSchema(payload)
|
|
48
|
-
const dataHashableFields = await PayloadBuilder.dataHashableFields(schema, fields)
|
|
49
|
-
const $hash = validate || incomingDataHash === undefined ? await PayloadHasher.hash(dataHashableFields) : incomingDataHash
|
|
50
|
-
const $meta: JsonObject = { ...incomingMeta }
|
|
51
|
-
if ($meta.timestamp === undefined && stamp) {
|
|
52
|
-
$meta.timestamp = Date.now()
|
|
53
|
-
}
|
|
54
|
-
const hashableFields: WithMeta<Payload> = {
|
|
55
|
-
...dataHashableFields, $hash, schema,
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
if (Object.keys($meta).length > 0) {
|
|
59
|
-
hashableFields.$meta = $meta
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
return hashableFields as WithMeta<T>
|
|
63
|
-
}
|
|
13
|
+
static async dataHash<T extends Payload>(payload: T): Promise<Hash> {
|
|
14
|
+
return await ObjectHasher.hash(this.omitMeta(payload))
|
|
64
15
|
}
|
|
65
16
|
|
|
66
|
-
static async
|
|
67
|
-
return (await this.build(payload, options)).$hash
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
static async dataHashPairs<T extends Payload>(payloads: T[], options?: BuildOptions): Promise<[WithMeta<T>, Hash][]> {
|
|
17
|
+
static async dataHashPairs<T extends Payload>(payloads: T[]): Promise<[T, Hash][]> {
|
|
71
18
|
return await Promise.all(
|
|
72
19
|
payloads.map(async (payload) => {
|
|
73
|
-
const
|
|
74
|
-
return [
|
|
20
|
+
const dataHash = await this.dataHash(payload)
|
|
21
|
+
return [payload, dataHash]
|
|
75
22
|
}),
|
|
76
23
|
)
|
|
77
24
|
}
|
|
78
25
|
|
|
79
|
-
static async dataHashes(payloads: undefined
|
|
80
|
-
static async dataHashes<T extends Payload>(payloads: T[]
|
|
81
|
-
static async dataHashes<T extends Payload>(payloads?: T[]
|
|
26
|
+
static async dataHashes(payloads: undefined): Promise<undefined>
|
|
27
|
+
static async dataHashes<T extends Payload>(payloads: T[]): Promise<Hash[]>
|
|
28
|
+
static async dataHashes<T extends Payload>(payloads?: T[]): Promise<Hash[] | undefined> {
|
|
82
29
|
return payloads
|
|
83
30
|
? await Promise.all(
|
|
84
31
|
payloads.map(async (payload) => {
|
|
85
|
-
|
|
86
|
-
return built.$hash
|
|
32
|
+
return await PayloadBuilder.dataHash(payload)
|
|
87
33
|
}),
|
|
88
34
|
)
|
|
89
35
|
: undefined
|
|
90
36
|
}
|
|
91
37
|
|
|
92
38
|
static async filterExclude<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {
|
|
93
|
-
return await
|
|
39
|
+
return await ObjectHasher.filterExcludeByHash(await this.filterExcludeByDataHash(payloads, hash), hash)
|
|
94
40
|
}
|
|
95
41
|
|
|
96
42
|
static async filterExcludeByDataHash<T extends Payload>(payloads: T[] = [], hash: Hash[] | Hash): Promise<T[]> {
|
|
@@ -107,8 +53,8 @@ export class PayloadBuilder<
|
|
|
107
53
|
return (await this.dataHashPairs(payloads)).find(([_, objHash]) => objHash === hash)?.[0]
|
|
108
54
|
}
|
|
109
55
|
|
|
110
|
-
static async hash<T extends Payload>(payload: T
|
|
111
|
-
return await
|
|
56
|
+
static async hash<T extends Payload>(payload: T): Promise<Hash> {
|
|
57
|
+
return await ObjectHasher.hash(this.omitStorageMeta(payload))
|
|
112
58
|
}
|
|
113
59
|
|
|
114
60
|
/**
|
|
@@ -116,68 +62,38 @@ export class PayloadBuilder<
|
|
|
116
62
|
* @param objs Any array of payloads
|
|
117
63
|
* @returns An array of payload/hash tuples
|
|
118
64
|
*/
|
|
119
|
-
static async hashPairs<T extends Payload>(payloads: T[]
|
|
65
|
+
static async hashPairs<T extends Payload>(payloads: T[]): Promise<[T, Hash][]> {
|
|
120
66
|
return await Promise.all(
|
|
121
|
-
payloads.map<Promise<[
|
|
122
|
-
|
|
123
|
-
return [built, await PayloadBuilder.hash(built)]
|
|
67
|
+
payloads.map<Promise<[T, Hash]>>(async (payload) => {
|
|
68
|
+
return [payload, await PayloadBuilder.hash(payload)]
|
|
124
69
|
}),
|
|
125
70
|
)
|
|
126
71
|
}
|
|
127
72
|
|
|
128
|
-
static
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
$hash?: Hash,
|
|
133
|
-
timestamp?: number,
|
|
134
|
-
stamp = false,
|
|
135
|
-
): Promise<WithMeta<T>> {
|
|
136
|
-
const dataFields = await this.dataHashableFields<T>(schema, fields)
|
|
137
|
-
assertEx($meta === undefined || isJsonObject($meta), () => '$meta must be JsonObject')
|
|
138
|
-
const result: WithMeta<T> = omitBy(
|
|
139
|
-
{
|
|
140
|
-
...dataFields,
|
|
141
|
-
$hash: $hash ?? (await PayloadBuilder.dataHash(dataFields)),
|
|
142
|
-
schema,
|
|
143
|
-
} as WithMeta<T>,
|
|
144
|
-
omitByPredicate('_'),
|
|
145
|
-
) as WithMeta<T>
|
|
146
|
-
|
|
147
|
-
const clonedMeta = { ...$meta }
|
|
148
|
-
|
|
149
|
-
if (timestamp) {
|
|
150
|
-
clonedMeta.timestamp = timestamp
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
if (clonedMeta.timestamp === undefined && stamp) {
|
|
154
|
-
clonedMeta.timestamp = Date.now()
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
if (Object.keys(clonedMeta).length > 0) {
|
|
158
|
-
result.$meta = clonedMeta
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
return result
|
|
73
|
+
static hashableFields<T extends Payload>(
|
|
74
|
+
payload: T,
|
|
75
|
+
): T {
|
|
76
|
+
return this.omitStorageMeta(payload)
|
|
162
77
|
}
|
|
163
78
|
|
|
164
79
|
static async hashes(payloads: undefined): Promise<undefined>
|
|
165
80
|
static async hashes<T extends Payload>(payloads: T[]): Promise<Hash[]>
|
|
166
81
|
static async hashes<T extends Payload>(payloads?: T[]): Promise<Hash[] | undefined> {
|
|
167
|
-
return await
|
|
82
|
+
return await ObjectHasher.hashes(payloads)
|
|
168
83
|
}
|
|
169
84
|
|
|
170
|
-
static async toAllHashMap<T extends Payload>(
|
|
171
|
-
const result: Record<Hash,
|
|
172
|
-
for (const pair of await this.hashPairs(
|
|
85
|
+
static async toAllHashMap<T extends Payload>(payloads: T[]): Promise<Record<Hash, T>> {
|
|
86
|
+
const result: Record<Hash, T> = {}
|
|
87
|
+
for (const pair of await this.hashPairs(payloads)) {
|
|
88
|
+
const dataHash = await this.dataHash(pair[0])
|
|
173
89
|
result[pair[1]] = pair[0]
|
|
174
|
-
result[
|
|
90
|
+
result[dataHash] = pair[0]
|
|
175
91
|
}
|
|
176
92
|
return result
|
|
177
93
|
}
|
|
178
94
|
|
|
179
|
-
static async toDataHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash,
|
|
180
|
-
const result: Record<Hash,
|
|
95
|
+
static async toDataHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, T>> {
|
|
96
|
+
const result: Record<Hash, T> = {}
|
|
181
97
|
for (const pair of await this.dataHashPairs(objs)) {
|
|
182
98
|
result[pair[1]] = pair[0]
|
|
183
99
|
}
|
|
@@ -189,41 +105,18 @@ export class PayloadBuilder<
|
|
|
189
105
|
* @param objs Any array of payloads
|
|
190
106
|
* @returns A map of hashes to payloads
|
|
191
107
|
*/
|
|
192
|
-
static async toHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash,
|
|
193
|
-
const result: Record<Hash,
|
|
108
|
+
static async toHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, T>> {
|
|
109
|
+
const result: Record<Hash, T> = {}
|
|
194
110
|
for (const pair of await this.hashPairs(objs)) {
|
|
195
111
|
result[pair[1]] = pair[0]
|
|
196
112
|
}
|
|
197
113
|
return result
|
|
198
114
|
}
|
|
199
115
|
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
return payloads.map(payload => this.withoutMeta(payload))
|
|
206
|
-
} else {
|
|
207
|
-
if (payloads) {
|
|
208
|
-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
209
|
-
const { $meta, ...result } = payloads
|
|
210
|
-
return result as Omit<T, '$meta'>
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
async build(options?: BuildOptions): Promise<WithMeta<T>> {
|
|
216
|
-
const dataHashableFields = await this.dataHashableFields()
|
|
217
|
-
return await PayloadBuilder.build<T>({
|
|
218
|
-
...dataHashableFields, $meta: this._$meta, schema: this._schema,
|
|
219
|
-
} as Payload as T, options)
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
async hashableFields() {
|
|
223
|
-
return await PayloadBuilder.hashableFields(
|
|
224
|
-
assertEx(this._schema, () => 'Payload: Missing Schema'),
|
|
225
|
-
this._fields,
|
|
226
|
-
this._$meta,
|
|
227
|
-
)
|
|
116
|
+
build(): T {
|
|
117
|
+
return {
|
|
118
|
+
schema: this._schema,
|
|
119
|
+
...this._fields,
|
|
120
|
+
} as T
|
|
228
121
|
}
|
|
229
122
|
}
|