@bopen-io/wallet-toolbox 1.7.18

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 (390) hide show
  1. package/.claude/settings.local.json +10 -0
  2. package/.env.template +22 -0
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +40 -0
  4. package/.github/ISSUE_TEMPLATE/discussion.md +24 -0
  5. package/.github/pull_request_template.md +22 -0
  6. package/.github/workflows/push.yaml +145 -0
  7. package/.prettierrc +10 -0
  8. package/CHANGELOG.md +280 -0
  9. package/CONTRIBUTING.md +89 -0
  10. package/README.md +43 -0
  11. package/docs/README.md +85 -0
  12. package/docs/client.md +19627 -0
  13. package/docs/monitor.md +953 -0
  14. package/docs/open-rpc/index.html +46 -0
  15. package/docs/services.md +6377 -0
  16. package/docs/setup.md +1268 -0
  17. package/docs/storage.md +5367 -0
  18. package/docs/wallet.md +19626 -0
  19. package/jest.config.ts +25 -0
  20. package/license.md +28 -0
  21. package/out/tsconfig.all.tsbuildinfo +1 -0
  22. package/package.json +63 -0
  23. package/src/CWIStyleWalletManager.ts +1999 -0
  24. package/src/Setup.ts +579 -0
  25. package/src/SetupClient.ts +322 -0
  26. package/src/SetupWallet.ts +108 -0
  27. package/src/SimpleWalletManager.ts +526 -0
  28. package/src/Wallet.ts +1169 -0
  29. package/src/WalletAuthenticationManager.ts +153 -0
  30. package/src/WalletLogger.ts +213 -0
  31. package/src/WalletPermissionsManager.ts +3660 -0
  32. package/src/WalletSettingsManager.ts +114 -0
  33. package/src/__tests/CWIStyleWalletManager.test.d.ts.map +1 -0
  34. package/src/__tests/CWIStyleWalletManager.test.js.map +1 -0
  35. package/src/__tests/CWIStyleWalletManager.test.ts +675 -0
  36. package/src/__tests/WalletPermissionsManager.callbacks.test.ts +323 -0
  37. package/src/__tests/WalletPermissionsManager.checks.test.ts +844 -0
  38. package/src/__tests/WalletPermissionsManager.encryption.test.ts +412 -0
  39. package/src/__tests/WalletPermissionsManager.fixtures.ts +307 -0
  40. package/src/__tests/WalletPermissionsManager.flows.test.ts +462 -0
  41. package/src/__tests/WalletPermissionsManager.initialization.test.ts +300 -0
  42. package/src/__tests/WalletPermissionsManager.pmodules.test.ts +798 -0
  43. package/src/__tests/WalletPermissionsManager.proxying.test.ts +724 -0
  44. package/src/__tests/WalletPermissionsManager.tokens.test.ts +503 -0
  45. package/src/index.all.ts +27 -0
  46. package/src/index.client.ts +25 -0
  47. package/src/index.mobile.ts +21 -0
  48. package/src/index.ts +1 -0
  49. package/src/monitor/Monitor.ts +412 -0
  50. package/src/monitor/MonitorDaemon.ts +188 -0
  51. package/src/monitor/README.md +3 -0
  52. package/src/monitor/__test/MonitorDaemon.man.test.ts +45 -0
  53. package/src/monitor/tasks/TaskCheckForProofs.ts +243 -0
  54. package/src/monitor/tasks/TaskCheckNoSends.ts +73 -0
  55. package/src/monitor/tasks/TaskClock.ts +33 -0
  56. package/src/monitor/tasks/TaskFailAbandoned.ts +54 -0
  57. package/src/monitor/tasks/TaskMonitorCallHistory.ts +26 -0
  58. package/src/monitor/tasks/TaskNewHeader.ts +93 -0
  59. package/src/monitor/tasks/TaskPurge.ts +68 -0
  60. package/src/monitor/tasks/TaskReorg.ts +89 -0
  61. package/src/monitor/tasks/TaskReviewStatus.ts +48 -0
  62. package/src/monitor/tasks/TaskSendWaiting.ts +122 -0
  63. package/src/monitor/tasks/TaskSyncWhenIdle.ts +26 -0
  64. package/src/monitor/tasks/TaskUnFail.ts +151 -0
  65. package/src/monitor/tasks/WalletMonitorTask.ts +47 -0
  66. package/src/sdk/CertOpsWallet.ts +18 -0
  67. package/src/sdk/PrivilegedKeyManager.ts +372 -0
  68. package/src/sdk/README.md +13 -0
  69. package/src/sdk/WERR_errors.ts +234 -0
  70. package/src/sdk/WalletError.ts +170 -0
  71. package/src/sdk/WalletErrorFromJson.ts +80 -0
  72. package/src/sdk/WalletServices.interfaces.ts +700 -0
  73. package/src/sdk/WalletSigner.interfaces.ts +11 -0
  74. package/src/sdk/WalletStorage.interfaces.ts +606 -0
  75. package/src/sdk/__test/CertificateLifeCycle.test.ts +131 -0
  76. package/src/sdk/__test/PrivilegedKeyManager.test.ts +738 -0
  77. package/src/sdk/__test/WalletError.test.ts +318 -0
  78. package/src/sdk/__test/validationHelpers.test.ts +21 -0
  79. package/src/sdk/index.ts +10 -0
  80. package/src/sdk/types.ts +226 -0
  81. package/src/services/README.md +11 -0
  82. package/src/services/ServiceCollection.ts +248 -0
  83. package/src/services/Services.ts +603 -0
  84. package/src/services/__tests/ARC.man.test.ts +123 -0
  85. package/src/services/__tests/ARC.timeout.man.test.ts +79 -0
  86. package/src/services/__tests/ArcGorillaPool.man.test.ts +108 -0
  87. package/src/services/__tests/arcServices.test.ts +8 -0
  88. package/src/services/__tests/bitrails.test.ts +56 -0
  89. package/src/services/__tests/getMerklePath.test.ts +15 -0
  90. package/src/services/__tests/getRawTx.test.ts +13 -0
  91. package/src/services/__tests/postBeef.test.ts +104 -0
  92. package/src/services/__tests/verifyBeef.test.ts +50 -0
  93. package/src/services/chaintracker/BHServiceClient.ts +212 -0
  94. package/src/services/chaintracker/ChaintracksChainTracker.ts +71 -0
  95. package/src/services/chaintracker/__tests/ChaintracksChainTracker.test.ts +33 -0
  96. package/src/services/chaintracker/__tests/ChaintracksServiceClient.test.ts +29 -0
  97. package/src/services/chaintracker/chaintracks/Api/BlockHeaderApi.ts +72 -0
  98. package/src/services/chaintracker/chaintracks/Api/BulkIngestorApi.ts +83 -0
  99. package/src/services/chaintracker/chaintracks/Api/BulkStorageApi.ts +92 -0
  100. package/src/services/chaintracker/chaintracks/Api/ChaintracksApi.ts +64 -0
  101. package/src/services/chaintracker/chaintracks/Api/ChaintracksClientApi.ts +189 -0
  102. package/src/services/chaintracker/chaintracks/Api/ChaintracksFetchApi.ts +18 -0
  103. package/src/services/chaintracker/chaintracks/Api/ChaintracksFsApi.ts +58 -0
  104. package/src/services/chaintracker/chaintracks/Api/ChaintracksStorageApi.ts +386 -0
  105. package/src/services/chaintracker/chaintracks/Api/LiveIngestorApi.ts +25 -0
  106. package/src/services/chaintracker/chaintracks/Chaintracks.ts +609 -0
  107. package/src/services/chaintracker/chaintracks/ChaintracksService.ts +199 -0
  108. package/src/services/chaintracker/chaintracks/ChaintracksServiceClient.ts +154 -0
  109. package/src/services/chaintracker/chaintracks/Ingest/BulkIngestorBase.ts +176 -0
  110. package/src/services/chaintracker/chaintracks/Ingest/BulkIngestorCDN.ts +174 -0
  111. package/src/services/chaintracker/chaintracks/Ingest/BulkIngestorCDNBabbage.ts +18 -0
  112. package/src/services/chaintracker/chaintracks/Ingest/BulkIngestorWhatsOnChainCdn.ts +113 -0
  113. package/src/services/chaintracker/chaintracks/Ingest/BulkIngestorWhatsOnChainWs.ts +81 -0
  114. package/src/services/chaintracker/chaintracks/Ingest/LiveIngestorBase.ts +86 -0
  115. package/src/services/chaintracker/chaintracks/Ingest/LiveIngestorTeranodeP2P.ts +59 -0
  116. package/src/services/chaintracker/chaintracks/Ingest/LiveIngestorWhatsOnChainPoll.ts +104 -0
  117. package/src/services/chaintracker/chaintracks/Ingest/LiveIngestorWhatsOnChainWs.ts +66 -0
  118. package/src/services/chaintracker/chaintracks/Ingest/WhatsOnChainIngestorWs.ts +566 -0
  119. package/src/services/chaintracker/chaintracks/Ingest/WhatsOnChainServices.ts +219 -0
  120. package/src/services/chaintracker/chaintracks/Ingest/__tests/BulkIngestorCDNBabbage.test.ts +54 -0
  121. package/src/services/chaintracker/chaintracks/Ingest/__tests/LiveIngestorWhatsOnChainPoll.test.ts +33 -0
  122. package/src/services/chaintracker/chaintracks/Ingest/__tests/WhatsOnChainServices.test.ts +124 -0
  123. package/src/services/chaintracker/chaintracks/Storage/BulkStorageBase.ts +92 -0
  124. package/src/services/chaintracker/chaintracks/Storage/ChaintracksKnexMigrations.ts +104 -0
  125. package/src/services/chaintracker/chaintracks/Storage/ChaintracksStorageBase.ts +382 -0
  126. package/src/services/chaintracker/chaintracks/Storage/ChaintracksStorageIdb.ts +574 -0
  127. package/src/services/chaintracker/chaintracks/Storage/ChaintracksStorageKnex.ts +438 -0
  128. package/src/services/chaintracker/chaintracks/Storage/ChaintracksStorageMemory.ts +29 -0
  129. package/src/services/chaintracker/chaintracks/Storage/ChaintracksStorageNoDb.ts +304 -0
  130. package/src/services/chaintracker/chaintracks/Storage/__tests/ChaintracksStorageIdb.test.ts +102 -0
  131. package/src/services/chaintracker/chaintracks/Storage/__tests/ChaintracksStorageKnex.test.ts +45 -0
  132. package/src/services/chaintracker/chaintracks/__tests/Chaintracks.test.ts +77 -0
  133. package/src/services/chaintracker/chaintracks/__tests/ChaintracksClientApi.test.ts +192 -0
  134. package/src/services/chaintracker/chaintracks/__tests/LocalCdnServer.ts +75 -0
  135. package/src/services/chaintracker/chaintracks/__tests/createIdbChaintracks.test.ts +62 -0
  136. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest349/mainNetBlockHeaders.json +1 -0
  137. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest349/mainNet_0.headers +0 -0
  138. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest349/mainNet_1.headers +0 -0
  139. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest349/mainNet_2.headers +0 -0
  140. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest349/mainNet_3.headers +0 -0
  141. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest379/mainNetBlockHeaders.json +1 -0
  142. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest379/mainNet_0.headers +0 -0
  143. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest379/mainNet_1.headers +0 -0
  144. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest379/mainNet_2.headers +0 -0
  145. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest379/mainNet_3.headers +0 -0
  146. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest399/mainNetBlockHeaders.json +1 -0
  147. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest399/mainNet_0.headers +0 -0
  148. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest399/mainNet_1.headers +0 -0
  149. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest399/mainNet_2.headers +0 -0
  150. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest399/mainNet_3.headers +0 -0
  151. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest402/mainNetBlockHeaders.json +1 -0
  152. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest402/mainNet_0.headers +0 -0
  153. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest402/mainNet_1.headers +0 -0
  154. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest402/mainNet_2.headers +0 -0
  155. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest402/mainNet_3.headers +0 -0
  156. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest402/mainNet_4.headers +0 -0
  157. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest499/mainNetBlockHeaders.json +1 -0
  158. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest499/mainNet_0.headers +0 -0
  159. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest499/mainNet_1.headers +0 -0
  160. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest499/mainNet_2.headers +0 -0
  161. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest499/mainNet_3.headers +0 -0
  162. package/src/services/chaintracker/chaintracks/__tests/data/cdnTest499/mainNet_4.headers +0 -0
  163. package/src/services/chaintracker/chaintracks/createDefaultIdbChaintracksOptions.ts +92 -0
  164. package/src/services/chaintracker/chaintracks/createDefaultKnexChaintracksOptions.ts +111 -0
  165. package/src/services/chaintracker/chaintracks/createDefaultNoDbChaintracksOptions.ts +91 -0
  166. package/src/services/chaintracker/chaintracks/createIdbChaintracks.ts +60 -0
  167. package/src/services/chaintracker/chaintracks/createKnexChaintracks.ts +65 -0
  168. package/src/services/chaintracker/chaintracks/createNoDbChaintracks.ts +60 -0
  169. package/src/services/chaintracker/chaintracks/index.all.ts +12 -0
  170. package/src/services/chaintracker/chaintracks/index.client.ts +4 -0
  171. package/src/services/chaintracker/chaintracks/index.mobile.ts +37 -0
  172. package/src/services/chaintracker/chaintracks/util/BulkFileDataManager.ts +975 -0
  173. package/src/services/chaintracker/chaintracks/util/BulkFileDataReader.ts +60 -0
  174. package/src/services/chaintracker/chaintracks/util/BulkFilesReader.ts +336 -0
  175. package/src/services/chaintracker/chaintracks/util/BulkHeaderFile.ts +247 -0
  176. package/src/services/chaintracker/chaintracks/util/ChaintracksFetch.ts +69 -0
  177. package/src/services/chaintracker/chaintracks/util/ChaintracksFs.ts +141 -0
  178. package/src/services/chaintracker/chaintracks/util/HeightRange.ts +153 -0
  179. package/src/services/chaintracker/chaintracks/util/SingleWriterMultiReaderLock.ts +76 -0
  180. package/src/services/chaintracker/chaintracks/util/__tests/BulkFileDataManager.test.ts +304 -0
  181. package/src/services/chaintracker/chaintracks/util/__tests/ChaintracksFetch.test.ts +60 -0
  182. package/src/services/chaintracker/chaintracks/util/__tests/HeightRange.test.ts +67 -0
  183. package/src/services/chaintracker/chaintracks/util/__tests/SingleWriterMultiReaderLock.test.ts +49 -0
  184. package/src/services/chaintracker/chaintracks/util/blockHeaderUtilities.ts +573 -0
  185. package/src/services/chaintracker/chaintracks/util/dirtyHashes.ts +29 -0
  186. package/src/services/chaintracker/chaintracks/util/validBulkHeaderFilesByFileHash.ts +432 -0
  187. package/src/services/chaintracker/index.all.ts +4 -0
  188. package/src/services/chaintracker/index.client.ts +4 -0
  189. package/src/services/chaintracker/index.mobile.ts +4 -0
  190. package/src/services/createDefaultWalletServicesOptions.ts +77 -0
  191. package/src/services/index.ts +1 -0
  192. package/src/services/processingErrors/arcSuccessError.json +76 -0
  193. package/src/services/providers/ARC.ts +350 -0
  194. package/src/services/providers/Bitails.ts +256 -0
  195. package/src/services/providers/SdkWhatsOnChain.ts +83 -0
  196. package/src/services/providers/WhatsOnChain.ts +883 -0
  197. package/src/services/providers/__tests/WhatsOnChain.test.ts +242 -0
  198. package/src/services/providers/__tests/exchangeRates.test.ts +18 -0
  199. package/src/services/providers/exchangeRates.ts +265 -0
  200. package/src/services/providers/getBeefForTxid.ts +369 -0
  201. package/src/signer/README.md +5 -0
  202. package/src/signer/WalletSigner.ts +17 -0
  203. package/src/signer/methods/acquireDirectCertificate.ts +52 -0
  204. package/src/signer/methods/buildSignableTransaction.ts +183 -0
  205. package/src/signer/methods/completeSignedTransaction.ts +117 -0
  206. package/src/signer/methods/createAction.ts +172 -0
  207. package/src/signer/methods/internalizeAction.ts +106 -0
  208. package/src/signer/methods/proveCertificate.ts +43 -0
  209. package/src/signer/methods/signAction.ts +54 -0
  210. package/src/storage/README.md +14 -0
  211. package/src/storage/StorageIdb.ts +2304 -0
  212. package/src/storage/StorageKnex.ts +1425 -0
  213. package/src/storage/StorageProvider.ts +810 -0
  214. package/src/storage/StorageReader.ts +194 -0
  215. package/src/storage/StorageReaderWriter.ts +432 -0
  216. package/src/storage/StorageSyncReader.ts +34 -0
  217. package/src/storage/WalletStorageManager.ts +943 -0
  218. package/src/storage/__test/StorageIdb.test.ts +43 -0
  219. package/src/storage/__test/WalletStorageManager.test.ts +275 -0
  220. package/src/storage/__test/adminStats.man.test.ts +89 -0
  221. package/src/storage/__test/getBeefForTransaction.test.ts +385 -0
  222. package/src/storage/index.all.ts +11 -0
  223. package/src/storage/index.client.ts +7 -0
  224. package/src/storage/index.mobile.ts +6 -0
  225. package/src/storage/methods/ListActionsSpecOp.ts +70 -0
  226. package/src/storage/methods/ListOutputsSpecOp.ts +129 -0
  227. package/src/storage/methods/__test/GenerateChange/generateChangeSdk.test.ts +1057 -0
  228. package/src/storage/methods/__test/GenerateChange/randomValsUsed1.ts +20 -0
  229. package/src/storage/methods/__test/offsetKey.test.ts +274 -0
  230. package/src/storage/methods/attemptToPostReqsToNetwork.ts +389 -0
  231. package/src/storage/methods/createAction.ts +947 -0
  232. package/src/storage/methods/generateChange.ts +556 -0
  233. package/src/storage/methods/getBeefForTransaction.ts +139 -0
  234. package/src/storage/methods/getSyncChunk.ts +293 -0
  235. package/src/storage/methods/internalizeAction.ts +562 -0
  236. package/src/storage/methods/listActionsIdb.ts +183 -0
  237. package/src/storage/methods/listActionsKnex.ts +226 -0
  238. package/src/storage/methods/listCertificates.ts +73 -0
  239. package/src/storage/methods/listOutputsIdb.ts +203 -0
  240. package/src/storage/methods/listOutputsKnex.ts +263 -0
  241. package/src/storage/methods/offsetKey.ts +89 -0
  242. package/src/storage/methods/processAction.ts +420 -0
  243. package/src/storage/methods/purgeData.ts +251 -0
  244. package/src/storage/methods/purgeDataIdb.ts +10 -0
  245. package/src/storage/methods/reviewStatus.ts +101 -0
  246. package/src/storage/methods/reviewStatusIdb.ts +43 -0
  247. package/src/storage/methods/utils.Buffer.ts +33 -0
  248. package/src/storage/methods/utils.ts +56 -0
  249. package/src/storage/remoting/StorageClient.ts +567 -0
  250. package/src/storage/remoting/StorageMobile.ts +544 -0
  251. package/src/storage/remoting/StorageServer.ts +291 -0
  252. package/src/storage/remoting/__test/StorageClient.test.ts +113 -0
  253. package/src/storage/schema/KnexMigrations.ts +489 -0
  254. package/src/storage/schema/StorageIdbSchema.ts +150 -0
  255. package/src/storage/schema/entities/EntityBase.ts +210 -0
  256. package/src/storage/schema/entities/EntityCertificate.ts +188 -0
  257. package/src/storage/schema/entities/EntityCertificateField.ts +136 -0
  258. package/src/storage/schema/entities/EntityCommission.ts +148 -0
  259. package/src/storage/schema/entities/EntityOutput.ts +290 -0
  260. package/src/storage/schema/entities/EntityOutputBasket.ts +153 -0
  261. package/src/storage/schema/entities/EntityOutputTag.ts +121 -0
  262. package/src/storage/schema/entities/EntityOutputTagMap.ts +123 -0
  263. package/src/storage/schema/entities/EntityProvenTx.ts +319 -0
  264. package/src/storage/schema/entities/EntityProvenTxReq.ts +580 -0
  265. package/src/storage/schema/entities/EntitySyncState.ts +389 -0
  266. package/src/storage/schema/entities/EntityTransaction.ts +306 -0
  267. package/src/storage/schema/entities/EntityTxLabel.ts +121 -0
  268. package/src/storage/schema/entities/EntityTxLabelMap.ts +123 -0
  269. package/src/storage/schema/entities/EntityUser.ts +112 -0
  270. package/src/storage/schema/entities/MergeEntity.ts +73 -0
  271. package/src/storage/schema/entities/__tests/CertificateFieldTests.test.ts +353 -0
  272. package/src/storage/schema/entities/__tests/CertificateTests.test.ts +354 -0
  273. package/src/storage/schema/entities/__tests/CommissionTests.test.ts +371 -0
  274. package/src/storage/schema/entities/__tests/OutputBasketTests.test.ts +278 -0
  275. package/src/storage/schema/entities/__tests/OutputTagMapTests.test.ts +242 -0
  276. package/src/storage/schema/entities/__tests/OutputTagTests.test.ts +288 -0
  277. package/src/storage/schema/entities/__tests/OutputTests.test.ts +464 -0
  278. package/src/storage/schema/entities/__tests/ProvenTxReqTests.test.ts +340 -0
  279. package/src/storage/schema/entities/__tests/ProvenTxTests.test.ts +504 -0
  280. package/src/storage/schema/entities/__tests/SyncStateTests.test.ts +288 -0
  281. package/src/storage/schema/entities/__tests/TransactionTests.test.ts +604 -0
  282. package/src/storage/schema/entities/__tests/TxLabelMapTests.test.ts +361 -0
  283. package/src/storage/schema/entities/__tests/TxLabelTests.test.ts +198 -0
  284. package/src/storage/schema/entities/__tests/stampLogTests.test.ts +90 -0
  285. package/src/storage/schema/entities/__tests/usersTests.test.ts +340 -0
  286. package/src/storage/schema/entities/index.ts +16 -0
  287. package/src/storage/schema/tables/TableCertificate.ts +21 -0
  288. package/src/storage/schema/tables/TableCertificateField.ts +12 -0
  289. package/src/storage/schema/tables/TableCommission.ts +13 -0
  290. package/src/storage/schema/tables/TableMonitorEvent.ts +9 -0
  291. package/src/storage/schema/tables/TableOutput.ts +64 -0
  292. package/src/storage/schema/tables/TableOutputBasket.ts +12 -0
  293. package/src/storage/schema/tables/TableOutputTag.ts +10 -0
  294. package/src/storage/schema/tables/TableOutputTagMap.ts +9 -0
  295. package/src/storage/schema/tables/TableProvenTx.ts +14 -0
  296. package/src/storage/schema/tables/TableProvenTxReq.ts +65 -0
  297. package/src/storage/schema/tables/TableSettings.ts +17 -0
  298. package/src/storage/schema/tables/TableSyncState.ts +18 -0
  299. package/src/storage/schema/tables/TableTransaction.ts +54 -0
  300. package/src/storage/schema/tables/TableTxLabel.ts +10 -0
  301. package/src/storage/schema/tables/TableTxLabelMap.ts +9 -0
  302. package/src/storage/schema/tables/TableUser.ts +16 -0
  303. package/src/storage/schema/tables/index.ts +16 -0
  304. package/src/storage/sync/StorageMySQLDojoReader.ts +696 -0
  305. package/src/storage/sync/index.ts +1 -0
  306. package/src/utility/Format.ts +133 -0
  307. package/src/utility/README.md +3 -0
  308. package/src/utility/ReaderUint8Array.ts +187 -0
  309. package/src/utility/ScriptTemplateBRC29.ts +73 -0
  310. package/src/utility/__tests/utilityHelpers.noBuffer.test.ts +109 -0
  311. package/src/utility/aggregateResults.ts +68 -0
  312. package/src/utility/identityUtils.ts +159 -0
  313. package/src/utility/index.all.ts +7 -0
  314. package/src/utility/index.client.ts +7 -0
  315. package/src/utility/parseTxScriptOffsets.ts +29 -0
  316. package/src/utility/stampLog.ts +69 -0
  317. package/src/utility/tscProofToMerklePath.ts +48 -0
  318. package/src/utility/utilityHelpers.buffer.ts +34 -0
  319. package/src/utility/utilityHelpers.noBuffer.ts +60 -0
  320. package/src/utility/utilityHelpers.ts +275 -0
  321. package/src/wab-client/WABClient.ts +94 -0
  322. package/src/wab-client/__tests/WABClient.man.test.ts +59 -0
  323. package/src/wab-client/auth-method-interactors/AuthMethodInteractor.ts +47 -0
  324. package/src/wab-client/auth-method-interactors/DevConsoleInteractor.ts +73 -0
  325. package/src/wab-client/auth-method-interactors/PersonaIDInteractor.ts +35 -0
  326. package/src/wab-client/auth-method-interactors/TwilioPhoneInteractor.ts +72 -0
  327. package/syncVersions.js +71 -0
  328. package/test/Wallet/StorageClient/storageClient.man.test.ts +75 -0
  329. package/test/Wallet/action/abortAction.test.ts +47 -0
  330. package/test/Wallet/action/createAction.test.ts +299 -0
  331. package/test/Wallet/action/createAction2.test.ts +1273 -0
  332. package/test/Wallet/action/createActionToGenerateBeefs.man.test.ts +293 -0
  333. package/test/Wallet/action/internalizeAction.a.test.ts +286 -0
  334. package/test/Wallet/action/internalizeAction.test.ts +682 -0
  335. package/test/Wallet/action/relinquishOutput.test.ts +37 -0
  336. package/test/Wallet/certificate/acquireCertificate.test.ts +298 -0
  337. package/test/Wallet/certificate/listCertificates.test.ts +346 -0
  338. package/test/Wallet/construct/Wallet.constructor.test.ts +57 -0
  339. package/test/Wallet/get/getHeaderForHeight.test.ts +82 -0
  340. package/test/Wallet/get/getHeight.test.ts +52 -0
  341. package/test/Wallet/get/getKnownTxids.test.ts +86 -0
  342. package/test/Wallet/get/getNetwork.test.ts +27 -0
  343. package/test/Wallet/get/getVersion.test.ts +27 -0
  344. package/test/Wallet/list/listActions.test.ts +279 -0
  345. package/test/Wallet/list/listActions2.test.ts +1381 -0
  346. package/test/Wallet/list/listCertificates.test.ts +118 -0
  347. package/test/Wallet/list/listOutputs.test.ts +447 -0
  348. package/test/Wallet/live/walletLive.man.test.ts +521 -0
  349. package/test/Wallet/local/localWallet.man.test.ts +93 -0
  350. package/test/Wallet/local/localWallet2.man.test.ts +277 -0
  351. package/test/Wallet/signAction/mountaintop.man.test.ts +130 -0
  352. package/test/Wallet/specOps/specOps.man.test.ts +220 -0
  353. package/test/Wallet/support/janitor.man.test.ts +40 -0
  354. package/test/Wallet/support/operations.man.test.ts +407 -0
  355. package/test/Wallet/support/reqErrorReview.2025.05.06.man.test.ts +347 -0
  356. package/test/Wallet/sync/Wallet.sync.test.ts +215 -0
  357. package/test/Wallet/sync/Wallet.updateWalletLegacyTestData.man.test.ts +203 -0
  358. package/test/Wallet/sync/setActive.test.ts +170 -0
  359. package/test/WalletClient/LocalKVStore.man.test.ts +114 -0
  360. package/test/WalletClient/WERR.man.test.ts +35 -0
  361. package/test/bsv-ts-sdk/LocalKVStore.test.ts +102 -0
  362. package/test/checkDB.ts +57 -0
  363. package/test/checkdb +0 -0
  364. package/test/examples/backup.man.test.ts +59 -0
  365. package/test/examples/pushdrop.test.ts +282 -0
  366. package/test/monitor/Monitor.test.ts +620 -0
  367. package/test/services/Services.test.ts +263 -0
  368. package/test/storage/KnexMigrations.test.ts +86 -0
  369. package/test/storage/StorageMySQLDojoReader.man.test.ts +60 -0
  370. package/test/storage/count.test.ts +177 -0
  371. package/test/storage/find.test.ts +195 -0
  372. package/test/storage/findLegacy.test.ts +67 -0
  373. package/test/storage/idb/allocateChange.test.ts +251 -0
  374. package/test/storage/idb/count.test.ts +158 -0
  375. package/test/storage/idb/find.test.ts +177 -0
  376. package/test/storage/idb/idbSpeed.test.ts +36 -0
  377. package/test/storage/idb/insert.test.ts +268 -0
  378. package/test/storage/idb/transactionAbort.test.ts +108 -0
  379. package/test/storage/idb/update.test.ts +999 -0
  380. package/test/storage/insert.test.ts +278 -0
  381. package/test/storage/update.test.ts +1021 -0
  382. package/test/storage/update2.test.ts +897 -0
  383. package/test/utils/TestUtilsWalletStorage.ts +2526 -0
  384. package/test/utils/localWalletMethods.ts +363 -0
  385. package/test/utils/removeFailedFromDatabase.sql +17 -0
  386. package/ts2md.json +44 -0
  387. package/tsconfig.all.json +31 -0
  388. package/tsconfig.client.json +29 -0
  389. package/tsconfig.json +17 -0
  390. package/tsconfig.mobile.json +28 -0
