mppx 0.6.30 → 0.7.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.
Files changed (483) hide show
  1. package/CHANGELOG.md +25 -0
  2. package/dist/Challenge.d.ts.map +1 -1
  3. package/dist/Challenge.js +9 -7
  4. package/dist/Challenge.js.map +1 -1
  5. package/dist/Constants.d.ts +46 -0
  6. package/dist/Constants.d.ts.map +1 -0
  7. package/dist/Constants.js +46 -0
  8. package/dist/Constants.js.map +1 -0
  9. package/dist/Credential.d.ts.map +1 -1
  10. package/dist/Credential.js +5 -4
  11. package/dist/Credential.js.map +1 -1
  12. package/dist/Method.d.ts +32 -4
  13. package/dist/Method.d.ts.map +1 -1
  14. package/dist/Method.js +5 -2
  15. package/dist/Method.js.map +1 -1
  16. package/dist/Receipt.d.ts.map +1 -1
  17. package/dist/Receipt.js +3 -2
  18. package/dist/Receipt.js.map +1 -1
  19. package/dist/cli/cli.d.ts.map +1 -1
  20. package/dist/cli/cli.js +19 -11
  21. package/dist/cli/cli.js.map +1 -1
  22. package/dist/cli/plugins/tempo.d.ts.map +1 -1
  23. package/dist/cli/plugins/tempo.js +17 -6
  24. package/dist/cli/plugins/tempo.js.map +1 -1
  25. package/dist/cli/utils.d.ts +5 -0
  26. package/dist/cli/utils.d.ts.map +1 -1
  27. package/dist/cli/utils.js +10 -0
  28. package/dist/cli/utils.js.map +1 -1
  29. package/dist/client/Methods.d.ts +5 -2
  30. package/dist/client/Methods.d.ts.map +1 -1
  31. package/dist/client/Methods.js +5 -2
  32. package/dist/client/Methods.js.map +1 -1
  33. package/dist/client/Transport.d.ts.map +1 -1
  34. package/dist/client/Transport.js +4 -5
  35. package/dist/client/Transport.js.map +1 -1
  36. package/dist/client/index.d.ts +2 -1
  37. package/dist/client/index.d.ts.map +1 -1
  38. package/dist/client/index.js +2 -1
  39. package/dist/client/index.js.map +1 -1
  40. package/dist/client/internal/Fetch.d.ts.map +1 -1
  41. package/dist/client/internal/Fetch.js +14 -6
  42. package/dist/client/internal/Fetch.js.map +1 -1
  43. package/dist/evm/server/Methods.d.ts +1 -1
  44. package/dist/evm/server/Methods.d.ts.map +1 -1
  45. package/dist/index.d.ts +1 -0
  46. package/dist/index.d.ts.map +1 -1
  47. package/dist/index.js +1 -0
  48. package/dist/index.js.map +1 -1
  49. package/dist/internal/AcceptPayment.d.ts +3 -0
  50. package/dist/internal/AcceptPayment.d.ts.map +1 -1
  51. package/dist/internal/AcceptPayment.js +15 -11
  52. package/dist/internal/AcceptPayment.js.map +1 -1
  53. package/dist/mcp-sdk/client/McpClient.d.ts +12 -5
  54. package/dist/mcp-sdk/client/McpClient.d.ts.map +1 -1
  55. package/dist/mcp-sdk/client/McpClient.js +55 -42
  56. package/dist/mcp-sdk/client/McpClient.js.map +1 -1
  57. package/dist/server/Mppx.d.ts +11 -3
  58. package/dist/server/Mppx.d.ts.map +1 -1
  59. package/dist/server/Mppx.js +76 -27
  60. package/dist/server/Mppx.js.map +1 -1
  61. package/dist/server/Request.js +24 -10
  62. package/dist/server/Request.js.map +1 -1
  63. package/dist/server/Response.d.ts.map +1 -1
  64. package/dist/server/Response.js +2 -1
  65. package/dist/server/Response.js.map +1 -1
  66. package/dist/server/Transport.d.ts.map +1 -1
  67. package/dist/server/Transport.js +4 -3
  68. package/dist/server/Transport.js.map +1 -1
  69. package/dist/server/index.d.ts +1 -0
  70. package/dist/server/index.d.ts.map +1 -1
  71. package/dist/server/index.js +1 -0
  72. package/dist/server/index.js.map +1 -1
  73. package/dist/stripe/client/Charge.d.ts +1 -1
  74. package/dist/stripe/client/Charge.d.ts.map +1 -1
  75. package/dist/stripe/client/Charge.js +3 -1
  76. package/dist/stripe/client/Charge.js.map +1 -1
  77. package/dist/stripe/server/Charge.d.ts +1 -1
  78. package/dist/stripe/server/Charge.d.ts.map +1 -1
  79. package/dist/stripe/server/Charge.js +9 -2
  80. package/dist/stripe/server/Charge.js.map +1 -1
  81. package/dist/stripe/server/Methods.d.ts +1 -1
  82. package/dist/stripe/server/Methods.d.ts.map +1 -1
  83. package/dist/stripe/server/internal/html.gen.d.ts +1 -1
  84. package/dist/stripe/server/internal/html.gen.d.ts.map +1 -1
  85. package/dist/stripe/server/internal/html.gen.js +1 -1
  86. package/dist/stripe/server/internal/html.gen.js.map +1 -1
  87. package/dist/tempo/Methods.d.ts +18 -0
  88. package/dist/tempo/Methods.d.ts.map +1 -1
  89. package/dist/tempo/Methods.js +16 -1
  90. package/dist/tempo/Methods.js.map +1 -1
  91. package/dist/tempo/client/Charge.d.ts +6 -0
  92. package/dist/tempo/client/Charge.d.ts.map +1 -1
  93. package/dist/tempo/client/Charge.js +9 -2
  94. package/dist/tempo/client/Charge.js.map +1 -1
  95. package/dist/tempo/client/Methods.d.ts +36 -7
  96. package/dist/tempo/client/Methods.d.ts.map +1 -1
  97. package/dist/tempo/client/Methods.js +12 -5
  98. package/dist/tempo/client/Methods.js.map +1 -1
  99. package/dist/tempo/client/index.d.ts +7 -4
  100. package/dist/tempo/client/index.d.ts.map +1 -1
  101. package/dist/tempo/client/index.js +5 -3
  102. package/dist/tempo/client/index.js.map +1 -1
  103. package/dist/tempo/index.d.ts +1 -0
  104. package/dist/tempo/index.d.ts.map +1 -1
  105. package/dist/tempo/index.js +1 -0
  106. package/dist/tempo/index.js.map +1 -1
  107. package/dist/tempo/internal/fee-payer.d.ts +21 -1
  108. package/dist/tempo/internal/fee-payer.d.ts.map +1 -1
  109. package/dist/tempo/internal/fee-payer.js +109 -4
  110. package/dist/tempo/internal/fee-payer.js.map +1 -1
  111. package/dist/tempo/{client → legacy/client}/ChannelOps.d.ts +19 -6
  112. package/dist/tempo/legacy/client/ChannelOps.d.ts.map +1 -0
  113. package/dist/tempo/{client → legacy/client}/ChannelOps.js +9 -3
  114. package/dist/tempo/legacy/client/ChannelOps.js.map +1 -0
  115. package/dist/tempo/{client → legacy/client}/Session.d.ts +23 -4
  116. package/dist/tempo/legacy/client/Session.d.ts.map +1 -0
  117. package/dist/tempo/{client → legacy/client}/Session.js +14 -7
  118. package/dist/tempo/legacy/client/Session.js.map +1 -0
  119. package/dist/tempo/{client → legacy/client}/SessionManager.d.ts +20 -5
  120. package/dist/tempo/legacy/client/SessionManager.d.ts.map +1 -0
  121. package/dist/tempo/{client → legacy/client}/SessionManager.js +20 -16
  122. package/dist/tempo/legacy/client/SessionManager.js.map +1 -0
  123. package/dist/tempo/legacy/client/index.d.ts +7 -0
  124. package/dist/tempo/legacy/client/index.d.ts.map +1 -0
  125. package/dist/tempo/legacy/client/index.js +5 -0
  126. package/dist/tempo/legacy/client/index.js.map +1 -0
  127. package/dist/tempo/legacy/index.d.ts +7 -0
  128. package/dist/tempo/legacy/index.d.ts.map +1 -0
  129. package/dist/tempo/legacy/index.js +7 -0
  130. package/dist/tempo/legacy/index.js.map +1 -0
  131. package/dist/tempo/{server → legacy/server}/Session.d.ts +28 -11
  132. package/dist/tempo/legacy/server/Session.d.ts.map +1 -0
  133. package/dist/tempo/{server → legacy/server}/Session.js +28 -23
  134. package/dist/tempo/legacy/server/Session.js.map +1 -0
  135. package/dist/tempo/legacy/server/index.d.ts +5 -0
  136. package/dist/tempo/legacy/server/index.d.ts.map +1 -0
  137. package/dist/tempo/legacy/server/index.js +5 -0
  138. package/dist/tempo/legacy/server/index.js.map +1 -0
  139. package/dist/tempo/{session → legacy/session}/Chain.d.ts +30 -23
  140. package/dist/tempo/legacy/session/Chain.d.ts.map +1 -0
  141. package/dist/tempo/{session → legacy/session}/Chain.js +12 -11
  142. package/dist/tempo/legacy/session/Chain.js.map +1 -0
  143. package/dist/tempo/{session → legacy/session}/Channel.d.ts +1 -0
  144. package/dist/tempo/legacy/session/Channel.d.ts.map +1 -0
  145. package/dist/tempo/legacy/session/Channel.js.map +1 -0
  146. package/dist/tempo/legacy/session/ChannelStore.d.ts +22 -0
  147. package/dist/tempo/legacy/session/ChannelStore.d.ts.map +1 -0
  148. package/dist/tempo/legacy/session/ChannelStore.js +6 -0
  149. package/dist/tempo/legacy/session/ChannelStore.js.map +1 -0
  150. package/dist/tempo/legacy/session/Types.d.ts +73 -0
  151. package/dist/tempo/legacy/session/Types.d.ts.map +1 -0
  152. package/dist/tempo/legacy/session/Types.js.map +1 -0
  153. package/dist/tempo/{session → legacy/session}/Voucher.d.ts +4 -4
  154. package/dist/tempo/legacy/session/Voucher.d.ts.map +1 -0
  155. package/dist/tempo/{session → legacy/session}/Voucher.js +1 -1
  156. package/dist/tempo/legacy/session/Voucher.js.map +1 -0
  157. package/dist/tempo/{session → legacy/session}/escrow.abi.d.ts +1 -0
  158. package/dist/tempo/{session → legacy/session}/escrow.abi.d.ts.map +1 -1
  159. package/dist/tempo/{session → legacy/session}/escrow.abi.js +1 -0
  160. package/dist/tempo/legacy/session/escrow.abi.js.map +1 -0
  161. package/dist/tempo/legacy/session/index.d.ts +9 -0
  162. package/dist/tempo/legacy/session/index.d.ts.map +1 -0
  163. package/dist/tempo/legacy/session/index.js +9 -0
  164. package/dist/tempo/legacy/session/index.js.map +1 -0
  165. package/dist/tempo/server/Charge.d.ts +1 -1
  166. package/dist/tempo/server/Charge.d.ts.map +1 -1
  167. package/dist/tempo/server/Charge.js +13 -16
  168. package/dist/tempo/server/Charge.js.map +1 -1
  169. package/dist/tempo/server/Methods.d.ts +63 -6
  170. package/dist/tempo/server/Methods.d.ts.map +1 -1
  171. package/dist/tempo/server/Methods.js +36 -8
  172. package/dist/tempo/server/Methods.js.map +1 -1
  173. package/dist/tempo/server/Subscription.d.ts +1 -1
  174. package/dist/tempo/server/Subscription.d.ts.map +1 -1
  175. package/dist/tempo/server/index.d.ts +6 -5
  176. package/dist/tempo/server/index.d.ts.map +1 -1
  177. package/dist/tempo/server/index.js +5 -5
  178. package/dist/tempo/server/index.js.map +1 -1
  179. package/dist/tempo/server/internal/html.gen.d.ts +1 -1
  180. package/dist/tempo/server/internal/html.gen.d.ts.map +1 -1
  181. package/dist/tempo/server/internal/html.gen.js +1 -1
  182. package/dist/tempo/server/internal/html.gen.js.map +1 -1
  183. package/dist/tempo/server/internal/request-body.d.ts +7 -2
  184. package/dist/tempo/server/internal/request-body.d.ts.map +1 -1
  185. package/dist/tempo/server/internal/request-body.js +20 -3
  186. package/dist/tempo/server/internal/request-body.js.map +1 -1
  187. package/dist/tempo/server/internal/transport.d.ts +8 -4
  188. package/dist/tempo/server/internal/transport.d.ts.map +1 -1
  189. package/dist/tempo/server/internal/transport.js +8 -7
  190. package/dist/tempo/server/internal/transport.js.map +1 -1
  191. package/dist/tempo/session/Snapshot.d.ts +32 -0
  192. package/dist/tempo/session/Snapshot.d.ts.map +1 -0
  193. package/dist/tempo/session/Snapshot.js +37 -0
  194. package/dist/tempo/session/Snapshot.js.map +1 -0
  195. package/dist/tempo/session/client/ChannelOps.d.ts +82 -0
  196. package/dist/tempo/session/client/ChannelOps.d.ts.map +1 -0
  197. package/dist/tempo/session/client/ChannelOps.js +204 -0
  198. package/dist/tempo/session/client/ChannelOps.js.map +1 -0
  199. package/dist/tempo/session/client/CredentialState.d.ts +262 -0
  200. package/dist/tempo/session/client/CredentialState.d.ts.map +1 -0
  201. package/dist/tempo/session/client/CredentialState.js +417 -0
  202. package/dist/tempo/session/client/CredentialState.js.map +1 -0
  203. package/dist/tempo/session/client/ReceiptCoordinator.d.ts +26 -0
  204. package/dist/tempo/session/client/ReceiptCoordinator.d.ts.map +1 -0
  205. package/dist/tempo/session/client/ReceiptCoordinator.js +61 -0
  206. package/dist/tempo/session/client/ReceiptCoordinator.js.map +1 -0
  207. package/dist/tempo/session/client/Runtime.d.ts +464 -0
  208. package/dist/tempo/session/client/Runtime.d.ts.map +1 -0
  209. package/dist/tempo/session/client/Runtime.js +499 -0
  210. package/dist/tempo/session/client/Runtime.js.map +1 -0
  211. package/dist/tempo/session/client/Session.d.ts +132 -0
  212. package/dist/tempo/session/client/Session.d.ts.map +1 -0
  213. package/dist/tempo/session/client/Session.js +55 -0
  214. package/dist/tempo/session/client/Session.js.map +1 -0
  215. package/dist/tempo/session/client/SessionManager.d.ts +120 -0
  216. package/dist/tempo/session/client/SessionManager.d.ts.map +1 -0
  217. package/dist/tempo/session/client/SessionManager.js +627 -0
  218. package/dist/tempo/session/client/SessionManager.js.map +1 -0
  219. package/dist/tempo/session/client/Transports.d.ts +449 -0
  220. package/dist/tempo/session/client/Transports.d.ts.map +1 -0
  221. package/dist/tempo/session/client/Transports.js +721 -0
  222. package/dist/tempo/session/client/Transports.js.map +1 -0
  223. package/dist/tempo/session/client/index.d.ts +12 -0
  224. package/dist/tempo/session/client/index.d.ts.map +1 -0
  225. package/dist/tempo/session/client/index.js +5 -0
  226. package/dist/tempo/session/client/index.js.map +1 -0
  227. package/dist/tempo/session/index.d.ts +7 -8
  228. package/dist/tempo/session/index.d.ts.map +1 -1
  229. package/dist/tempo/session/index.js +7 -8
  230. package/dist/tempo/session/index.js.map +1 -1
  231. package/dist/tempo/session/precompile/Chain.d.ts +319 -0
  232. package/dist/tempo/session/precompile/Chain.d.ts.map +1 -0
  233. package/dist/tempo/session/precompile/Chain.js +492 -0
  234. package/dist/tempo/session/precompile/Chain.js.map +1 -0
  235. package/dist/tempo/session/precompile/Channel.d.ts +46 -0
  236. package/dist/tempo/session/precompile/Channel.d.ts.map +1 -0
  237. package/dist/tempo/session/precompile/Channel.js +56 -0
  238. package/dist/tempo/session/precompile/Channel.js.map +1 -0
  239. package/dist/tempo/session/precompile/Protocol.d.ts +308 -0
  240. package/dist/tempo/session/precompile/Protocol.d.ts.map +1 -0
  241. package/dist/tempo/session/precompile/Protocol.js +264 -0
  242. package/dist/tempo/session/precompile/Protocol.js.map +1 -0
  243. package/dist/tempo/session/precompile/Voucher.d.ts +40 -0
  244. package/dist/tempo/session/precompile/Voucher.d.ts.map +1 -0
  245. package/dist/tempo/session/precompile/Voucher.js +126 -0
  246. package/dist/tempo/session/precompile/Voucher.js.map +1 -0
  247. package/dist/tempo/session/precompile/escrow.abi.d.ts +522 -0
  248. package/dist/tempo/session/precompile/escrow.abi.d.ts.map +1 -0
  249. package/dist/tempo/session/precompile/escrow.abi.js +224 -0
  250. package/dist/tempo/session/precompile/escrow.abi.js.map +1 -0
  251. package/dist/tempo/session/precompile/index.d.ts +24 -0
  252. package/dist/tempo/session/precompile/index.d.ts.map +1 -0
  253. package/dist/tempo/session/precompile/index.js +22 -0
  254. package/dist/tempo/session/precompile/index.js.map +1 -0
  255. package/dist/tempo/session/server/ChannelOps.d.ts +56 -0
  256. package/dist/tempo/session/server/ChannelOps.d.ts.map +1 -0
  257. package/dist/tempo/session/server/ChannelOps.js +91 -0
  258. package/dist/tempo/session/server/ChannelOps.js.map +1 -0
  259. package/dist/tempo/session/server/ChannelStore.d.ts +347 -0
  260. package/dist/tempo/session/server/ChannelStore.d.ts.map +1 -0
  261. package/dist/tempo/session/server/ChannelStore.js +404 -0
  262. package/dist/tempo/session/server/ChannelStore.js.map +1 -0
  263. package/dist/tempo/session/server/CredentialVerification.d.ts +85 -0
  264. package/dist/tempo/session/server/CredentialVerification.d.ts.map +1 -0
  265. package/dist/tempo/session/server/CredentialVerification.js +494 -0
  266. package/dist/tempo/session/server/CredentialVerification.js.map +1 -0
  267. package/dist/tempo/session/server/MeteredStream.d.ts +40 -0
  268. package/dist/tempo/session/server/MeteredStream.d.ts.map +1 -0
  269. package/dist/tempo/session/server/MeteredStream.js +42 -0
  270. package/dist/tempo/session/server/MeteredStream.js.map +1 -0
  271. package/dist/tempo/session/server/RequestState.d.ts +208 -0
  272. package/dist/tempo/session/server/RequestState.d.ts.map +1 -0
  273. package/dist/tempo/session/server/RequestState.js +252 -0
  274. package/dist/tempo/session/server/RequestState.js.map +1 -0
  275. package/dist/tempo/session/server/Session.d.ts +169 -0
  276. package/dist/tempo/session/server/Session.d.ts.map +1 -0
  277. package/dist/tempo/session/server/Session.js +351 -0
  278. package/dist/tempo/session/server/Session.js.map +1 -0
  279. package/dist/tempo/session/server/Settlement.d.ts +185 -0
  280. package/dist/tempo/session/server/Settlement.d.ts.map +1 -0
  281. package/dist/tempo/session/server/Settlement.js +250 -0
  282. package/dist/tempo/session/server/Settlement.js.map +1 -0
  283. package/dist/tempo/session/{Sse.d.ts → server/Sse.d.ts} +9 -56
  284. package/dist/tempo/session/server/Sse.d.ts.map +1 -0
  285. package/dist/tempo/session/server/Sse.js +184 -0
  286. package/dist/tempo/session/server/Sse.js.map +1 -0
  287. package/dist/tempo/session/server/Transports.d.ts +89 -0
  288. package/dist/tempo/session/server/Transports.d.ts.map +1 -0
  289. package/dist/tempo/session/server/Transports.js +149 -0
  290. package/dist/tempo/session/server/Transports.js.map +1 -0
  291. package/dist/tempo/session/server/Ws.d.ts +48 -0
  292. package/dist/tempo/session/server/Ws.d.ts.map +1 -0
  293. package/dist/tempo/session/server/Ws.js +244 -0
  294. package/dist/tempo/session/server/Ws.js.map +1 -0
  295. package/dist/tempo/session/server/index.d.ts +4 -0
  296. package/dist/tempo/session/server/index.d.ts.map +1 -0
  297. package/dist/tempo/session/server/index.js +2 -0
  298. package/dist/tempo/session/server/index.js.map +1 -0
  299. package/package.json +8 -3
  300. package/src/Challenge.ts +9 -7
  301. package/src/Constants.ts +58 -0
  302. package/src/Credential.ts +5 -4
  303. package/src/Method.ts +46 -5
  304. package/src/Receipt.ts +3 -2
  305. package/src/cli/cli.test.ts +23 -28
  306. package/src/cli/cli.ts +23 -10
  307. package/src/cli/mcp.test.ts +21 -7
  308. package/src/cli/plugins/tempo.ts +21 -8
  309. package/src/cli/utils.test.ts +25 -1
  310. package/src/cli/utils.ts +10 -0
  311. package/src/client/Methods.ts +5 -2
  312. package/src/client/Mppx.test-d.ts +10 -0
  313. package/src/client/Mppx.test.ts +75 -0
  314. package/src/client/Transport.ts +4 -5
  315. package/src/client/index.ts +11 -1
  316. package/src/client/internal/Fetch.test.ts +29 -4
  317. package/src/client/internal/Fetch.ts +17 -5
  318. package/src/env.d.ts +1 -1
  319. package/src/index.ts +1 -0
  320. package/src/internal/AcceptPayment.test.ts +61 -0
  321. package/src/internal/AcceptPayment.ts +21 -14
  322. package/src/mcp-sdk/client/McpClient.integration.test.ts +8 -7
  323. package/src/mcp-sdk/client/McpClient.test-d.ts +7 -0
  324. package/src/mcp-sdk/client/McpClient.ts +99 -67
  325. package/src/mcp-sdk/client/McpClient.unit.test.ts +131 -0
  326. package/src/middlewares/elysia.test.ts +8 -4
  327. package/src/middlewares/express.test.ts +8 -4
  328. package/src/middlewares/hono.test.ts +4 -4
  329. package/src/middlewares/nextjs.test.ts +8 -4
  330. package/src/proxy/Proxy.test.ts +8 -8
  331. package/src/server/Mppx.test-d.ts +54 -0
  332. package/src/server/Mppx.test.ts +274 -7
  333. package/src/server/Mppx.ts +487 -406
  334. package/src/server/Request.test.ts +81 -0
  335. package/src/server/Request.ts +23 -9
  336. package/src/server/Response.ts +2 -1
  337. package/src/server/Transport.ts +4 -3
  338. package/src/server/index.ts +1 -0
  339. package/src/stripe/client/Charge.test.ts +20 -5
  340. package/src/stripe/client/Charge.ts +6 -2
  341. package/src/stripe/server/Charge.test.ts +114 -1
  342. package/src/stripe/server/Charge.ts +13 -2
  343. package/src/stripe/server/internal/html/package.json +1 -1
  344. package/src/stripe/server/internal/html.gen.ts +1 -1
  345. package/src/tempo/AccessKeyAuthorization.test.ts +4 -94
  346. package/src/tempo/Methods.test.ts +45 -17
  347. package/src/tempo/Methods.ts +22 -0
  348. package/src/tempo/PublicExports.test-d.ts +105 -0
  349. package/src/tempo/client/Charge.test.ts +85 -0
  350. package/src/tempo/client/Charge.ts +19 -2
  351. package/src/tempo/client/Methods.ts +18 -6
  352. package/src/tempo/client/index.ts +15 -4
  353. package/src/tempo/index.ts +1 -0
  354. package/src/tempo/internal/fee-payer.test.ts +241 -17
  355. package/src/tempo/internal/fee-payer.ts +150 -4
  356. package/src/tempo/internal/fee-token.test.ts +14 -9
  357. package/src/tempo/legacy/AccessKeyAuthorization.test.ts +162 -0
  358. package/src/tempo/legacy/README.md +9 -0
  359. package/src/tempo/{client → legacy/client}/ChannelOps.test.ts +6 -7
  360. package/src/tempo/{client → legacy/client}/ChannelOps.ts +22 -9
  361. package/src/tempo/{client → legacy/client}/Session.test.ts +51 -9
  362. package/src/tempo/{client → legacy/client}/Session.ts +25 -11
  363. package/src/tempo/{client → legacy/client}/SessionManager.test.ts +81 -9
  364. package/src/tempo/{client → legacy/client}/SessionManager.ts +41 -20
  365. package/src/tempo/legacy/client/index.ts +6 -0
  366. package/src/tempo/legacy/index.ts +6 -0
  367. package/src/tempo/{server → legacy/server}/Session.test.ts +162 -63
  368. package/src/tempo/{server → legacy/server}/Session.ts +49 -37
  369. package/src/tempo/legacy/server/index.ts +4 -0
  370. package/src/tempo/{session → legacy/session}/Chain.test.ts +3 -4
  371. package/src/tempo/{session → legacy/session}/Chain.ts +94 -63
  372. package/src/tempo/{session → legacy/session}/Channel.ts +1 -0
  373. package/src/tempo/legacy/session/ChannelStore.test.ts +58 -0
  374. package/src/tempo/legacy/session/ChannelStore.ts +39 -0
  375. package/src/tempo/legacy/session/Types.ts +91 -0
  376. package/src/tempo/{session → legacy/session}/Voucher.ts +12 -8
  377. package/src/tempo/{session → legacy/session}/escrow.abi.ts +1 -0
  378. package/src/tempo/legacy/session/index.ts +8 -0
  379. package/src/tempo/server/AtomicStore.test-d.ts +16 -11
  380. package/src/tempo/server/Charge.test.ts +92 -14
  381. package/src/tempo/server/Charge.ts +18 -16
  382. package/src/tempo/server/Methods.ts +54 -8
  383. package/src/tempo/server/Sse.test.ts +2 -2
  384. package/src/tempo/server/index.ts +6 -5
  385. package/src/tempo/server/internal/html/package.json +1 -1
  386. package/src/tempo/server/internal/html.gen.ts +1 -1
  387. package/src/tempo/server/internal/request-body.test.ts +37 -4
  388. package/src/tempo/server/internal/request-body.ts +25 -6
  389. package/src/tempo/server/internal/transport.test.ts +4 -4
  390. package/src/tempo/server/internal/transport.ts +19 -10
  391. package/src/tempo/session/Snapshot.test.ts +41 -0
  392. package/src/tempo/session/Snapshot.ts +74 -0
  393. package/src/tempo/session/client/ChannelOps.test.ts +163 -0
  394. package/src/tempo/session/client/ChannelOps.ts +344 -0
  395. package/src/tempo/session/client/CredentialState.test.ts +645 -0
  396. package/src/tempo/session/client/CredentialState.ts +814 -0
  397. package/src/tempo/session/client/ReceiptCoordinator.ts +95 -0
  398. package/src/tempo/session/client/Runtime.test.ts +1092 -0
  399. package/src/tempo/session/client/Runtime.ts +986 -0
  400. package/src/tempo/session/client/Session.test.ts +734 -0
  401. package/src/tempo/session/client/Session.ts +97 -0
  402. package/src/tempo/session/client/SessionManager.test.ts +1308 -0
  403. package/src/tempo/session/client/SessionManager.ts +845 -0
  404. package/src/tempo/session/client/Transports.test.ts +837 -0
  405. package/src/tempo/session/client/Transports.ts +1292 -0
  406. package/src/tempo/session/client/index.ts +37 -0
  407. package/src/tempo/session/index.ts +7 -8
  408. package/src/tempo/session/precompile/Chain.integration.test.ts +321 -0
  409. package/src/tempo/session/precompile/Chain.test.ts +1258 -0
  410. package/src/tempo/session/precompile/Chain.ts +979 -0
  411. package/src/tempo/session/precompile/Channel.test.ts +138 -0
  412. package/src/tempo/session/precompile/Channel.ts +103 -0
  413. package/src/tempo/session/precompile/Protocol.test.ts +358 -0
  414. package/src/tempo/session/precompile/Protocol.ts +520 -0
  415. package/src/tempo/session/precompile/Voucher.test.ts +316 -0
  416. package/src/tempo/session/precompile/Voucher.ts +160 -0
  417. package/src/tempo/session/precompile/escrow.abi.ts +226 -0
  418. package/src/tempo/session/precompile/index.ts +33 -0
  419. package/src/tempo/session/server/ChannelOps.test.ts +129 -0
  420. package/src/tempo/session/server/ChannelOps.ts +157 -0
  421. package/src/tempo/session/{ChannelStore.test.ts → server/ChannelStore.test.ts} +536 -29
  422. package/src/tempo/session/server/ChannelStore.ts +835 -0
  423. package/src/tempo/session/server/CredentialVerification.test.ts +146 -0
  424. package/src/tempo/session/server/CredentialVerification.ts +710 -0
  425. package/src/tempo/session/server/MeteredStream.ts +88 -0
  426. package/src/tempo/session/server/RequestState.test.ts +531 -0
  427. package/src/tempo/session/server/RequestState.ts +499 -0
  428. package/src/tempo/session/server/Session.integration.test.ts +444 -0
  429. package/src/tempo/session/server/Session.test.ts +3253 -0
  430. package/src/tempo/session/server/Session.ts +543 -0
  431. package/src/tempo/session/server/Settlement.test.ts +242 -0
  432. package/src/tempo/session/server/Settlement.ts +470 -0
  433. package/src/tempo/session/{Sse.test.ts → server/Sse.test.ts} +37 -3
  434. package/src/tempo/session/server/Sse.ts +256 -0
  435. package/src/tempo/session/server/Transports.test.ts +346 -0
  436. package/src/tempo/session/server/Transports.ts +255 -0
  437. package/src/tempo/session/{Ws.test.ts → server/Ws.test.ts} +4 -4
  438. package/src/tempo/session/server/Ws.ts +384 -0
  439. package/src/tempo/session/server/index.ts +8 -0
  440. package/dist/tempo/client/ChannelOps.d.ts.map +0 -1
  441. package/dist/tempo/client/ChannelOps.js.map +0 -1
  442. package/dist/tempo/client/Session.d.ts.map +0 -1
  443. package/dist/tempo/client/Session.js.map +0 -1
  444. package/dist/tempo/client/SessionManager.d.ts.map +0 -1
  445. package/dist/tempo/client/SessionManager.js.map +0 -1
  446. package/dist/tempo/server/Session.d.ts.map +0 -1
  447. package/dist/tempo/server/Session.js.map +0 -1
  448. package/dist/tempo/session/Chain.d.ts.map +0 -1
  449. package/dist/tempo/session/Chain.js.map +0 -1
  450. package/dist/tempo/session/Channel.d.ts.map +0 -1
  451. package/dist/tempo/session/Channel.js.map +0 -1
  452. package/dist/tempo/session/ChannelStore.d.ts +0 -117
  453. package/dist/tempo/session/ChannelStore.d.ts.map +0 -1
  454. package/dist/tempo/session/ChannelStore.js +0 -172
  455. package/dist/tempo/session/ChannelStore.js.map +0 -1
  456. package/dist/tempo/session/Receipt.d.ts +0 -22
  457. package/dist/tempo/session/Receipt.d.ts.map +0 -1
  458. package/dist/tempo/session/Receipt.js +0 -34
  459. package/dist/tempo/session/Receipt.js.map +0 -1
  460. package/dist/tempo/session/Sse.d.ts.map +0 -1
  461. package/dist/tempo/session/Sse.js +0 -363
  462. package/dist/tempo/session/Sse.js.map +0 -1
  463. package/dist/tempo/session/Types.d.ts +0 -78
  464. package/dist/tempo/session/Types.d.ts.map +0 -1
  465. package/dist/tempo/session/Types.js.map +0 -1
  466. package/dist/tempo/session/Voucher.d.ts.map +0 -1
  467. package/dist/tempo/session/Voucher.js.map +0 -1
  468. package/dist/tempo/session/Ws.d.ts +0 -87
  469. package/dist/tempo/session/Ws.d.ts.map +0 -1
  470. package/dist/tempo/session/Ws.js +0 -443
  471. package/dist/tempo/session/Ws.js.map +0 -1
  472. package/dist/tempo/session/escrow.abi.js.map +0 -1
  473. package/src/tempo/session/ChannelStore.ts +0 -308
  474. package/src/tempo/session/Receipt.test.ts +0 -89
  475. package/src/tempo/session/Receipt.ts +0 -46
  476. package/src/tempo/session/Sse.ts +0 -462
  477. package/src/tempo/session/Types.ts +0 -86
  478. package/src/tempo/session/Ws.ts +0 -576
  479. /package/dist/tempo/{session → legacy/session}/Channel.js +0 -0
  480. /package/dist/tempo/{session → legacy/session}/Types.js +0 -0
  481. /package/src/tempo/{session → legacy/session}/Channel.test.ts +0 -0
  482. /package/src/tempo/{session → legacy/session}/Voucher.test.ts +0 -0
  483. /package/src/tempo/session/{Sse.fuzz.test.ts → server/Sse.fuzz.test.ts} +0 -0
