@xyo-network/payment-plugin 3.1.1 → 3.2.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.
@@ -11,10 +11,29 @@ export declare class PaymentDiscountDiviner<TParams extends PaymentDiscountDivin
11
11
  static defaultConfigSchema: string;
12
12
  protected get couponAuthorities(): Address[];
13
13
  protected divineHandler(payloads?: TIn[]): Promise<TOut[]>;
14
+ /**
15
+ * Filters the supplied list of coupons to only those that are signed by
16
+ * addresses specified in the couponAuthorities
17
+ * @param coupons The list of coupons to filter
18
+ * @returns The filtered list of coupons that are signed by the couponAuthorities
19
+ */
14
20
  protected filterToSigned(coupons: Coupon[]): Promise<Coupon[]>;
15
21
  protected getDiscountsArchivist(): Promise<ArchivistInstance>;
16
22
  protected getDiscountsBoundWitnessDiviner(): Promise<DivinerInstance>;
23
+ /**
24
+ * Finds the appraisals specified by the escrow terms from the supplied payloads
25
+ * @param terms The escrow terms
26
+ * @param payloads The payloads to search for the appraisals
27
+ * @returns The appraisals found in the payloads
28
+ */
17
29
  protected getEscrowAppraisals(terms: EscrowTerms, hashMap: Record<Hash, Payload>): HashLeaseEstimate[];
30
+ /**
31
+ * Finds the discounts specified by the escrow terms from the supplied payloads
32
+ * @param terms The escrow terms
33
+ * @param hashMap The payloads to search for the discounts
34
+ * @returns A tuple containing all the escrow coupons and conditions referenced in those coupons
35
+ * that were found in the either the supplied payloads or the archivist
36
+ */
18
37
  protected getEscrowDiscounts(terms: EscrowTerms, hashMap: Record<Hash, Payload>): Promise<[Coupon[], Condition[]]>;
19
38
  protected isCouponCurrent(coupon: Coupon): boolean;
20
39
  }