@@ -0,0 +1,183 @@
1
+ import {
2
+ Transaction as BsvTransaction,
3
+ ActionStatus,
4
+ ListActionsResult,
5
+ WalletAction,
6
+ WalletActionOutput,
7
+ WalletActionInput,
8
+ Validation
9
+ } from '@bsv/sdk'
10
+ import { StorageIdb } from '../StorageIdb'
11
+ import { getLabelToSpecOp, ListActionsSpecOp } from './ListActionsSpecOp'
12
+ import { AuthId } from '../../sdk/WalletStorage.interfaces'
13
+ import { isListActionsSpecOp, TransactionStatus } from '../../sdk/types'
14
+ import { TableOutputX } from '../schema/tables/TableOutput'
15
+ import { asString } from '../../utility/utilityHelpers.noBuffer'
16
+
17
+ export async function listActionsIdb(
18
+ storage: StorageIdb,
19
+ auth: AuthId,
20
+ vargs: Validation.ValidListActionsArgs
21
+ ): Promise<ListActionsResult> {
22
+ const limit = vargs.limit
23
+ const offset = vargs.offset
24
+
25
+ const r: ListActionsResult = {
26
+ totalActions: 0,
27
+ actions: []
28
+ }
29
+
30
+ let specOp: ListActionsSpecOp | undefined = undefined
31
+ let specOpLabels: string[] = []
32
+ let labels: string[] = []
33
+ for (const label of vargs.labels) {
34
+ if (isListActionsSpecOp(label)) {
35
+ specOp = getLabelToSpecOp()[label]
36
+ } else {
37
+ labels.push(label)
38
+ }
39
+ }
40
+ if (specOp?.labelsToIntercept !== undefined) {
41
+ const intercept = specOp.labelsToIntercept!
42
+ const labels2 = labels
43
+ labels = []
44
+ if (intercept.length === 0) {
45
+ specOpLabels = labels2
46
+ }
47
+ for (const label of labels2) {
48
+ if (intercept.indexOf(label) >= 0) {
49
+ specOpLabels.push(label)
50
+ } else {
51
+ labels.push(label)
52
+ }
53
+ }
54
+ }
55
+
56
+ let labelIds: number[] = []
57
+ if (labels.length > 0) {
58
+ await storage.filterTxLabels({ partial: { userId: auth.userId, isDeleted: false } }, tl => {
59
+ if (labels.includes(tl.label)) {
60
+ labelIds.push(tl.txLabelId)
61
+ }
62
+ })
63
+ }
64
+
65
+ const isQueryModeAll = vargs.labelQueryMode === 'all'
66
+ if (isQueryModeAll && labelIds.length < labels.length)
67
+ // all the required labels don't exist, impossible to satisfy.
68
+ return r
69
+
70
+ if (!isQueryModeAll && labelIds.length === 0 && labels.length > 0)
71
+ // any and only non-existing labels, impossible to satisfy.
72
+ return r
73
+
74
+ const stati: TransactionStatus[] = specOp?.setStatusFilter
75
+ ? specOp.setStatusFilter()
76
+ : ['completed', 'unprocessed', 'sending', 'unproven', 'unsigned', 'nosend', 'nonfinal']
77
+
78
+ const noLabels = labelIds.length === 0
79
+
80
+ const txs = await storage.findTransactions(
81
+ {
82
+ partial: { userId: auth.userId },
83
+ status: stati,
84
+ paged: { limit: vargs.limit, offset: vargs.offset },
85
+ noRawTx: true
86
+ },
87
+ labelIds,
88
+ isQueryModeAll
89
+ )
90
+ if (txs.length === vargs.limit) {
91
+ r.totalActions = await storage.countTransactions(
92
+ { partial: { userId: auth.userId }, status: stati },
93
+ labelIds,
94
+ isQueryModeAll
95
+ )
96
+ } else {
97
+ r.totalActions = txs.length
98
+ }
99
+
100
+ if (specOp?.postProcess) {
101
+ await specOp.postProcess(storage, auth, vargs, specOpLabels, txs)
102
+ }
103
+
104
+ for (const tx of txs) {
105
+ const wtx: WalletAction = {
106
+ txid: tx.txid || '',
107
+ satoshis: tx.satoshis || 0,
108
+ status: <ActionStatus>tx.status!,
109
+ isOutgoing: !!tx.isOutgoing,
110
+ description: tx.description || '',
111
+ version: tx.version || 0,
112
+ lockTime: tx.lockTime || 0
113
+ }
114
+ r.actions.push(wtx)
115
+ }
116
+
117
+ if (vargs.includeLabels || vargs.includeInputs || vargs.includeOutputs) {
118
+ await Promise.all(
119
+ txs.map(async (tx, i) => {
120
+ //let i = -1
121
+ //for (const tx of txs) {
122
+ // i++
123
+ const action = r.actions[i]
124
+ if (vargs.includeLabels) {
125
+ action.labels = (await storage.getLabelsForTransactionId(tx.transactionId)).map(l => l.label)
126
+ }
127
+ if (vargs.includeOutputs) {
128
+ const outputs: TableOutputX[] = await storage.findOutputs({
129
+ partial: { transactionId: tx.transactionId },
130
+ noScript: !vargs.includeOutputLockingScripts
131
+ })
132
+ action.outputs = []
133
+ for (const o of outputs) {
134
+ await storage.extendOutput(o, true, true)
135
+ const wo: WalletActionOutput = {
136
+ satoshis: o.satoshis || 0,
137
+ spendable: !!o.spendable,
138
+ tags: o.tags?.map(t => t.tag) || [],
139
+ outputIndex: Number(o.vout),
140
+ outputDescription: o.outputDescription || '',
141
+ basket: o.basket?.name || ''
142
+ }
143
+ if (vargs.includeOutputLockingScripts) wo.lockingScript = asString(o.lockingScript || [])
144
+ action.outputs.push(wo)
145
+ }
146
+ }
147
+ if (vargs.includeInputs) {
148
+ const inputs: TableOutputX[] = await storage.findOutputs({
149
+ partial: { spentBy: tx.transactionId },
150
+ noScript: !vargs.includeInputSourceLockingScripts
151
+ })
152
+ action.inputs = []
153
+ if (inputs.length > 0) {
154
+ const rawTx = await storage.getRawTxOfKnownValidTransaction(tx.txid)
155
+ let bsvTx: BsvTransaction | undefined = undefined
156
+ if (rawTx) {
157
+ bsvTx = BsvTransaction.fromBinary(rawTx)
158
+ }
159
+ for (const o of inputs) {
160
+ await storage.extendOutput(o, true, true)
161
+ const input = bsvTx?.inputs.find(v => v.sourceTXID === o.txid && v.sourceOutputIndex === o.vout)
162
+ const wo: WalletActionInput = {
163
+ sourceOutpoint: `${o.txid}.${o.vout}`,
164
+ sourceSatoshis: o.satoshis || 0,
165
+ inputDescription: o.outputDescription || '',
166
+ sequenceNumber: input?.sequence || 0
167
+ }
168
+ action.inputs.push(wo)
169
+ if (vargs.includeInputSourceLockingScripts) {
170
+ wo.sourceLockingScript = asString(o.lockingScript || [])
171
+ }
172
+ if (vargs.includeInputUnlockingScripts) {
173
+ wo.unlockingScript = input?.unlockingScript?.toHex()
174
+ }
175
+ }
176
+ }
177
+ }
178
+ //}
179
+ })
180
+ )
181
+ }
182
+ return r
183
+ }
@@ -0,0 +1,226 @@
1
+ import {
2
+ Transaction as BsvTransaction,
3
+ ActionStatus,
4
+ ListActionsResult,
5
+ WalletAction,
6
+ WalletActionOutput,
7
+ WalletActionInput,
8
+ Validation
9
+ } from '@bsv/sdk'
10
+ import { StorageKnex } from '../StorageKnex'
11
+ import { getLabelToSpecOp, ListActionsSpecOp } from './ListActionsSpecOp'
12
+ import { AuthId } from '../../sdk/WalletStorage.interfaces'
13
+ import { isListActionsSpecOp } from '../../sdk/types'
14
+ import { TableTxLabel } from '../schema/tables/TableTxLabel'
15
+ import { TableTransaction } from '../schema/tables/TableTransaction'
16
+ import { verifyOne } from '../../utility/utilityHelpers'
17
+ import { TableOutputX } from '../schema/tables/TableOutput'
18
+ import { asString } from '../../utility/utilityHelpers.noBuffer'
19
+
20
+ export async function listActions(
21
+ storage: StorageKnex,
22
+ auth: AuthId,
23
+ vargs: Validation.ValidListActionsArgs
24
+ ): Promise<ListActionsResult> {
25
+ const limit = vargs.limit
26
+ const offset = vargs.offset
27
+
28
+ const k = storage.toDb(undefined)
29
+
30
+ const r: ListActionsResult = {
31
+ totalActions: 0,
32
+ actions: []
33
+ }
34
+
35
+ let specOp: ListActionsSpecOp | undefined = undefined
36
+ let specOpLabels: string[] = []
37
+ let labels: string[] = []
38
+ for (const label of vargs.labels) {
39
+ if (isListActionsSpecOp(label)) {
40
+ specOp = getLabelToSpecOp()[label]
41
+ } else {
42
+ labels.push(label)
43
+ }
44
+ }
45
+ if (specOp?.labelsToIntercept !== undefined) {
46
+ const intercept = specOp.labelsToIntercept!
47
+ const labels2 = labels
48
+ labels = []
49
+ if (intercept.length === 0) {
50
+ specOpLabels = labels2
51
+ }
52
+ for (const label of labels2) {
53
+ if (intercept.indexOf(label) >= 0) {
54
+ specOpLabels.push(label)
55
+ } else {
56
+ labels.push(label)
57
+ }
58
+ }
59
+ }
60
+
61
+ let labelIds: number[] = []
62
+ if (labels.length > 0) {
63
+ const q = k<TableTxLabel>('tx_labels')
64
+ .where({
65
+ userId: auth.userId,
66
+ isDeleted: false
67
+ })
68
+ .whereNotNull('txLabelId')
69
+ .whereIn('label', labels)
70
+ .select('txLabelId')
71
+ const r = await q
72
+ labelIds = r.map(r => r.txLabelId!)
73
+ }
74
+
75
+ const isQueryModeAll = vargs.labelQueryMode === 'all'
76
+ if (isQueryModeAll && labelIds.length < labels.length)
77
+ // all the required labels don't exist, impossible to satisfy.
78
+ return r
79
+
80
+ if (!isQueryModeAll && labelIds.length === 0 && labels.length > 0)
81
+ // any and only non-existing labels, impossible to satisfy.
82
+ return r
83
+
84
+ const columns: string[] = [
85
+ 'transactionId',
86
+ 'reference',
87
+ 'txid',
88
+ 'satoshis',
89
+ 'status',
90
+ 'isOutgoing',
91
+ 'description',
92
+ 'version',
93
+ 'lockTime'
94
+ ]
95
+
96
+ const stati: string[] = specOp?.setStatusFilter
97
+ ? specOp.setStatusFilter()
98
+ : ['completed', 'unprocessed', 'sending', 'unproven', 'unsigned', 'nosend', 'nonfinal']
99
+
100
+ const noLabels = labelIds.length === 0
101
+
102
+ const makeWithLabelsQueries = () => {
103
+ const cteq = k.raw(`
104
+ SELECT ${columns.map(c => 't.' + c).join(',')},
105
+ (SELECT COUNT(*)
106
+ FROM tx_labels_map AS m
107
+ WHERE m.transactionId = t.transactionId
108
+ AND m.txLabelId IN (${labelIds.join(',')})
109
+ ) AS lc
110
+ FROM transactions AS t
111
+ WHERE t.userId = ${auth.userId}
112
+ AND t.status in (${stati.map(s => `'${s}'`).join(',')})
113
+ `)
114
+
115
+ const q = k.with('tlc', cteq)
116
+ q.from('tlc')
117
+ if (isQueryModeAll) q.where('lc', labelIds.length)
118
+ else q.where('lc', '>', 0)
119
+ const qcount = q.clone()
120
+ q.select(columns)
121
+ qcount.count('transactionId as total')
122
+ return { q, qcount }
123
+ }
124
+
125
+ const makeWithoutLabelsQueries = () => {
126
+ const q = k('transactions').where('userId', auth.userId).whereIn('status', stati)
127
+ const qcount = q.clone().count('transactionId as total')
128
+ return { q, qcount }
129
+ }
130
+
131
+ const { q, qcount } = noLabels ? makeWithoutLabelsQueries() : makeWithLabelsQueries()
132
+
133
+ q.limit(limit).offset(offset).orderBy('transactionId', 'asc')
134
+
135
+ const txs: Partial<TableTransaction>[] = await q
136
+
137
+ if (specOp?.postProcess) {
138
+ await specOp.postProcess(storage, auth, vargs, specOpLabels, txs)
139
+ }
140
+
141
+ if (!limit || txs.length < limit) r.totalActions = txs.length
142
+ else {
143
+ const total = verifyOne(await qcount)['total']
144
+ r.totalActions = Number(total)
145
+ }
146
+
147
+ for (const tx of txs) {
148
+ const wtx: WalletAction = {
149
+ txid: tx.txid || '',
150
+ satoshis: tx.satoshis || 0,
151
+ status: <ActionStatus>tx.status!,
152
+ isOutgoing: !!tx.isOutgoing,
153
+ description: tx.description || '',
154
+ version: tx.version || 0,
155
+ lockTime: tx.lockTime || 0
156
+ }
157
+ r.actions.push(wtx)
158
+ }
159
+
160
+ if (vargs.includeLabels || vargs.includeInputs || vargs.includeOutputs) {
161
+ await Promise.all(
162
+ txs.map(async (tx, i) => {
163
+ //let i = -1
164
+ //for (const tx of txs) {
165
+ // i++
166
+ const action = r.actions[i]
167
+ if (vargs.includeLabels) {
168
+ action.labels = (await storage.getLabelsForTransactionId(tx.transactionId)).map(l => l.label)
169
+ }
170
+ if (vargs.includeOutputs) {
171
+ const outputs: TableOutputX[] = await storage.findOutputs({
172
+ partial: { transactionId: tx.transactionId },
173
+ noScript: !vargs.includeOutputLockingScripts
174
+ })
175
+ action.outputs = []
176
+ for (const o of outputs) {
177
+ await storage.extendOutput(o, true, true)
178
+ const wo: WalletActionOutput = {
179
+ satoshis: o.satoshis || 0,
180
+ spendable: !!o.spendable,
181
+ tags: o.tags?.map(t => t.tag) || [],
182
+ outputIndex: Number(o.vout),
183
+ outputDescription: o.outputDescription || '',
184
+ basket: o.basket?.name || ''
185
+ }
186
+ if (vargs.includeOutputLockingScripts) wo.lockingScript = asString(o.lockingScript || [])
187
+ action.outputs.push(wo)
188
+ }
189
+ }
190
+ if (vargs.includeInputs) {
191
+ const inputs: TableOutputX[] = await storage.findOutputs({
192
+ partial: { spentBy: tx.transactionId },
193
+ noScript: !vargs.includeInputSourceLockingScripts
194
+ })
195
+ action.inputs = []
196
+ if (inputs.length > 0) {
197
+ const rawTx = await storage.getRawTxOfKnownValidTransaction(tx.txid)
198
+ let bsvTx: BsvTransaction | undefined = undefined
199
+ if (rawTx) {
200
+ bsvTx = BsvTransaction.fromBinary(rawTx)
201
+ }
202
+ for (const o of inputs) {
203
+ await storage.extendOutput(o, true, true)
204
+ const input = bsvTx?.inputs.find(v => v.sourceTXID === o.txid && v.sourceOutputIndex === o.vout)
205
+ const wo: WalletActionInput = {
206
+ sourceOutpoint: `${o.txid}.${o.vout}`,
207
+ sourceSatoshis: o.satoshis || 0,
208
+ inputDescription: o.outputDescription || '',
209
+ sequenceNumber: input?.sequence || 0
210
+ }
211
+ action.inputs.push(wo)
212
+ if (vargs.includeInputSourceLockingScripts) {
213
+ wo.sourceLockingScript = asString(o.lockingScript || [])
214
+ }
215
+ if (vargs.includeInputUnlockingScripts) {
216
+ wo.unlockingScript = input?.unlockingScript?.toHex()
217
+ }
218
+ }
219
+ }
220
+ }
221
+ //}
222
+ })
223
+ )
224
+ }
225
+ return r
226
+ }
@@ -0,0 +1,73 @@
1
+ import { ListCertificatesResult, OriginatorDomainNameStringUnder250Bytes, Validation } from '@bsv/sdk'
2
+ import { StorageProvider, TableCertificate } from '../index.client'
3
+ import { AuthId, FindCertificatesArgs } from '../../sdk/WalletStorage.interfaces'
4
+ import { Paged } from '../../sdk/types'
5
+
6
+ export async function listCertificates(
7
+ storage: StorageProvider,
8
+ auth: AuthId,
9
+ vargs: Validation.ValidListCertificatesArgs,
10
+ originator?: OriginatorDomainNameStringUnder250Bytes
11
+ ): Promise<ListCertificatesResult> {
12
+ const paged: Paged = { limit: vargs.limit, offset: vargs.offset }
13
+
14
+ const partial: Partial<TableCertificate> = {
15
+ userId: auth.userId,
16
+ isDeleted: false
17
+ }
18
+
19
+ if (vargs.partial) {
20
+ const vp = vargs.partial
21
+ if (vp.type) partial['type'] = vp.type
22
+ if (vp.subject) partial['subject'] = vp.subject
23
+ if (vp.serialNumber) partial['serialNumber'] = vp.serialNumber
24
+ if (vp.certifier) partial['certifier'] = vp.certifier
25
+ if (vp.revocationOutpoint) partial['revocationOutpoint'] = vp.revocationOutpoint
26
+ if (vp.signature) partial['signature'] = vp.signature
27
+ }
28
+
29
+ const r = await storage.transaction(async trx => {
30
+ const findCertsArgs: FindCertificatesArgs = {
31
+ partial,
32
+ certifiers: vargs.certifiers,
33
+ types: vargs.types,
34
+ paged,
35
+ trx
36
+ }
37
+ const certs = await storage.findCertificates(findCertsArgs)
38
+ const certsWithFields = await Promise.all(
39
+ certs.map(async cert => {
40
+ const fields = await storage.findCertificateFields({
41
+ partial: { certificateId: cert.certificateId, userId: auth.userId },
42
+ trx
43
+ })
44
+ return {
45
+ ...cert,
46
+ fields: Object.fromEntries(fields.map(f => [f.fieldName, f.fieldValue])),
47
+ masterKeyring: Object.fromEntries(fields.map(f => [f.fieldName, f.masterKey]))
48
+ }
49
+ })
50
+ )
51
+ const r: ListCertificatesResult = {
52
+ totalCertificates: 0,
53
+ certificates: certsWithFields.map(cwf => ({
54
+ type: cwf.type,
55
+ subject: cwf.subject,
56
+ serialNumber: cwf.serialNumber,
57
+ certifier: cwf.certifier,
58
+ revocationOutpoint: cwf.revocationOutpoint,
59
+ signature: cwf.signature,
60
+ fields: cwf.fields,
61
+ verifier: cwf.verifier,
62
+ keyring: cwf.masterKeyring
63
+ }))
64
+ }
65
+ if (r.certificates.length < paged.limit) r.totalCertificates = r.certificates.length
66
+ else {
67
+ r.totalCertificates = await storage.countCertificates(findCertsArgs)
68
+ }
69
+ return r
70
+ })
71
+
72
+ return r
73
+ }
@@ -0,0 +1,203 @@
1
+ import { Beef, ListOutputsResult, OriginatorDomainNameStringUnder250Bytes, WalletOutput, Validation } from '@bsv/sdk'
2
+ import { getBasketToSpecOp, ListOutputsSpecOp } from './ListOutputsSpecOp'
3
+ import { StorageIdb } from '../StorageIdb'
4
+ import { AuthId, FindOutputsArgs } from '../../sdk/WalletStorage.interfaces'
5
+ import { verifyId } from '../../utility/utilityHelpers'
6
+ import { WERR_NOT_IMPLEMENTED } from '../../sdk/WERR_errors'
7
+ import { TableOutputBasket } from '../schema/tables/TableOutputBasket'
8
+ import { TransactionStatus } from '../../sdk/types'
9
+ import { asString } from '../../utility/utilityHelpers.noBuffer'
10
+
11
+ export async function listOutputsIdb(
12
+ storage: StorageIdb,
13
+ auth: AuthId,
14
+ vargs: Validation.ValidListOutputsArgs,
15
+ originator?: OriginatorDomainNameStringUnder250Bytes
16
+ ): Promise<ListOutputsResult> {
17
+ const userId = verifyId(auth.userId)
18
+ const limit = vargs.limit
19
+ const offset = vargs.offset
20
+ if (offset < 0) throw new WERR_NOT_IMPLEMENTED('Negative offset not supported in IndexedDB')
21
+
22
+ const r: ListOutputsResult = {
23
+ totalOutputs: 0,
24
+ outputs: []
25
+ }
26
+
27
+ /*
28
+ ListOutputsArgs {
29
+ basket: BasketStringUnder300Bytes
30
+
31
+ tags?: OutputTagStringUnder300Bytes[]
32
+ tagQueryMode?: 'all' | 'any' // default any
33
+
34
+ limit?: PositiveIntegerDefault10Max10000
35
+ offset?: PositiveIntegerOrZero
36
+ }
37
+ */
38
+
39
+ let specOp: ListOutputsSpecOp | undefined = undefined
40
+ let basketId: number | undefined = undefined
41
+ const basketsById: Record<number, TableOutputBasket> = {}
42
+ if (vargs.basket) {
43
+ let b = vargs.basket
44
+ specOp = getBasketToSpecOp()[b]
45
+ b = specOp ? (specOp.useBasket ? specOp.useBasket : '') : b
46
+ if (b) {
47
+ const baskets = await storage.findOutputBaskets({
48
+ partial: { userId, name: b }
49
+ })
50
+ if (baskets.length !== 1) {
51
+ // If basket does not exist, result is no outputs.
52
+ return r
53
+ }
54
+ const basket = baskets[0]
55
+ basketId = basket.basketId!
56
+ basketsById[basketId!] = basket
57
+ }
58
+ }
59
+
60
+ let tags = [...vargs.tags]
61
+ const specOpTags: string[] = []
62
+ if (specOp && specOp.tagsParamsCount) {
63
+ specOpTags.push(...tags.splice(0, Math.min(tags.length, specOp.tagsParamsCount)))
64
+ }
65
+ if (specOp && specOp.tagsToIntercept) {
66
+ // Pull out tags used by current specOp
67
+ const ts = tags
68
+ tags = []
69
+ for (const t of ts) {
70
+ if (specOp.tagsToIntercept.length === 0 || specOp.tagsToIntercept.indexOf(t) >= 0) {
71
+ specOpTags.push(t)
72
+ if (t === 'all') {
73
+ basketId = undefined
74
+ }
75
+ } else {
76
+ tags.push(t)
77
+ }
78
+ }
79
+ }
80
+
81
+ if (specOp && specOp.resultFromTags) {
82
+ const r = await specOp.resultFromTags(storage, auth, vargs, specOpTags)
83
+ return r
84
+ }
85
+
86
+ let tagIds: number[] = []
87
+ if (tags && tags.length > 0) {
88
+ await storage.filterOutputTags({ partial: { userId, isDeleted: false } }, ot => {
89
+ if (tags.includes(ot.tag)) {
90
+ tagIds.push(ot.outputTagId)
91
+ }
92
+ })
93
+ }
94
+
95
+ const isQueryModeAll = vargs.tagQueryMode === 'all'
96
+ if (isQueryModeAll && tagIds.length < tags.length)
97
+ // all the required tags don't exist, impossible to satisfy.
98
+ return r
99
+
100
+ if (!isQueryModeAll && tagIds.length === 0 && tags.length > 0)
101
+ // any and only non-existing labels, impossible to satisfy.
102
+ return r
103
+
104
+ const noTags = tagIds.length === 0
105
+ const includeSpent = false
106
+
107
+ const stati: TransactionStatus[] = ['completed', 'unproven', 'nosend']
108
+
109
+ const args: FindOutputsArgs = {
110
+ partial: {
111
+ userId,
112
+ basketId,
113
+ spendable: !includeSpent ? true : undefined
114
+ },
115
+ txStatus: stati,
116
+ noScript: true
117
+ }
118
+ if (!specOp || !specOp.ignoreLimit) args.paged = { limit, offset }
119
+
120
+ let outputs = await storage.findOutputs(args, tagIds, isQueryModeAll)
121
+ if (outputs.length === vargs.limit) {
122
+ args.paged = undefined
123
+ r.totalOutputs = await storage.countOutputs(args, tagIds, isQueryModeAll)
124
+ } else {
125
+ r.totalOutputs = outputs.length
126
+ }
127
+
128
+ if (specOp) {
129
+ if (specOp.filterOutputs) outputs = await specOp.filterOutputs(storage, auth, vargs, specOpTags, outputs)
130
+ if (specOp.resultFromOutputs) {
131
+ const r = await specOp.resultFromOutputs(storage, auth, vargs, specOpTags, outputs)
132
+ return r
133
+ }
134
+ }
135
+
136
+ /*
137
+ ListOutputsArgs {
138
+ include?: 'locking scripts' | 'entire transactions'
139
+ includeCustomInstructions?: BooleanDefaultFalse
140
+ includeTags?: BooleanDefaultFalse
141
+ includeLabels?: BooleanDefaultFalse
142
+ }
143
+
144
+ ListOutputsResult {
145
+ totalOutputs: PositiveIntegerOrZero
146
+ BEEF?: BEEF
147
+ outputs: Array<WalletOutput>
148
+ }
149
+
150
+ WalletOutput {
151
+ satoshis: SatoshiValue
152
+ spendable: boolean
153
+ outpoint: OutpointString
154
+
155
+ customInstructions?: string
156
+ lockingScript?: HexString
157
+ tags?: OutputTagStringUnder300Bytes[]
158
+ labels?: LabelStringUnder300Bytes[]
159
+ }
160
+ */
161
+
162
+ const labelsByTxid: Record<string, string[]> = {}
163
+
164
+ const beef = new Beef()
165
+
166
+ for (const o of outputs) {
167
+ const wo: WalletOutput = {
168
+ satoshis: Number(o.satoshis),
169
+ spendable: !!o.spendable,
170
+ outpoint: `${o.txid}.${o.vout}`
171
+ }
172
+ r.outputs.push(wo)
173
+ //if (vargs.includeBasket && o.basketId) {
174
+ // if (!basketsById[o.basketId]) {
175
+ // basketsById[o.basketId] = verifyTruthy(await dsk.findOutputBasketId(o.basketId!, trx))
176
+ // }
177
+ // wo.basket = basketsById[o.basketId].name
178
+ //}
179
+ if (vargs.includeCustomInstructions && o.customInstructions) wo.customInstructions = o.customInstructions
180
+ if (vargs.includeLabels && o.txid) {
181
+ if (labelsByTxid[o.txid] === undefined) {
182
+ labelsByTxid[o.txid] = (await storage.getLabelsForTransactionId(o.transactionId)).map(l => l.label)
183
+ }
184
+ wo.labels = labelsByTxid[o.txid]
185
+ }
186
+ if (vargs.includeTags) {
187
+ wo.tags = (await storage.getTagsForOutputId(o.outputId)).map(t => t.tag)
188
+ }
189
+ if (vargs.includeLockingScripts) {
190
+ await storage.validateOutputScript(o)
191
+ if (o.lockingScript) wo.lockingScript = asString(o.lockingScript)
192
+ }
193
+ if (vargs.includeTransactions && !beef.findTxid(o.txid!)) {
194
+ await storage.getValidBeefForKnownTxid(o.txid!, beef, undefined, vargs.knownTxids)
195
+ }
196
+ }
197
+
198
+ if (vargs.includeTransactions) {
199
+ r.BEEF = beef.toBinary()
200
+ }
201
+
202
+ return r
203
+ }