@@ -0,0 +1,645 @@
1
+ import { createClient, custom, type Address, type Client, type Hex } from 'viem'
2
+ import { privateKeyToAccount } from 'viem/accounts'
3
+ import { describe, expect, test } from 'vp/test'
4
+
5
+ import * as Challenge from '../../../Challenge.js'
6
+ import * as Constants from '../../../Constants.js'
7
+ import type * as z from '../../../zod.js'
8
+ import * as Channel from '../precompile/Channel.js'
9
+ import { tip20ChannelEscrow, type SessionCredentialPayload } from '../precompile/Protocol.js'
10
+ import type { SessionSnapshot } from '../Snapshot.js'
11
+ import type { ChannelEntry } from './ChannelOps.js'
12
+ import {
13
+ channelKey,
14
+ createChannelCache,
15
+ executeCredentialPlan,
16
+ hasCredentialCumulativeAmount,
17
+ hasManualSessionDescriptor,
18
+ hasSessionAction,
19
+ hasSessionDescriptor,
20
+ parseOptionalContextAmount,
21
+ planCredential,
22
+ readCredentialCumulativeAmount,
23
+ requireContextAmount,
24
+ resolveChallengeContext,
25
+ resolveRecoveredCumulative,
26
+ resolveRecoverContext,
27
+ resolveReusableChannel,
28
+ sessionContextSchema,
29
+ storeChannelEntry,
30
+ updateCachedCumulative,
31
+ type ChallengeContext,
32
+ type SessionContext,
33
+ } from './CredentialState.js'
34
+
35
+ describe('ChannelCache', () => {
36
+ const channelId = `0x${'11'.repeat(32)}` as Hex
37
+
38
+ function channel(overrides: Partial<ChannelEntry> = {}): ChannelEntry {
39
+ return {
40
+ channelId,
41
+ cumulativeAmount: 10n,
42
+ deposit: 20n,
43
+ descriptor: {
44
+ payer: '0x0000000000000000000000000000000000000001',
45
+ payee: '0x0000000000000000000000000000000000000002',
46
+ operator: '0x0000000000000000000000000000000000000000',
47
+ token: '0x20c0000000000000000000000000000000000001',
48
+ salt: `0x${'22'.repeat(32)}`,
49
+ authorizedSigner: '0x0000000000000000000000000000000000000001',
50
+ expiringNonceHash: `0x${'33'.repeat(32)}`,
51
+ },
52
+ escrow: '0x4D50500000000000000000000000000000000000',
53
+ chainId: 4217,
54
+ opened: true,
55
+ ...overrides,
56
+ }
57
+ }
58
+
59
+ function voucher(cumulativeAmount: string): SessionCredentialPayload {
60
+ return {
61
+ action: 'voucher',
62
+ channelId,
63
+ descriptor: channel().descriptor,
64
+ cumulativeAmount,
65
+ signature: '0x1234',
66
+ }
67
+ }
68
+
69
+ function close(cumulativeAmount: string): SessionCredentialPayload {
70
+ return {
71
+ action: 'close',
72
+ channelId,
73
+ descriptor: channel().descriptor,
74
+ cumulativeAmount,
75
+ signature: '0x1234',
76
+ }
77
+ }
78
+
79
+ function topUp(additionalDeposit: string): SessionCredentialPayload {
80
+ return {
81
+ action: 'topUp',
82
+ type: 'transaction',
83
+ channelId,
84
+ descriptor: channel().descriptor,
85
+ transaction: '0x1234',
86
+ additionalDeposit,
87
+ }
88
+ }
89
+
90
+ describe('precompile client ChannelCache', () => {
91
+ test('creates stable case-insensitive reusable channel keys', () => {
92
+ expect(
93
+ channelKey(
94
+ '0x00000000000000000000000000000000000000AA' as Address,
95
+ '0x20C0000000000000000000000000000000000001' as Address,
96
+ '0x4D50500000000000000000000000000000000000' as Address,
97
+ ),
98
+ ).toBe(
99
+ '0x00000000000000000000000000000000000000aa:0x20c0000000000000000000000000000000000001:0x4d50500000000000000000000000000000000000',
100
+ )
101
+ })
102
+
103
+ test('stores entries by key and channel ID and notifies observers', () => {
104
+ const updates: ChannelEntry[] = []
105
+ const cache = createChannelCache((entry) => updates.push(entry))
106
+ const entry = channel()
107
+
108
+ storeChannelEntry(cache, 'payee:token:escrow', entry)
109
+
110
+ expect(cache.channels.get('payee:token:escrow')).toBe(entry)
111
+ expect(cache.channelIdToKey.get(channelId)).toBe('payee:token:escrow')
112
+ expect(updates).toEqual([entry])
113
+ })
114
+
115
+ test('updates cached cumulative amounts monotonically', () => {
116
+ const cache = createChannelCache()
117
+ const entry = channel({ cumulativeAmount: 10n })
118
+ storeChannelEntry(cache, 'payee:token:escrow', entry)
119
+
120
+ updateCachedCumulative(cache, channelId, voucher('8'))
121
+ expect(entry.cumulativeAmount).toBe(10n)
122
+
123
+ updateCachedCumulative(cache, channelId, voucher('12'))
124
+ expect(entry.cumulativeAmount).toBe(12n)
125
+ })
126
+
127
+ test('reads cumulative amounts only from cumulative credential payloads', () => {
128
+ expect(hasCredentialCumulativeAmount(voucher('12'))).toBe(true)
129
+ expect(readCredentialCumulativeAmount(voucher('12'))).toBe(12n)
130
+ expect(hasCredentialCumulativeAmount(topUp('12'))).toBe(false)
131
+ expect(readCredentialCumulativeAmount(topUp('12'))).toBeUndefined()
132
+ })
133
+
134
+ test('ignores non-cumulative top-up credentials when updating cached cumulative amount', () => {
135
+ const cache = createChannelCache()
136
+ const entry = channel({ cumulativeAmount: 10n })
137
+ storeChannelEntry(cache, 'payee:token:escrow', entry)
138
+
139
+ updateCachedCumulative(cache, channelId, topUp('12'))
140
+
141
+ expect(entry.cumulativeAmount).toBe(10n)
142
+ })
143
+
144
+ test('marks cached channels closed from close credentials', () => {
145
+ const cache = createChannelCache()
146
+ const entry = channel({ opened: true })
147
+ storeChannelEntry(cache, 'payee:token:escrow', entry)
148
+
149
+ updateCachedCumulative(cache, channelId, close('12'))
150
+
151
+ expect(entry.cumulativeAmount).toBe(12n)
152
+ expect(entry.opened).toBe(false)
153
+ })
154
+ })
155
+ })
156
+
157
+ describe('Context', () => {
158
+ const descriptor = {
159
+ payer: '0x0000000000000000000000000000000000000001' as Address,
160
+ payee: '0x0000000000000000000000000000000000000002' as Address,
161
+ operator: '0x0000000000000000000000000000000000000000' as Address,
162
+ token: '0x0000000000000000000000000000000000000003' as Address,
163
+ salt: `0x${'11'.repeat(32)}` as const,
164
+ authorizedSigner: '0x0000000000000000000000000000000000000001' as Address,
165
+ expiringNonceHash: `0x${'22'.repeat(32)}` as const,
166
+ }
167
+
168
+ describe('precompile session client context helpers', () => {
169
+ test('keeps runtime schema aligned with documented context type', () => {
170
+ type SchemaContext = z.infer<typeof sessionContextSchema>
171
+ expectTypeOf<SchemaContext>().toEqualTypeOf<SessionContext>()
172
+ })
173
+
174
+ test('narrows manual action context', () => {
175
+ expect(hasSessionAction({ action: 'voucher' })).toBe(true)
176
+ expect(hasSessionAction({})).toBe(false)
177
+ expect(hasSessionAction(undefined)).toBe(false)
178
+ })
179
+
180
+ test('narrows descriptor recovery context', () => {
181
+ expect(hasSessionDescriptor({ descriptor })).toBe(true)
182
+ expect(hasSessionDescriptor({})).toBe(false)
183
+ expect(hasSessionDescriptor(undefined)).toBe(false)
184
+ })
185
+
186
+ test('narrows manual action context with descriptor', () => {
187
+ expect(hasManualSessionDescriptor({ action: 'voucher', descriptor })).toBe(true)
188
+ expect(hasManualSessionDescriptor({ action: 'voucher' })).toBe(false)
189
+ expect(hasManualSessionDescriptor({ descriptor })).toBe(false)
190
+ })
191
+
192
+ test('prefers raw amount fields over human-readable amount fields', () => {
193
+ expect(
194
+ parseOptionalContextAmount(
195
+ {
196
+ cumulativeAmount: '100',
197
+ cumulativeAmountRaw: '42',
198
+ },
199
+ 6,
200
+ 'cumulativeAmount',
201
+ ),
202
+ ).toBe(42n)
203
+ })
204
+
205
+ test('parses human-readable amount fields with caller decimals', () => {
206
+ expect(parseOptionalContextAmount({ additionalDeposit: '1.5' }, 6, 'additionalDeposit')).toBe(
207
+ 1_500_000n,
208
+ )
209
+ })
210
+
211
+ test('returns undefined for absent optional amounts', () => {
212
+ expect(parseOptionalContextAmount({}, 6, 'cumulativeAmount')).toBeUndefined()
213
+ })
214
+
215
+ test('throws action-specific errors for absent required amounts', () => {
216
+ expect(() => requireContextAmount({}, 6, 'cumulativeAmount', 'voucher')).toThrow(
217
+ 'cumulativeAmount required for voucher action',
218
+ )
219
+ })
220
+ })
221
+ })
222
+
223
+ describe('CredentialPlan', () => {
224
+ const account = privateKeyToAccount(
225
+ '0x1000000000000000000000000000000000000000000000000000000000000000',
226
+ )
227
+ const channelId = `0x${'11'.repeat(32)}` as Hex
228
+ const snapshotChannelId = `0x${'12'.repeat(32)}` as Hex
229
+ const escrow = '0x4D50500000000000000000000000000000000000' as Address
230
+ const payee = '0x0000000000000000000000000000000000000002' as Address
231
+ const token = '0x20c0000000000000000000000000000000000001' as Address
232
+
233
+ const descriptor = {
234
+ payer: account.address,
235
+ payee,
236
+ operator: '0x0000000000000000000000000000000000000000' as Address,
237
+ token,
238
+ salt: `0x${'22'.repeat(32)}` as Hex,
239
+ authorizedSigner: account.address,
240
+ expiringNonceHash: `0x${'33'.repeat(32)}` as Hex,
241
+ }
242
+
243
+ const snapshotDescriptor = {
244
+ ...descriptor,
245
+ salt: `0x${'44'.repeat(32)}` as Hex,
246
+ }
247
+
248
+ const client = createClient({
249
+ transport: custom({
250
+ async request() {
251
+ return null
252
+ },
253
+ }),
254
+ })
255
+
256
+ function challengeContext(overrides: Partial<ChallengeContext> = {}): ChallengeContext {
257
+ return {
258
+ amount: 5n,
259
+ challenge: Challenge.from({
260
+ id: 'challenge-1',
261
+ realm: 'example.test',
262
+ method: 'tempo',
263
+ intent: 'session',
264
+ request: {},
265
+ }),
266
+ chainId: 4217,
267
+ client,
268
+ escrow,
269
+ key: 'payee:token:escrow',
270
+ payee,
271
+ token,
272
+ ...overrides,
273
+ }
274
+ }
275
+
276
+ function channel(overrides: Partial<ChannelEntry> = {}): ChannelEntry {
277
+ return {
278
+ channelId,
279
+ cumulativeAmount: 10n,
280
+ deposit: 20n,
281
+ descriptor,
282
+ escrow,
283
+ chainId: 4217,
284
+ opened: true,
285
+ ...overrides,
286
+ }
287
+ }
288
+
289
+ function snapshot(overrides: Partial<SessionSnapshot> = {}): SessionSnapshot {
290
+ return {
291
+ acceptedCumulative: '10',
292
+ chainId: 42431,
293
+ channelId: snapshotChannelId,
294
+ deposit: '20',
295
+ descriptor: snapshotDescriptor,
296
+ escrow,
297
+ requiredCumulative: '10',
298
+ settled: '0',
299
+ spent: '0',
300
+ ...overrides,
301
+ }
302
+ }
303
+
304
+ function paymentChallenge(
305
+ overrides: Partial<Challenge.Challenge['request']> = {},
306
+ ): Challenge.Challenge {
307
+ return {
308
+ id: 'challenge-1',
309
+ realm: 'test',
310
+ method: 'tempo',
311
+ intent: 'session',
312
+ request: {
313
+ amount: '10',
314
+ currency: token,
315
+ methodDetails: {
316
+ chainId: 42431,
317
+ escrowContract: escrow,
318
+ feePayer: true,
319
+ [Constants.MethodDetailKeys.sessionSnapshot]: snapshot(),
320
+ },
321
+ recipient: payee,
322
+ suggestedDeposit: '100',
323
+ unitType: 'request',
324
+ ...overrides,
325
+ },
326
+ }
327
+ }
328
+
329
+ describe('precompile client CredentialPlan', () => {
330
+ test('resolves typed credential-planning fields from challenge details', async () => {
331
+ const client = { chain: { id: 42431 } } as Client
332
+ const resolved = await resolveChallengeContext({
333
+ challenge: paymentChallenge(),
334
+ getClient: async () => client,
335
+ })
336
+
337
+ expect(resolved).toMatchObject({
338
+ amount: 10n,
339
+ chainId: 42431,
340
+ client,
341
+ escrow,
342
+ feePayer: true,
343
+ payee,
344
+ snapshot: snapshot(),
345
+ suggestedDepositRaw: '100',
346
+ token,
347
+ })
348
+ expect(resolved.key).toBe(
349
+ `${payee.toLowerCase()}:${token.toLowerCase()}:${escrow.toLowerCase()}`,
350
+ )
351
+ })
352
+
353
+ test('uses client chain ID when the challenge omits one', async () => {
354
+ const resolved = await resolveChallengeContext({
355
+ challenge: paymentChallenge({ methodDetails: { escrowContract: escrow } }),
356
+ getClient: async () => ({ chain: { id: 4217 } }) as Client,
357
+ })
358
+
359
+ expect(resolved.chainId).toBe(4217)
360
+ })
361
+
362
+ test('rejects challenges without required payment fields', async () => {
363
+ await expect(
364
+ resolveChallengeContext({
365
+ challenge: paymentChallenge({ recipient: undefined }),
366
+ getClient: async () => ({ chain: { id: 4217 } }) as Client,
367
+ }),
368
+ ).rejects.toThrow('tempo session challenge missing recipient')
369
+
370
+ await expect(
371
+ resolveChallengeContext({
372
+ challenge: paymentChallenge({ currency: 'pathUSD' }),
373
+ getClient: async () => ({ chain: { id: 4217 } }) as Client,
374
+ }),
375
+ ).rejects.toThrow('tempo session challenge missing currency')
376
+ })
377
+
378
+ test('ignores invalid optional escrow hints', async () => {
379
+ const resolved = await resolveChallengeContext({
380
+ challenge: paymentChallenge({
381
+ methodDetails: {
382
+ chainId: 42431,
383
+ escrowContract: 'not-an-address',
384
+ },
385
+ }),
386
+ getClient: async () => ({ chain: { id: 42431 } }) as Client,
387
+ })
388
+
389
+ expect(resolved.escrow).toBe(tip20ChannelEscrow)
390
+ })
391
+
392
+ test('recover context prefers caller descriptor and channel ID over server snapshot', () => {
393
+ expect(
394
+ resolveRecoverContext({
395
+ context: { channelId, descriptor },
396
+ snapshot: snapshot(),
397
+ }),
398
+ ).toMatchObject({
399
+ channelId,
400
+ descriptor,
401
+ })
402
+ })
403
+
404
+ test('recover context can be bootstrapped entirely from server snapshot', () => {
405
+ expect(resolveRecoverContext({ snapshot: snapshot() })).toMatchObject({
406
+ channelId: snapshotChannelId,
407
+ descriptor: snapshotDescriptor,
408
+ })
409
+ })
410
+
411
+ test('recovery cumulative ignores server-advertised unused voucher headroom', () => {
412
+ expect(
413
+ resolveRecoveredCumulative({
414
+ context: { descriptor, channelId },
415
+ decimals: 6,
416
+ requestAmount: 5n,
417
+ settled: 0n,
418
+ snapshot: snapshot({
419
+ acceptedCumulative: '1000000',
420
+ requiredCumulative: '1000000',
421
+ spent: '10',
422
+ }),
423
+ }),
424
+ ).toBe(15n)
425
+ })
426
+
427
+ test('plans manual credentials only when an explicit action includes descriptor', () => {
428
+ const cache = createChannelCache()
429
+ const plan = planCredential({
430
+ account,
431
+ cache,
432
+ context: { action: 'voucher', descriptor, cumulativeAmountRaw: '10' },
433
+ decimals: 6,
434
+ resolved: challengeContext(),
435
+ })
436
+
437
+ expect(plan.type).toBe('manual')
438
+ if (plan.type !== 'manual') throw new Error('expected manual plan')
439
+ expect(plan.context.descriptor).toBe(descriptor)
440
+ })
441
+
442
+ test('rejects manual actions without a descriptor', () => {
443
+ expect(() =>
444
+ planCredential({
445
+ account,
446
+ cache: createChannelCache(),
447
+ context: { action: 'voucher', cumulativeAmountRaw: '10' },
448
+ decimals: 6,
449
+ resolved: challengeContext(),
450
+ }),
451
+ ).toThrow('descriptor required for TIP-1034 session action')
452
+ })
453
+
454
+ test('rejects manual descriptors that do not match the active challenge', async () => {
455
+ const plan = planCredential({
456
+ account,
457
+ cache: createChannelCache(),
458
+ context: {
459
+ action: 'voucher',
460
+ cumulativeAmountRaw: '10',
461
+ descriptor: {
462
+ ...descriptor,
463
+ payee: '0x0000000000000000000000000000000000000003' as Address,
464
+ },
465
+ },
466
+ decimals: 6,
467
+ resolved: challengeContext(),
468
+ })
469
+
470
+ await expect(executeCredentialPlan(plan, createChannelCache())).rejects.toThrow(
471
+ 'context descriptor payee does not match challenge',
472
+ )
473
+ })
474
+
475
+ test('plans recovery from server snapshot when no reusable cache entry exists', () => {
476
+ const plan = planCredential({
477
+ account,
478
+ cache: createChannelCache(),
479
+ decimals: 6,
480
+ resolved: challengeContext({ snapshot: snapshot() }),
481
+ })
482
+
483
+ expect(plan.type).toBe('recover')
484
+ if (plan.type !== 'recover') throw new Error('expected recover plan')
485
+ expect(plan.context.channelId).toBe(snapshotChannelId)
486
+ expect(plan.context.descriptor).toBe(snapshotDescriptor)
487
+ })
488
+
489
+ test('plans voucher reuse before snapshot recovery when cache entry is open', () => {
490
+ const cache = createChannelCache()
491
+ const entry = channel()
492
+ storeChannelEntry(cache, 'payee:token:escrow', entry)
493
+
494
+ const plan = planCredential({
495
+ account,
496
+ cache,
497
+ decimals: 6,
498
+ resolved: challengeContext({ snapshot: snapshot() }),
499
+ })
500
+
501
+ expect(plan).toMatchObject({ type: 'voucher', entry })
502
+ })
503
+
504
+ test('rejects channel ID reuse without descriptor or cache entry', () => {
505
+ expect(() =>
506
+ planCredential({
507
+ account,
508
+ cache: createChannelCache(),
509
+ context: { channelId },
510
+ decimals: 6,
511
+ resolved: challengeContext(),
512
+ }),
513
+ ).toThrow('descriptor required to reuse TIP-1034 channel')
514
+ })
515
+
516
+ test('returns descriptor-derived channel ID and reusable on-chain state', async () => {
517
+ const state = { deposit: 1_000n, settled: 250n, closeRequestedAt: 0 }
518
+ const reusableChannelId = Channel.computeId({ ...descriptor, chainId: 42431, escrow })
519
+
520
+ await expect(
521
+ resolveReusableChannel({
522
+ channelId: reusableChannelId,
523
+ client,
524
+ descriptor,
525
+ expected: {
526
+ chainId: 42431,
527
+ escrow,
528
+ payee,
529
+ payer: account.address,
530
+ authorizedSigner: account.address,
531
+ token,
532
+ },
533
+ readChannelState: async () => state,
534
+ }),
535
+ ).resolves.toEqual({ channelId: reusableChannelId, state })
536
+ })
537
+
538
+ test('rejects reusable channel descriptor mismatches before reading chain state', async () => {
539
+ const cases = [
540
+ {
541
+ name: 'channel ID',
542
+ parameters: { channelId: `0x${'ff'.repeat(32)}` },
543
+ message: 'context channelId does not match descriptor',
544
+ },
545
+ {
546
+ name: 'payee',
547
+ parameters: {
548
+ descriptor: {
549
+ ...descriptor,
550
+ payee: '0x0000000000000000000000000000000000000003' as Address,
551
+ },
552
+ },
553
+ message: 'context descriptor payee does not match challenge',
554
+ },
555
+ {
556
+ name: 'token',
557
+ parameters: {
558
+ descriptor: {
559
+ ...descriptor,
560
+ token: '0x0000000000000000000000000000000000000004' as Address,
561
+ },
562
+ },
563
+ message: 'context descriptor token does not match challenge',
564
+ },
565
+ {
566
+ name: 'payer',
567
+ parameters: {
568
+ descriptor: {
569
+ ...descriptor,
570
+ payer: '0x0000000000000000000000000000000000000005' as Address,
571
+ },
572
+ },
573
+ message: 'context descriptor payer does not match account',
574
+ },
575
+ {
576
+ name: 'authorizedSigner',
577
+ parameters: {
578
+ descriptor: {
579
+ ...descriptor,
580
+ authorizedSigner: '0x0000000000000000000000000000000000000006' as Address,
581
+ },
582
+ },
583
+ message: 'context descriptor authorizedSigner does not match account',
584
+ },
585
+ ] as const
586
+
587
+ for (const item of cases) {
588
+ let reads = 0
589
+ await expect(
590
+ resolveReusableChannel({
591
+ client,
592
+ descriptor,
593
+ expected: {
594
+ chainId: 42431,
595
+ escrow,
596
+ payee,
597
+ payer: account.address,
598
+ authorizedSigner: account.address,
599
+ token,
600
+ },
601
+ readChannelState: async () => {
602
+ reads += 1
603
+ return { deposit: 1n, settled: 0n, closeRequestedAt: 0 }
604
+ },
605
+ ...item.parameters,
606
+ }),
607
+ ).rejects.toThrow(item.message)
608
+ expect(reads, item.name).toBe(0)
609
+ }
610
+ })
611
+
612
+ test('rejects non-reusable on-chain state', async () => {
613
+ const reusableChannelId = Channel.computeId({ ...descriptor, chainId: 42431, escrow })
614
+ const cases = [
615
+ {
616
+ state: { deposit: 0n, settled: 0n, closeRequestedAt: 0 },
617
+ message: /cannot be reused \(closed or not found on-chain\)/,
618
+ },
619
+ {
620
+ state: { deposit: 1_000n, settled: 0n, closeRequestedAt: 123 },
621
+ message: /cannot be reused \(pending close request\)/,
622
+ },
623
+ ] as const
624
+
625
+ for (const item of cases) {
626
+ await expect(
627
+ resolveReusableChannel({
628
+ channelId: reusableChannelId,
629
+ client,
630
+ descriptor,
631
+ expected: {
632
+ chainId: 42431,
633
+ escrow,
634
+ payee,
635
+ payer: account.address,
636
+ authorizedSigner: account.address,
637
+ token,
638
+ },
639
+ readChannelState: async () => item.state,
640
+ }),
641
+ ).rejects.toThrow(item.message)
642
+ }
643
+ })
644
+ })
645
+ })