@@ -1 +1 @@
1
- {"version":3,"file":"Diviner.d.ts","sourceRoot":"","sources":["../../../src/Discount/Diviner.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAC3C,OAAO,EAAE,iBAAiB,EAAuB,MAAM,8BAA8B,CAAA;AACrF,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAE/D,OAAO,EACL,iBAAiB,EAElB,MAAM,iCAAiC,CAAA;AACxC,OAAO,EACc,eAAe,EAAE,sBAAsB,EAC3D,MAAM,4BAA4B,CAAA;AAGnC,OAAO,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACpD,OAAO,EACL,SAAS,EACT,MAAM,EACN,QAAQ,EACR,WAAW,EAEqD,4BAA4B,EAC7F,MAAM,sCAAsC,CAAA;AAU7C,MAAM,MAAM,+BAA+B,GAAG,WAAW,GAAG,MAAM,GAAG,iBAAiB,GAAG,OAAO,CAAA;AAEhG,qBACa,sBAAsB,CACjC,OAAO,SAAS,4BAA4B,GAAG,4BAA4B,EAC3E,GAAG,SAAS,+BAA+B,GAAG,+BAA+B,EAC7E,IAAI,SAAS,QAAQ,GAAG,QAAQ,EAChC,UAAU,SAAS,sBAAsB,CAAC,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,sBAAsB,CAChH,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EACnC,GAAG,EACH,IAAI,CACL,CACD,SAAQ,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,CAAC;IACvD,OAAgB,aAAa,WAAuC;IACpE,OAAgB,mBAAmB,SAAqC;IAExE,SAAS,KAAK,iBAAiB,IAAI,OAAO,EAAE,CAE3C;cAEe,aAAa,CAAC,QAAQ,GAAE,GAAG,EAAO,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;cA6CpD,cAAc,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;cAqBpD,qBAAqB,IAAI,OAAO,CAAC,iBAAiB,CAAC;cAMnD,+BAA+B,IAAI,OAAO,CAAC,eAAe,CAAC;IAY3E,SAAS,CAAC,mBAAmB,CAAC,KAAK,EAAE,WAAW,EAAE,OAAO,EAAE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,iBAAiB,EAAE;cAatF,kBAAkB,CAAC,KAAK,EAAE,WAAW,EAAE,OAAO,EAAE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;IAiDxH,SAAS,CAAC,eAAe,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;CAInD"}
1
+ {"version":3,"file":"Diviner.d.ts","sourceRoot":"","sources":["../../../src/Discount/Diviner.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAC3C,OAAO,EAAE,iBAAiB,EAAuB,MAAM,8BAA8B,CAAA;AACrF,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAE/D,OAAO,EACL,iBAAiB,EAElB,MAAM,iCAAiC,CAAA;AACxC,OAAO,EACc,eAAe,EAAE,sBAAsB,EAC3D,MAAM,4BAA4B,CAAA;AAGnC,OAAO,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACpD,OAAO,EACL,SAAS,EACT,MAAM,EACN,QAAQ,EACR,WAAW,EACqD,4BAA4B,EAC7F,MAAM,sCAAsC,CAAA;AAU7C,MAAM,MAAM,+BAA+B,GAAG,WAAW,GAAG,MAAM,GAAG,iBAAiB,GAAG,OAAO,CAAA;AAEhG,qBACa,sBAAsB,CACjC,OAAO,SAAS,4BAA4B,GAAG,4BAA4B,EAC3E,GAAG,SAAS,+BAA+B,GAAG,+BAA+B,EAC7E,IAAI,SAAS,QAAQ,GAAG,QAAQ,EAChC,UAAU,SAAS,sBAAsB,CAAC,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,sBAAsB,CAChH,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EACnC,GAAG,EACH,IAAI,CACL,CACD,SAAQ,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,CAAC;IACvD,OAAgB,aAAa,WAAuC;IACpE,OAAgB,mBAAmB,SAAqC;IAExE,SAAS,KAAK,iBAAiB,IAAI,OAAO,EAAE,CAE3C;cAEe,aAAa,CAAC,QAAQ,GAAE,GAAG,EAAO,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;IAuCpE;;;;;OAKG;cACa,cAAc,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;cAqBpD,qBAAqB,IAAI,OAAO,CAAC,iBAAiB,CAAC;cAMnD,+BAA+B,IAAI,OAAO,CAAC,eAAe,CAAC;IAM3E;;;;;OAKG;IACH,SAAS,CAAC,mBAAmB,CAAC,KAAK,EAAE,WAAW,EAAE,OAAO,EAAE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,iBAAiB,EAAE;IAMtG;;;;;;OAMG;cACa,kBAAkB,CAAC,KAAK,EAAE,WAAW,EAAE,OAAO,EAAE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;IAiDxH,SAAS,CAAC,eAAe,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;CAInD"}
@@ -1,6 +1,20 @@
1
1
  import type { Hash } from '@xylabs/hex';
2
2
  import type { Payload } from '@xyo-network/payload-model';
3
3
  import { type Condition, type Coupon } from '@xyo-network/payment-payload-plugins';
4
+ /**
5
+ * Validates the conditions of a coupon against the provided payloads
6
+ * @param coupon The coupon to check
7
+ * @param conditions The conditions associated with the coupon
8
+ * @param payloads The associated payloads (containing the conditions and data to validate the conditions against)
9
+ * @returns True if all conditions are fulfilled, false otherwise
10
+ */
4
11
  export declare const areConditionsFulfilled: (coupon: Coupon, conditions?: Condition[], payloads?: Payload[]) => Promise<boolean>;
12
+ /**
13
+ * Validates the conditions of a coupon against the provided payloads
14
+ * @param coupon The coupon to check
15
+ * @param conditions The conditions associated with the coupon
16
+ * @param payloads The associated payloads (containing the conditions and data to validate the conditions against)
17
+ * @returns The unfulfilled condition hashes
18
+ */
5
19
  export declare const findUnfulfilledConditions: (coupon: Coupon, conditions?: Condition[], payloads?: Payload[]) => Promise<Hash[]>;
6
20
  //# sourceMappingURL=findUnfulfilledConditions.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"findUnfulfilledConditions.d.ts","sourceRoot":"","sources":["../../../../src/Discount/lib/findUnfulfilledConditions.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAEvC,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACzD,OAAO,EACL,KAAK,SAAS,EAAE,KAAK,MAAM,EAC5B,MAAM,sCAAsC,CAAA;AAgB7C,eAAO,MAAM,sBAAsB,WAAkB,MAAM,eAAc,SAAS,EAAE,aAAiB,OAAO,EAAE,KAAQ,OAAO,CAAC,OAAO,CACvD,CAAA;AAS9E,eAAO,MAAM,yBAAyB,WAAkB,MAAM,eAAc,SAAS,EAAE,aAAiB,OAAO,EAAE,KAAQ,OAAO,CAAC,IAAI,EAAE,CAwCtI,CAAA"}
1
+ {"version":3,"file":"findUnfulfilledConditions.d.ts","sourceRoot":"","sources":["../../../../src/Discount/lib/findUnfulfilledConditions.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAEvC,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACzD,OAAO,EACL,KAAK,SAAS,EAAE,KAAK,MAAM,EAC5B,MAAM,sCAAsC,CAAA;AAS7C;;;;;;GAMG;AACH,eAAO,MAAM,sBAAsB,WAAkB,MAAM,eAAc,SAAS,EAAE,aAAiB,OAAO,EAAE,KAAQ,OAAO,CAAC,OAAO,CACvD,CAAA;AAE9E;;;;;;GAMG;AACH,eAAO,MAAM,yBAAyB,WAAkB,MAAM,eAAc,SAAS,EAAE,aAAiB,OAAO,EAAE,KAAQ,OAAO,CAAC,IAAI,EAAE,CAwCtI,CAAA"}
@@ -2,5 +2,9 @@ import type { Hash } from '@xylabs/hex';
2
2
  import type { DivinerInstance } from '@xyo-network/diviner-model';
3
3
  import type { Payload } from '@xyo-network/payload-model';
4
4
  import type { EscrowTerms, Invoice } from '@xyo-network/payment-payload-plugins';
5
+ /**
6
+ * Validates the escrow terms to ensure they are valid for a purchase
7
+ * @returns A payment if the terms are valid for a purchase, undefined otherwise
8
+ */
5
9
  export declare const getInvoiceForEscrow: (terms: EscrowTerms, dataHashMap: Record<Hash, Payload>, paymentTotalDiviner: DivinerInstance) => Promise<Invoice | undefined>;
6
10
  //# sourceMappingURL=getInvoiceForEscrow.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"getInvoiceForEscrow.d.ts","sourceRoot":"","sources":["../../../src/Invoice/getInvoiceForEscrow.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AACvC,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAA;AAEjE,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACzD,OAAO,KAAK,EACA,WAAW,EAAE,OAAO,EAC/B,MAAM,sCAAsC,CAAA;AAS7C,eAAO,MAAM,mBAAmB,UACvB,WAAW,eACL,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,uBACb,eAAe,KACnC,OAAO,CAAC,OAAO,GAAG,SAAS,CAc7B,CAAA"}
1
+ {"version":3,"file":"getInvoiceForEscrow.d.ts","sourceRoot":"","sources":["../../../src/Invoice/getInvoiceForEscrow.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AACvC,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAA;AAEjE,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACzD,OAAO,KAAK,EACA,WAAW,EAAE,OAAO,EAC/B,MAAM,sCAAsC,CAAA;AAK7C;;;GAGG;AACH,eAAO,MAAM,mBAAmB,UACvB,WAAW,eACL,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,uBACb,eAAe,KACnC,OAAO,CAAC,OAAO,GAAG,SAAS,CAc7B,CAAA"}
@@ -3,6 +3,9 @@ import { HashLeaseEstimate } from '@xyo-network/diviner-hash-lease';
3
3
  import { DivinerInstance, DivinerModuleEventData } from '@xyo-network/diviner-model';
4
4
  import { Payload } from '@xyo-network/payload-model';
5
5
  import { EscrowTerms, PaymentSubtotalDivinerParams, Subtotal } from '@xyo-network/payment-payload-plugins';
6
+ /**
7
+ * Escrow terms that contain all the valid fields for calculating a subtotal
8
+ */
6
9
  export type PaymentSubtotalDivinerInputType = EscrowTerms | HashLeaseEstimate | Payload;
7
10
  export declare class PaymentSubtotalDiviner<TParams extends PaymentSubtotalDivinerParams = PaymentSubtotalDivinerParams, TIn extends PaymentSubtotalDivinerInputType = PaymentSubtotalDivinerInputType, TOut extends Subtotal = Subtotal, TEventData extends DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut> = DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut>> extends AbstractDiviner<TParams, TIn, TOut, TEventData> {
8
11
  static configSchemas: string[];
@@ -1 +1 @@
1
- {"version":3,"file":"Diviner.d.ts","sourceRoot":"","sources":["../../../src/Subtotal/Diviner.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAC/D,OAAO,EAAE,iBAAiB,EAAuB,MAAM,iCAAiC,CAAA;AACxF,OAAO,EAAE,eAAe,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAA;AAGpF,OAAO,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACpD,OAAO,EACL,WAAW,EAAqD,4BAA4B,EAAE,QAAQ,EACvG,MAAM,sCAAsC,CAAA;AAW7C,MAAM,MAAM,+BAA+B,GAAG,WAAW,GAAG,iBAAiB,GAAG,OAAO,CAAA;AAEvF,qBACa,sBAAsB,CACjC,OAAO,SAAS,4BAA4B,GAAG,4BAA4B,EAC3E,GAAG,SAAS,+BAA+B,GAAG,+BAA+B,EAC7E,IAAI,SAAS,QAAQ,GAAG,QAAQ,EAChC,UAAU,SAAS,sBAAsB,CAAC,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,sBAAsB,CAChH,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EACnC,GAAG,EACH,IAAI,CACL,CACD,SAAQ,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,CAAC;IACvD,OAAgB,aAAa,WAAuC;IACpE,OAAgB,mBAAmB,SAAqC;cAExD,aAAa,CAAC,QAAQ,GAAE,GAAG,EAAO,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;CAwBrE"}
1
+ {"version":3,"file":"Diviner.d.ts","sourceRoot":"","sources":["../../../src/Subtotal/Diviner.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAC/D,OAAO,EAAE,iBAAiB,EAAuB,MAAM,iCAAiC,CAAA;AACxF,OAAO,EAAE,eAAe,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAA;AAGpF,OAAO,EAAE,OAAO,EAAE,MAAM,4BAA4B,CAAA;AACpD,OAAO,EACL,WAAW,EAAqD,4BAA4B,EAAE,QAAQ,EACvG,MAAM,sCAAsC,CAAA;AAQ7C;;GAEG;AACH,MAAM,MAAM,+BAA+B,GAAG,WAAW,GAAG,iBAAiB,GAAG,OAAO,CAAA;AAEvF,qBACa,sBAAsB,CACjC,OAAO,SAAS,4BAA4B,GAAG,4BAA4B,EAC3E,GAAG,SAAS,+BAA+B,GAAG,+BAA+B,EAC7E,IAAI,SAAS,QAAQ,GAAG,QAAQ,EAChC,UAAU,SAAS,sBAAsB,CAAC,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,sBAAsB,CAChH,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EACnC,GAAG,EACH,IAAI,CACL,CACD,SAAQ,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,CAAC;IACvD,OAAgB,aAAa,WAAuC;IACpE,OAAgB,mBAAmB,SAAqC;cAExD,aAAa,CAAC,QAAQ,GAAE,GAAG,EAAO,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;CAwBrE"}
@@ -1,3 +1,9 @@
1
1
  import type { DurationFields } from '@xyo-network/xns-record-payload-plugins';
2
+ /**
3
+ * Validates that the current time is within the duration window, within a configurable a buffer
4
+ * @param value The duration value
5
+ * @param windowMs The window in milliseconds to allow for a buffer
6
+ * @returns True if the duration is valid, false otherwise
7
+ */
2
8
  export declare const validateDuration: (value: Partial<DurationFields>, windowMs?: number) => boolean;
3
9
  //# sourceMappingURL=durationValidators.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"durationValidators.d.ts","sourceRoot":"","sources":["../../../../src/Subtotal/lib/durationValidators.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,yCAAyC,CAAA;AAU7E,eAAO,MAAM,gBAAgB,UAAW,OAAO,CAAC,cAAc,CAAC,wBAA4B,OAO1F,CAAA"}
1
+ {"version":3,"file":"durationValidators.d.ts","sourceRoot":"","sources":["../../../../src/Subtotal/lib/durationValidators.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,yCAAyC,CAAA;AAI7E;;;;;GAKG;AACH,eAAO,MAAM,gBAAgB,UAAW,OAAO,CAAC,cAAc,CAAC,wBAA4B,OAO1F,CAAA"}
@@ -1 +1 @@
1
- {"version":3,"file":"Diviner.d.ts","sourceRoot":"","sources":["../../../src/Total/Diviner.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAC/D,OAAO,EACc,eAAe,EAAE,sBAAsB,EAC3D,MAAM,4BAA4B,CAAA;AAEnC,OAAO,EACL,QAAQ,EAGR,+BAA+B,EAAE,yBAAyB,EAAE,QAAQ,EAAE,KAAK,EAC5E,MAAM,sCAAsC,CAAA;AAE7C,OAAO,EAAE,sBAAsB,EAAE,+BAA+B,EAAE,MAAM,sBAAsB,CAAA;AAC9F,OAAO,EAAE,sBAAsB,EAAE,+BAA+B,EAAE,MAAM,sBAAsB,CAAA;AAE9F,KAAK,SAAS,GAAG,+BAA+B,GAAG,+BAA+B,CAAA;AAClF,KAAK,UAAU,GAAG,QAAQ,GAAG,QAAQ,GAAG,KAAK,CAAA;AAE7C,qBACa,mBAAmB,CAC9B,OAAO,SAAS,yBAAyB,GAAG,yBAAyB,EACrE,GAAG,SAAS,SAAS,GAAG,SAAS,EACjC,IAAI,SAAS,UAAU,GAAG,UAAU,EACpC,UAAU,SAAS,sBAAsB,CAAC,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,sBAAsB,CAChH,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EACnC,GAAG,EACH,IAAI,CACL,CACD,SAAQ,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,CAAC;IACvD,OAAgB,aAAa,WAAoC;IACjE,OAAgB,mBAAmB,EAAE,+BAA+B,CAAkC;cAEtF,aAAa,CAAC,QAAQ,GAAE,GAAG,EAAO,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;cAqBpD,0BAA0B,IAAI,OAAO,CAAC,sBAAsB,CAAC;cAM7D,yBAAyB,IAAI,OAAO,CAAC,sBAAsB,CAAC;CAK7E"}
1
+ {"version":3,"file":"Diviner.d.ts","sourceRoot":"","sources":["../../../src/Total/Diviner.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAC/D,OAAO,EACc,eAAe,EAAE,sBAAsB,EAC3D,MAAM,4BAA4B,CAAA;AAGnC,OAAO,EACL,QAAQ,EAGR,+BAA+B,EAAE,yBAAyB,EAAE,QAAQ,EAAE,KAAK,EAC5E,MAAM,sCAAsC,CAAA;AAE7C,OAAO,EAAE,sBAAsB,EAAE,+BAA+B,EAAE,MAAM,sBAAsB,CAAA;AAC9F,OAAO,EAAE,sBAAsB,EAAE,+BAA+B,EAAE,MAAM,sBAAsB,CAAA;AAE9F,KAAK,SAAS,GAAG,+BAA+B,GAAG,+BAA+B,CAAA;AAClF,KAAK,UAAU,GAAG,QAAQ,GAAG,QAAQ,GAAG,KAAK,CAAA;AAE7C,qBACa,mBAAmB,CAC9B,OAAO,SAAS,yBAAyB,GAAG,yBAAyB,EACrE,GAAG,SAAS,SAAS,GAAG,SAAS,EACjC,IAAI,SAAS,UAAU,GAAG,UAAU,EACpC,UAAU,SAAS,sBAAsB,CAAC,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,sBAAsB,CAChH,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,CAAC,EACnC,GAAG,EACH,IAAI,CACL,CACD,SAAQ,eAAe,CAAC,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,CAAC;IACvD,OAAgB,aAAa,WAAoC;IACjE,OAAgB,mBAAmB,EAAE,+BAA+B,CAAkC;cAEtF,aAAa,CAAC,QAAQ,GAAE,GAAG,EAAO,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;cAqBpD,0BAA0B,IAAI,OAAO,CAAC,sBAAsB,CAAC;cAM7D,yBAAyB,IAAI,OAAO,CAAC,sBAAsB,CAAC;CAK7E"}
@@ -26,9 +26,8 @@ import {
26
26
  import { creatableModule } from "@xyo-network/module-model";
27
27
  import { PayloadBuilder as PayloadBuilder2 } from "@xyo-network/payload-builder";
28
28
  import {
29
- isConditionWithMeta,
29
+ isCondition as isCondition2,
30
30
  isCoupon,
31
- isCouponWithMeta,
32
31
  isEscrowTerms,
33
32
  NO_DISCOUNT,
34
33
  PaymentDiscountDivinerConfigSchema
@@ -89,7 +88,7 @@ import { PayloadBuilder } from "@xyo-network/payload-builder";
89
88
  import {
90
89
  isCondition
91
90
  } from "@xyo-network/payment-payload-plugins";
92
- import { isSchemaPayloadWithMeta } from "@xyo-network/schema-payload-plugin";
91
+ import { isSchemaPayload } from "@xyo-network/schema-payload-plugin";
93
92
  import { Ajv } from "ajv";
94
93
  var ajv = new Ajv({ strict: false });
95
94
  var schemaCache = /* @__PURE__ */ new Map();
@@ -110,7 +109,7 @@ var findUnfulfilledConditions = async (coupon, conditions = [], payloads = []) =
110
109
  validator = schemaCache.get(hash);
111
110
  } else {
112
111
  const payload = hashMap[hash];
113
- const definition = isSchemaPayloadWithMeta(payload) ? payload.definition : void 0;
112
+ const definition = isSchemaPayload(payload) ? payload.definition : void 0;
114
113
  if (definition) {
115
114
  validator = ajv.compile(definition);
116
115
  schemaCache.set(hash, validator);
@@ -217,7 +216,7 @@ var PaymentDiscountDiviner = class extends AbstractDiviner {
217
216
  if (missingDiscounts.length > 0) {
218
217
  const discountsArchivist = await this.getDiscountsArchivist();
219
218
  const payloads = await discountsArchivist.get(missingDiscounts);
220
- discounts.push(...payloads.filter(isCouponWithMeta));
219
+ discounts.push(...payloads.filter(isCoupon));
221
220
  }
222
221
  if (discounts.length !== discountsHashes.length) {
223
222
  const termsHash = await PayloadBuilder2.hash(terms);
@@ -227,12 +226,12 @@ var PaymentDiscountDiviner = class extends AbstractDiviner {
227
226
  }
228
227
  }
229
228
  const conditionsHashes = discounts.flatMap((discount) => discount.conditions ?? []);
230
- const conditions = conditionsHashes.map((hash) => hashMap[hash]).filter(exists2).filter(isConditionWithMeta);
229
+ const conditions = conditionsHashes.map((hash) => hashMap[hash]).filter(exists2).filter(isCondition2);
231
230
  const missingConditions = conditionsHashes.filter((hash) => !hashMap[hash]);
232
231
  if (missingConditions.length > 0) {
233
232
  const discountsArchivist = await this.getDiscountsArchivist();
234
233
  const payloads = await discountsArchivist.get(missingConditions);
235
- conditions.push(...payloads.filter(isConditionWithMeta));
234
+ conditions.push(...payloads.filter(isCondition2));
236
235
  }
237
236
  if (conditions.length !== conditionsHashes.length) {
238
237
  const termsHash = await PayloadBuilder2.hash(terms);
@@ -384,9 +383,10 @@ import {
384
383
  asDivinerInstance as asDivinerInstance2
385
384
  } from "@xyo-network/diviner-model";
386
385
  import { creatableModule as creatableModule3 } from "@xyo-network/module-model";
386
+ import { PayloadBuilder as PayloadBuilder5 } from "@xyo-network/payload-builder";
387
387
  import {
388
- isDiscountWithMeta,
389
- isSubtotalWithMeta,
388
+ isDiscount as isDiscount2,
389
+ isSubtotal as isSubtotal2,
390
390
  PaymentTotalDivinerConfigSchema,
391
391
  TotalSchema
392
392
  } from "@xyo-network/payment-payload-plugins";
@@ -394,18 +394,18 @@ var PaymentTotalDiviner = class extends AbstractDiviner3 {
394
394
  async divineHandler(payloads = []) {
395
395
  const subtotalDiviner = await this.getPaymentSubtotalDiviner();
396
396
  const subtotalResult = await subtotalDiviner.divine(payloads);
397
- const subtotal = subtotalResult.find(isSubtotalWithMeta);
397
+ const subtotal = subtotalResult.find(isSubtotal2);
398
398
  if (!subtotal) return [];
399
399
  const discountDiviner = await this.getPaymentDiscountsDiviner();
400
400
  const discountResult = await discountDiviner.divine(payloads);
401
- const discount = discountResult.find(isDiscountWithMeta);
401
+ const discount = discountResult.find(isDiscount2);
402
402
  if (!discount) return [];
403
403
  const { currency: subtotalCurrency } = subtotal;
404
404
  const { currency: discountCurrency } = discount;
405
405
  assertEx3(subtotalCurrency === discountCurrency, () => `Subtotal currency ${subtotalCurrency} does not match discount currency ${discountCurrency}`);
406
406
  const amount = Math.max(0, subtotal.amount - discount.amount);
407
407
  const currency2 = subtotalCurrency;
408
- const sources = [subtotal.$hash, discount.$hash];
408
+ const sources = [await PayloadBuilder5.dataHash(subtotal), await PayloadBuilder5.dataHash(discount)];
409
409
  const total = {
410
410
  amount,
411
411
  currency: currency2,
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/Discount/Diviner.ts","../../src/Discount/lib/applyCoupons.ts","../../src/Discount/lib/findUnfulfilledConditions.ts","../../src/Invoice/getInvoiceForEscrow.ts","../../src/Subtotal/Diviner.ts","../../src/Subtotal/lib/appraisalValidators.ts","../../src/Subtotal/lib/durationValidators.ts","../../src/Subtotal/lib/termsValidators.ts","../../src/Total/Diviner.ts"],"sourcesContent":["import { assertEx } from '@xylabs/assert'\nimport { exists } from '@xylabs/exists'\nimport { Address, Hash } from '@xylabs/hex'\nimport { ArchivistInstance, asArchivistInstance } from '@xyo-network/archivist-model'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { BoundWitnessDivinerQueryPayload, BoundWitnessDivinerQuerySchema } from '@xyo-network/diviner-boundwitness-model'\nimport {\n HashLeaseEstimate,\n isHashLeaseEstimate,\n} from '@xyo-network/diviner-hash-lease'\nimport {\n asDivinerInstance, DivinerInstance, DivinerModuleEventData,\n} from '@xyo-network/diviner-model'\nimport { creatableModule } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload } from '@xyo-network/payload-model'\nimport {\n Condition,\n Coupon,\n Discount,\n EscrowTerms, isConditionWithMeta, isCoupon,\n isCouponWithMeta,\n isEscrowTerms, NO_DISCOUNT, PaymentDiscountDivinerConfigSchema, PaymentDiscountDivinerParams,\n} from '@xyo-network/payment-payload-plugins'\n\nimport { applyCoupons, areConditionsFulfilled } from './lib/index.ts'\n\nconst DEFAULT_BOUND_WITNESS_DIVINER_QUERY_PROPS: Readonly<BoundWitnessDivinerQueryPayload> = {\n limit: 1,\n order: 'desc',\n schema: BoundWitnessDivinerQuerySchema,\n}\n\nexport type PaymentDiscountDivinerInputType = EscrowTerms | Coupon | HashLeaseEstimate | Payload\n\n@creatableModule()\nexport class PaymentDiscountDiviner<\n TParams extends PaymentDiscountDivinerParams = PaymentDiscountDivinerParams,\n TIn extends PaymentDiscountDivinerInputType = PaymentDiscountDivinerInputType,\n TOut extends Discount = Discount,\n TEventData extends DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut> = DivinerModuleEventData<\n DivinerInstance<TParams, TIn, TOut>,\n TIn,\n TOut\n >,\n> extends AbstractDiviner<TParams, TIn, TOut, TEventData> {\n static override configSchemas = [PaymentDiscountDivinerConfigSchema]\n static override defaultConfigSchema = PaymentDiscountDivinerConfigSchema\n\n protected get couponAuthorities(): Address[] {\n return [...(this.config.couponAuthorities ?? []), ...(this.params.couponAuthorities ?? [])]\n }\n\n protected async divineHandler(payloads: TIn[] = []): Promise<TOut[]> {\n const sources: Hash[] = []\n\n // Parse terms\n const terms = payloads.find(isEscrowTerms) as EscrowTerms | undefined\n if (!terms) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n sources.push(await PayloadBuilder.hash(terms))\n\n // Parse appraisals\n const termsAppraisals = terms?.appraisals\n // If the escrow terms do not have appraisals, return no discount\n if (!termsAppraisals || termsAppraisals.length === 0) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n const hashMap = await PayloadBuilder.toAllHashMap(payloads) as Record<Hash, Payload>\n const appraisals = this.getEscrowAppraisals(terms, hashMap)\n // Add the appraisals that were found to the sources\n sources.push(...termsAppraisals)\n // If not all appraisals are found, return no discount\n if (appraisals.length !== termsAppraisals.length) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n\n // Parse coupons\n const [coupons, conditions] = await this.getEscrowDiscounts(terms, hashMap)\n // Add the coupons that were found to the sources\n // NOTE: Should we throw if not all coupons are found?\n const couponHashes = await PayloadBuilder.hashes(coupons)\n sources.push(...couponHashes)\n\n const currentCoupons = coupons.filter(this.isCouponCurrent)\n const conditionsMetCoupons = (\n await Promise.all(currentCoupons.map(async coupon => await areConditionsFulfilled(coupon, conditions, payloads) ? coupon : undefined))).filter(exists)\n\n const validCoupons = await this.filterToSigned(conditionsMetCoupons)\n // NOTE: Should we throw if not all coupons are valid?\n if (validCoupons.length === 0) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n\n // TODO: Call paymentSubtotalDiviner to get the subtotal to centralize the logic\n const discount = applyCoupons(appraisals, validCoupons)\n return [{ ...discount, sources }] as TOut[]\n }\n\n /**\n * Filters the supplied list of coupons to only those that are signed by\n * addresses specified in the couponAuthorities\n * @param coupons The list of coupons to filter\n * @returns The filtered list of coupons that are signed by the couponAuthorities\n */\n protected async filterToSigned(coupons: Coupon[]): Promise<Coupon[]> {\n const signed: Coupon[] = []\n const dataHashMap = await PayloadBuilder.toDataHashMap(coupons)\n const boundWitnessDiviner = await this.getDiscountsBoundWitnessDiviner()\n const hashes = Object.keys(dataHashMap)\n const addresses = this.couponAuthorities\n // TODO: Keep an in memory cache of the hashes queried and their results\n // to avoid querying the same hash multiple times\n await Promise.all(hashes.map((h) => {\n const hash = h as Hash\n return Promise.all(addresses.map(async (address) => {\n const query: BoundWitnessDivinerQueryPayload = {\n ...DEFAULT_BOUND_WITNESS_DIVINER_QUERY_PROPS, addresses: [address], payload_hashes: [hash],\n }\n const result = await boundWitnessDiviner.divine([query])\n if (result.length > 0) signed.push(dataHashMap[hash])\n }))\n }))\n return signed\n }\n\n protected async getDiscountsArchivist(): Promise<ArchivistInstance> {\n const name = assertEx(this.config.archivist, () => 'Missing archivist in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving archivist: ${name}`)\n return assertEx(asArchivistInstance(mod), () => `Resolved module ${mod.address} not a valid Archivist`)\n }\n\n protected async getDiscountsBoundWitnessDiviner(): Promise<DivinerInstance> {\n const name = assertEx(this.config.boundWitnessDiviner, () => 'Missing boundWitnessDiviner in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving boundWitnessDiviner: ${name}`)\n return assertEx(asDivinerInstance(mod), () => `Resolved module ${mod.address} not a valid Diviner`)\n }\n\n /**\n * Finds the appraisals specified by the escrow terms from the supplied payloads\n * @param terms The escrow terms\n * @param payloads The payloads to search for the appraisals\n * @returns The appraisals found in the payloads\n */\n protected getEscrowAppraisals(terms: EscrowTerms, hashMap: Record<Hash, Payload>): HashLeaseEstimate[] {\n const hashes = terms?.appraisals ?? []\n if (hashes.length === 0) return []\n return hashes.map(hash => hashMap[hash]).filter(exists).filter(isHashLeaseEstimate)\n }\n\n /**\n * Finds the discounts specified by the escrow terms from the supplied payloads\n * @param terms The escrow terms\n * @param hashMap The payloads to search for the discounts\n * @returns A tuple containing all the escrow coupons and conditions referenced in those coupons\n * that were found in the either the supplied payloads or the archivist\n */\n protected async getEscrowDiscounts(terms: EscrowTerms, hashMap: Record<Hash, Payload>): Promise<[Coupon[], Condition[]]> {\n // Parse discounts\n const discountsHashes = terms.discounts ?? []\n if (discountsHashes.length === 0) return [[], []]\n\n // Use the supplied payloads to find the discounts\n const discounts: Coupon[] = discountsHashes.map(hash => hashMap[hash]).filter(exists).filter(isCoupon)\n const missingDiscounts = discountsHashes.filter(hash => !hashMap[hash])\n // If not all discounts are found\n if (missingDiscounts.length > 0) {\n // Find any remaining from discounts archivist\n const discountsArchivist = await this.getDiscountsArchivist()\n const payloads = await discountsArchivist.get(missingDiscounts)\n discounts.push(...payloads.filter(isCouponWithMeta))\n }\n // If not all discounts are found\n if (discounts.length !== discountsHashes.length) {\n const termsHash = await PayloadBuilder.hash(terms)\n const foundHashes = await PayloadBuilder.hashes(discounts)\n // Log individual discounts that were not found\n for (const hash of discountsHashes) {\n if (!foundHashes.includes(hash)) console.warn(`Discount ${hash} not found for terms ${termsHash}`)\n }\n }\n\n const conditionsHashes: Hash[] = discounts.flatMap(discount => discount.conditions ?? [])\n const conditions: Condition[] = conditionsHashes.map(hash => hashMap[hash]).filter(exists).filter(isConditionWithMeta)\n const missingConditions = conditionsHashes.filter(hash => !hashMap[hash])\n\n // If not all conditions are found\n if (missingConditions.length > 0) {\n // Find any remaining from discounts archivist\n const discountsArchivist = await this.getDiscountsArchivist()\n const payloads = await discountsArchivist.get(missingConditions)\n conditions.push(...payloads.filter(isConditionWithMeta))\n }\n // If not all conditions are found\n if (conditions.length !== conditionsHashes.length) {\n const termsHash = await PayloadBuilder.hash(terms)\n const foundHashes = await PayloadBuilder.hashes(conditions)\n // Log individual conditions that were not found\n for (const hash of discountsHashes) {\n if (!foundHashes.includes(hash)) console.warn(`Coupon condition ${hash} not found for terms ${termsHash}`)\n }\n }\n\n return [discounts, conditions]\n }\n\n protected isCouponCurrent(coupon: Coupon): boolean {\n const now = Date.now()\n return coupon.exp > now && coupon.nbf < now\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { exists } from '@xylabs/exists'\nimport type { HashLeaseEstimate } from '@xyo-network/diviner-hash-lease'\nimport type {\n AmountFields,\n Coupon, Discount, FixedAmountCoupon,\n FixedPercentageCoupon,\n FixedPriceCoupon,\n} from '@xyo-network/payment-payload-plugins'\nimport {\n DiscountSchema, isFixedAmountCoupon, isFixedPercentageCoupon,\n isFixedPriceCoupon,\n isStackable,\n} from '@xyo-network/payment-payload-plugins'\n\nexport const applyCoupons = (appraisals: HashLeaseEstimate[], coupons: Coupon[]): Discount => {\n // Ensure all appraisals and coupons are in USD\n const allAppraisalsAreUSD = appraisals.every(appraisal => appraisal.currency === 'USD')\n assertEx(allAppraisalsAreUSD, 'All appraisals must be in USD')\n const allCouponsAreUSD = coupons.map(coupon => (coupon as Partial<AmountFields>)?.currency).filter(exists).every(currency => currency === 'USD')\n assertEx(allCouponsAreUSD, 'All coupons must be in USD')\n const total = appraisals.reduce((acc, appraisal) => acc + appraisal.price, 0)\n\n // Calculated non-stackable discount coupons\n const singularFixedDiscount = Math.max(...coupons\n .filter(coupon => isFixedAmountCoupon(coupon) && !isStackable(coupon))\n .map(coupon => (coupon as FixedAmountCoupon).amount), 0)\n const singularPercentageDiscount = (Math.max(...coupons\n .filter(coupon => isFixedPercentageCoupon(coupon) && !isStackable(coupon))\n .map(coupon => (coupon as FixedPercentageCoupon).percentage), 0)) * total\n const singularFixedPriceDiscount = calculateSingularFixedPriceDiscount(total, appraisals, coupons, false)\n\n // Calculate stackable discount coupons\n // First calculate the total discount from fixed amount coupons\n const stackedFixedDiscount = coupons\n .filter(coupon => isFixedAmountCoupon(coupon) && isStackable(coupon))\n .reduce((acc, coupon) => acc + (coupon as FixedAmountCoupon).amount, 0)\n // Then calculate the total discount from percentage coupons and apply\n // the percentage discount to the remaining total after fixed discounts\n const stackedPercentageDiscount = coupons\n .filter(coupon => isFixedPercentageCoupon(coupon) && isStackable(coupon))\n .reduce((acc, coupon) => acc + (coupon as FixedPercentageCoupon).percentage, 0) * (total - stackedFixedDiscount)\n // Then calculate the total discount from fixed price coupons\n const stackedFixedPriceDiscount = calculateSingularFixedPriceDiscount(total, appraisals, coupons, true)\n\n // Sum all stackable discounts\n const stackedDiscount = stackedFixedDiscount + stackedPercentageDiscount + stackedFixedPriceDiscount\n\n // Find the best coupon(s) to apply\n const maxDiscount = Math.max(\n singularFixedDiscount,\n singularFixedPriceDiscount,\n singularPercentageDiscount,\n stackedDiscount,\n 0,\n )\n\n // Ensure discount is not more than the total\n const amount = Math.min(maxDiscount, total)\n\n // Return single discount payload\n return {\n amount, schema: DiscountSchema, currency: 'USD',\n }\n}\n\nconst calculateSingularFixedPriceDiscount = (total: number, appraisals: HashLeaseEstimate[], coupons: Coupon[], stackable = false): number => {\n // Find all singular fixed price discounts\n const singularFixedPriceDiscounts = coupons\n .filter(isFixedPriceCoupon)\n .filter(coupon => stackable ? isStackable(coupon) : !isStackable(coupon))\n .map(coupon => (coupon as FixedPriceCoupon).amount)\n // Ensure all fixed price discounts are positive\n .filter(amount => amount > 0)\n\n // If there are no singular fixed price discounts, return no discount\n if (singularFixedPriceDiscounts.length === 0) return 0\n\n // Find the maximum discount (the lowest fixed price)\n const lowestFixedPrice = Math.min(...singularFixedPriceDiscounts)\n\n // Apply the fixed price to all appraisals to get the reduced prices\n const reducedPrices = appraisals.map(appraisal =>\n // If the appraisal price is less than the fixed price\n Math.min(appraisal.price, lowestFixedPrice))\n\n // Calculate the reduced total using the reduced prices\n const reducedTotal = reducedPrices.reduce((acc, price) => acc + price, 0)\n\n // Calculate the discount\n const discount = total - reducedTotal\n\n // Return the discount or 0 if the discount would have resulted in a negative value\n return Math.max(discount, 0)\n}\n","import type { Hash } from '@xylabs/hex'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type { Payload } from '@xyo-network/payload-model'\nimport {\n type Condition, type Coupon, isCondition,\n} from '@xyo-network/payment-payload-plugins'\nimport { isSchemaPayloadWithMeta } from '@xyo-network/schema-payload-plugin'\nimport type { ValidateFunction } from 'ajv'\nimport { Ajv } from 'ajv'\n\n// TODO: Use our schema cache\nconst ajv = new Ajv({ strict: false }) // Create the Ajv instance once\nconst schemaCache = new Map() // Cache to store compiled validators\n\n/**\n * Validates the conditions of a coupon against the provided payloads\n * @param coupon The coupon to check\n * @param conditions The conditions associated with the coupon\n * @param payloads The associated payloads (containing the conditions and data to validate the conditions against)\n * @returns True if all conditions are fulfilled, false otherwise\n */\nexport const areConditionsFulfilled = async (coupon: Coupon, conditions: Condition[] = [], payloads: Payload[] = []): Promise<boolean> =>\n (await findUnfulfilledConditions(coupon, conditions, payloads)).length === 0\n\n/**\n * Validates the conditions of a coupon against the provided payloads\n * @param coupon The coupon to check\n * @param conditions The conditions associated with the coupon\n * @param payloads The associated payloads (containing the conditions and data to validate the conditions against)\n * @returns The unfulfilled condition hashes\n */\nexport const findUnfulfilledConditions = async (coupon: Coupon, conditions: Condition[] = [], payloads: Payload[] = []): Promise<Hash[]> => {\n const unfulfilledConditions: Hash[] = []\n // If there are no conditions, then they are fulfilled\n if (!coupon.conditions || coupon.conditions.length === 0) return unfulfilledConditions\n const hashMap = await PayloadBuilder.toAllHashMap([...conditions, ...payloads])\n // Find all the conditions\n const foundConditions = coupon.conditions.map(hash => hashMap[hash]).filter(isCondition)\n // Not all conditions were found\n if (foundConditions.length !== coupon.conditions.length) {\n const missing = coupon.conditions.filter(hash => !hashMap[hash])\n unfulfilledConditions.push(...missing)\n return unfulfilledConditions\n }\n\n // Test each condition\n for (const hash of coupon.conditions) {\n let validator: ValidateFunction\n\n // Check if the schema is already cached\n if (schemaCache.has(hash)) {\n validator = schemaCache.get(hash)\n } else {\n const payload = hashMap[hash]\n const definition = isSchemaPayloadWithMeta(payload) ? payload.definition : undefined\n if (definition) {\n // Compile and cache the validator\n validator = ajv.compile(definition)\n schemaCache.set(hash, validator)\n\n // Validate the payload\n } else {\n unfulfilledConditions.push(hash)\n continue\n }\n }\n if (!validator(payloads)) unfulfilledConditions.push(hash)\n }\n\n // All conditions passed\n return unfulfilledConditions\n}\n","import type { Hash } from '@xylabs/hex'\nimport type { DivinerInstance } from '@xyo-network/diviner-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type { Payload } from '@xyo-network/payload-model'\nimport type {\n Discount, EscrowTerms, Invoice, Payment, Subtotal, Total,\n} from '@xyo-network/payment-payload-plugins'\nimport {\n isDiscount, isSubtotal, isTotal, PaymentSchema,\n} from '@xyo-network/payment-payload-plugins'\n\n/**\n * Validates the escrow terms to ensure they are valid for a purchase\n * @returns A payment if the terms are valid for a purchase, undefined otherwise\n */\nexport const getInvoiceForEscrow = async (\n terms: EscrowTerms,\n dataHashMap: Record<Hash, Payload>,\n paymentTotalDiviner: DivinerInstance,\n): Promise<Invoice | undefined> => {\n const payloads = Object.values(dataHashMap)\n const results = await paymentTotalDiviner.divine([terms, ...payloads])\n const subtotal = results.find(isSubtotal) as Subtotal | undefined\n const discount = results.find(isDiscount) as Discount | undefined\n const total = results.find(isTotal) as Total | undefined\n if (!subtotal || !total) return undefined\n const { amount, currency } = total\n if (currency !== 'USD') return undefined\n const sources = await getSources(terms, subtotal, total, discount)\n const payment: Payment = {\n amount, currency, schema: PaymentSchema, sources,\n }\n return discount ? [subtotal, total, payment, discount] : [subtotal, total, payment]\n}\n\nconst getSources = async (terms: EscrowTerms, subtotal: Subtotal, total: Total, discount?: Discount): Promise<Hash[]> => {\n const sources = discount ? [terms, subtotal, total, discount] : [terms, subtotal, total]\n return await Promise.all(sources.map(p => PayloadBuilder.dataHash(p)))\n}\n","import { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { HashLeaseEstimate, isHashLeaseEstimate } from '@xyo-network/diviner-hash-lease'\nimport { DivinerInstance, DivinerModuleEventData } from '@xyo-network/diviner-model'\nimport { creatableModule } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload } from '@xyo-network/payload-model'\nimport {\n EscrowTerms, isEscrowTerms, PaymentSubtotalDivinerConfigSchema, PaymentSubtotalDivinerParams, Subtotal, SubtotalSchema,\n} from '@xyo-network/payment-payload-plugins'\n\nimport {\n appraisalValidators, termsValidators, ValidEscrowTerms,\n} from './lib/index.ts'\n\nconst currency = 'USD'\n\n/**\n * Escrow terms that contain all the valid fields for calculating a subtotal\n */\nexport type PaymentSubtotalDivinerInputType = EscrowTerms | HashLeaseEstimate | Payload\n\n@creatableModule()\nexport class PaymentSubtotalDiviner<\n TParams extends PaymentSubtotalDivinerParams = PaymentSubtotalDivinerParams,\n TIn extends PaymentSubtotalDivinerInputType = PaymentSubtotalDivinerInputType,\n TOut extends Subtotal = Subtotal,\n TEventData extends DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut> = DivinerModuleEventData<\n DivinerInstance<TParams, TIn, TOut>,\n TIn,\n TOut\n >,\n> extends AbstractDiviner<TParams, TIn, TOut, TEventData> {\n static override configSchemas = [PaymentSubtotalDivinerConfigSchema]\n static override defaultConfigSchema = PaymentSubtotalDivinerConfigSchema\n\n protected async divineHandler(payloads: TIn[] = []): Promise<TOut[]> {\n // Find the escrow terms\n const terms = payloads.find(isEscrowTerms) as EscrowTerms | undefined\n if (!terms) return []\n\n // Run all terms validations\n if (!termsValidators.every(validator => validator(terms))) return []\n const validTerms = terms as ValidEscrowTerms\n\n // Retrieve all appraisals from terms\n const hashMap = await PayloadBuilder.toAllHashMap(payloads)\n const appraisals = validTerms.appraisals.map(appraisal => hashMap[appraisal]).filter(isHashLeaseEstimate) as unknown as HashLeaseEstimate[]\n\n // Ensure all appraisals are present\n if (appraisals.length !== validTerms.appraisals.length) return []\n\n // Run all appraisal validations\n if (!appraisalValidators.every(validator => validator(appraisals))) return []\n const amount = calculateSubtotal(appraisals)\n const sources = [await PayloadBuilder.dataHash(validTerms), ...validTerms.appraisals]\n return [{\n amount, currency, schema: SubtotalSchema, sources,\n }] as TOut[]\n }\n}\n\n// TODO: Add support for other currencies\nconst calculateSubtotal = (appraisals: HashLeaseEstimate[]): number => {\n return appraisals.reduce((sum, appraisal) => sum + appraisal.price, 0)\n}\n","import type { HashLeaseEstimate } from '@xyo-network/diviner-hash-lease'\nimport { isIso4217CurrencyCode } from '@xyo-network/payment-payload-plugins'\n\nimport { validateDuration } from './durationValidators.ts'\n\nconst validateAppraisalAmount = (appraisals: HashLeaseEstimate[]): boolean => {\n // Ensure all appraisals are numeric\n if (appraisals.some(appraisal => typeof appraisal.price !== 'number')) return false\n // Ensure all appraisals are positive numbers\n if (appraisals.some(appraisal => appraisal.price < 0)) return false\n return true\n}\n\nconst validateAppraisalCurrency = (appraisals: HashLeaseEstimate[]): boolean => {\n // NOTE: Only supporting USD for now, the remaining checks are for future-proofing.\n if (!appraisals.every(appraisal => appraisal.currency == 'USD')) return false\n\n // Check every object in the array to ensure they all are in a supported currency.\n if (!appraisals.every(appraisal => isIso4217CurrencyCode(appraisal.currency))) return false\n\n return true\n}\n\nconst validateAppraisalConsistentCurrency = (appraisals: HashLeaseEstimate[]): boolean => {\n // Check if the array is empty or contains only one element, no need to compare.\n if (appraisals.length <= 1) return true\n\n // Get the currency of the first element to compare with others.\n const { currency } = appraisals[0]\n if (!currency) return false\n\n // Check every object in the array to ensure they all have the same currency.\n if (!appraisals.every(item => item.currency === currency)) return false\n\n return true\n}\n\nconst validateAppraisalWindow = (appraisals: HashLeaseEstimate[]): boolean => appraisals.every(validateDuration)\n\nexport const appraisalValidators = [\n validateAppraisalAmount,\n validateAppraisalCurrency,\n validateAppraisalConsistentCurrency,\n validateAppraisalWindow,\n]\n","import type { DurationFields } from '@xyo-network/xns-record-payload-plugins'\n\nconst FIVE_MINUTES = 1000 * 60 * 5\n\n/**\n * Validates that the current time is within the duration window, within a configurable a buffer\n * @param value The duration value\n * @param windowMs The window in milliseconds to allow for a buffer\n * @returns True if the duration is valid, false otherwise\n */\nexport const validateDuration = (value: Partial<DurationFields>, windowMs = FIVE_MINUTES): boolean => {\n const now = Date.now()\n if (!value.nbf || value.nbf > now) return false\n // If already expired (include for a 5 minute buffer to allow for a reasonable\n // minimum amount of time for the transaction to be processed)\n if (!value.exp || value.exp - now < windowMs) return false\n return true\n}\n","import type { Hash } from '@xylabs/hex'\nimport type { EscrowTerms } from '@xyo-network/payment-payload-plugins'\n\nimport { validateDuration } from './durationValidators.ts'\n\nexport type ValidEscrowTerms = Required<EscrowTerms>\n\nconst validateTermsAppraisals = (terms: EscrowTerms): terms is Required<EscrowTerms & { appraisals: Hash[] }> => {\n if (!terms.appraisals) return false\n if (terms.appraisals.length === 0) return false\n return true\n}\nconst validateTermsWindow = (terms: EscrowTerms): boolean => validateDuration(terms)\n\nexport const termsValidators = [\n validateTermsAppraisals,\n validateTermsWindow,\n]\n","import { assertEx } from '@xylabs/assert'\nimport { Hash } from '@xylabs/hex'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport {\n asDivinerInstance, DivinerInstance, DivinerModuleEventData,\n} from '@xyo-network/diviner-model'\nimport { creatableModule } from '@xyo-network/module-model'\nimport {\n Discount,\n isDiscountWithMeta,\n isSubtotalWithMeta,\n PaymentTotalDivinerConfigSchema, PaymentTotalDivinerParams, Subtotal, Total, TotalSchema,\n} from '@xyo-network/payment-payload-plugins'\n\nimport { PaymentDiscountDiviner, PaymentDiscountDivinerInputType } from '../Discount/index.ts'\nimport { PaymentSubtotalDiviner, PaymentSubtotalDivinerInputType } from '../Subtotal/index.ts'\n\ntype InputType = PaymentDiscountDivinerInputType | PaymentSubtotalDivinerInputType\ntype OutputType = Subtotal | Discount | Total\n\n@creatableModule()\nexport class PaymentTotalDiviner<\n TParams extends PaymentTotalDivinerParams = PaymentTotalDivinerParams,\n TIn extends InputType = InputType,\n TOut extends OutputType = OutputType,\n TEventData extends DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut> = DivinerModuleEventData<\n DivinerInstance<TParams, TIn, TOut>,\n TIn,\n TOut\n >,\n> extends AbstractDiviner<TParams, TIn, TOut, TEventData> {\n static override configSchemas = [PaymentTotalDivinerConfigSchema]\n static override defaultConfigSchema: PaymentTotalDivinerConfigSchema = PaymentTotalDivinerConfigSchema\n\n protected async divineHandler(payloads: TIn[] = []): Promise<TOut[]> {\n const subtotalDiviner = await this.getPaymentSubtotalDiviner()\n const subtotalResult = await subtotalDiviner.divine(payloads)\n const subtotal = subtotalResult.find(isSubtotalWithMeta)\n if (!subtotal) return []\n const discountDiviner = await this.getPaymentDiscountsDiviner()\n const discountResult = await discountDiviner.divine(payloads)\n const discount = discountResult.find(isDiscountWithMeta)\n if (!discount) return []\n const { currency: subtotalCurrency } = subtotal\n const { currency: discountCurrency } = discount\n assertEx(subtotalCurrency === discountCurrency, () => `Subtotal currency ${subtotalCurrency} does not match discount currency ${discountCurrency}`)\n const amount = Math.max(0, subtotal.amount - discount.amount)\n const currency = subtotalCurrency\n const sources = [subtotal.$hash, discount.$hash] as Hash[]\n const total: Total = {\n amount, currency, sources, schema: TotalSchema,\n }\n return [subtotal, discount, total] as TOut[]\n }\n\n protected async getPaymentDiscountsDiviner(): Promise<PaymentDiscountDiviner> {\n const name = assertEx(this.config.paymentDiscountDiviner, () => 'Missing paymentDiscountDiviner in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving paymentDiscountDiviner: ${name}`)\n return assertEx(asDivinerInstance(mod), () => `Resolved module ${mod.address} not a valid Diviner`) as PaymentDiscountDiviner\n }\n\n protected async getPaymentSubtotalDiviner(): Promise<PaymentSubtotalDiviner> {\n const name = assertEx(this.config.paymentSubtotalDiviner, () => 'Missing paymentSubtotalDiviner in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving paymentSubtotalDiviner: ${name}`)\n return assertEx(asDivinerInstance(mod), () => `Resolved module ${mod.address} not a valid Diviner`) as PaymentSubtotalDiviner\n }\n}\n"],"mappings":"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","names":["assertEx","exists","PayloadBuilder","currency","PayloadBuilder","exists","assertEx","PayloadBuilder","currency","AbstractDiviner","isHashLeaseEstimate","creatableModule","PayloadBuilder","isEscrowTerms","currency","AbstractDiviner","isEscrowTerms","PayloadBuilder","isHashLeaseEstimate","creatableModule","assertEx","AbstractDiviner","asDivinerInstance","creatableModule","AbstractDiviner","assertEx","currency","asDivinerInstance","creatableModule"]}
1
+ {"version":3,"sources":["../../src/Discount/Diviner.ts","../../src/Discount/lib/applyCoupons.ts","../../src/Discount/lib/findUnfulfilledConditions.ts","../../src/Invoice/getInvoiceForEscrow.ts","../../src/Subtotal/Diviner.ts","../../src/Subtotal/lib/appraisalValidators.ts","../../src/Subtotal/lib/durationValidators.ts","../../src/Subtotal/lib/termsValidators.ts","../../src/Total/Diviner.ts"],"sourcesContent":["import { assertEx } from '@xylabs/assert'\nimport { exists } from '@xylabs/exists'\nimport { Address, Hash } from '@xylabs/hex'\nimport { ArchivistInstance, asArchivistInstance } from '@xyo-network/archivist-model'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { BoundWitnessDivinerQueryPayload, BoundWitnessDivinerQuerySchema } from '@xyo-network/diviner-boundwitness-model'\nimport {\n HashLeaseEstimate,\n isHashLeaseEstimate,\n} from '@xyo-network/diviner-hash-lease'\nimport {\n asDivinerInstance, DivinerInstance, DivinerModuleEventData,\n} from '@xyo-network/diviner-model'\nimport { creatableModule } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload } from '@xyo-network/payload-model'\nimport {\n Condition,\n Coupon,\n Discount,\n EscrowTerms, isCondition, isCoupon,\n isEscrowTerms, NO_DISCOUNT, PaymentDiscountDivinerConfigSchema, PaymentDiscountDivinerParams,\n} from '@xyo-network/payment-payload-plugins'\n\nimport { applyCoupons, areConditionsFulfilled } from './lib/index.ts'\n\nconst DEFAULT_BOUND_WITNESS_DIVINER_QUERY_PROPS: Readonly<BoundWitnessDivinerQueryPayload> = {\n limit: 1,\n order: 'desc',\n schema: BoundWitnessDivinerQuerySchema,\n}\n\nexport type PaymentDiscountDivinerInputType = EscrowTerms | Coupon | HashLeaseEstimate | Payload\n\n@creatableModule()\nexport class PaymentDiscountDiviner<\n TParams extends PaymentDiscountDivinerParams = PaymentDiscountDivinerParams,\n TIn extends PaymentDiscountDivinerInputType = PaymentDiscountDivinerInputType,\n TOut extends Discount = Discount,\n TEventData extends DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut> = DivinerModuleEventData<\n DivinerInstance<TParams, TIn, TOut>,\n TIn,\n TOut\n >,\n> extends AbstractDiviner<TParams, TIn, TOut, TEventData> {\n static override configSchemas = [PaymentDiscountDivinerConfigSchema]\n static override defaultConfigSchema = PaymentDiscountDivinerConfigSchema\n\n protected get couponAuthorities(): Address[] {\n return [...(this.config.couponAuthorities ?? []), ...(this.params.couponAuthorities ?? [])]\n }\n\n protected async divineHandler(payloads: TIn[] = []): Promise<TOut[]> {\n const sources: Hash[] = []\n\n // Parse terms\n const terms = payloads.find(isEscrowTerms) as EscrowTerms | undefined\n if (!terms) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n sources.push(await PayloadBuilder.hash(terms))\n\n // Parse appraisals\n const termsAppraisals = terms?.appraisals\n // If the escrow terms do not have appraisals, return no discount\n if (!termsAppraisals || termsAppraisals.length === 0) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n const hashMap = await PayloadBuilder.toAllHashMap(payloads) as Record<Hash, Payload>\n const appraisals = this.getEscrowAppraisals(terms, hashMap)\n // Add the appraisals that were found to the sources\n sources.push(...termsAppraisals)\n // If not all appraisals are found, return no discount\n if (appraisals.length !== termsAppraisals.length) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n\n // Parse coupons\n const [coupons, conditions] = await this.getEscrowDiscounts(terms, hashMap)\n // Add the coupons that were found to the sources\n // NOTE: Should we throw if not all coupons are found?\n const couponHashes = await PayloadBuilder.hashes(coupons)\n sources.push(...couponHashes)\n\n const currentCoupons = coupons.filter(this.isCouponCurrent)\n const conditionsMetCoupons = (\n await Promise.all(currentCoupons.map(async coupon => await areConditionsFulfilled(coupon, conditions, payloads) ? coupon : undefined))).filter(exists)\n\n const validCoupons = await this.filterToSigned(conditionsMetCoupons)\n // NOTE: Should we throw if not all coupons are valid?\n if (validCoupons.length === 0) return [{ ...NO_DISCOUNT, sources }] as TOut[]\n\n // TODO: Call paymentSubtotalDiviner to get the subtotal to centralize the logic\n const discount = applyCoupons(appraisals, validCoupons)\n return [{ ...discount, sources }] as TOut[]\n }\n\n /**\n * Filters the supplied list of coupons to only those that are signed by\n * addresses specified in the couponAuthorities\n * @param coupons The list of coupons to filter\n * @returns The filtered list of coupons that are signed by the couponAuthorities\n */\n protected async filterToSigned(coupons: Coupon[]): Promise<Coupon[]> {\n const signed: Coupon[] = []\n const dataHashMap = await PayloadBuilder.toDataHashMap(coupons)\n const boundWitnessDiviner = await this.getDiscountsBoundWitnessDiviner()\n const hashes = Object.keys(dataHashMap)\n const addresses = this.couponAuthorities\n // TODO: Keep an in memory cache of the hashes queried and their results\n // to avoid querying the same hash multiple times\n await Promise.all(hashes.map((h) => {\n const hash = h as Hash\n return Promise.all(addresses.map(async (address) => {\n const query: BoundWitnessDivinerQueryPayload = {\n ...DEFAULT_BOUND_WITNESS_DIVINER_QUERY_PROPS, addresses: [address], payload_hashes: [hash],\n }\n const result = await boundWitnessDiviner.divine([query])\n if (result.length > 0) signed.push(dataHashMap[hash])\n }))\n }))\n return signed\n }\n\n protected async getDiscountsArchivist(): Promise<ArchivistInstance> {\n const name = assertEx(this.config.archivist, () => 'Missing archivist in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving archivist: ${name}`)\n return assertEx(asArchivistInstance(mod), () => `Resolved module ${mod.address} not a valid Archivist`)\n }\n\n protected async getDiscountsBoundWitnessDiviner(): Promise<DivinerInstance> {\n const name = assertEx(this.config.boundWitnessDiviner, () => 'Missing boundWitnessDiviner in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving boundWitnessDiviner: ${name}`)\n return assertEx(asDivinerInstance(mod), () => `Resolved module ${mod.address} not a valid Diviner`)\n }\n\n /**\n * Finds the appraisals specified by the escrow terms from the supplied payloads\n * @param terms The escrow terms\n * @param payloads The payloads to search for the appraisals\n * @returns The appraisals found in the payloads\n */\n protected getEscrowAppraisals(terms: EscrowTerms, hashMap: Record<Hash, Payload>): HashLeaseEstimate[] {\n const hashes = terms?.appraisals ?? []\n if (hashes.length === 0) return []\n return hashes.map(hash => hashMap[hash]).filter(exists).filter(isHashLeaseEstimate)\n }\n\n /**\n * Finds the discounts specified by the escrow terms from the supplied payloads\n * @param terms The escrow terms\n * @param hashMap The payloads to search for the discounts\n * @returns A tuple containing all the escrow coupons and conditions referenced in those coupons\n * that were found in the either the supplied payloads or the archivist\n */\n protected async getEscrowDiscounts(terms: EscrowTerms, hashMap: Record<Hash, Payload>): Promise<[Coupon[], Condition[]]> {\n // Parse discounts\n const discountsHashes = terms.discounts ?? []\n if (discountsHashes.length === 0) return [[], []]\n\n // Use the supplied payloads to find the discounts\n const discounts: Coupon[] = discountsHashes.map(hash => hashMap[hash]).filter(exists).filter(isCoupon)\n const missingDiscounts = discountsHashes.filter(hash => !hashMap[hash])\n // If not all discounts are found\n if (missingDiscounts.length > 0) {\n // Find any remaining from discounts archivist\n const discountsArchivist = await this.getDiscountsArchivist()\n const payloads = await discountsArchivist.get(missingDiscounts)\n discounts.push(...payloads.filter(isCoupon))\n }\n // If not all discounts are found\n if (discounts.length !== discountsHashes.length) {\n const termsHash = await PayloadBuilder.hash(terms)\n const foundHashes = await PayloadBuilder.hashes(discounts)\n // Log individual discounts that were not found\n for (const hash of discountsHashes) {\n if (!foundHashes.includes(hash)) console.warn(`Discount ${hash} not found for terms ${termsHash}`)\n }\n }\n\n const conditionsHashes: Hash[] = discounts.flatMap(discount => discount.conditions ?? [])\n const conditions: Condition[] = conditionsHashes.map(hash => hashMap[hash]).filter(exists).filter(isCondition)\n const missingConditions = conditionsHashes.filter(hash => !hashMap[hash])\n\n // If not all conditions are found\n if (missingConditions.length > 0) {\n // Find any remaining from discounts archivist\n const discountsArchivist = await this.getDiscountsArchivist()\n const payloads = await discountsArchivist.get(missingConditions)\n conditions.push(...payloads.filter(isCondition))\n }\n // If not all conditions are found\n if (conditions.length !== conditionsHashes.length) {\n const termsHash = await PayloadBuilder.hash(terms)\n const foundHashes = await PayloadBuilder.hashes(conditions)\n // Log individual conditions that were not found\n for (const hash of discountsHashes) {\n if (!foundHashes.includes(hash)) console.warn(`Coupon condition ${hash} not found for terms ${termsHash}`)\n }\n }\n\n return [discounts, conditions]\n }\n\n protected isCouponCurrent(coupon: Coupon): boolean {\n const now = Date.now()\n return coupon.exp > now && coupon.nbf < now\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { exists } from '@xylabs/exists'\nimport type { HashLeaseEstimate } from '@xyo-network/diviner-hash-lease'\nimport type {\n AmountFields,\n Coupon, Discount, FixedAmountCoupon,\n FixedPercentageCoupon,\n FixedPriceCoupon,\n} from '@xyo-network/payment-payload-plugins'\nimport {\n DiscountSchema, isFixedAmountCoupon, isFixedPercentageCoupon,\n isFixedPriceCoupon,\n isStackable,\n} from '@xyo-network/payment-payload-plugins'\n\nexport const applyCoupons = (appraisals: HashLeaseEstimate[], coupons: Coupon[]): Discount => {\n // Ensure all appraisals and coupons are in USD\n const allAppraisalsAreUSD = appraisals.every(appraisal => appraisal.currency === 'USD')\n assertEx(allAppraisalsAreUSD, 'All appraisals must be in USD')\n const allCouponsAreUSD = coupons.map(coupon => (coupon as Partial<AmountFields>)?.currency).filter(exists).every(currency => currency === 'USD')\n assertEx(allCouponsAreUSD, 'All coupons must be in USD')\n const total = appraisals.reduce((acc, appraisal) => acc + appraisal.price, 0)\n\n // Calculated non-stackable discount coupons\n const singularFixedDiscount = Math.max(...coupons\n .filter(coupon => isFixedAmountCoupon(coupon) && !isStackable(coupon))\n .map(coupon => (coupon as FixedAmountCoupon).amount), 0)\n const singularPercentageDiscount = (Math.max(...coupons\n .filter(coupon => isFixedPercentageCoupon(coupon) && !isStackable(coupon))\n .map(coupon => (coupon as FixedPercentageCoupon).percentage), 0)) * total\n const singularFixedPriceDiscount = calculateSingularFixedPriceDiscount(total, appraisals, coupons, false)\n\n // Calculate stackable discount coupons\n // First calculate the total discount from fixed amount coupons\n const stackedFixedDiscount = coupons\n .filter(coupon => isFixedAmountCoupon(coupon) && isStackable(coupon))\n .reduce((acc, coupon) => acc + (coupon as FixedAmountCoupon).amount, 0)\n // Then calculate the total discount from percentage coupons and apply\n // the percentage discount to the remaining total after fixed discounts\n const stackedPercentageDiscount = coupons\n .filter(coupon => isFixedPercentageCoupon(coupon) && isStackable(coupon))\n .reduce((acc, coupon) => acc + (coupon as FixedPercentageCoupon).percentage, 0) * (total - stackedFixedDiscount)\n // Then calculate the total discount from fixed price coupons\n const stackedFixedPriceDiscount = calculateSingularFixedPriceDiscount(total, appraisals, coupons, true)\n\n // Sum all stackable discounts\n const stackedDiscount = stackedFixedDiscount + stackedPercentageDiscount + stackedFixedPriceDiscount\n\n // Find the best coupon(s) to apply\n const maxDiscount = Math.max(\n singularFixedDiscount,\n singularFixedPriceDiscount,\n singularPercentageDiscount,\n stackedDiscount,\n 0,\n )\n\n // Ensure discount is not more than the total\n const amount = Math.min(maxDiscount, total)\n\n // Return single discount payload\n return {\n amount, schema: DiscountSchema, currency: 'USD',\n }\n}\n\nconst calculateSingularFixedPriceDiscount = (total: number, appraisals: HashLeaseEstimate[], coupons: Coupon[], stackable = false): number => {\n // Find all singular fixed price discounts\n const singularFixedPriceDiscounts = coupons\n .filter(isFixedPriceCoupon)\n .filter(coupon => stackable ? isStackable(coupon) : !isStackable(coupon))\n .map(coupon => (coupon as FixedPriceCoupon).amount)\n // Ensure all fixed price discounts are positive\n .filter(amount => amount > 0)\n\n // If there are no singular fixed price discounts, return no discount\n if (singularFixedPriceDiscounts.length === 0) return 0\n\n // Find the maximum discount (the lowest fixed price)\n const lowestFixedPrice = Math.min(...singularFixedPriceDiscounts)\n\n // Apply the fixed price to all appraisals to get the reduced prices\n const reducedPrices = appraisals.map(appraisal =>\n // If the appraisal price is less than the fixed price\n Math.min(appraisal.price, lowestFixedPrice))\n\n // Calculate the reduced total using the reduced prices\n const reducedTotal = reducedPrices.reduce((acc, price) => acc + price, 0)\n\n // Calculate the discount\n const discount = total - reducedTotal\n\n // Return the discount or 0 if the discount would have resulted in a negative value\n return Math.max(discount, 0)\n}\n","import type { Hash } from '@xylabs/hex'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type { Payload } from '@xyo-network/payload-model'\nimport {\n type Condition, type Coupon, isCondition,\n} from '@xyo-network/payment-payload-plugins'\nimport { isSchemaPayload } from '@xyo-network/schema-payload-plugin'\nimport type { ValidateFunction } from 'ajv'\nimport { Ajv } from 'ajv'\n\n// TODO: Use our schema cache\nconst ajv = new Ajv({ strict: false }) // Create the Ajv instance once\nconst schemaCache = new Map() // Cache to store compiled validators\n\n/**\n * Validates the conditions of a coupon against the provided payloads\n * @param coupon The coupon to check\n * @param conditions The conditions associated with the coupon\n * @param payloads The associated payloads (containing the conditions and data to validate the conditions against)\n * @returns True if all conditions are fulfilled, false otherwise\n */\nexport const areConditionsFulfilled = async (coupon: Coupon, conditions: Condition[] = [], payloads: Payload[] = []): Promise<boolean> =>\n (await findUnfulfilledConditions(coupon, conditions, payloads)).length === 0\n\n/**\n * Validates the conditions of a coupon against the provided payloads\n * @param coupon The coupon to check\n * @param conditions The conditions associated with the coupon\n * @param payloads The associated payloads (containing the conditions and data to validate the conditions against)\n * @returns The unfulfilled condition hashes\n */\nexport const findUnfulfilledConditions = async (coupon: Coupon, conditions: Condition[] = [], payloads: Payload[] = []): Promise<Hash[]> => {\n const unfulfilledConditions: Hash[] = []\n // If there are no conditions, then they are fulfilled\n if (!coupon.conditions || coupon.conditions.length === 0) return unfulfilledConditions\n const hashMap = await PayloadBuilder.toAllHashMap([...conditions, ...payloads])\n // Find all the conditions\n const foundConditions = coupon.conditions.map(hash => hashMap[hash]).filter(isCondition)\n // Not all conditions were found\n if (foundConditions.length !== coupon.conditions.length) {\n const missing = coupon.conditions.filter(hash => !hashMap[hash])\n unfulfilledConditions.push(...missing)\n return unfulfilledConditions\n }\n\n // Test each condition\n for (const hash of coupon.conditions) {\n let validator: ValidateFunction\n\n // Check if the schema is already cached\n if (schemaCache.has(hash)) {\n validator = schemaCache.get(hash)\n } else {\n const payload = hashMap[hash]\n const definition = isSchemaPayload(payload) ? payload.definition : undefined\n if (definition) {\n // Compile and cache the validator\n validator = ajv.compile(definition)\n schemaCache.set(hash, validator)\n\n // Validate the payload\n } else {\n unfulfilledConditions.push(hash)\n continue\n }\n }\n if (!validator(payloads)) unfulfilledConditions.push(hash)\n }\n\n // All conditions passed\n return unfulfilledConditions\n}\n","import type { Hash } from '@xylabs/hex'\nimport type { DivinerInstance } from '@xyo-network/diviner-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type { Payload } from '@xyo-network/payload-model'\nimport type {\n Discount, EscrowTerms, Invoice, Payment, Subtotal, Total,\n} from '@xyo-network/payment-payload-plugins'\nimport {\n isDiscount, isSubtotal, isTotal, PaymentSchema,\n} from '@xyo-network/payment-payload-plugins'\n\n/**\n * Validates the escrow terms to ensure they are valid for a purchase\n * @returns A payment if the terms are valid for a purchase, undefined otherwise\n */\nexport const getInvoiceForEscrow = async (\n terms: EscrowTerms,\n dataHashMap: Record<Hash, Payload>,\n paymentTotalDiviner: DivinerInstance,\n): Promise<Invoice | undefined> => {\n const payloads = Object.values(dataHashMap)\n const results = await paymentTotalDiviner.divine([terms, ...payloads])\n const subtotal = results.find(isSubtotal) as Subtotal | undefined\n const discount = results.find(isDiscount) as Discount | undefined\n const total = results.find(isTotal) as Total | undefined\n if (!subtotal || !total) return undefined\n const { amount, currency } = total\n if (currency !== 'USD') return undefined\n const sources = await getSources(terms, subtotal, total, discount)\n const payment: Payment = {\n amount, currency, schema: PaymentSchema, sources,\n }\n return discount ? [subtotal, total, payment, discount] : [subtotal, total, payment]\n}\n\nconst getSources = async (terms: EscrowTerms, subtotal: Subtotal, total: Total, discount?: Discount): Promise<Hash[]> => {\n const sources = discount ? [terms, subtotal, total, discount] : [terms, subtotal, total]\n return await Promise.all(sources.map(p => PayloadBuilder.dataHash(p)))\n}\n","import { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { HashLeaseEstimate, isHashLeaseEstimate } from '@xyo-network/diviner-hash-lease'\nimport { DivinerInstance, DivinerModuleEventData } from '@xyo-network/diviner-model'\nimport { creatableModule } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload } from '@xyo-network/payload-model'\nimport {\n EscrowTerms, isEscrowTerms, PaymentSubtotalDivinerConfigSchema, PaymentSubtotalDivinerParams, Subtotal, SubtotalSchema,\n} from '@xyo-network/payment-payload-plugins'\n\nimport {\n appraisalValidators, termsValidators, ValidEscrowTerms,\n} from './lib/index.ts'\n\nconst currency = 'USD'\n\n/**\n * Escrow terms that contain all the valid fields for calculating a subtotal\n */\nexport type PaymentSubtotalDivinerInputType = EscrowTerms | HashLeaseEstimate | Payload\n\n@creatableModule()\nexport class PaymentSubtotalDiviner<\n TParams extends PaymentSubtotalDivinerParams = PaymentSubtotalDivinerParams,\n TIn extends PaymentSubtotalDivinerInputType = PaymentSubtotalDivinerInputType,\n TOut extends Subtotal = Subtotal,\n TEventData extends DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut> = DivinerModuleEventData<\n DivinerInstance<TParams, TIn, TOut>,\n TIn,\n TOut\n >,\n> extends AbstractDiviner<TParams, TIn, TOut, TEventData> {\n static override configSchemas = [PaymentSubtotalDivinerConfigSchema]\n static override defaultConfigSchema = PaymentSubtotalDivinerConfigSchema\n\n protected async divineHandler(payloads: TIn[] = []): Promise<TOut[]> {\n // Find the escrow terms\n const terms = payloads.find(isEscrowTerms) as EscrowTerms | undefined\n if (!terms) return []\n\n // Run all terms validations\n if (!termsValidators.every(validator => validator(terms))) return []\n const validTerms = terms as ValidEscrowTerms\n\n // Retrieve all appraisals from terms\n const hashMap = await PayloadBuilder.toAllHashMap(payloads)\n const appraisals = validTerms.appraisals.map(appraisal => hashMap[appraisal]).filter(isHashLeaseEstimate) as unknown as HashLeaseEstimate[]\n\n // Ensure all appraisals are present\n if (appraisals.length !== validTerms.appraisals.length) return []\n\n // Run all appraisal validations\n if (!appraisalValidators.every(validator => validator(appraisals))) return []\n const amount = calculateSubtotal(appraisals)\n const sources = [await PayloadBuilder.dataHash(validTerms), ...validTerms.appraisals]\n return [{\n amount, currency, schema: SubtotalSchema, sources,\n }] as TOut[]\n }\n}\n\n// TODO: Add support for other currencies\nconst calculateSubtotal = (appraisals: HashLeaseEstimate[]): number => {\n return appraisals.reduce((sum, appraisal) => sum + appraisal.price, 0)\n}\n","import type { HashLeaseEstimate } from '@xyo-network/diviner-hash-lease'\nimport { isIso4217CurrencyCode } from '@xyo-network/payment-payload-plugins'\n\nimport { validateDuration } from './durationValidators.ts'\n\nconst validateAppraisalAmount = (appraisals: HashLeaseEstimate[]): boolean => {\n // Ensure all appraisals are numeric\n if (appraisals.some(appraisal => typeof appraisal.price !== 'number')) return false\n // Ensure all appraisals are positive numbers\n if (appraisals.some(appraisal => appraisal.price < 0)) return false\n return true\n}\n\nconst validateAppraisalCurrency = (appraisals: HashLeaseEstimate[]): boolean => {\n // NOTE: Only supporting USD for now, the remaining checks are for future-proofing.\n if (!appraisals.every(appraisal => appraisal.currency == 'USD')) return false\n\n // Check every object in the array to ensure they all are in a supported currency.\n if (!appraisals.every(appraisal => isIso4217CurrencyCode(appraisal.currency))) return false\n\n return true\n}\n\nconst validateAppraisalConsistentCurrency = (appraisals: HashLeaseEstimate[]): boolean => {\n // Check if the array is empty or contains only one element, no need to compare.\n if (appraisals.length <= 1) return true\n\n // Get the currency of the first element to compare with others.\n const { currency } = appraisals[0]\n if (!currency) return false\n\n // Check every object in the array to ensure they all have the same currency.\n if (!appraisals.every(item => item.currency === currency)) return false\n\n return true\n}\n\nconst validateAppraisalWindow = (appraisals: HashLeaseEstimate[]): boolean => appraisals.every(validateDuration)\n\nexport const appraisalValidators = [\n validateAppraisalAmount,\n validateAppraisalCurrency,\n validateAppraisalConsistentCurrency,\n validateAppraisalWindow,\n]\n","import type { DurationFields } from '@xyo-network/xns-record-payload-plugins'\n\nconst FIVE_MINUTES = 1000 * 60 * 5\n\n/**\n * Validates that the current time is within the duration window, within a configurable a buffer\n * @param value The duration value\n * @param windowMs The window in milliseconds to allow for a buffer\n * @returns True if the duration is valid, false otherwise\n */\nexport const validateDuration = (value: Partial<DurationFields>, windowMs = FIVE_MINUTES): boolean => {\n const now = Date.now()\n if (!value.nbf || value.nbf > now) return false\n // If already expired (include for a 5 minute buffer to allow for a reasonable\n // minimum amount of time for the transaction to be processed)\n if (!value.exp || value.exp - now < windowMs) return false\n return true\n}\n","import type { Hash } from '@xylabs/hex'\nimport type { EscrowTerms } from '@xyo-network/payment-payload-plugins'\n\nimport { validateDuration } from './durationValidators.ts'\n\nexport type ValidEscrowTerms = Required<EscrowTerms>\n\nconst validateTermsAppraisals = (terms: EscrowTerms): terms is Required<EscrowTerms & { appraisals: Hash[] }> => {\n if (!terms.appraisals) return false\n if (terms.appraisals.length === 0) return false\n return true\n}\nconst validateTermsWindow = (terms: EscrowTerms): boolean => validateDuration(terms)\n\nexport const termsValidators = [\n validateTermsAppraisals,\n validateTermsWindow,\n]\n","import { assertEx } from '@xylabs/assert'\nimport { Hash } from '@xylabs/hex'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport {\n asDivinerInstance, DivinerInstance, DivinerModuleEventData,\n} from '@xyo-network/diviner-model'\nimport { creatableModule } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport {\n Discount,\n isDiscount,\n isSubtotal,\n PaymentTotalDivinerConfigSchema, PaymentTotalDivinerParams, Subtotal, Total, TotalSchema,\n} from '@xyo-network/payment-payload-plugins'\n\nimport { PaymentDiscountDiviner, PaymentDiscountDivinerInputType } from '../Discount/index.ts'\nimport { PaymentSubtotalDiviner, PaymentSubtotalDivinerInputType } from '../Subtotal/index.ts'\n\ntype InputType = PaymentDiscountDivinerInputType | PaymentSubtotalDivinerInputType\ntype OutputType = Subtotal | Discount | Total\n\n@creatableModule()\nexport class PaymentTotalDiviner<\n TParams extends PaymentTotalDivinerParams = PaymentTotalDivinerParams,\n TIn extends InputType = InputType,\n TOut extends OutputType = OutputType,\n TEventData extends DivinerModuleEventData<DivinerInstance<TParams, TIn, TOut>, TIn, TOut> = DivinerModuleEventData<\n DivinerInstance<TParams, TIn, TOut>,\n TIn,\n TOut\n >,\n> extends AbstractDiviner<TParams, TIn, TOut, TEventData> {\n static override configSchemas = [PaymentTotalDivinerConfigSchema]\n static override defaultConfigSchema: PaymentTotalDivinerConfigSchema = PaymentTotalDivinerConfigSchema\n\n protected async divineHandler(payloads: TIn[] = []): Promise<TOut[]> {\n const subtotalDiviner = await this.getPaymentSubtotalDiviner()\n const subtotalResult = await subtotalDiviner.divine(payloads)\n const subtotal = subtotalResult.find(isSubtotal)\n if (!subtotal) return []\n const discountDiviner = await this.getPaymentDiscountsDiviner()\n const discountResult = await discountDiviner.divine(payloads)\n const discount = discountResult.find(isDiscount)\n if (!discount) return []\n const { currency: subtotalCurrency } = subtotal\n const { currency: discountCurrency } = discount\n assertEx(subtotalCurrency === discountCurrency, () => `Subtotal currency ${subtotalCurrency} does not match discount currency ${discountCurrency}`)\n const amount = Math.max(0, subtotal.amount - discount.amount)\n const currency = subtotalCurrency\n const sources = [await PayloadBuilder.dataHash(subtotal), await PayloadBuilder.dataHash(discount)] as Hash[]\n const total: Total = {\n amount, currency, sources, schema: TotalSchema,\n }\n return [subtotal, discount, total] as TOut[]\n }\n\n protected async getPaymentDiscountsDiviner(): Promise<PaymentDiscountDiviner> {\n const name = assertEx(this.config.paymentDiscountDiviner, () => 'Missing paymentDiscountDiviner in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving paymentDiscountDiviner: ${name}`)\n return assertEx(asDivinerInstance(mod), () => `Resolved module ${mod.address} not a valid Diviner`) as PaymentDiscountDiviner\n }\n\n protected async getPaymentSubtotalDiviner(): Promise<PaymentSubtotalDiviner> {\n const name = assertEx(this.config.paymentSubtotalDiviner, () => 'Missing paymentSubtotalDiviner in config')\n const mod = assertEx(await this.resolve(name), () => `Error resolving paymentSubtotalDiviner: ${name}`)\n return assertEx(asDivinerInstance(mod), () => `Resolved module ${mod.address} not a valid Diviner`) as PaymentSubtotalDiviner\n }\n}\n"],"mappings":"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","names":["assertEx","exists","PayloadBuilder","isCondition","currency","PayloadBuilder","exists","assertEx","isCondition","PayloadBuilder","currency","AbstractDiviner","isHashLeaseEstimate","creatableModule","PayloadBuilder","isEscrowTerms","currency","AbstractDiviner","isEscrowTerms","PayloadBuilder","isHashLeaseEstimate","creatableModule","assertEx","AbstractDiviner","asDivinerInstance","creatableModule","PayloadBuilder","isDiscount","isSubtotal","AbstractDiviner","isSubtotal","isDiscount","assertEx","currency","PayloadBuilder","asDivinerInstance","creatableModule"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@xyo-network/payment-plugin",
3
- "version": "3.1.1",
3
+ "version": "3.2.0-rc.1",
4
4
  "description": "Typescript/Javascript Plugins for XYO Platform",
5
5
  "homepage": "https://xyo.network",
6
6
  "bugs": {
@@ -29,36 +29,37 @@
29
29
  "module": "dist/neutral/index.mjs",
30
30
  "types": "dist/neutral/index.d.ts",
31
31
  "dependencies": {
32
- "@xylabs/assert": "^4.1.0",
33
- "@xylabs/exists": "^4.1.0",
34
- "@xylabs/hex": "^4.1.0",
35
- "@xyo-network/archivist-model": "^3.2.0",
36
- "@xyo-network/diviner-abstract": "^3.2.0",
37
- "@xyo-network/diviner-boundwitness-model": "^3.2.0",
38
- "@xyo-network/diviner-hash-lease": "^3.2.0",
39
- "@xyo-network/diviner-model": "^3.2.0",
40
- "@xyo-network/module-model": "^3.2.0",
41
- "@xyo-network/payload-builder": "^3.2.0",
42
- "@xyo-network/payload-model": "^3.2.0",
43
- "@xyo-network/payment-payload-plugins": "^3.1.1",
44
- "@xyo-network/schema-payload-plugin": "^3.2.0",
45
- "@xyo-network/xns-record-payload-plugins": "^3.1.1",
32
+ "@xylabs/assert": "^4.4.12",
33
+ "@xylabs/exists": "^4.4.12",
34
+ "@xylabs/hex": "^4.4.12",
35
+ "@xyo-network/archivist-model": "^3.6.0-rc.1",
36
+ "@xyo-network/diviner-abstract": "^3.6.0-rc.1",
37
+ "@xyo-network/diviner-boundwitness-model": "^3.6.0-rc.1",
38
+ "@xyo-network/diviner-hash-lease": "^3.6.0-rc.1",
39
+ "@xyo-network/diviner-model": "^3.6.0-rc.1",
40
+ "@xyo-network/module-model": "^3.6.0-rc.1",
41
+ "@xyo-network/payload-builder": "^3.6.0-rc.1",
42
+ "@xyo-network/payload-model": "^3.6.0-rc.1",
43
+ "@xyo-network/payment-payload-plugins": "^3.2.0-rc.1",
44
+ "@xyo-network/schema-payload-plugin": "^3.6.0-rc.1",
45
+ "@xyo-network/xns-record-payload-plugins": "^3.2.0-rc.1",
46
46
  "ajv": "^8.17.1"
47
47
  },
48
48
  "devDependencies": {
49
- "@xylabs/ts-scripts-yarn3": "^4.1.0",
50
- "@xylabs/tsconfig": "^4.1.0",
51
- "@xyo-network/account": "^3.2.0",
52
- "@xyo-network/archivist-memory": "^3.2.0",
53
- "@xyo-network/boundwitness-builder": "^3.2.0",
54
- "@xyo-network/diviner-boundwitness-memory": "^3.2.0",
55
- "@xyo-network/id-payload-plugin": "^3.2.0",
56
- "@xyo-network/node-memory": "^3.2.0",
57
- "jest": "^29.7.0",
58
- "typescript": "^5.6.3",
59
- "vitest": "^2.1.2"
49
+ "@xylabs/ts-scripts-yarn3": "^4.2.4",
50
+ "@xylabs/tsconfig": "^4.2.4",
51
+ "@xylabs/vitest-extended": "^4.4.12",
52
+ "@xyo-network/account": "^3.6.0-rc.1",
53
+ "@xyo-network/archivist-memory": "^3.6.0-rc.1",
54
+ "@xyo-network/boundwitness-builder": "^3.6.0-rc.1",
55
+ "@xyo-network/diviner-boundwitness-memory": "^3.6.0-rc.1",
56
+ "@xyo-network/id-payload-plugin": "^3.6.0-rc.1",
57
+ "@xyo-network/node-memory": "^3.6.0-rc.1",
58
+ "typescript": "^5.7.2",
59
+ "vitest": "^2.1.8"
60
60
  },
61
61
  "publishConfig": {
62
62
  "access": "public"
63
- }
63
+ },
64
+ "stableVersion": "3.1.2"
64
65
  }
@@ -18,8 +18,7 @@ import {
18
18
  Condition,
19
19
  Coupon,
20
20
  Discount,
21
- EscrowTerms, isConditionWithMeta, isCoupon,
22
- isCouponWithMeta,
21
+ EscrowTerms, isCondition, isCoupon,
23
22
  isEscrowTerms, NO_DISCOUNT, PaymentDiscountDivinerConfigSchema, PaymentDiscountDivinerParams,
24
23
  } from '@xyo-network/payment-payload-plugins'
25
24
 
@@ -161,7 +160,7 @@ export class PaymentDiscountDiviner<
161
160
  // Find any remaining from discounts archivist
162
161
  const discountsArchivist = await this.getDiscountsArchivist()
163
162
  const payloads = await discountsArchivist.get(missingDiscounts)
164
- discounts.push(...payloads.filter(isCouponWithMeta))
163
+ discounts.push(...payloads.filter(isCoupon))
165
164
  }
166
165
  // If not all discounts are found
167
166
  if (discounts.length !== discountsHashes.length) {
@@ -174,7 +173,7 @@ export class PaymentDiscountDiviner<
174
173
  }
175
174
 
176
175
  const conditionsHashes: Hash[] = discounts.flatMap(discount => discount.conditions ?? [])
177
- const conditions: Condition[] = conditionsHashes.map(hash => hashMap[hash]).filter(exists).filter(isConditionWithMeta)
176
+ const conditions: Condition[] = conditionsHashes.map(hash => hashMap[hash]).filter(exists).filter(isCondition)
178
177
  const missingConditions = conditionsHashes.filter(hash => !hashMap[hash])
179
178
 
180
179
  // If not all conditions are found
@@ -182,7 +181,7 @@ export class PaymentDiscountDiviner<
182
181
  // Find any remaining from discounts archivist
183
182
  const discountsArchivist = await this.getDiscountsArchivist()
184
183
  const payloads = await discountsArchivist.get(missingConditions)
185
- conditions.push(...payloads.filter(isConditionWithMeta))
184
+ conditions.push(...payloads.filter(isCondition))
186
185
  }
187
186
  // If not all conditions are found
188
187
  if (conditions.length !== conditionsHashes.length) {
@@ -4,7 +4,7 @@ import type { Payload } from '@xyo-network/payload-model'
4
4
  import {
5
5
  type Condition, type Coupon, isCondition,
6
6
  } from '@xyo-network/payment-payload-plugins'
7
- import { isSchemaPayloadWithMeta } from '@xyo-network/schema-payload-plugin'
7
+ import { isSchemaPayload } from '@xyo-network/schema-payload-plugin'
8
8
  import type { ValidateFunction } from 'ajv'
9
9
  import { Ajv } from 'ajv'
10
10
 
@@ -52,7 +52,7 @@ export const findUnfulfilledConditions = async (coupon: Coupon, conditions: Cond
52
52
  validator = schemaCache.get(hash)
53
53
  } else {
54
54
  const payload = hashMap[hash]
55
- const definition = isSchemaPayloadWithMeta(payload) ? payload.definition : undefined
55
+ const definition = isSchemaPayload(payload) ? payload.definition : undefined
56
56
  if (definition) {
57
57
  // Compile and cache the validator
58
58
  validator = ajv.compile(definition)
@@ -5,10 +5,11 @@ import {
5
5
  asDivinerInstance, DivinerInstance, DivinerModuleEventData,
6
6
  } from '@xyo-network/diviner-model'
7
7
  import { creatableModule } from '@xyo-network/module-model'
8
+ import { PayloadBuilder } from '@xyo-network/payload-builder'
8
9
  import {
9
10
  Discount,
10
- isDiscountWithMeta,
11
- isSubtotalWithMeta,
11
+ isDiscount,
12
+ isSubtotal,
12
13
  PaymentTotalDivinerConfigSchema, PaymentTotalDivinerParams, Subtotal, Total, TotalSchema,
13
14
  } from '@xyo-network/payment-payload-plugins'
14
15
 
@@ -35,18 +36,18 @@ export class PaymentTotalDiviner<
35
36
  protected async divineHandler(payloads: TIn[] = []): Promise<TOut[]> {
36
37
  const subtotalDiviner = await this.getPaymentSubtotalDiviner()
37
38
  const subtotalResult = await subtotalDiviner.divine(payloads)
38
- const subtotal = subtotalResult.find(isSubtotalWithMeta)
39
+ const subtotal = subtotalResult.find(isSubtotal)
39
40
  if (!subtotal) return []
40
41
  const discountDiviner = await this.getPaymentDiscountsDiviner()
41
42
  const discountResult = await discountDiviner.divine(payloads)
42
- const discount = discountResult.find(isDiscountWithMeta)
43
+ const discount = discountResult.find(isDiscount)
43
44
  if (!discount) return []
44
45
  const { currency: subtotalCurrency } = subtotal
45
46
  const { currency: discountCurrency } = discount
46
47
  assertEx(subtotalCurrency === discountCurrency, () => `Subtotal currency ${subtotalCurrency} does not match discount currency ${discountCurrency}`)
47
48
  const amount = Math.max(0, subtotal.amount - discount.amount)
48
49
  const currency = subtotalCurrency
49
- const sources = [subtotal.$hash, discount.$hash] as Hash[]
50
+ const sources = [await PayloadBuilder.dataHash(subtotal), await PayloadBuilder.dataHash(discount)] as Hash[]
50
51
  const total: Total = {
51
52
  amount, currency, sources, schema: TotalSchema,
52
53
  }