@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.
@@ -1,7 +1,5 @@
1
1
  // src/Builder.ts
2
- import { assertEx as assertEx2 } from "@xylabs/assert";
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 omitByPredicate = (prefix) => (_, key) => {
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, fields) {
41
- const cleanFields = fields ? removeEmptyFields(fields) : void 0;
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 omitBy(omitBy({ schema, ...cleanFields }, omitByPredicate("$")), omitByPredicate("_"));
36
+ return this.omitMeta(cleanFields);
47
37
  }
48
- static metaFields(dataHash, otherMeta, stamp = true) {
49
- const meta = { ...otherMeta };
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
- $meta(meta) {
56
- this._$meta = meta ?? this._fields.$meta;
57
- return this;
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
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
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(fields);
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 build(payload, options = {}) {
100
- if (Array.isArray(payload)) {
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, options) {
75
+ static async dataHashPairs(payloads) {
135
76
  return await Promise.all(
136
77
  payloads.map(async (payload) => {
137
- const built = await _PayloadBuilder.build(payload, options);
138
- return [built, built.$hash];
78
+ const dataHash = await this.dataHash(payload);
79
+ return [payload, dataHash];
139
80
  })
140
81
  );
141
82
  }
142
- static async dataHashes(payloads, options) {
83
+ static async dataHashes(payloads) {
143
84
  return payloads ? await Promise.all(
144
85
  payloads.map(async (payload) => {
145
- const built = await _PayloadBuilder.build(payload, options);
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 PayloadHasher.filterExcludeByHash(await this.filterExcludeByDataHash(payloads, hash), hash);
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, options) {
165
- return await PayloadHasher.hash(await _PayloadBuilder.build(payload, options));
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, options) {
112
+ static async hashPairs(payloads) {
173
113
  return await Promise.all(
174
114
  payloads.map(async (payload) => {
175
- const built = await _PayloadBuilder.build(payload, options);
176
- return [built, await _PayloadBuilder.hash(built)];
115
+ return [payload, await _PayloadBuilder.hash(payload)];
177
116
  })
178
117
  );
179
118
  }
180
- static async hashableFields(schema, fields, $meta, $hash, timestamp, stamp = false) {
181
- const dataFields = await this.dataHashableFields(schema, fields);
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 PayloadHasher.hashes(payloads);
123
+ return await ObjectHasher.hashes(payloads);
205
124
  }
206
- static async toAllHashMap(objs) {
125
+ static async toAllHashMap(payloads) {
207
126
  const result = {};
208
- for (const pair of await this.hashPairs(objs)) {
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[pair[0].$hash] = pair[0];
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
- static withoutMeta(payloads) {
234
- if (Array.isArray(payloads)) {
235
- return payloads.map((payload) => this.withoutMeta(payload));
236
- } else {
237
- if (payloads) {
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 {
@@ -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":";AAAA,SAAS,YAAAA,iBAAgB;AAKzB,SAAS,gBAAAC,eAAc,UAAAC,eAAc;AACrC,SAAS,qBAAqB;;;ACN9B,SAAS,gBAAgB;AAGzB;AAAA,EACE;AAAA,EAAc;AAAA,EAAQ;AAAA,OACjB;AAEP,SAAS,yBAAyB;AAa3B,IAAM,sBAAsB,CAAoB,YAAoF;AACzI,QAAM,EAAE,GAAG,OAAO,IAAI;AACtB,SAAO,OAAO;AACd,SAAO,OAAO;AACd,SAAO,OAAO;AACd,SAAO;AACT;AAEA,IAAM,kBAAkB,CAAC,WAAmB,CAAC,GAAY,QAAgB;AACvE,WAAS,OAAO,QAAQ,UAAU,MAAM,qBAAqB,GAAG,KAAK,OAAO,GAAG,GAAG;AAClF,SAAO,IAAI,WAAW,MAAM;AAC9B;AAEO,IAAM,qBAAN,MAAM,oBAA0H;AAAA,EAKrI,YAAqB,SAAY;AAAZ;AACnB,UAAM;AAAA,MACJ;AAAA,MAAQ;AAAA,MAAQ;AAAA,IAClB,IAAI;AACJ,SAAK,UAAU;AACf,SAAK,UAAU,kBAAkB,UAAU,CAAC,CAAC;AAC7C,SAAK,SAAS;AAAA,EAChB;AAAA,EAXU;AAAA,EACA;AAAA,EACA;AAAA,EAWV,OAAO,mBACL,QACA,QACwC;AACxC,UAAM,cAAc,SAAS,kBAAkB,MAAM,IAAI;AACzD;AAAA,MACE,gBAAgB,UAAa,aAAa,WAAW;AAAA,MACrD,MAAM,8BAA8B,KAAK,UAAU,OAAO,WAAW,GAAG,MAAM,CAAC,CAAC;AAAA,IAClF;AACA,WAAO,OAAO,OAAO,EAAE,QAAQ,GAAG,YAAY,GAAG,gBAAgB,GAAG,CAAC,GAAG,gBAAgB,GAAG,CAAC;AAAA,EAC9F;AAAA,EAEA,OAAiB,WAAW,UAAgB,WAAwB,QAAQ,MAA8B;AACxG,UAAM,OAAmB,EAAE,GAAG,UAAU;AAExC,QAAI,CAAC,KAAK,aAAa,OAAO;AAC5B,WAAK,YAAY,KAAK,aAAa,KAAK,IAAI;AAAA,IAC9C;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,MAAmB;AACvB,SAAK,SAAS,QAAS,KAAK,QAAwB;AACpD,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,qBAAqB;AACzB,WAAO,MAAM,oBAAmB;AAAA,MAC9B,SAAS,KAAK,SAAS,MAAM,yBAAyB;AAAA,MACtD,KAAK;AAAA,IACP;AAAA,EACF;AAAA;AAAA,EAGA,OAAO,QAAiD;AACtD,QAAI,QAAQ;AACV,YAAM;AAAA;AAAA,QAEJ;AAAA,QAAO;AAAA,QAAO;AAAA,QAAQ,GAAG;AAAA,MAC3B,IAAI;AACJ,UAAI,OAAO;AACT,aAAK,MAAM,KAAK;AAAA,MAClB;AACA,UAAI,QAAQ;AACV,aAAK,OAAO,MAAM;AAAA,MACpB;AACA,WAAK,UAAU,oBAAuB,MAAM;AAAA,IAC9C;AACA,WAAO;AAAA,EACT;AAAA,EAEA,OAAO,OAAe;AACpB,SAAK,UAAU;AAAA,EACjB;AAAA,EAEA,MAAgB,WAAW,UAAgB,QAAQ,MAA2B;AAC5E,WAAO,MAAM,oBAAmB,WAAW,UAAU,KAAK,QAAQ,KAAK;AAAA,EACzE;AACF;;;ADtFA,IAAMC,mBAAkB,CAAmB,WAAmB,CAAC,GAAe,QAAiB;AAC7F,EAAAC,UAAS,OAAO,QAAQ,UAAU,MAAM,qBAAqB,OAAO,GAAG,CAAC,KAAK,OAAO,GAAG,GAAG;AAC1F,SAAQ,IAAe,WAAW,MAAM;AAC1C;AAEO,IAAM,iBAAN,MAAM,wBAGH,mBAAyB;AAAA,EAGjC,aAAa,MAA8C,SAAkB,UAAwB,CAAC,GAAG;AACvG,QAAI,MAAM,QAAQ,OAAO,GAAG;AAC1B,aAAO,MAAM,QAAQ,IAAI,QAAQ,IAAI,CAAAC,aAAW,KAAK,MAAMA,UAAS,OAAO,CAAC,CAAC;AAAA,IAC/E,OAAO;AACL,YAAM,EAAE,QAAQ,OAAO,WAAW,KAAK,IAAI;AAC3C,YAAM;AAAA,QACJ;AAAA,QAAQ,OAAO;AAAA,QAAkB,OAAO,eAAe,CAAC;AAAA,MAC1D,IAAI;AAGJ,YAAM,EAAE,YAAY,IAAI;AACxB,UAAI,eAAe,CAAC,aAAa,YAAY;AAC3C,qBAAa,aAAa;AAAA,MAC5B;AAEA,YAAM,SAAS,oBAAoB,OAAO;AAC1C,YAAM,qBAAqB,MAAM,gBAAe,mBAAmB,QAAQ,MAAM;AACjF,YAAM,QAAQ,YAAY,qBAAqB,SAAY,MAAM,cAAc,KAAK,kBAAkB,IAAI;AAC1G,YAAM,QAAoB,EAAE,GAAG,aAAa;AAC5C,UAAI,MAAM,cAAc,UAAa,OAAO;AAC1C,cAAM,YAAY,KAAK,IAAI;AAAA,MAC7B;AACA,YAAM,iBAAoC;AAAA,QACxC,GAAG;AAAA,QAAoB;AAAA,QAAO;AAAA,MAChC;AAEA,UAAI,OAAO,KAAK,KAAK,EAAE,SAAS,GAAG;AACjC,uBAAe,QAAQ;AAAA,MACzB;AAEA,aAAO;AAAA,IACT;AAAA,EACF;AAAA,EAEA,aAAa,SAA4B,SAAY,SAAuC;AAC1F,YAAQ,MAAM,KAAK,MAAM,SAAS,OAAO,GAAG;AAAA,EAC9C;AAAA,EAEA,aAAa,cAAiC,UAAe,SAAwD;AACnH,WAAO,MAAM,QAAQ;AAAA,MACnB,SAAS,IAAI,OAAO,YAAY;AAC9B,cAAM,QAAQ,MAAM,gBAAe,MAAM,SAAS,OAAO;AACzD,eAAO,CAAC,OAAO,MAAM,KAAK;AAAA,MAC5B,CAAC;AAAA,IACH;AAAA,EACF;AAAA,EAIA,aAAa,WAA8B,UAAgB,SAAqD;AAC9G,WAAO,WACH,MAAM,QAAQ;AAAA,MACd,SAAS,IAAI,OAAO,YAAY;AAC9B,cAAM,QAAQ,MAAM,gBAAe,MAAM,SAAS,OAAO;AACzD,eAAO,MAAM;AAAA,MACf,CAAC;AAAA,IACH,IACE;AAAA,EACN;AAAA,EAEA,aAAa,cAAiC,WAAgB,CAAC,GAAG,MAAmC;AACnG,WAAO,MAAM,cAAc,oBAAoB,MAAM,KAAK,wBAAwB,UAAU,IAAI,GAAG,IAAI;AAAA,EACzG;AAAA,EAEA,aAAa,wBAA2C,WAAgB,CAAC,GAAG,MAAmC;AAC7G,UAAM,SAAS,MAAM,QAAQ,IAAI,IAAI,OAAO,CAAC,IAAI;AACjD,YAAQ,MAAM,KAAK,cAAc,QAAQ,GAAG,OAAO,CAAC,CAAC,GAAG,OAAO,MAAM,CAAC,OAAO,SAAS,OAAO,CAAC,GAAG,IAAI,UAAQ,KAAK,CAAC,CAAC;AAAA,EACtH;AAAA,EAEA,aAAa,wBAA2C,WAAgB,CAAC,GAAG,MAAmC;AAC7G,UAAM,SAAS,MAAM,QAAQ,IAAI,IAAI,OAAO,CAAC,IAAI;AACjD,YAAQ,MAAM,KAAK,cAAc,QAAQ,GAAG,OAAO,CAAC,CAAC,GAAG,OAAO,MAAM,OAAO,SAAS,OAAO,CAAC,GAAG,IAAI,UAAQ,KAAK,CAAC,CAAC;AAAA,EACrH;AAAA,EAEA,aAAa,eAAkC,WAAgB,CAAC,GAAG,MAAoC;AACrG,YAAQ,MAAM,KAAK,cAAc,QAAQ,GAAG,KAAK,CAAC,CAAC,GAAG,OAAO,MAAM,YAAY,IAAI,IAAI,CAAC;AAAA,EAC1F;AAAA,EAEA,aAAa,KAAwB,SAAY,SAAuC;AACtF,WAAO,MAAM,cAAc,KAAK,MAAM,gBAAe,MAAM,SAAS,OAAO,CAAC;AAAA,EAC9E;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,aAAa,UAA6B,UAAe,SAAwD;AAC/G,WAAO,MAAM,QAAQ;AAAA,MACnB,SAAS,IAAkC,OAAO,YAAY;AAC5D,cAAM,QAAQ,MAAM,gBAAe,MAAM,SAAS,OAAO;AACzD,eAAO,CAAC,OAAO,MAAM,gBAAe,KAAK,KAAK,CAAC;AAAA,MACjD,CAAC;AAAA,IACH;AAAA,EACF;AAAA,EAEA,aAAa,eACX,QACA,QACA,OACA,OACA,WACA,QAAQ,OACc;AACtB,UAAM,aAAa,MAAM,KAAK,mBAAsB,QAAQ,MAAM;AAClE,IAAAD,UAAS,UAAU,UAAaE,cAAa,KAAK,GAAG,MAAM,0BAA0B;AACrF,UAAM,SAAsBC;AAAA,MAC1B;AAAA,QACE,GAAG;AAAA,QACH,OAAO,SAAU,MAAM,gBAAe,SAAS,UAAU;AAAA,QACzD;AAAA,MACF;AAAA,MACAJ,iBAAgB,GAAG;AAAA,IACrB;AAEA,UAAM,aAAa,EAAE,GAAG,MAAM;AAE9B,QAAI,WAAW;AACb,iBAAW,YAAY;AAAA,IACzB;AAEA,QAAI,WAAW,cAAc,UAAa,OAAO;AAC/C,iBAAW,YAAY,KAAK,IAAI;AAAA,IAClC;AAEA,QAAI,OAAO,KAAK,UAAU,EAAE,SAAS,GAAG;AACtC,aAAO,QAAQ;AAAA,IACjB;AAEA,WAAO;AAAA,EACT;AAAA,EAIA,aAAa,OAA0B,UAA6C;AAClF,WAAO,MAAM,cAAc,OAAO,QAAQ;AAAA,EAC5C;AAAA,EAEA,aAAa,aAAgC,MAA+C;AAC1F,UAAM,SAAoC,CAAC;AAC3C,eAAW,QAAQ,MAAM,KAAK,UAAU,IAAI,GAAG;AAC7C,aAAO,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC;AACxB,aAAO,KAAK,CAAC,EAAE,KAAK,IAAI,KAAK,CAAC;AAAA,IAChC;AACA,WAAO;AAAA,EACT;AAAA,EAEA,aAAa,cAAiC,MAA+C;AAC3F,UAAM,SAAoC,CAAC;AAC3C,eAAW,QAAQ,MAAM,KAAK,cAAc,IAAI,GAAG;AACjD,aAAO,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC;AAAA,IAC1B;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,aAAa,UAA6B,MAA+C;AACvF,UAAM,SAAoC,CAAC;AAC3C,eAAW,QAAQ,MAAM,KAAK,UAAU,IAAI,GAAG;AAC7C,aAAO,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC;AAAA,IAC1B;AACA,WAAO;AAAA,EACT;AAAA,EAKA,OAAO,YAAuC,UAAmB;AAC/D,QAAI,MAAM,QAAQ,QAAQ,GAAG;AAC3B,aAAO,SAAS,IAAI,aAAW,KAAK,YAAY,OAAO,CAAC;AAAA,IAC1D,OAAO;AACL,UAAI,UAAU;AAEZ,cAAM,EAAE,OAAO,GAAG,OAAO,IAAI;AAC7B,eAAO;AAAA,MACT;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAM,MAAM,SAA8C;AACxD,UAAM,qBAAqB,MAAM,KAAK,mBAAmB;AACzD,WAAO,MAAM,gBAAe,MAAS;AAAA,MACnC,GAAG;AAAA,MAAoB,OAAO,KAAK;AAAA,MAAQ,QAAQ,KAAK;AAAA,IAC1D,GAAmB,OAAO;AAAA,EAC5B;AAAA,EAEA,MAAM,iBAAiB;AACrB,WAAO,MAAM,gBAAe;AAAA,MAC1BC,UAAS,KAAK,SAAS,MAAM,yBAAyB;AAAA,MACtD,KAAK;AAAA,MACL,KAAK;AAAA,IACP;AAAA,EACF;AACF;","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":";AAEA,SAAS,oBAAoB;;;ACF7B,SAAS,gBAAgB;AAEzB;AAAA,EACE;AAAA,EAAc;AAAA,EAAQ;AAAA,OACjB;AAEP,SAAS,yBAAyB;AAU3B,IAAM,sBAAsB,CAAoB,YAA8D;AACnH,QAAM,EAAE,GAAG,OAAO,IAAI,eAAe,SAAS,OAAY;AAC1D,SAAO,OAAO;AACd,SAAO;AACT;AAEA,IAAM,wBAAwB,CAAC,WAAmB,CAAC,GAAY,QAAgB;AAC7E,WAAS,OAAO,QAAQ,UAAU,MAAM,qBAAqB,GAAG,KAAK,OAAO,GAAG,GAAG;AAClF,SAAO,IAAI,WAAW,MAAM;AAC9B;AAEO,IAAM,qBAAN,MAAM,oBAA0H;AAAA,EAIrI,YAAqB,SAAY;AAAZ;AACnB,UAAM,EAAE,QAAQ,OAAO,IAAI;AAC3B,SAAK,UAAU;AACf,SAAK,UAAU,oBAAoB,kBAAkB,gBAAgB,UAAU,CAAC,CAAC,CAAC,CAAC;AAAA,EACrF;AAAA,EAPU;AAAA,EACA;AAAA,EAQV,OAAO,mBACL,QACA,SAEqB;AACrB,UAAM,cAAc,kBAAkB,EAAE,GAAG,SAAS,OAAO,CAAC;AAC5D;AAAA,MACE,gBAAgB,UAAa,aAAa,WAAW;AAAA,MACrD,MAAM,8BAA8B,KAAK,UAAU,OAAO,WAAW,GAAG,MAAM,CAAC,CAAC;AAAA,IAClF;AACA,WAAO,KAAK,SAAS,WAAW;AAAA,EAClC;AAAA,EAEA,OAAO,eAAkC,SAAY,WAAW,KAAQ;AACtE,WAAO,OAAO,SAAS,sBAAsB,GAAG,GAAG,QAAQ;AAAA,EAC7D;AAAA,EAEA,OAAO,SAA4B,SAAY,WAAW,KAAQ;AAChE,WAAO,KAAK,gBAAgB,KAAK,eAAe,SAAS,QAAQ,GAAG,QAAQ;AAAA,EAC9E;AAAA,EAEA,OAAO,gBAAmC,SAAY,WAAW,KAAQ;AACvE,WAAO,OAAO,SAAS,sBAAsB,GAAG,GAAG,QAAQ;AAAA,EAC7D;AAAA,EAEA,MAAM,qBAAqB;AACzB,WAAO,MAAM,oBAAmB;AAAA,MAC9B,SAAS,KAAK,SAAS,MAAM,yBAAyB;AAAA;AAAA,MAEtD,KAAK;AAAA,IACP;AAAA,EACF;AAAA,EAEA,OAAO,QAA+B;AACpC,QAAI,QAAQ;AACV,YAAM,cAAc,gBAAgB,MAAM;AAC1C,YAAM,EAAE,OAAO,IAAI;AACnB,UAAI,QAAQ;AACV,aAAK,OAAO,MAAM;AAAA,MACpB;AACA,WAAK,UAAU,kBAAkB,oBAAuB,WAAW,CAAC;AAAA,IACtE;AACA,WAAO;AAAA,EACT;AAAA,EAEA,OAAO,OAAe;AACpB,SAAK,UAAU;AAAA,EACjB;AACF;;;AD7EO,IAAM,iBAAN,MAAM,wBAGH,mBAAyB;AAAA,EACjC,aAAa,SAA4B,SAA2B;AAClE,WAAO,MAAM,aAAa,KAAK,KAAK,SAAS,OAAO,CAAC;AAAA,EACvD;AAAA,EAEA,aAAa,cAAiC,UAAqC;AACjF,WAAO,MAAM,QAAQ;AAAA,MACnB,SAAS,IAAI,OAAO,YAAY;AAC9B,cAAM,WAAW,MAAM,KAAK,SAAS,OAAO;AAC5C,eAAO,CAAC,SAAS,QAAQ;AAAA,MAC3B,CAAC;AAAA,IACH;AAAA,EACF;AAAA,EAIA,aAAa,WAA8B,UAA6C;AACtF,WAAO,WACH,MAAM,QAAQ;AAAA,MACd,SAAS,IAAI,OAAO,YAAY;AAC9B,eAAO,MAAM,gBAAe,SAAS,OAAO;AAAA,MAC9C,CAAC;AAAA,IACH,IACE;AAAA,EACN;AAAA,EAEA,aAAa,cAAiC,WAAgB,CAAC,GAAG,MAAmC;AACnG,WAAO,MAAM,aAAa,oBAAoB,MAAM,KAAK,wBAAwB,UAAU,IAAI,GAAG,IAAI;AAAA,EACxG;AAAA,EAEA,aAAa,wBAA2C,WAAgB,CAAC,GAAG,MAAmC;AAC7G,UAAM,SAAS,MAAM,QAAQ,IAAI,IAAI,OAAO,CAAC,IAAI;AACjD,YAAQ,MAAM,KAAK,cAAc,QAAQ,GAAG,OAAO,CAAC,CAAC,GAAG,OAAO,MAAM,CAAC,OAAO,SAAS,OAAO,CAAC,GAAG,IAAI,UAAQ,KAAK,CAAC,CAAC;AAAA,EACtH;AAAA,EAEA,aAAa,wBAA2C,WAAgB,CAAC,GAAG,MAAmC;AAC7G,UAAM,SAAS,MAAM,QAAQ,IAAI,IAAI,OAAO,CAAC,IAAI;AACjD,YAAQ,MAAM,KAAK,cAAc,QAAQ,GAAG,OAAO,CAAC,CAAC,GAAG,OAAO,MAAM,OAAO,SAAS,OAAO,CAAC,GAAG,IAAI,UAAQ,KAAK,CAAC,CAAC;AAAA,EACrH;AAAA,EAEA,aAAa,eAAkC,WAAgB,CAAC,GAAG,MAAoC;AACrG,YAAQ,MAAM,KAAK,cAAc,QAAQ,GAAG,KAAK,CAAC,CAAC,GAAG,OAAO,MAAM,YAAY,IAAI,IAAI,CAAC;AAAA,EAC1F;AAAA,EAEA,aAAa,KAAwB,SAA2B;AAC9D,WAAO,MAAM,aAAa,KAAK,KAAK,gBAAgB,OAAO,CAAC;AAAA,EAC9D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,aAAa,UAA6B,UAAqC;AAC7E,WAAO,MAAM,QAAQ;AAAA,MACnB,SAAS,IAAwB,OAAO,YAAY;AAClD,eAAO,CAAC,SAAS,MAAM,gBAAe,KAAK,OAAO,CAAC;AAAA,MACrD,CAAC;AAAA,IACH;AAAA,EACF;AAAA,EAEA,OAAO,eACL,SACG;AACH,WAAO,KAAK,gBAAgB,OAAO;AAAA,EACrC;AAAA,EAIA,aAAa,OAA0B,UAA6C;AAClF,WAAO,MAAM,aAAa,OAAO,QAAQ;AAAA,EAC3C;AAAA,EAEA,aAAa,aAAgC,UAAyC;AACpF,UAAM,SAA0B,CAAC;AACjC,eAAW,QAAQ,MAAM,KAAK,UAAU,QAAQ,GAAG;AACjD,YAAM,WAAW,MAAM,KAAK,SAAS,KAAK,CAAC,CAAC;AAC5C,aAAO,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC;AACxB,aAAO,QAAQ,IAAI,KAAK,CAAC;AAAA,IAC3B;AACA,WAAO;AAAA,EACT;AAAA,EAEA,aAAa,cAAiC,MAAqC;AACjF,UAAM,SAA0B,CAAC;AACjC,eAAW,QAAQ,MAAM,KAAK,cAAc,IAAI,GAAG;AACjD,aAAO,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC;AAAA,IAC1B;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,aAAa,UAA6B,MAAqC;AAC7E,UAAM,SAA0B,CAAC;AACjC,eAAW,QAAQ,MAAM,KAAK,UAAU,IAAI,GAAG;AAC7C,aAAO,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC;AAAA,IAC1B;AACA,WAAO;AAAA,EACT;AAAA,EAEA,QAAW;AACT,WAAO;AAAA,MACL,QAAQ,KAAK;AAAA,MACb,GAAG,KAAK;AAAA,IACV;AAAA,EACF;AACF;","names":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@xyo-network/payload-builder",
3
- "version": "3.5.2",
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.9",
33
- "@xylabs/hex": "^4.4.9",
34
- "@xylabs/logger": "^4.4.9",
35
- "@xylabs/object": "^4.4.9",
36
- "@xylabs/promise": "^4.4.9",
37
- "@xyo-network/hash": "^3.5.2",
38
- "@xyo-network/payload-model": "^3.5.2"
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.9",
43
+ "@xylabs/vitest-extended": "^4.4.12",
44
44
  "typescript": "^5.7.2",
45
- "vitest": "^2.1.5"
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
- AnyObject, JsonArray, JsonObject,
5
- } from '@xylabs/object'
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
- const omitByPredicate = <T extends object>(prefix: string) => (_: T[keyof T], key: keyof T) => {
22
- assertEx(typeof key === 'string', () => `Invalid key type [${String(key)}, ${typeof key}]`)
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 build<T extends Payload = Payload<AnyObject>>(payload: T, options?: BuildOptions): Promise<WithMeta<T>>
31
- static async build<T extends Payload = Payload<AnyObject>>(payload: T[], options?: BuildOptions): Promise<WithMeta<T>[]>
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 dataHash<T extends Payload>(payload: T, options?: BuildOptions): Promise<Hash> {
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 built = await PayloadBuilder.build(payload, options)
74
- return [built, built.$hash]
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, options?: BuildOptions): Promise<undefined>
80
- static async dataHashes<T extends Payload>(payloads: T[], options?: BuildOptions): Promise<Hash[]>
81
- static async dataHashes<T extends Payload>(payloads?: T[], options?: BuildOptions): Promise<Hash[] | undefined> {
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
- const built = await PayloadBuilder.build(payload, options)
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 PayloadHasher.filterExcludeByHash(await this.filterExcludeByDataHash(payloads, hash), hash)
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, options?: BuildOptions): Promise<Hash> {
111
- return await PayloadHasher.hash(await PayloadBuilder.build(payload, options))
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[], options?: BuildOptions): Promise<[WithMeta<T>, Hash][]> {
65
+ static async hashPairs<T extends Payload>(payloads: T[]): Promise<[T, Hash][]> {
120
66
  return await Promise.all(
121
- payloads.map<Promise<[WithMeta<T>, Hash]>>(async (payload) => {
122
- const built = await PayloadBuilder.build(payload, options)
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 async hashableFields<T extends Payload = Payload<AnyObject>>(
129
- schema: string,
130
- fields?: WithoutSchema<WithoutMeta<T>>,
131
- $meta?: JsonObject,
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 PayloadHasher.hashes(payloads)
82
+ return await ObjectHasher.hashes(payloads)
168
83
  }
169
84
 
170
- static async toAllHashMap<T extends Payload>(objs: T[]): Promise<Record<Hash, WithMeta<T>>> {
171
- const result: Record<Hash, WithMeta<T>> = {}
172
- for (const pair of await this.hashPairs(objs)) {
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[pair[0].$hash] = pair[0]
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, WithMeta<T>>> {
180
- const result: Record<Hash, WithMeta<T>> = {}
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, WithMeta<T>>> {
193
- const result: Record<Hash, WithMeta<T>> = {}
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
- static withoutMeta(payload: undefined): undefined
201
- static withoutMeta<T extends PayloadWithMeta>(payload: T): Omit<T, '$meta'>
202
- static withoutMeta<T extends PayloadWithMeta>(payloads: T[]): Omit<T, '$meta'>[]
203
- static withoutMeta<T extends PayloadWithMeta>(payloads: T | T[]) {
204
- if (Array.isArray(payloads)) {
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
  }