envoy-pay 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +99 -0
- package/LICENSE +200 -0
- package/README.md +529 -0
- package/dist/cjs/adapters/base.js +174 -0
- package/dist/cjs/adapters/base.js.map +1 -0
- package/dist/cjs/adapters/evm.js +339 -0
- package/dist/cjs/adapters/evm.js.map +1 -0
- package/dist/cjs/adapters/index.js +15 -0
- package/dist/cjs/adapters/index.js.map +1 -0
- package/dist/cjs/adapters/ows.js +204 -0
- package/dist/cjs/adapters/ows.js.map +1 -0
- package/dist/cjs/adapters/solana.js +210 -0
- package/dist/cjs/adapters/solana.js.map +1 -0
- package/dist/cjs/adapters/stellar.js +173 -0
- package/dist/cjs/adapters/stellar.js.map +1 -0
- package/dist/cjs/adapters/stripe.js +338 -0
- package/dist/cjs/adapters/stripe.js.map +1 -0
- package/dist/cjs/adapters/types.js +3 -0
- package/dist/cjs/adapters/types.js.map +1 -0
- package/dist/cjs/client.js +309 -0
- package/dist/cjs/client.js.map +1 -0
- package/dist/cjs/contracts/abis/EnvoyFacilitator.js +197 -0
- package/dist/cjs/contracts/abis/EnvoyFacilitator.js.map +1 -0
- package/dist/cjs/contracts/addresses.js +42 -0
- package/dist/cjs/contracts/addresses.js.map +1 -0
- package/dist/cjs/contracts/facilitator.js +225 -0
- package/dist/cjs/contracts/facilitator.js.map +1 -0
- package/dist/cjs/contracts/index.js +21 -0
- package/dist/cjs/contracts/index.js.map +1 -0
- package/dist/cjs/facilitator/facilitator-service.js +305 -0
- package/dist/cjs/facilitator/facilitator-service.js.map +1 -0
- package/dist/cjs/facilitator/fee-calculator.js +134 -0
- package/dist/cjs/facilitator/fee-calculator.js.map +1 -0
- package/dist/cjs/facilitator/index.js +12 -0
- package/dist/cjs/facilitator/index.js.map +1 -0
- package/dist/cjs/facilitator/types.js +99 -0
- package/dist/cjs/facilitator/types.js.map +1 -0
- package/dist/cjs/identity/agent-card.js +143 -0
- package/dist/cjs/identity/agent-card.js.map +1 -0
- package/dist/cjs/identity/agent-identity.js +166 -0
- package/dist/cjs/identity/agent-identity.js.map +1 -0
- package/dist/cjs/identity/did-resolver.js +149 -0
- package/dist/cjs/identity/did-resolver.js.map +1 -0
- package/dist/cjs/identity/erc8004/abis.js +218 -0
- package/dist/cjs/identity/erc8004/abis.js.map +1 -0
- package/dist/cjs/identity/erc8004/identity.js +268 -0
- package/dist/cjs/identity/erc8004/identity.js.map +1 -0
- package/dist/cjs/identity/erc8004/index.js +36 -0
- package/dist/cjs/identity/erc8004/index.js.map +1 -0
- package/dist/cjs/identity/erc8004/reputation.js +110 -0
- package/dist/cjs/identity/erc8004/reputation.js.map +1 -0
- package/dist/cjs/identity/erc8004/types.js +3 -0
- package/dist/cjs/identity/erc8004/types.js.map +1 -0
- package/dist/cjs/identity/index.js +61 -0
- package/dist/cjs/identity/index.js.map +1 -0
- package/dist/cjs/identity/owner-registry.js +140 -0
- package/dist/cjs/identity/owner-registry.js.map +1 -0
- package/dist/cjs/identity/reputation.js +193 -0
- package/dist/cjs/identity/reputation.js.map +1 -0
- package/dist/cjs/identity/types.js +16 -0
- package/dist/cjs/identity/types.js.map +1 -0
- package/dist/cjs/index.js +72 -0
- package/dist/cjs/index.js.map +1 -0
- package/dist/cjs/logger.js +7 -0
- package/dist/cjs/logger.js.map +1 -0
- package/dist/cjs/monitor/evm-watcher.js +165 -0
- package/dist/cjs/monitor/evm-watcher.js.map +1 -0
- package/dist/cjs/monitor/index.js +32 -0
- package/dist/cjs/monitor/index.js.map +1 -0
- package/dist/cjs/monitor/multi-watcher.js +94 -0
- package/dist/cjs/monitor/multi-watcher.js.map +1 -0
- package/dist/cjs/monitor/solana-watcher.js +172 -0
- package/dist/cjs/monitor/solana-watcher.js.map +1 -0
- package/dist/cjs/monitor/stellar-watcher.js +125 -0
- package/dist/cjs/monitor/stellar-watcher.js.map +1 -0
- package/dist/cjs/mpp.js +187 -0
- package/dist/cjs/mpp.js.map +1 -0
- package/dist/cjs/policy.js +71 -0
- package/dist/cjs/policy.js.map +1 -0
- package/dist/cjs/providers/bridge.js +323 -0
- package/dist/cjs/providers/bridge.js.map +1 -0
- package/dist/cjs/providers/onchainos.js +210 -0
- package/dist/cjs/providers/onchainos.js.map +1 -0
- package/dist/cjs/requests/eip681.js +77 -0
- package/dist/cjs/requests/eip681.js.map +1 -0
- package/dist/cjs/requests/index.js +18 -0
- package/dist/cjs/requests/index.js.map +1 -0
- package/dist/cjs/requests/sep7.js +61 -0
- package/dist/cjs/requests/sep7.js.map +1 -0
- package/dist/cjs/requests/solana-pay.js +51 -0
- package/dist/cjs/requests/solana-pay.js.map +1 -0
- package/dist/cjs/requests/universal.js +40 -0
- package/dist/cjs/requests/universal.js.map +1 -0
- package/dist/cjs/server/index.js +35 -0
- package/dist/cjs/server/index.js.map +1 -0
- package/dist/cjs/server/mpp-gate.js +118 -0
- package/dist/cjs/server/mpp-gate.js.map +1 -0
- package/dist/cjs/server/payment-gate.js +92 -0
- package/dist/cjs/server/payment-gate.js.map +1 -0
- package/dist/cjs/server/receipt.js +52 -0
- package/dist/cjs/server/receipt.js.map +1 -0
- package/dist/cjs/server/webhook.js +122 -0
- package/dist/cjs/server/webhook.js.map +1 -0
- package/dist/cjs/server/x402-gate.js +103 -0
- package/dist/cjs/server/x402-gate.js.map +1 -0
- package/dist/cjs/solana.js +14 -0
- package/dist/cjs/solana.js.map +1 -0
- package/dist/cjs/stellar.js +14 -0
- package/dist/cjs/stellar.js.map +1 -0
- package/dist/cjs/wallet/balance-aggregator.js +149 -0
- package/dist/cjs/wallet/balance-aggregator.js.map +1 -0
- package/dist/cjs/wallet/chain-router.js +254 -0
- package/dist/cjs/wallet/chain-router.js.map +1 -0
- package/dist/cjs/wallet/index.js +15 -0
- package/dist/cjs/wallet/index.js.map +1 -0
- package/dist/cjs/wallet/intent-resolver.js +145 -0
- package/dist/cjs/wallet/intent-resolver.js.map +1 -0
- package/dist/cjs/wallet/session-manager.js +170 -0
- package/dist/cjs/wallet/session-manager.js.map +1 -0
- package/dist/cjs/wallet/types.js +14 -0
- package/dist/cjs/wallet/types.js.map +1 -0
- package/dist/cjs/wallet/unified-wallet.js +253 -0
- package/dist/cjs/wallet/unified-wallet.js.map +1 -0
- package/dist/esm/adapters/base.js +170 -0
- package/dist/esm/adapters/base.js.map +1 -0
- package/dist/esm/adapters/evm.js +334 -0
- package/dist/esm/adapters/evm.js.map +1 -0
- package/dist/esm/adapters/index.js +6 -0
- package/dist/esm/adapters/index.js.map +1 -0
- package/dist/esm/adapters/ows.js +194 -0
- package/dist/esm/adapters/ows.js.map +1 -0
- package/dist/esm/adapters/solana.js +206 -0
- package/dist/esm/adapters/solana.js.map +1 -0
- package/dist/esm/adapters/stellar.js +136 -0
- package/dist/esm/adapters/stellar.js.map +1 -0
- package/dist/esm/adapters/stripe.js +334 -0
- package/dist/esm/adapters/stripe.js.map +1 -0
- package/dist/esm/adapters/types.js +2 -0
- package/dist/esm/adapters/types.js.map +1 -0
- package/dist/esm/client.js +302 -0
- package/dist/esm/client.js.map +1 -0
- package/dist/esm/contracts/abis/EnvoyFacilitator.js +194 -0
- package/dist/esm/contracts/abis/EnvoyFacilitator.js.map +1 -0
- package/dist/esm/contracts/addresses.js +38 -0
- package/dist/esm/contracts/addresses.js.map +1 -0
- package/dist/esm/contracts/facilitator.js +218 -0
- package/dist/esm/contracts/facilitator.js.map +1 -0
- package/dist/esm/contracts/index.js +8 -0
- package/dist/esm/contracts/index.js.map +1 -0
- package/dist/esm/facilitator/facilitator-service.js +301 -0
- package/dist/esm/facilitator/facilitator-service.js.map +1 -0
- package/dist/esm/facilitator/fee-calculator.js +130 -0
- package/dist/esm/facilitator/fee-calculator.js.map +1 -0
- package/dist/esm/facilitator/index.js +5 -0
- package/dist/esm/facilitator/index.js.map +1 -0
- package/dist/esm/facilitator/types.js +96 -0
- package/dist/esm/facilitator/types.js.map +1 -0
- package/dist/esm/identity/agent-card.js +139 -0
- package/dist/esm/identity/agent-card.js.map +1 -0
- package/dist/esm/identity/agent-identity.js +162 -0
- package/dist/esm/identity/agent-identity.js.map +1 -0
- package/dist/esm/identity/did-resolver.js +145 -0
- package/dist/esm/identity/did-resolver.js.map +1 -0
- package/dist/esm/identity/erc8004/abis.js +215 -0
- package/dist/esm/identity/erc8004/abis.js.map +1 -0
- package/dist/esm/identity/erc8004/identity.js +250 -0
- package/dist/esm/identity/erc8004/identity.js.map +1 -0
- package/dist/esm/identity/erc8004/index.js +13 -0
- package/dist/esm/identity/erc8004/index.js.map +1 -0
- package/dist/esm/identity/erc8004/reputation.js +105 -0
- package/dist/esm/identity/erc8004/reputation.js.map +1 -0
- package/dist/esm/identity/erc8004/types.js +2 -0
- package/dist/esm/identity/erc8004/types.js.map +1 -0
- package/dist/esm/identity/index.js +19 -0
- package/dist/esm/identity/index.js.map +1 -0
- package/dist/esm/identity/owner-registry.js +136 -0
- package/dist/esm/identity/owner-registry.js.map +1 -0
- package/dist/esm/identity/reputation.js +189 -0
- package/dist/esm/identity/reputation.js.map +1 -0
- package/dist/esm/identity/types.js +15 -0
- package/dist/esm/identity/types.js.map +1 -0
- package/dist/esm/index.js +37 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/logger.js +3 -0
- package/dist/esm/logger.js.map +1 -0
- package/dist/esm/monitor/evm-watcher.js +162 -0
- package/dist/esm/monitor/evm-watcher.js.map +1 -0
- package/dist/esm/monitor/index.js +25 -0
- package/dist/esm/monitor/index.js.map +1 -0
- package/dist/esm/monitor/multi-watcher.js +91 -0
- package/dist/esm/monitor/multi-watcher.js.map +1 -0
- package/dist/esm/monitor/solana-watcher.js +169 -0
- package/dist/esm/monitor/solana-watcher.js.map +1 -0
- package/dist/esm/monitor/stellar-watcher.js +122 -0
- package/dist/esm/monitor/stellar-watcher.js.map +1 -0
- package/dist/esm/mpp.js +175 -0
- package/dist/esm/mpp.js.map +1 -0
- package/dist/esm/policy.js +67 -0
- package/dist/esm/policy.js.map +1 -0
- package/dist/esm/providers/bridge.js +319 -0
- package/dist/esm/providers/bridge.js.map +1 -0
- package/dist/esm/providers/onchainos.js +172 -0
- package/dist/esm/providers/onchainos.js.map +1 -0
- package/dist/esm/requests/eip681.js +74 -0
- package/dist/esm/requests/eip681.js.map +1 -0
- package/dist/esm/requests/index.js +11 -0
- package/dist/esm/requests/index.js.map +1 -0
- package/dist/esm/requests/sep7.js +58 -0
- package/dist/esm/requests/sep7.js.map +1 -0
- package/dist/esm/requests/solana-pay.js +48 -0
- package/dist/esm/requests/solana-pay.js.map +1 -0
- package/dist/esm/requests/universal.js +37 -0
- package/dist/esm/requests/universal.js.map +1 -0
- package/dist/esm/server/index.js +27 -0
- package/dist/esm/server/index.js.map +1 -0
- package/dist/esm/server/mpp-gate.js +115 -0
- package/dist/esm/server/mpp-gate.js.map +1 -0
- package/dist/esm/server/payment-gate.js +89 -0
- package/dist/esm/server/payment-gate.js.map +1 -0
- package/dist/esm/server/receipt.js +48 -0
- package/dist/esm/server/receipt.js.map +1 -0
- package/dist/esm/server/webhook.js +119 -0
- package/dist/esm/server/webhook.js.map +1 -0
- package/dist/esm/server/x402-gate.js +100 -0
- package/dist/esm/server/x402-gate.js.map +1 -0
- package/dist/esm/solana.js +8 -0
- package/dist/esm/solana.js.map +1 -0
- package/dist/esm/stellar.js +8 -0
- package/dist/esm/stellar.js.map +1 -0
- package/dist/esm/wallet/balance-aggregator.js +145 -0
- package/dist/esm/wallet/balance-aggregator.js.map +1 -0
- package/dist/esm/wallet/chain-router.js +250 -0
- package/dist/esm/wallet/chain-router.js.map +1 -0
- package/dist/esm/wallet/index.js +7 -0
- package/dist/esm/wallet/index.js.map +1 -0
- package/dist/esm/wallet/intent-resolver.js +141 -0
- package/dist/esm/wallet/intent-resolver.js.map +1 -0
- package/dist/esm/wallet/session-manager.js +166 -0
- package/dist/esm/wallet/session-manager.js.map +1 -0
- package/dist/esm/wallet/types.js +13 -0
- package/dist/esm/wallet/types.js.map +1 -0
- package/dist/esm/wallet/unified-wallet.js +249 -0
- package/dist/esm/wallet/unified-wallet.js.map +1 -0
- package/dist/types/adapters/base.d.ts +53 -0
- package/dist/types/adapters/base.d.ts.map +1 -0
- package/dist/types/adapters/evm.d.ts +93 -0
- package/dist/types/adapters/evm.d.ts.map +1 -0
- package/dist/types/adapters/index.d.ts +7 -0
- package/dist/types/adapters/index.d.ts.map +1 -0
- package/dist/types/adapters/ows.d.ts +123 -0
- package/dist/types/adapters/ows.d.ts.map +1 -0
- package/dist/types/adapters/solana.d.ts +77 -0
- package/dist/types/adapters/solana.d.ts.map +1 -0
- package/dist/types/adapters/stellar.d.ts +67 -0
- package/dist/types/adapters/stellar.d.ts.map +1 -0
- package/dist/types/adapters/stripe.d.ts +206 -0
- package/dist/types/adapters/stripe.d.ts.map +1 -0
- package/dist/types/adapters/types.d.ts +110 -0
- package/dist/types/adapters/types.d.ts.map +1 -0
- package/dist/types/client.d.ts +89 -0
- package/dist/types/client.d.ts.map +1 -0
- package/dist/types/contracts/abis/EnvoyFacilitator.d.ts +296 -0
- package/dist/types/contracts/abis/EnvoyFacilitator.d.ts.map +1 -0
- package/dist/types/contracts/addresses.d.ts +24 -0
- package/dist/types/contracts/addresses.d.ts.map +1 -0
- package/dist/types/contracts/facilitator.d.ts +172 -0
- package/dist/types/contracts/facilitator.d.ts.map +1 -0
- package/dist/types/contracts/index.d.ts +6 -0
- package/dist/types/contracts/index.d.ts.map +1 -0
- package/dist/types/facilitator/facilitator-service.d.ts +69 -0
- package/dist/types/facilitator/facilitator-service.d.ts.map +1 -0
- package/dist/types/facilitator/fee-calculator.d.ts +48 -0
- package/dist/types/facilitator/fee-calculator.d.ts.map +1 -0
- package/dist/types/facilitator/index.d.ts +5 -0
- package/dist/types/facilitator/index.d.ts.map +1 -0
- package/dist/types/facilitator/types.d.ts +221 -0
- package/dist/types/facilitator/types.d.ts.map +1 -0
- package/dist/types/identity/agent-card.d.ts +83 -0
- package/dist/types/identity/agent-card.d.ts.map +1 -0
- package/dist/types/identity/agent-identity.d.ts +102 -0
- package/dist/types/identity/agent-identity.d.ts.map +1 -0
- package/dist/types/identity/did-resolver.d.ts +70 -0
- package/dist/types/identity/did-resolver.d.ts.map +1 -0
- package/dist/types/identity/erc8004/abis.d.ts +336 -0
- package/dist/types/identity/erc8004/abis.d.ts.map +1 -0
- package/dist/types/identity/erc8004/identity.d.ts +109 -0
- package/dist/types/identity/erc8004/identity.d.ts.map +1 -0
- package/dist/types/identity/erc8004/index.d.ts +15 -0
- package/dist/types/identity/erc8004/index.d.ts.map +1 -0
- package/dist/types/identity/erc8004/reputation.d.ts +52 -0
- package/dist/types/identity/erc8004/reputation.d.ts.map +1 -0
- package/dist/types/identity/erc8004/types.d.ts +31 -0
- package/dist/types/identity/erc8004/types.d.ts.map +1 -0
- package/dist/types/identity/index.d.ts +10 -0
- package/dist/types/identity/index.d.ts.map +1 -0
- package/dist/types/identity/owner-registry.d.ts +68 -0
- package/dist/types/identity/owner-registry.d.ts.map +1 -0
- package/dist/types/identity/reputation.d.ts +66 -0
- package/dist/types/identity/reputation.d.ts.map +1 -0
- package/dist/types/identity/types.d.ts +182 -0
- package/dist/types/identity/types.d.ts.map +1 -0
- package/dist/types/index.d.ts +17 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/logger.d.ts +10 -0
- package/dist/types/logger.d.ts.map +1 -0
- package/dist/types/monitor/evm-watcher.d.ts +40 -0
- package/dist/types/monitor/evm-watcher.d.ts.map +1 -0
- package/dist/types/monitor/index.d.ts +25 -0
- package/dist/types/monitor/index.d.ts.map +1 -0
- package/dist/types/monitor/multi-watcher.d.ts +48 -0
- package/dist/types/monitor/multi-watcher.d.ts.map +1 -0
- package/dist/types/monitor/solana-watcher.d.ts +36 -0
- package/dist/types/monitor/solana-watcher.d.ts.map +1 -0
- package/dist/types/monitor/stellar-watcher.d.ts +34 -0
- package/dist/types/monitor/stellar-watcher.d.ts.map +1 -0
- package/dist/types/mpp.d.ts +153 -0
- package/dist/types/mpp.d.ts.map +1 -0
- package/dist/types/policy.d.ts +40 -0
- package/dist/types/policy.d.ts.map +1 -0
- package/dist/types/providers/bridge.d.ts +162 -0
- package/dist/types/providers/bridge.d.ts.map +1 -0
- package/dist/types/providers/onchainos.d.ts +128 -0
- package/dist/types/providers/onchainos.d.ts.map +1 -0
- package/dist/types/requests/eip681.d.ts +51 -0
- package/dist/types/requests/eip681.d.ts.map +1 -0
- package/dist/types/requests/index.d.ts +11 -0
- package/dist/types/requests/index.d.ts.map +1 -0
- package/dist/types/requests/sep7.d.ts +56 -0
- package/dist/types/requests/sep7.d.ts.map +1 -0
- package/dist/types/requests/solana-pay.d.ts +50 -0
- package/dist/types/requests/solana-pay.d.ts.map +1 -0
- package/dist/types/requests/universal.d.ts +28 -0
- package/dist/types/requests/universal.d.ts.map +1 -0
- package/dist/types/server/index.d.ts +27 -0
- package/dist/types/server/index.d.ts.map +1 -0
- package/dist/types/server/mpp-gate.d.ts +66 -0
- package/dist/types/server/mpp-gate.d.ts.map +1 -0
- package/dist/types/server/payment-gate.d.ts +33 -0
- package/dist/types/server/payment-gate.d.ts.map +1 -0
- package/dist/types/server/receipt.d.ts +52 -0
- package/dist/types/server/receipt.d.ts.map +1 -0
- package/dist/types/server/webhook.d.ts +46 -0
- package/dist/types/server/webhook.d.ts.map +1 -0
- package/dist/types/server/x402-gate.d.ts +70 -0
- package/dist/types/server/x402-gate.d.ts.map +1 -0
- package/dist/types/solana.d.ts +7 -0
- package/dist/types/solana.d.ts.map +1 -0
- package/dist/types/stellar.d.ts +7 -0
- package/dist/types/stellar.d.ts.map +1 -0
- package/dist/types/wallet/balance-aggregator.d.ts +51 -0
- package/dist/types/wallet/balance-aggregator.d.ts.map +1 -0
- package/dist/types/wallet/chain-router.d.ts +44 -0
- package/dist/types/wallet/chain-router.d.ts.map +1 -0
- package/dist/types/wallet/index.d.ts +7 -0
- package/dist/types/wallet/index.d.ts.map +1 -0
- package/dist/types/wallet/intent-resolver.d.ts +39 -0
- package/dist/types/wallet/intent-resolver.d.ts.map +1 -0
- package/dist/types/wallet/session-manager.d.ts +57 -0
- package/dist/types/wallet/session-manager.d.ts.map +1 -0
- package/dist/types/wallet/types.d.ts +161 -0
- package/dist/types/wallet/types.d.ts.map +1 -0
- package/dist/types/wallet/unified-wallet.d.ts +122 -0
- package/dist/types/wallet/unified-wallet.d.ts.map +1 -0
- package/package.json +194 -0
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SEP-7 — Stellar Payment URI builder.
|
|
3
|
+
*
|
|
4
|
+
* Generates `web+stellar:` URIs per SEP-0007 for requesting
|
|
5
|
+
* XLM or custom asset payments on Stellar.
|
|
6
|
+
*
|
|
7
|
+
* @see https://github.com/nicohman/sep-0007
|
|
8
|
+
*
|
|
9
|
+
* @example
|
|
10
|
+
* ```
|
|
11
|
+
* web+stellar:pay?destination=GABCD...&amount=100&asset_code=USDC&asset_issuer=GA5Z...&memo=invoice123
|
|
12
|
+
* ```
|
|
13
|
+
*/
|
|
14
|
+
/**
|
|
15
|
+
* Build a SEP-7 Stellar payment URI.
|
|
16
|
+
*
|
|
17
|
+
* @example
|
|
18
|
+
* ```ts
|
|
19
|
+
* // Native XLM
|
|
20
|
+
* buildSep7Uri({ destination: 'GABCD...', amount: '100' });
|
|
21
|
+
* // => 'web+stellar:pay?destination=GABCD...&amount=100'
|
|
22
|
+
*
|
|
23
|
+
* // USDC on Stellar
|
|
24
|
+
* buildSep7Uri({
|
|
25
|
+
* destination: 'GABCD...',
|
|
26
|
+
* amount: '50',
|
|
27
|
+
* assetCode: 'USDC',
|
|
28
|
+
* assetIssuer: 'GA5ZSEJYB37JRC5AVCIA5MOP4RHTM335X2KGX3IHOJAPP5RE34K4KZVN',
|
|
29
|
+
* memo: 'invoice-42',
|
|
30
|
+
* });
|
|
31
|
+
* ```
|
|
32
|
+
*/
|
|
33
|
+
export function buildSep7Uri(options) {
|
|
34
|
+
const params = new URLSearchParams();
|
|
35
|
+
params.set('destination', options.destination);
|
|
36
|
+
params.set('amount', options.amount);
|
|
37
|
+
const isNative = !options.assetCode || options.assetCode === 'XLM';
|
|
38
|
+
if (!isNative) {
|
|
39
|
+
params.set('asset_code', options.assetCode);
|
|
40
|
+
if (options.assetIssuer) {
|
|
41
|
+
params.set('asset_issuer', options.assetIssuer);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
if (options.memo) {
|
|
45
|
+
params.set('memo', options.memo);
|
|
46
|
+
params.set('memo_type', options.memoType ?? 'MEMO_TEXT');
|
|
47
|
+
}
|
|
48
|
+
if (options.callback)
|
|
49
|
+
params.set('callback', options.callback);
|
|
50
|
+
if (options.msg)
|
|
51
|
+
params.set('msg', options.msg);
|
|
52
|
+
if (options.networkPassphrase)
|
|
53
|
+
params.set('network_passphrase', options.networkPassphrase);
|
|
54
|
+
if (options.originDomain)
|
|
55
|
+
params.set('origin_domain', options.originDomain);
|
|
56
|
+
return `web+stellar:pay?${params.toString()}`;
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=sep7.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sep7.js","sourceRoot":"","sources":["../../../src/requests/sep7.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAyBH;;;;;;;;;;;;;;;;;;GAkBG;AACH,MAAM,UAAU,YAAY,CAAC,OAAoB;IAC/C,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;IAErC,MAAM,CAAC,GAAG,CAAC,aAAa,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC;IAC/C,MAAM,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IAErC,MAAM,QAAQ,GAAG,CAAC,OAAO,CAAC,SAAS,IAAI,OAAO,CAAC,SAAS,KAAK,KAAK,CAAC;IAEnE,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,MAAM,CAAC,GAAG,CAAC,YAAY,EAAE,OAAO,CAAC,SAAU,CAAC,CAAC;QAC7C,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;YACxB,MAAM,CAAC,GAAG,CAAC,cAAc,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC;QAClD,CAAC;IACH,CAAC;IAED,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;QACjB,MAAM,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;QACjC,MAAM,CAAC,GAAG,CAAC,WAAW,EAAE,OAAO,CAAC,QAAQ,IAAI,WAAW,CAAC,CAAC;IAC3D,CAAC;IAED,IAAI,OAAO,CAAC,QAAQ;QAAE,MAAM,CAAC,GAAG,CAAC,UAAU,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;IAC/D,IAAI,OAAO,CAAC,GAAG;QAAE,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;IAChD,IAAI,OAAO,CAAC,iBAAiB;QAAE,MAAM,CAAC,GAAG,CAAC,oBAAoB,EAAE,OAAO,CAAC,iBAAiB,CAAC,CAAC;IAC3F,IAAI,OAAO,CAAC,YAAY;QAAE,MAAM,CAAC,GAAG,CAAC,eAAe,EAAE,OAAO,CAAC,YAAY,CAAC,CAAC;IAE5E,OAAO,mBAAmB,MAAM,CAAC,QAAQ,EAAE,EAAE,CAAC;AAChD,CAAC"}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Solana Pay — Payment URI builder.
|
|
3
|
+
*
|
|
4
|
+
* Generates `solana:` URIs per the Solana Pay specification
|
|
5
|
+
* for requesting SOL or SPL token payments.
|
|
6
|
+
*
|
|
7
|
+
* @see https://docs.solanapay.com/spec
|
|
8
|
+
*
|
|
9
|
+
* @example
|
|
10
|
+
* ```
|
|
11
|
+
* solana:7abc...?amount=1.5&spl-token=EPjFW...&reference=ref123&label=MyStore&message=Order42
|
|
12
|
+
* ```
|
|
13
|
+
*/
|
|
14
|
+
/**
|
|
15
|
+
* Build a Solana Pay payment URI.
|
|
16
|
+
*
|
|
17
|
+
* @example
|
|
18
|
+
* ```ts
|
|
19
|
+
* // Native SOL
|
|
20
|
+
* buildSolanaPayUri({ recipient: '7abc...', amount: '1.5' });
|
|
21
|
+
* // => 'solana:7abc...?amount=1.5'
|
|
22
|
+
*
|
|
23
|
+
* // USDC SPL token
|
|
24
|
+
* buildSolanaPayUri({
|
|
25
|
+
* recipient: '7abc...',
|
|
26
|
+
* amount: '25.00',
|
|
27
|
+
* splToken: 'EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v',
|
|
28
|
+
* label: 'envoy',
|
|
29
|
+
* message: 'Agent service payment',
|
|
30
|
+
* });
|
|
31
|
+
* ```
|
|
32
|
+
*/
|
|
33
|
+
export function buildSolanaPayUri(options) {
|
|
34
|
+
const params = new URLSearchParams();
|
|
35
|
+
params.set('amount', options.amount);
|
|
36
|
+
if (options.splToken)
|
|
37
|
+
params.set('spl-token', options.splToken);
|
|
38
|
+
if (options.reference)
|
|
39
|
+
params.set('reference', options.reference);
|
|
40
|
+
if (options.label)
|
|
41
|
+
params.set('label', options.label);
|
|
42
|
+
if (options.message)
|
|
43
|
+
params.set('message', options.message);
|
|
44
|
+
if (options.memo)
|
|
45
|
+
params.set('memo', options.memo);
|
|
46
|
+
return `solana:${options.recipient}?${params.toString()}`;
|
|
47
|
+
}
|
|
48
|
+
//# sourceMappingURL=solana-pay.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"solana-pay.js","sourceRoot":"","sources":["../../../src/requests/solana-pay.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAmBH;;;;;;;;;;;;;;;;;;GAkBG;AACH,MAAM,UAAU,iBAAiB,CAAC,OAAyB;IACzD,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;IAErC,MAAM,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IAErC,IAAI,OAAO,CAAC,QAAQ;QAAE,MAAM,CAAC,GAAG,CAAC,WAAW,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;IAChE,IAAI,OAAO,CAAC,SAAS;QAAE,MAAM,CAAC,GAAG,CAAC,WAAW,EAAE,OAAO,CAAC,SAAS,CAAC,CAAC;IAClE,IAAI,OAAO,CAAC,KAAK;QAAE,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;IACtD,IAAI,OAAO,CAAC,OAAO;QAAE,MAAM,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC;IAC5D,IAAI,OAAO,CAAC,IAAI;QAAE,MAAM,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;IAEnD,OAAO,UAAU,OAAO,CAAC,SAAS,IAAI,MAAM,CAAC,QAAQ,EAAE,EAAE,CAAC;AAC5D,CAAC"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Universal Payment URI builder — auto-detects chain and generates
|
|
3
|
+
* the appropriate payment URI format.
|
|
4
|
+
*/
|
|
5
|
+
import { buildEip681Uri } from './eip681.js';
|
|
6
|
+
import { buildSep7Uri } from './sep7.js';
|
|
7
|
+
import { buildSolanaPayUri } from './solana-pay.js';
|
|
8
|
+
/**
|
|
9
|
+
* Build a payment URI for any supported chain.
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* ```ts
|
|
13
|
+
* const uri = buildPaymentUri({
|
|
14
|
+
* chain: 'evm',
|
|
15
|
+
* evm: { to: '0x...', amount: '0.5', chainId: 8453 },
|
|
16
|
+
* });
|
|
17
|
+
* ```
|
|
18
|
+
*/
|
|
19
|
+
export function buildPaymentUri(options) {
|
|
20
|
+
switch (options.chain) {
|
|
21
|
+
case 'evm':
|
|
22
|
+
if (!options.evm)
|
|
23
|
+
throw new Error('EVM options required for chain=evm');
|
|
24
|
+
return buildEip681Uri(options.evm);
|
|
25
|
+
case 'stellar':
|
|
26
|
+
if (!options.stellar)
|
|
27
|
+
throw new Error('Stellar options required for chain=stellar');
|
|
28
|
+
return buildSep7Uri(options.stellar);
|
|
29
|
+
case 'solana':
|
|
30
|
+
if (!options.solana)
|
|
31
|
+
throw new Error('Solana options required for chain=solana');
|
|
32
|
+
return buildSolanaPayUri(options.solana);
|
|
33
|
+
default:
|
|
34
|
+
throw new Error(`Unsupported chain: ${options.chain}`);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
//# sourceMappingURL=universal.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"universal.js","sourceRoot":"","sources":["../../../src/requests/universal.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EAAE,cAAc,EAAsB,MAAM,UAAU,CAAC;AAC9D,OAAO,EAAE,YAAY,EAAoB,MAAM,QAAQ,CAAC;AACxD,OAAO,EAAE,iBAAiB,EAAyB,MAAM,cAAc,CAAC;AAWxE;;;;;;;;;;GAUG;AACH,MAAM,UAAU,eAAe,CAAC,OAAmC;IACjE,QAAQ,OAAO,CAAC,KAAK,EAAE,CAAC;QACtB,KAAK,KAAK;YACR,IAAI,CAAC,OAAO,CAAC,GAAG;gBAAE,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;YACxE,OAAO,cAAc,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACrC,KAAK,SAAS;YACZ,IAAI,CAAC,OAAO,CAAC,OAAO;gBAAE,MAAM,IAAI,KAAK,CAAC,4CAA4C,CAAC,CAAC;YACpF,OAAO,YAAY,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QACvC,KAAK,QAAQ;YACX,IAAI,CAAC,OAAO,CAAC,MAAM;gBAAE,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;YACjF,OAAO,iBAAiB,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAC3C;YACE,MAAM,IAAI,KAAK,CAAC,sBAAsB,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* envoy — Server-Side Payment Gating
|
|
3
|
+
*
|
|
4
|
+
* Express/Connect-compatible middleware for monetizing API endpoints.
|
|
5
|
+
* Supports both x402 and MPP (Machine Payment Protocol) standards.
|
|
6
|
+
*
|
|
7
|
+
* @example
|
|
8
|
+
* ```ts
|
|
9
|
+
* import { createPaymentGate } from 'envoy-pay/server';
|
|
10
|
+
*
|
|
11
|
+
* app.post('/api/premium',
|
|
12
|
+
* createPaymentGate({
|
|
13
|
+
* payTo: '0xYOUR_WALLET',
|
|
14
|
+
* amount: '500000', // 0.50 USDC
|
|
15
|
+
* asset: 'USDC',
|
|
16
|
+
* network: 'eip155:8453',
|
|
17
|
+
* }),
|
|
18
|
+
* (req, res) => { res.json({ data: 'premium content' }); }
|
|
19
|
+
* );
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
export { createX402Gate } from './x402-gate.js';
|
|
23
|
+
export { createMppGate } from './mpp-gate.js';
|
|
24
|
+
export { createPaymentGate } from './payment-gate.js';
|
|
25
|
+
export { createWebhookHandler } from './webhook.js';
|
|
26
|
+
export { buildReceipt } from './receipt.js';
|
|
27
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/server/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,OAAO,EAAE,cAAc,EAAuC,MAAM,aAAa,CAAC;AAClF,OAAO,EAAE,aAAa,EAAsB,MAAM,YAAY,CAAC;AAC/D,OAAO,EAAE,iBAAiB,EAA0B,MAAM,gBAAgB,CAAC;AAC3E,OAAO,EAAE,oBAAoB,EAAsB,MAAM,WAAW,CAAC;AACrE,OAAO,EAAE,YAAY,EAA4C,MAAM,WAAW,CAAC"}
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
import { noopLogger } from '../logger.js';
|
|
2
|
+
let challengeCounter = 0;
|
|
3
|
+
/**
|
|
4
|
+
* Generate a unique challenge ID.
|
|
5
|
+
*/
|
|
6
|
+
function generateChallengeId() {
|
|
7
|
+
challengeCounter++;
|
|
8
|
+
return `ch_${Date.now()}_${challengeCounter}`;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Creates an Express/Connect middleware that gates access behind MPP payments.
|
|
12
|
+
*
|
|
13
|
+
* When a request lacks a valid `Authorization: Payment` header, the middleware
|
|
14
|
+
* responds with 402 and a `WWW-Authenticate: Payment` challenge header.
|
|
15
|
+
*
|
|
16
|
+
* @example
|
|
17
|
+
* ```ts
|
|
18
|
+
* app.post('/api/premium', createMppGate({
|
|
19
|
+
* realm: 'api.example.com',
|
|
20
|
+
* method: 'stripe',
|
|
21
|
+
* amount: '50',
|
|
22
|
+
* currency: 'usd',
|
|
23
|
+
* recipient: 'acct_xxx',
|
|
24
|
+
* }), handler);
|
|
25
|
+
* ```
|
|
26
|
+
*/
|
|
27
|
+
export function createMppGate(config) {
|
|
28
|
+
const log = config.logger ?? noopLogger;
|
|
29
|
+
const intent = config.intent ?? 'payment';
|
|
30
|
+
const currency = config.currency ?? 'usd';
|
|
31
|
+
const decimals = config.decimals ?? 2;
|
|
32
|
+
const ttlMs = config.ttlMs ?? 300000;
|
|
33
|
+
return async (req, res, next) => {
|
|
34
|
+
// Check for Authorization: Payment header
|
|
35
|
+
const authHeader = req.headers['authorization'];
|
|
36
|
+
if (!authHeader || !authHeader.startsWith('Payment ')) {
|
|
37
|
+
// No credential — return 402 with WWW-Authenticate challenge
|
|
38
|
+
log('[mpp-gate] 💰 No Authorization: Payment — returning 402 challenge');
|
|
39
|
+
const challengeId = generateChallengeId();
|
|
40
|
+
const expires = new Date(Date.now() + ttlMs).toISOString();
|
|
41
|
+
const requestObj = {
|
|
42
|
+
amount: config.amount,
|
|
43
|
+
currency,
|
|
44
|
+
decimals,
|
|
45
|
+
recipient: config.recipient,
|
|
46
|
+
};
|
|
47
|
+
const requestEncoded = Buffer.from(JSON.stringify(requestObj))
|
|
48
|
+
.toString('base64')
|
|
49
|
+
.replace(/\+/g, '-')
|
|
50
|
+
.replace(/\//g, '_')
|
|
51
|
+
.replace(/=+$/, '');
|
|
52
|
+
const challengeStr = `Payment id="${challengeId}",realm="${config.realm}",method="${config.method}",intent="${intent}",request="${requestEncoded}",expires="${expires}"`;
|
|
53
|
+
res.writeHead(402, {
|
|
54
|
+
'WWW-Authenticate': challengeStr,
|
|
55
|
+
'Content-Type': 'text/plain',
|
|
56
|
+
});
|
|
57
|
+
res.end('Payment Required');
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
// Decode Payment credential
|
|
61
|
+
try {
|
|
62
|
+
const credentialBase64 = authHeader.slice('Payment '.length);
|
|
63
|
+
// Re-add base64 padding
|
|
64
|
+
const padded = credentialBase64
|
|
65
|
+
.replace(/-/g, '+')
|
|
66
|
+
.replace(/_/g, '/');
|
|
67
|
+
const decoded = Buffer.from(padded, 'base64').toString('utf-8');
|
|
68
|
+
const credential = JSON.parse(decoded);
|
|
69
|
+
// Validate credential structure
|
|
70
|
+
if (!credential.challenge?.id || !credential.payload?.transaction) {
|
|
71
|
+
log('[mpp-gate] ❌ Malformed credential');
|
|
72
|
+
res.writeHead(400, { 'Content-Type': 'application/json' });
|
|
73
|
+
res.end(JSON.stringify({ error: 'Malformed payment credential' }));
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
// Check realm matches
|
|
77
|
+
if (credential.challenge.realm !== config.realm) {
|
|
78
|
+
log('[mpp-gate] ❌ Realm mismatch');
|
|
79
|
+
res.writeHead(403, { 'Content-Type': 'application/json' });
|
|
80
|
+
res.end(JSON.stringify({ error: 'Challenge realm mismatch' }));
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
// Check expiration
|
|
84
|
+
if (credential.challenge.expires) {
|
|
85
|
+
const expiresAt = new Date(credential.challenge.expires).getTime();
|
|
86
|
+
if (Date.now() > expiresAt) {
|
|
87
|
+
log('[mpp-gate] ❌ Challenge expired');
|
|
88
|
+
res.writeHead(402, { 'Content-Type': 'application/json' });
|
|
89
|
+
res.end(JSON.stringify({ error: 'Challenge expired' }));
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
// Custom verification
|
|
94
|
+
if (config.verifyCredential) {
|
|
95
|
+
const isValid = await config.verifyCredential(credential, credential.challenge.id);
|
|
96
|
+
if (!isValid) {
|
|
97
|
+
log('[mpp-gate] ❌ Credential verification failed');
|
|
98
|
+
res.writeHead(402, { 'Content-Type': 'application/json' });
|
|
99
|
+
res.end(JSON.stringify({ error: 'Payment verification failed' }));
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
// Attach credential to request
|
|
104
|
+
req.mppPayment = credential;
|
|
105
|
+
log(`[mpp-gate] ✅ Payment verified: tx=${credential.payload.transaction.slice(0, 16)}…`);
|
|
106
|
+
next();
|
|
107
|
+
}
|
|
108
|
+
catch (err) {
|
|
109
|
+
log(`[mpp-gate] ❌ Failed to decode credential: ${err.message}`);
|
|
110
|
+
res.writeHead(400, { 'Content-Type': 'application/json' });
|
|
111
|
+
res.end(JSON.stringify({ error: 'Invalid Authorization: Payment header' }));
|
|
112
|
+
}
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
//# sourceMappingURL=mpp-gate.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mpp-gate.js","sourceRoot":"","sources":["../../../src/server/mpp-gate.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import { createX402Gate } from './x402-gate.js';
|
|
2
|
+
import { createMppGate } from './mpp-gate.js';
|
|
3
|
+
import { noopLogger } from '../logger.js';
|
|
4
|
+
/**
|
|
5
|
+
* Creates a dual-protocol payment gate that accepts both x402 and MPP.
|
|
6
|
+
*
|
|
7
|
+
* Detection logic:
|
|
8
|
+
* - `X-PAYMENT` header → x402 flow
|
|
9
|
+
* - `Authorization: Payment` header → MPP flow
|
|
10
|
+
* - Neither present → returns both challenge types simultaneously
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```ts
|
|
14
|
+
* app.post('/api/premium', createPaymentGate({
|
|
15
|
+
* x402: { payTo: '0x...', amount: '500000', asset: 'USDC', network: 'eip155:8453' },
|
|
16
|
+
* mpp: { realm: 'api.example.com', method: 'stripe', amount: '50', recipient: 'acct_xxx' },
|
|
17
|
+
* }), handler);
|
|
18
|
+
* ```
|
|
19
|
+
*/
|
|
20
|
+
export function createPaymentGate(config) {
|
|
21
|
+
const log = config.logger ?? noopLogger;
|
|
22
|
+
const x402Middleware = createX402Gate({
|
|
23
|
+
...config.x402,
|
|
24
|
+
logger: log,
|
|
25
|
+
});
|
|
26
|
+
const mppMiddleware = createMppGate({
|
|
27
|
+
...config.mpp,
|
|
28
|
+
logger: log,
|
|
29
|
+
});
|
|
30
|
+
return async (req, res, next) => {
|
|
31
|
+
// Detect which protocol the client is using
|
|
32
|
+
const hasX402 = !!(req.headers['x-payment'] || req.headers['X-PAYMENT']);
|
|
33
|
+
const hasAuth = !!(req.headers['authorization'] &&
|
|
34
|
+
req.headers['authorization'].startsWith('Payment '));
|
|
35
|
+
if (hasX402) {
|
|
36
|
+
log('[payment-gate] 🔍 Detected x402 protocol (X-PAYMENT header)');
|
|
37
|
+
return x402Middleware(req, res, next);
|
|
38
|
+
}
|
|
39
|
+
if (hasAuth) {
|
|
40
|
+
log('[payment-gate] 🔍 Detected MPP protocol (Authorization: Payment header)');
|
|
41
|
+
return mppMiddleware(req, res, next);
|
|
42
|
+
}
|
|
43
|
+
// No payment header — return dual-protocol 402 challenge
|
|
44
|
+
log('[payment-gate] 💰 No payment header — returning dual-protocol 402');
|
|
45
|
+
// Build MPP challenge header
|
|
46
|
+
const mppIntent = config.mpp.intent ?? 'payment';
|
|
47
|
+
const mppCurrency = config.mpp.currency ?? 'usd';
|
|
48
|
+
const mppDecimals = config.mpp.decimals ?? 2;
|
|
49
|
+
const ttlMs = config.mpp.ttlMs ?? 300000;
|
|
50
|
+
const challengeId = `ch_${Date.now()}_dual`;
|
|
51
|
+
const expires = new Date(Date.now() + ttlMs).toISOString();
|
|
52
|
+
const requestObj = {
|
|
53
|
+
amount: config.mpp.amount,
|
|
54
|
+
currency: mppCurrency,
|
|
55
|
+
decimals: mppDecimals,
|
|
56
|
+
recipient: config.mpp.recipient,
|
|
57
|
+
};
|
|
58
|
+
const requestEncoded = Buffer.from(JSON.stringify(requestObj))
|
|
59
|
+
.toString('base64')
|
|
60
|
+
.replace(/\+/g, '-')
|
|
61
|
+
.replace(/\//g, '_')
|
|
62
|
+
.replace(/=+$/, '');
|
|
63
|
+
const wwwAuth = `Payment id="${challengeId}",realm="${config.mpp.realm}",method="${config.mpp.method}",intent="${mppIntent}",request="${requestEncoded}",expires="${expires}"`;
|
|
64
|
+
// Build x402 challenge body
|
|
65
|
+
const x402Body = {
|
|
66
|
+
x402Version: config.x402.x402Version ?? 2,
|
|
67
|
+
resource: {
|
|
68
|
+
url: req.url || '/',
|
|
69
|
+
description: config.x402.description || 'Payment required',
|
|
70
|
+
...(config.x402.usdAmount ? { usdAmount: config.x402.usdAmount } : {}),
|
|
71
|
+
},
|
|
72
|
+
accepts: [
|
|
73
|
+
{
|
|
74
|
+
scheme: config.x402.scheme ?? 'exact',
|
|
75
|
+
network: config.x402.network,
|
|
76
|
+
amount: config.x402.amount,
|
|
77
|
+
payTo: config.x402.payTo,
|
|
78
|
+
asset: config.x402.asset,
|
|
79
|
+
},
|
|
80
|
+
],
|
|
81
|
+
};
|
|
82
|
+
res.writeHead(402, {
|
|
83
|
+
'WWW-Authenticate': wwwAuth,
|
|
84
|
+
'Content-Type': 'application/json',
|
|
85
|
+
});
|
|
86
|
+
res.end(JSON.stringify(x402Body));
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
//# sourceMappingURL=payment-gate.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"payment-gate.js","sourceRoot":"","sources":["../../../src/server/payment-gate.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MPP Payment-Receipt builder.
|
|
3
|
+
*
|
|
4
|
+
* Generates base64url-encoded receipt JSON that can be attached
|
|
5
|
+
* as a `Payment-Receipt` response header after successful payment verification.
|
|
6
|
+
*/
|
|
7
|
+
/**
|
|
8
|
+
* Build a Payment-Receipt header value.
|
|
9
|
+
*
|
|
10
|
+
* @returns base64url-encoded JSON receipt.
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```ts
|
|
14
|
+
* const receipt = buildReceipt({
|
|
15
|
+
* challengeId: 'ch_123',
|
|
16
|
+
* status: 'settled',
|
|
17
|
+
* amount: '500000',
|
|
18
|
+
* asset: 'USDC',
|
|
19
|
+
* transactionHash: '0xabc...',
|
|
20
|
+
* });
|
|
21
|
+
*
|
|
22
|
+
* res.setHeader('Payment-Receipt', receipt);
|
|
23
|
+
* ```
|
|
24
|
+
*/
|
|
25
|
+
export function buildReceipt(options) {
|
|
26
|
+
const receipt = {
|
|
27
|
+
challengeId: options.challengeId,
|
|
28
|
+
status: options.status,
|
|
29
|
+
amount: options.amount,
|
|
30
|
+
asset: options.asset,
|
|
31
|
+
...(options.transactionHash ? { transactionHash: options.transactionHash } : {}),
|
|
32
|
+
issuedAt: (options.timestamp ?? new Date()).toISOString(),
|
|
33
|
+
};
|
|
34
|
+
return Buffer.from(JSON.stringify(receipt))
|
|
35
|
+
.toString('base64')
|
|
36
|
+
.replace(/\+/g, '-')
|
|
37
|
+
.replace(/\//g, '_')
|
|
38
|
+
.replace(/=+$/, '');
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Parse a Payment-Receipt header value back into a receipt object.
|
|
42
|
+
*/
|
|
43
|
+
export function parseReceipt(encoded) {
|
|
44
|
+
const padded = encoded.replace(/-/g, '+').replace(/_/g, '/');
|
|
45
|
+
const decoded = Buffer.from(padded, 'base64').toString('utf-8');
|
|
46
|
+
return JSON.parse(decoded);
|
|
47
|
+
}
|
|
48
|
+
//# sourceMappingURL=receipt.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"receipt.js","sourceRoot":"","sources":["../../../src/server/receipt.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AA0BH;;;;;;;;;;;;;;;;;GAiBG;AACH,MAAM,UAAU,YAAY,CAAC,OAAuB;IAClD,MAAM,OAAO,GAAmB;QAC9B,WAAW,EAAE,OAAO,CAAC,WAAW;QAChC,MAAM,EAAE,OAAO,CAAC,MAAM;QACtB,MAAM,EAAE,OAAO,CAAC,MAAM;QACtB,KAAK,EAAE,OAAO,CAAC,KAAK;QACpB,GAAG,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC,EAAE,eAAe,EAAE,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QAChF,QAAQ,EAAE,CAAC,OAAO,CAAC,SAAS,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE;KAC1D,CAAC;IAEF,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;SACxC,QAAQ,CAAC,QAAQ,CAAC;SAClB,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC;SACnB,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC;SACnB,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;AACxB,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,YAAY,CAAC,OAAe;IAC1C,MAAM,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IAC7D,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IAChE,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;AAC7B,CAAC"}
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
import { noopLogger } from '../logger.js';
|
|
2
|
+
/**
|
|
3
|
+
* Creates a Connect/Express-compatible webhook handler for Stripe events.
|
|
4
|
+
*
|
|
5
|
+
* Uses HMAC signature verification via the raw request body.
|
|
6
|
+
* Supports idempotency guard to prevent duplicate processing.
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```ts
|
|
10
|
+
* const handler = createWebhookHandler({
|
|
11
|
+
* signingSecret: process.env.STRIPE_WEBHOOK_SECRET!,
|
|
12
|
+
* onPaymentSuccess: (event) => {
|
|
13
|
+
* console.log(`Payment received: ${event.amount} ${event.currency}`);
|
|
14
|
+
* },
|
|
15
|
+
* });
|
|
16
|
+
*
|
|
17
|
+
* app.post('/webhook', express.raw({ type: 'application/json' }), handler);
|
|
18
|
+
* ```
|
|
19
|
+
*/
|
|
20
|
+
export function createWebhookHandler(config) {
|
|
21
|
+
const log = config.logger ?? noopLogger;
|
|
22
|
+
const processedIds = config.processedEventIds ?? new Set();
|
|
23
|
+
return async (req, res, _next) => {
|
|
24
|
+
const sig = req.headers['stripe-signature'];
|
|
25
|
+
if (!sig) {
|
|
26
|
+
log('[webhook] ❌ Missing stripe-signature header');
|
|
27
|
+
res.writeHead(400, { 'Content-Type': 'application/json' });
|
|
28
|
+
res.end(JSON.stringify({ error: 'Missing stripe-signature header' }));
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
// Get raw body
|
|
32
|
+
const rawBody = typeof req.body === 'string'
|
|
33
|
+
? req.body
|
|
34
|
+
: req.body instanceof Buffer
|
|
35
|
+
? req.body.toString('utf-8')
|
|
36
|
+
: JSON.stringify(req.body);
|
|
37
|
+
// Verify HMAC signature (simplified — in production use stripe.webhooks.constructEvent)
|
|
38
|
+
if (!verifySignature(rawBody, sig, config.signingSecret)) {
|
|
39
|
+
log('[webhook] ❌ Invalid signature');
|
|
40
|
+
res.writeHead(400, { 'Content-Type': 'application/json' });
|
|
41
|
+
res.end(JSON.stringify({ error: 'Invalid webhook signature' }));
|
|
42
|
+
return;
|
|
43
|
+
}
|
|
44
|
+
let event;
|
|
45
|
+
try {
|
|
46
|
+
event = typeof req.body === 'string' ? JSON.parse(req.body) : req.body;
|
|
47
|
+
}
|
|
48
|
+
catch {
|
|
49
|
+
log('[webhook] ❌ Failed to parse event body');
|
|
50
|
+
res.writeHead(400, { 'Content-Type': 'application/json' });
|
|
51
|
+
res.end(JSON.stringify({ error: 'Invalid JSON body' }));
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
// Idempotency guard
|
|
55
|
+
if (processedIds.has(event.id)) {
|
|
56
|
+
log(`[webhook] ⏭️ Duplicate event: ${event.id}`);
|
|
57
|
+
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
58
|
+
res.end(JSON.stringify({ received: true, duplicate: true }));
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
processedIds.add(event.id);
|
|
62
|
+
// Route based on event type
|
|
63
|
+
const webhookEvent = {
|
|
64
|
+
id: event.id,
|
|
65
|
+
type: event.type,
|
|
66
|
+
amount: event.data?.object?.amount,
|
|
67
|
+
currency: event.data?.object?.currency,
|
|
68
|
+
paymentIntentId: event.data?.object?.id,
|
|
69
|
+
metadata: event.data?.object?.metadata,
|
|
70
|
+
timestamp: new Date(),
|
|
71
|
+
};
|
|
72
|
+
try {
|
|
73
|
+
if (event.type === 'payment_intent.succeeded' && config.onPaymentSuccess) {
|
|
74
|
+
log(`[webhook] ✅ Payment succeeded: ${webhookEvent.paymentIntentId}`);
|
|
75
|
+
await config.onPaymentSuccess(webhookEvent);
|
|
76
|
+
}
|
|
77
|
+
else if (event.type === 'payment_intent.payment_failed' && config.onPaymentFailure) {
|
|
78
|
+
log(`[webhook] ❌ Payment failed: ${webhookEvent.paymentIntentId}`);
|
|
79
|
+
await config.onPaymentFailure(webhookEvent);
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
log(`[webhook] 📋 Unhandled event type: ${event.type}`);
|
|
83
|
+
}
|
|
84
|
+
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
85
|
+
res.end(JSON.stringify({ received: true }));
|
|
86
|
+
}
|
|
87
|
+
catch (err) {
|
|
88
|
+
log(`[webhook] ❌ Handler error: ${err.message}`);
|
|
89
|
+
res.writeHead(500, { 'Content-Type': 'application/json' });
|
|
90
|
+
res.end(JSON.stringify({ error: 'Webhook handler error' }));
|
|
91
|
+
}
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Simplified HMAC verification for webhook signatures.
|
|
96
|
+
* In production, use Stripe's official `stripe.webhooks.constructEvent()`.
|
|
97
|
+
*/
|
|
98
|
+
function verifySignature(payload, sig, secret) {
|
|
99
|
+
try {
|
|
100
|
+
// Parse Stripe signature header format: t=timestamp,v1=hash
|
|
101
|
+
const parts = sig.split(',');
|
|
102
|
+
const timestamp = parts.find((p) => p.startsWith('t='))?.slice(2);
|
|
103
|
+
const hash = parts.find((p) => p.startsWith('v1='))?.slice(3);
|
|
104
|
+
if (!timestamp || !hash)
|
|
105
|
+
return false;
|
|
106
|
+
// In production: compute HMAC-SHA256 of `${timestamp}.${payload}` with secret
|
|
107
|
+
// For SDK purposes, we trust the Stripe library to do this
|
|
108
|
+
const crypto = require('crypto');
|
|
109
|
+
const expected = crypto
|
|
110
|
+
.createHmac('sha256', secret)
|
|
111
|
+
.update(`${timestamp}.${payload}`)
|
|
112
|
+
.digest('hex');
|
|
113
|
+
return crypto.timingSafeEqual(Buffer.from(hash, 'hex'), Buffer.from(expected, 'hex'));
|
|
114
|
+
}
|
|
115
|
+
catch {
|
|
116
|
+
return false;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
//# sourceMappingURL=webhook.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"webhook.js","sourceRoot":"","sources":["../../../src/server/webhook.ts"],"names":[],"mappings":"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"}
|