gnoman 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (606) hide show
  1. package/.eslintrc.cjs +24 -0
  2. package/.gnoman/contracts.json +4094 -0
  3. package/.gnoman/exec_package/runtime-debug.jsonl +45 -0
  4. package/.gnoman/holds.sqlite +0 -0
  5. package/.gnoman/license.json +7 -0
  6. package/.gnoman/safes.json +37 -0
  7. package/.gnoman/vanity-jobs.json +67 -0
  8. package/.gnoman/wallets.db +0 -0
  9. package/.prettierrc.json +6 -0
  10. package/CODex_TASKS.md +124 -0
  11. package/LICENSE.md +164 -0
  12. package/README.md +95 -0
  13. package/assets/GnoLogo.jpg +0 -0
  14. package/assets/self.png +0 -0
  15. package/backend/controllers/contractController.ts +49 -0
  16. package/backend/controllers/devToolsController.ts +76 -0
  17. package/backend/controllers/etherscanController.ts +59 -0
  18. package/backend/controllers/historyController.ts +7 -0
  19. package/backend/controllers/keyringController.ts +134 -0
  20. package/backend/controllers/robinhoodController.ts +80 -0
  21. package/backend/controllers/safeController.ts +167 -0
  22. package/backend/controllers/sandboxController.ts +63 -0
  23. package/backend/controllers/settingsController.ts +38 -0
  24. package/backend/controllers/walletController.ts +151 -0
  25. package/backend/index.ts +133 -0
  26. package/backend/licenses/license_public.pem +3 -0
  27. package/backend/licenses/verify_license.py +43 -0
  28. package/backend/routes/contractRoutes.ts +11 -0
  29. package/backend/routes/devToolsRoutes.ts +11 -0
  30. package/backend/routes/etherscanRoutes.ts +11 -0
  31. package/backend/routes/historyRoutes.ts +8 -0
  32. package/backend/routes/keyringRoutes.ts +25 -0
  33. package/backend/routes/licenseRoutes.ts +35 -0
  34. package/backend/routes/robinhoodRoutes.ts +22 -0
  35. package/backend/routes/runtimeRoutes.ts +29 -0
  36. package/backend/routes/safeRoutes.ts +28 -0
  37. package/backend/routes/sandboxRoutes.ts +17 -0
  38. package/backend/routes/settingsRoutes.ts +14 -0
  39. package/backend/routes/walletRoutes.ts +21 -0
  40. package/backend/services/chainlinkService.ts +65 -0
  41. package/backend/services/contractRegistryService.ts +205 -0
  42. package/backend/services/devToolsService.ts +251 -0
  43. package/backend/services/diagnosticsService.ts +350 -0
  44. package/backend/services/etherscanService.ts +152 -0
  45. package/backend/services/historyService.ts +89 -0
  46. package/backend/services/keyringAccessor.ts +4 -0
  47. package/backend/services/licenseService.ts +163 -0
  48. package/backend/services/onchain/abiRegistry.ts +57 -0
  49. package/backend/services/onchain/chainlinkClient.ts +56 -0
  50. package/backend/services/onchain/errors.ts +16 -0
  51. package/backend/services/onchain/etherscanClient.ts +94 -0
  52. package/backend/services/onchain/index.ts +76 -0
  53. package/backend/services/onchain/tenderlyRpcClient.ts +74 -0
  54. package/backend/services/onchain/types.ts +33 -0
  55. package/backend/services/onchainAutomationService.ts +424 -0
  56. package/backend/services/robinhood/auth.ts +42 -0
  57. package/backend/services/robinhood/client.ts +123 -0
  58. package/backend/services/robinhood/integrationService.ts +140 -0
  59. package/backend/services/robinhood/provider.ts +22 -0
  60. package/backend/services/robinhood/unofficialClient.ts +66 -0
  61. package/backend/services/rpcService.ts +44 -0
  62. package/backend/services/runtimeTelemetryService.ts +158 -0
  63. package/backend/services/safeConfigRepository.ts +205 -0
  64. package/backend/services/safeService.ts +588 -0
  65. package/backend/services/sandboxService.ts +157 -0
  66. package/backend/services/secureSettingsService.ts +45 -0
  67. package/backend/services/transactionHoldService.ts +223 -0
  68. package/backend/services/vanityService.ts +293 -0
  69. package/backend/services/walletService.ts +290 -0
  70. package/backend/services/walletStore.ts +179 -0
  71. package/backend/types/express-async-handler.d.ts +13 -0
  72. package/backend/types/keyring.d.ts +19 -0
  73. package/backend/utils/abiResolver.ts +208 -0
  74. package/backend/utils/http.ts +6 -0
  75. package/backend/utils/secretsResolver.ts +150 -0
  76. package/backend/utils/signer.ts +11 -0
  77. package/backend/workers/vanityWorker.ts +76 -0
  78. package/capacitor.config.ts +13 -0
  79. package/cli/gnoman.ts +424 -0
  80. package/contracts/OracleConsumer.sol +20 -0
  81. package/contracts/PriceFeedConsumer.sol +22 -0
  82. package/dist/backend/backend/controllers/contractController.js +41 -0
  83. package/dist/backend/backend/controllers/contractController.js.map +1 -0
  84. package/dist/backend/backend/controllers/devToolsController.js +63 -0
  85. package/dist/backend/backend/controllers/devToolsController.js.map +1 -0
  86. package/dist/backend/backend/controllers/etherscanController.js +53 -0
  87. package/dist/backend/backend/controllers/etherscanController.js.map +1 -0
  88. package/dist/backend/backend/controllers/historyController.js +12 -0
  89. package/dist/backend/backend/controllers/historyController.js.map +1 -0
  90. package/dist/backend/backend/controllers/keyringController.js +126 -0
  91. package/dist/backend/backend/controllers/keyringController.js.map +1 -0
  92. package/dist/backend/backend/controllers/robinhoodController.js +69 -0
  93. package/dist/backend/backend/controllers/robinhoodController.js.map +1 -0
  94. package/dist/backend/backend/controllers/safeController.js +137 -0
  95. package/dist/backend/backend/controllers/safeController.js.map +1 -0
  96. package/dist/backend/backend/controllers/sandboxController.js +48 -0
  97. package/dist/backend/backend/controllers/sandboxController.js.map +1 -0
  98. package/dist/backend/backend/controllers/settingsController.js +34 -0
  99. package/dist/backend/backend/controllers/settingsController.js.map +1 -0
  100. package/dist/backend/backend/controllers/walletController.js +140 -0
  101. package/dist/backend/backend/controllers/walletController.js.map +1 -0
  102. package/dist/backend/backend/index.js +119 -0
  103. package/dist/backend/backend/index.js.map +1 -0
  104. package/dist/backend/backend/routes/contractRoutes.js +44 -0
  105. package/dist/backend/backend/routes/contractRoutes.js.map +1 -0
  106. package/dist/backend/backend/routes/devToolsRoutes.js +44 -0
  107. package/dist/backend/backend/routes/devToolsRoutes.js.map +1 -0
  108. package/dist/backend/backend/routes/etherscanRoutes.js +44 -0
  109. package/dist/backend/backend/routes/etherscanRoutes.js.map +1 -0
  110. package/dist/backend/backend/routes/historyRoutes.js +41 -0
  111. package/dist/backend/backend/routes/historyRoutes.js.map +1 -0
  112. package/dist/backend/backend/routes/keyringRoutes.js +18 -0
  113. package/dist/backend/backend/routes/keyringRoutes.js.map +1 -0
  114. package/dist/backend/backend/routes/licenseRoutes.js +30 -0
  115. package/dist/backend/backend/routes/licenseRoutes.js.map +1 -0
  116. package/dist/backend/backend/routes/robinhoodRoutes.js +14 -0
  117. package/dist/backend/backend/routes/robinhoodRoutes.js.map +1 -0
  118. package/dist/backend/backend/routes/runtimeRoutes.js +26 -0
  119. package/dist/backend/backend/routes/runtimeRoutes.js.map +1 -0
  120. package/dist/backend/backend/routes/safeRoutes.js +61 -0
  121. package/dist/backend/backend/routes/safeRoutes.js.map +1 -0
  122. package/dist/backend/backend/routes/sandboxRoutes.js +50 -0
  123. package/dist/backend/backend/routes/sandboxRoutes.js.map +1 -0
  124. package/dist/backend/backend/routes/settingsRoutes.js +10 -0
  125. package/dist/backend/backend/routes/settingsRoutes.js.map +1 -0
  126. package/dist/backend/backend/routes/walletRoutes.js +54 -0
  127. package/dist/backend/backend/routes/walletRoutes.js.map +1 -0
  128. package/dist/backend/backend/services/chainlinkService.js +48 -0
  129. package/dist/backend/backend/services/chainlinkService.js.map +1 -0
  130. package/dist/backend/backend/services/contractRegistryService.js +138 -0
  131. package/dist/backend/backend/services/contractRegistryService.js.map +1 -0
  132. package/dist/backend/backend/services/devToolsService.js +213 -0
  133. package/dist/backend/backend/services/devToolsService.js.map +1 -0
  134. package/dist/backend/backend/services/diagnosticsService.js +286 -0
  135. package/dist/backend/backend/services/diagnosticsService.js.map +1 -0
  136. package/dist/backend/backend/services/etherscanService.js +125 -0
  137. package/dist/backend/backend/services/etherscanService.js.map +1 -0
  138. package/dist/backend/backend/services/historyService.js +75 -0
  139. package/dist/backend/backend/services/historyService.js.map +1 -0
  140. package/dist/backend/backend/services/keyringAccessor.js +40 -0
  141. package/dist/backend/backend/services/keyringAccessor.js.map +1 -0
  142. package/dist/backend/backend/services/licenseService.js +130 -0
  143. package/dist/backend/backend/services/licenseService.js.map +1 -0
  144. package/dist/backend/backend/services/onchain/abiRegistry.js +47 -0
  145. package/dist/backend/backend/services/onchain/abiRegistry.js.map +1 -0
  146. package/dist/backend/backend/services/onchain/chainlinkClient.js +43 -0
  147. package/dist/backend/backend/services/onchain/chainlinkClient.js.map +1 -0
  148. package/dist/backend/backend/services/onchain/errors.js +13 -0
  149. package/dist/backend/backend/services/onchain/errors.js.map +1 -0
  150. package/dist/backend/backend/services/onchain/etherscanClient.js +82 -0
  151. package/dist/backend/backend/services/onchain/etherscanClient.js.map +1 -0
  152. package/dist/backend/backend/services/onchain/index.js +79 -0
  153. package/dist/backend/backend/services/onchain/index.js.map +1 -0
  154. package/dist/backend/backend/services/onchain/tenderlyRpcClient.js +60 -0
  155. package/dist/backend/backend/services/onchain/tenderlyRpcClient.js.map +1 -0
  156. package/dist/backend/backend/services/onchain/types.js +14 -0
  157. package/dist/backend/backend/services/onchain/types.js.map +1 -0
  158. package/dist/backend/backend/services/onchainAutomationService.js +316 -0
  159. package/dist/backend/backend/services/onchainAutomationService.js.map +1 -0
  160. package/dist/backend/backend/services/robinhood/auth.js +26 -0
  161. package/dist/backend/backend/services/robinhood/auth.js.map +1 -0
  162. package/dist/backend/backend/services/robinhood/client.js +73 -0
  163. package/dist/backend/backend/services/robinhood/client.js.map +1 -0
  164. package/dist/backend/backend/services/robinhood/integrationService.js +119 -0
  165. package/dist/backend/backend/services/robinhood/integrationService.js.map +1 -0
  166. package/dist/backend/backend/services/robinhood/provider.js +17 -0
  167. package/dist/backend/backend/services/robinhood/provider.js.map +1 -0
  168. package/dist/backend/backend/services/robinhood/unofficialClient.js +61 -0
  169. package/dist/backend/backend/services/robinhood/unofficialClient.js.map +1 -0
  170. package/dist/backend/backend/services/rpcService.js +48 -0
  171. package/dist/backend/backend/services/rpcService.js.map +1 -0
  172. package/dist/backend/backend/services/runtimeTelemetryService.js +96 -0
  173. package/dist/backend/backend/services/runtimeTelemetryService.js.map +1 -0
  174. package/dist/backend/backend/services/safeConfigRepository.js +147 -0
  175. package/dist/backend/backend/services/safeConfigRepository.js.map +1 -0
  176. package/dist/backend/backend/services/safeService.js +527 -0
  177. package/dist/backend/backend/services/safeService.js.map +1 -0
  178. package/dist/backend/backend/services/sandboxService.js +135 -0
  179. package/dist/backend/backend/services/sandboxService.js.map +1 -0
  180. package/dist/backend/backend/services/secureSettingsService.js +50 -0
  181. package/dist/backend/backend/services/secureSettingsService.js.map +1 -0
  182. package/dist/backend/backend/services/transactionHoldService.js +184 -0
  183. package/dist/backend/backend/services/transactionHoldService.js.map +1 -0
  184. package/dist/backend/backend/services/vanityService.js +235 -0
  185. package/dist/backend/backend/services/vanityService.js.map +1 -0
  186. package/dist/backend/backend/services/walletService.js +202 -0
  187. package/dist/backend/backend/services/walletService.js.map +1 -0
  188. package/dist/backend/backend/services/walletStore.js +132 -0
  189. package/dist/backend/backend/services/walletStore.js.map +1 -0
  190. package/dist/backend/backend/utils/abiResolver.js +182 -0
  191. package/dist/backend/backend/utils/abiResolver.js.map +1 -0
  192. package/dist/backend/backend/utils/http.js +12 -0
  193. package/dist/backend/backend/utils/http.js.map +1 -0
  194. package/dist/backend/backend/utils/secretsResolver.js +137 -0
  195. package/dist/backend/backend/utils/secretsResolver.js.map +1 -0
  196. package/dist/backend/backend/utils/signer.js +15 -0
  197. package/dist/backend/backend/utils/signer.js.map +1 -0
  198. package/dist/backend/backend/workers/vanityWorker.js +63 -0
  199. package/dist/backend/backend/workers/vanityWorker.js.map +1 -0
  200. package/dist/backend/cli/gnoman.js +387 -0
  201. package/dist/backend/cli/gnoman.js.map +1 -0
  202. package/dist/backend/modules/sandbox/abiLoader.js +78 -0
  203. package/dist/backend/modules/sandbox/abiLoader.js.map +1 -0
  204. package/dist/backend/modules/sandbox/contractSimulator.js +205 -0
  205. package/dist/backend/modules/sandbox/contractSimulator.js.map +1 -0
  206. package/dist/backend/modules/sandbox/formBuilder.js +14 -0
  207. package/dist/backend/modules/sandbox/formBuilder.js.map +1 -0
  208. package/dist/backend/modules/sandbox/index.js +24 -0
  209. package/dist/backend/modules/sandbox/index.js.map +1 -0
  210. package/dist/backend/modules/sandbox/localFork.js +103 -0
  211. package/dist/backend/modules/sandbox/localFork.js.map +1 -0
  212. package/dist/backend/modules/sandbox/sandboxManager.js +130 -0
  213. package/dist/backend/modules/sandbox/sandboxManager.js.map +1 -0
  214. package/dist/backend/modules/sandbox/types.js +3 -0
  215. package/dist/backend/modules/sandbox/types.js.map +1 -0
  216. package/dist/backend/src/core/backends/fileBackend.js +136 -0
  217. package/dist/backend/src/core/backends/fileBackend.js.map +1 -0
  218. package/dist/backend/src/core/backends/memoryBackend.js +26 -0
  219. package/dist/backend/src/core/backends/memoryBackend.js.map +1 -0
  220. package/dist/backend/src/core/backends/systemBackend.js +86 -0
  221. package/dist/backend/src/core/backends/systemBackend.js.map +1 -0
  222. package/dist/backend/src/core/backends/types.js +12 -0
  223. package/dist/backend/src/core/backends/types.js.map +1 -0
  224. package/dist/backend/src/core/keyringManager.js +178 -0
  225. package/dist/backend/src/core/keyringManager.js.map +1 -0
  226. package/dist/backend/src/utils/abiResolver.js +180 -0
  227. package/dist/backend/src/utils/abiResolver.js.map +1 -0
  228. package/dist/backend/src/utils/runtimeObservability.js +78 -0
  229. package/dist/backend/src/utils/runtimeObservability.js.map +1 -0
  230. package/dist/backend/src/utils/secretsResolver.js +138 -0
  231. package/dist/backend/src/utils/secretsResolver.js.map +1 -0
  232. package/dist/cli/backend/services/diagnosticsService.js +286 -0
  233. package/dist/cli/backend/services/diagnosticsService.js.map +1 -0
  234. package/dist/cli/backend/services/keyringAccessor.js +40 -0
  235. package/dist/cli/backend/services/keyringAccessor.js.map +1 -0
  236. package/dist/cli/backend/services/rpcService.js +48 -0
  237. package/dist/cli/backend/services/rpcService.js.map +1 -0
  238. package/dist/cli/backend/services/runtimeTelemetryService.js +96 -0
  239. package/dist/cli/backend/services/runtimeTelemetryService.js.map +1 -0
  240. package/dist/cli/backend/services/walletService.js +202 -0
  241. package/dist/cli/backend/services/walletService.js.map +1 -0
  242. package/dist/cli/backend/services/walletStore.js +132 -0
  243. package/dist/cli/backend/services/walletStore.js.map +1 -0
  244. package/dist/cli/backend/utils/http.js +12 -0
  245. package/dist/cli/backend/utils/http.js.map +1 -0
  246. package/dist/cli/backend/utils/secretsResolver.js +137 -0
  247. package/dist/cli/backend/utils/secretsResolver.js.map +1 -0
  248. package/dist/cli/cli/gnoman.js +387 -0
  249. package/dist/cli/cli/gnoman.js.map +1 -0
  250. package/dist/cli/src/core/backends/fileBackend.js +136 -0
  251. package/dist/cli/src/core/backends/fileBackend.js.map +1 -0
  252. package/dist/cli/src/core/backends/memoryBackend.js +26 -0
  253. package/dist/cli/src/core/backends/memoryBackend.js.map +1 -0
  254. package/dist/cli/src/core/backends/systemBackend.js +86 -0
  255. package/dist/cli/src/core/backends/systemBackend.js.map +1 -0
  256. package/dist/cli/src/core/backends/types.js +12 -0
  257. package/dist/cli/src/core/backends/types.js.map +1 -0
  258. package/dist/cli/src/core/keyringManager.js +178 -0
  259. package/dist/cli/src/core/keyringManager.js.map +1 -0
  260. package/dist/cli/src/utils/abiResolver.js +180 -0
  261. package/dist/cli/src/utils/abiResolver.js.map +1 -0
  262. package/dist/cli/src/utils/runtimeObservability.js +78 -0
  263. package/dist/cli/src/utils/runtimeObservability.js.map +1 -0
  264. package/dist/cli/src/utils/secretsResolver.js +138 -0
  265. package/dist/cli/src/utils/secretsResolver.js.map +1 -0
  266. package/dist/main/backend/services/keyringAccessor.js +40 -0
  267. package/dist/main/backend/services/keyringAccessor.js.map +1 -0
  268. package/dist/main/backend/utils/http.js +12 -0
  269. package/dist/main/backend/utils/http.js.map +1 -0
  270. package/dist/main/main/ipcHandlers/index.js +26 -0
  271. package/dist/main/main/ipcHandlers/index.js.map +1 -0
  272. package/dist/main/main/keyring/keyringmanager.js +101 -0
  273. package/dist/main/main/keyring/keyringmanager.js.map +1 -0
  274. package/dist/main/main/main.js +224 -0
  275. package/dist/main/main/main.js.map +1 -0
  276. package/dist/main/main/preload/index.js +19 -0
  277. package/dist/main/main/preload/index.js.map +1 -0
  278. package/dist/main/main/preload/licenseBridge.js +105 -0
  279. package/dist/main/main/preload/licenseBridge.js.map +1 -0
  280. package/dist/main/src/core/backends/fileBackend.js +136 -0
  281. package/dist/main/src/core/backends/fileBackend.js.map +1 -0
  282. package/dist/main/src/core/backends/memoryBackend.js +26 -0
  283. package/dist/main/src/core/backends/memoryBackend.js.map +1 -0
  284. package/dist/main/src/core/backends/systemBackend.js +86 -0
  285. package/dist/main/src/core/backends/systemBackend.js.map +1 -0
  286. package/dist/main/src/core/backends/types.js +12 -0
  287. package/dist/main/src/core/backends/types.js.map +1 -0
  288. package/dist/main/src/core/keyringManager.js +178 -0
  289. package/dist/main/src/core/keyringManager.js.map +1 -0
  290. package/dist/main/src/utils/abiResolver.js +180 -0
  291. package/dist/main/src/utils/abiResolver.js.map +1 -0
  292. package/dist/main/src/utils/runtimeObservability.js +78 -0
  293. package/dist/main/src/utils/runtimeObservability.js.map +1 -0
  294. package/dist/main/src/utils/secretsResolver.js +138 -0
  295. package/dist/main/src/utils/secretsResolver.js.map +1 -0
  296. package/docs/development-guide.md +203 -0
  297. package/docs/etherscan-chainlink-integration.md +44 -0
  298. package/docs/gnoman-20-user-manual-STANDARD-PRINT-READY.pdf +0 -0
  299. package/docs/gnoman-20-user-manual-STANDARD.pdf +0 -0
  300. package/docs/license-dev-guide.md +106 -0
  301. package/docs/robinhood-integration.md +30 -0
  302. package/docs/system-audit-gpt-guide.md +208 -0
  303. package/docs/system-robustness-audit.md +50 -0
  304. package/docs/user-guide.md +73 -0
  305. package/docs/wiki/development-guide.md +203 -0
  306. package/docs/wiki/license-dev-guide.md +106 -0
  307. package/docs/wiki/user-guide.md +73 -0
  308. package/eslint.config.js +85 -0
  309. package/gnoman2.0/.eslintrc.cjs +24 -0
  310. package/gnoman2.0/.prettierrc.json +6 -0
  311. package/gnoman2.0/CODex_TASKS.md +124 -0
  312. package/gnoman2.0/LICENSE.md +164 -0
  313. package/gnoman2.0/README.md +95 -0
  314. package/gnoman2.0/assets/GnoLogo.jpg +0 -0
  315. package/gnoman2.0/assets/self.png +0 -0
  316. package/gnoman2.0/backend/controllers/contractController.ts +49 -0
  317. package/gnoman2.0/backend/controllers/devToolsController.ts +76 -0
  318. package/gnoman2.0/backend/controllers/etherscanController.ts +59 -0
  319. package/gnoman2.0/backend/controllers/historyController.ts +7 -0
  320. package/gnoman2.0/backend/controllers/keyringController.ts +134 -0
  321. package/gnoman2.0/backend/controllers/robinhoodController.ts +80 -0
  322. package/gnoman2.0/backend/controllers/safeController.ts +167 -0
  323. package/gnoman2.0/backend/controllers/sandboxController.ts +63 -0
  324. package/gnoman2.0/backend/controllers/settingsController.ts +38 -0
  325. package/gnoman2.0/backend/controllers/walletController.ts +151 -0
  326. package/gnoman2.0/backend/index.ts +133 -0
  327. package/gnoman2.0/backend/licenses/license_public.pem +3 -0
  328. package/gnoman2.0/backend/licenses/verify_license.py +43 -0
  329. package/gnoman2.0/backend/routes/contractRoutes.ts +11 -0
  330. package/gnoman2.0/backend/routes/devToolsRoutes.ts +11 -0
  331. package/gnoman2.0/backend/routes/etherscanRoutes.ts +11 -0
  332. package/gnoman2.0/backend/routes/historyRoutes.ts +8 -0
  333. package/gnoman2.0/backend/routes/keyringRoutes.ts +25 -0
  334. package/gnoman2.0/backend/routes/licenseRoutes.ts +35 -0
  335. package/gnoman2.0/backend/routes/robinhoodRoutes.ts +22 -0
  336. package/gnoman2.0/backend/routes/runtimeRoutes.ts +29 -0
  337. package/gnoman2.0/backend/routes/safeRoutes.ts +28 -0
  338. package/gnoman2.0/backend/routes/sandboxRoutes.ts +17 -0
  339. package/gnoman2.0/backend/routes/settingsRoutes.ts +14 -0
  340. package/gnoman2.0/backend/routes/walletRoutes.ts +21 -0
  341. package/gnoman2.0/backend/services/chainlinkService.ts +65 -0
  342. package/gnoman2.0/backend/services/contractRegistryService.ts +205 -0
  343. package/gnoman2.0/backend/services/devToolsService.ts +251 -0
  344. package/gnoman2.0/backend/services/diagnosticsService.ts +350 -0
  345. package/gnoman2.0/backend/services/etherscanService.ts +152 -0
  346. package/gnoman2.0/backend/services/historyService.ts +89 -0
  347. package/gnoman2.0/backend/services/keyringAccessor.ts +4 -0
  348. package/gnoman2.0/backend/services/licenseService.ts +163 -0
  349. package/gnoman2.0/backend/services/onchain/abiRegistry.ts +57 -0
  350. package/gnoman2.0/backend/services/onchain/chainlinkClient.ts +56 -0
  351. package/gnoman2.0/backend/services/onchain/errors.ts +16 -0
  352. package/gnoman2.0/backend/services/onchain/etherscanClient.ts +94 -0
  353. package/gnoman2.0/backend/services/onchain/index.ts +76 -0
  354. package/gnoman2.0/backend/services/onchain/tenderlyRpcClient.ts +74 -0
  355. package/gnoman2.0/backend/services/onchain/types.ts +33 -0
  356. package/gnoman2.0/backend/services/onchainAutomationService.ts +424 -0
  357. package/gnoman2.0/backend/services/robinhood/auth.ts +42 -0
  358. package/gnoman2.0/backend/services/robinhood/client.ts +123 -0
  359. package/gnoman2.0/backend/services/robinhood/integrationService.ts +140 -0
  360. package/gnoman2.0/backend/services/robinhood/provider.ts +22 -0
  361. package/gnoman2.0/backend/services/robinhood/unofficialClient.ts +66 -0
  362. package/gnoman2.0/backend/services/rpcService.ts +44 -0
  363. package/gnoman2.0/backend/services/runtimeTelemetryService.ts +158 -0
  364. package/gnoman2.0/backend/services/safeConfigRepository.ts +205 -0
  365. package/gnoman2.0/backend/services/safeService.ts +588 -0
  366. package/gnoman2.0/backend/services/sandboxService.ts +157 -0
  367. package/gnoman2.0/backend/services/secureSettingsService.ts +45 -0
  368. package/gnoman2.0/backend/services/transactionHoldService.ts +223 -0
  369. package/gnoman2.0/backend/services/vanityService.ts +293 -0
  370. package/gnoman2.0/backend/services/walletService.ts +290 -0
  371. package/gnoman2.0/backend/services/walletStore.ts +179 -0
  372. package/gnoman2.0/backend/types/express-async-handler.d.ts +13 -0
  373. package/gnoman2.0/backend/types/keyring.d.ts +19 -0
  374. package/gnoman2.0/backend/utils/abiResolver.ts +208 -0
  375. package/gnoman2.0/backend/utils/http.ts +6 -0
  376. package/gnoman2.0/backend/utils/secretsResolver.ts +150 -0
  377. package/gnoman2.0/backend/utils/signer.ts +11 -0
  378. package/gnoman2.0/backend/workers/vanityWorker.ts +76 -0
  379. package/gnoman2.0/capacitor.config.ts +13 -0
  380. package/gnoman2.0/cli/gnoman.ts +424 -0
  381. package/gnoman2.0/contracts/OracleConsumer.sol +20 -0
  382. package/gnoman2.0/contracts/PriceFeedConsumer.sol +22 -0
  383. package/gnoman2.0/docs/development-guide.md +203 -0
  384. package/gnoman2.0/docs/etherscan-chainlink-integration.md +44 -0
  385. package/gnoman2.0/docs/gnoman-20-user-manual-STANDARD-PRINT-READY.pdf +0 -0
  386. package/gnoman2.0/docs/gnoman-20-user-manual-STANDARD.pdf +0 -0
  387. package/gnoman2.0/docs/license-dev-guide.md +106 -0
  388. package/gnoman2.0/docs/robinhood-integration.md +30 -0
  389. package/gnoman2.0/docs/system-audit-gpt-guide.md +208 -0
  390. package/gnoman2.0/docs/system-robustness-audit.md +50 -0
  391. package/gnoman2.0/docs/user-guide.md +73 -0
  392. package/gnoman2.0/docs/wiki/development-guide.md +203 -0
  393. package/gnoman2.0/docs/wiki/license-dev-guide.md +106 -0
  394. package/gnoman2.0/docs/wiki/user-guide.md +73 -0
  395. package/gnoman2.0/eslint.config.js +85 -0
  396. package/gnoman2.0/gnomon/__init__.py +0 -0
  397. package/gnoman2.0/gnomon/api/__init__.py +0 -0
  398. package/gnoman2.0/gnomon/api/etherscan_tracker.py +72 -0
  399. package/gnoman2.0/gnomon/core/__init__.py +0 -0
  400. package/gnoman2.0/gnomon/core/safe_manager.py +111 -0
  401. package/gnoman2.0/gnomon/tests/test_abi_resolver.py +181 -0
  402. package/gnoman2.0/gnomon/tests/test_safe_persistence_and_etherscan.py +97 -0
  403. package/gnoman2.0/gnomon/utils/__init__.py +5 -0
  404. package/gnoman2.0/gnomon/utils/abi_resolver.py +255 -0
  405. package/gnoman2.0/ios/ExportOptions.plist +16 -0
  406. package/gnoman2.0/ios/README.md +33 -0
  407. package/gnoman2.0/jest.config.ts +18 -0
  408. package/gnoman2.0/keyring/__init__.py +17 -0
  409. package/gnoman2.0/licensingServer/package.json +23 -0
  410. package/gnoman2.0/licensingServer/src/config/keys.ts +84 -0
  411. package/gnoman2.0/licensingServer/src/index.ts +30 -0
  412. package/gnoman2.0/licensingServer/src/lib/canonicalize.ts +5 -0
  413. package/gnoman2.0/licensingServer/src/lib/crypto.ts +25 -0
  414. package/gnoman2.0/licensingServer/src/lib/validate.ts +62 -0
  415. package/gnoman2.0/licensingServer/src/middleware/auth.ts +20 -0
  416. package/gnoman2.0/licensingServer/src/routes/licenses.ts +110 -0
  417. package/gnoman2.0/licensingServer/tsconfig.json +12 -0
  418. package/gnoman2.0/main/ipcHandlers/index.ts +23 -0
  419. package/gnoman2.0/main/keyring/keyringmanager.ts +154 -0
  420. package/gnoman2.0/main/main.ts +234 -0
  421. package/gnoman2.0/main/preload/index.ts +31 -0
  422. package/gnoman2.0/main/preload/licenseBridge.ts +73 -0
  423. package/gnoman2.0/modules/sandbox/abiLoader.ts +78 -0
  424. package/gnoman2.0/modules/sandbox/contractSimulator.ts +241 -0
  425. package/gnoman2.0/modules/sandbox/formBuilder.ts +16 -0
  426. package/gnoman2.0/modules/sandbox/index.ts +6 -0
  427. package/gnoman2.0/modules/sandbox/localFork.ts +129 -0
  428. package/gnoman2.0/modules/sandbox/safe.abi.json +82 -0
  429. package/gnoman2.0/modules/sandbox/sandboxManager.ts +154 -0
  430. package/gnoman2.0/modules/sandbox/types.ts +84 -0
  431. package/gnoman2.0/modules/sandbox/ui/LogViewer.tsx +30 -0
  432. package/gnoman2.0/modules/sandbox/ui/ParameterForm.tsx +49 -0
  433. package/gnoman2.0/modules/sandbox/ui/SandboxPanel.tsx +568 -0
  434. package/gnoman2.0/package-lock.json +10904 -0
  435. package/gnoman2.0/package.json +82 -0
  436. package/gnoman2.0/renderer/components/LicenseScreen.tsx +134 -0
  437. package/gnoman2.0/renderer/index.html +12 -0
  438. package/gnoman2.0/renderer/package-lock.json +4104 -0
  439. package/gnoman2.0/renderer/package.json +35 -0
  440. package/gnoman2.0/renderer/postcss.config.cjs +6 -0
  441. package/gnoman2.0/renderer/src/App.tsx +229 -0
  442. package/gnoman2.0/renderer/src/context/KeyringContext.tsx +217 -0
  443. package/gnoman2.0/renderer/src/context/SafeContext.tsx +49 -0
  444. package/gnoman2.0/renderer/src/context/ThemeContext.tsx +60 -0
  445. package/gnoman2.0/renderer/src/context/WalletContext.tsx +50 -0
  446. package/gnoman2.0/renderer/src/context/main.tsx +18 -0
  447. package/gnoman2.0/renderer/src/main.tsx +18 -0
  448. package/gnoman2.0/renderer/src/pages/Contracts.tsx +482 -0
  449. package/gnoman2.0/renderer/src/pages/Dashboard.tsx +653 -0
  450. package/gnoman2.0/renderer/src/pages/DeveloperTools.tsx +270 -0
  451. package/gnoman2.0/renderer/src/pages/History.tsx +149 -0
  452. package/gnoman2.0/renderer/src/pages/Keyring.tsx +449 -0
  453. package/gnoman2.0/renderer/src/pages/Safes.tsx +1089 -0
  454. package/gnoman2.0/renderer/src/pages/Sandbox.tsx +146 -0
  455. package/gnoman2.0/renderer/src/pages/Settings.tsx +871 -0
  456. package/gnoman2.0/renderer/src/pages/Wallets.tsx +752 -0
  457. package/gnoman2.0/renderer/src/pages/WikiGuide.tsx +75 -0
  458. package/gnoman2.0/renderer/src/styles.css +32 -0
  459. package/gnoman2.0/renderer/src/types/gnoman.d.ts +9 -0
  460. package/gnoman2.0/renderer/src/types/license.ts +8 -0
  461. package/gnoman2.0/renderer/src/types/safevault.d.ts +17 -0
  462. package/gnoman2.0/renderer/src/utils/backend.ts +88 -0
  463. package/gnoman2.0/renderer/tailwind.config.cjs +8 -0
  464. package/gnoman2.0/renderer/tsconfig.json +13 -0
  465. package/gnoman2.0/renderer/tsconfig.node.json +9 -0
  466. package/gnoman2.0/renderer/vite.config.ts +19 -0
  467. package/gnoman2.0/requests/__init__.py +35 -0
  468. package/gnoman2.0/scripts/build-ios.sh +30 -0
  469. package/gnoman2.0/scripts/copyBackendAssets.js +24 -0
  470. package/gnoman2.0/scripts/copyRenderer.js +87 -0
  471. package/gnoman2.0/scripts/launchElectron.js +51 -0
  472. package/gnoman2.0/src/core/backends/fileBackend.ts +154 -0
  473. package/gnoman2.0/src/core/backends/memoryBackend.ts +27 -0
  474. package/gnoman2.0/src/core/backends/systemBackend.ts +66 -0
  475. package/gnoman2.0/src/core/backends/types.ts +17 -0
  476. package/gnoman2.0/src/core/keyringManager.ts +208 -0
  477. package/gnoman2.0/src/utils/abiCache/.gitkeep +0 -0
  478. package/gnoman2.0/src/utils/abiResolver.ts +200 -0
  479. package/gnoman2.0/src/utils/runtimeObservability.ts +110 -0
  480. package/gnoman2.0/src/utils/secretsResolver.ts +144 -0
  481. package/gnoman2.0/tests/chainlinkService.test.ts +32 -0
  482. package/gnoman2.0/tests/diagnosticsService.test.ts +68 -0
  483. package/gnoman2.0/tests/etherscanController.test.ts +99 -0
  484. package/gnoman2.0/tests/etherscanService.test.ts +116 -0
  485. package/gnoman2.0/tests/keyringManager.test.ts +135 -0
  486. package/gnoman2.0/tests/onchainToolkit.test.ts +71 -0
  487. package/gnoman2.0/tests/robinhoodClient.test.ts +54 -0
  488. package/gnoman2.0/tests/robinhoodController.test.ts +81 -0
  489. package/gnoman2.0/tests/robinhoodIntegrationService.test.ts +50 -0
  490. package/gnoman2.0/tests/safeServicePersistence.test.ts +81 -0
  491. package/gnoman2.0/tests/test_contract_sandbox/sandbox.test.js +407 -0
  492. package/gnoman2.0/tests/walletController.test.ts +57 -0
  493. package/gnoman2.0/tsconfig.backend.json +7 -0
  494. package/gnoman2.0/tsconfig.cli.json +7 -0
  495. package/gnoman2.0/tsconfig.json +18 -0
  496. package/gnoman2.0/tsconfig.main.json +7 -0
  497. package/gnomon/__init__.py +0 -0
  498. package/gnomon/__pycache__/__init__.cpython-310.pyc +0 -0
  499. package/gnomon/api/__init__.py +0 -0
  500. package/gnomon/api/__pycache__/__init__.cpython-310.pyc +0 -0
  501. package/gnomon/api/__pycache__/etherscan_tracker.cpython-310.pyc +0 -0
  502. package/gnomon/api/etherscan_tracker.py +72 -0
  503. package/gnomon/core/__init__.py +0 -0
  504. package/gnomon/core/safe_manager.py +111 -0
  505. package/gnomon/tests/__pycache__/test_safe_persistence_and_etherscan.cpython-310-pytest-8.3.3.pyc +0 -0
  506. package/gnomon/tests/test_abi_resolver.py +181 -0
  507. package/gnomon/tests/test_safe_persistence_and_etherscan.py +97 -0
  508. package/gnomon/utils/__init__.py +5 -0
  509. package/gnomon/utils/abi_resolver.py +255 -0
  510. package/ios/ExportOptions.plist +16 -0
  511. package/ios/README.md +33 -0
  512. package/jest.config.ts +18 -0
  513. package/keyring/__init__.py +17 -0
  514. package/launcher.sh +57 -0
  515. package/license.env +2 -0
  516. package/licensingServer/package.json +23 -0
  517. package/licensingServer/src/config/keys.ts +84 -0
  518. package/licensingServer/src/index.ts +30 -0
  519. package/licensingServer/src/lib/canonicalize.ts +5 -0
  520. package/licensingServer/src/lib/crypto.ts +25 -0
  521. package/licensingServer/src/lib/validate.ts +62 -0
  522. package/licensingServer/src/middleware/auth.ts +20 -0
  523. package/licensingServer/src/routes/licenses.ts +110 -0
  524. package/licensingServer/tsconfig.json +12 -0
  525. package/main/ipcHandlers/index.ts +23 -0
  526. package/main/keyring/keyringmanager.ts +154 -0
  527. package/main/main.ts +234 -0
  528. package/main/preload/index.ts +31 -0
  529. package/main/preload/licenseBridge.ts +73 -0
  530. package/modules/sandbox/abiLoader.ts +78 -0
  531. package/modules/sandbox/contractSimulator.ts +241 -0
  532. package/modules/sandbox/formBuilder.ts +16 -0
  533. package/modules/sandbox/index.ts +6 -0
  534. package/modules/sandbox/localFork.ts +129 -0
  535. package/modules/sandbox/safe.abi.json +82 -0
  536. package/modules/sandbox/sandboxManager.ts +154 -0
  537. package/modules/sandbox/types.ts +84 -0
  538. package/modules/sandbox/ui/LogViewer.tsx +30 -0
  539. package/modules/sandbox/ui/ParameterForm.tsx +49 -0
  540. package/modules/sandbox/ui/SandboxPanel.tsx +568 -0
  541. package/package.json +82 -0
  542. package/renderer/components/LicenseScreen.tsx +134 -0
  543. package/renderer/index.html +12 -0
  544. package/renderer/package-lock.json +4104 -0
  545. package/renderer/package.json +35 -0
  546. package/renderer/postcss.config.cjs +6 -0
  547. package/renderer/src/App.tsx +229 -0
  548. package/renderer/src/context/KeyringContext.tsx +217 -0
  549. package/renderer/src/context/SafeContext.tsx +49 -0
  550. package/renderer/src/context/ThemeContext.tsx +60 -0
  551. package/renderer/src/context/WalletContext.tsx +50 -0
  552. package/renderer/src/context/main.tsx +18 -0
  553. package/renderer/src/main.tsx +18 -0
  554. package/renderer/src/pages/Contracts.tsx +482 -0
  555. package/renderer/src/pages/Dashboard.tsx +653 -0
  556. package/renderer/src/pages/DeveloperTools.tsx +270 -0
  557. package/renderer/src/pages/History.tsx +149 -0
  558. package/renderer/src/pages/Keyring.tsx +449 -0
  559. package/renderer/src/pages/Safes.tsx +1089 -0
  560. package/renderer/src/pages/Sandbox.tsx +146 -0
  561. package/renderer/src/pages/Settings.tsx +871 -0
  562. package/renderer/src/pages/Wallets.tsx +752 -0
  563. package/renderer/src/pages/WikiGuide.tsx +75 -0
  564. package/renderer/src/styles.css +32 -0
  565. package/renderer/src/types/gnoman.d.ts +9 -0
  566. package/renderer/src/types/license.ts +8 -0
  567. package/renderer/src/types/safevault.d.ts +17 -0
  568. package/renderer/src/utils/backend.ts +88 -0
  569. package/renderer/tailwind.config.cjs +8 -0
  570. package/renderer/tsconfig.json +13 -0
  571. package/renderer/tsconfig.node.json +9 -0
  572. package/renderer/vite.config.ts +19 -0
  573. package/requests/__init__.py +35 -0
  574. package/requests/__pycache__/__init__.cpython-310.pyc +0 -0
  575. package/scripts/build-ios.sh +30 -0
  576. package/scripts/copyBackendAssets.js +24 -0
  577. package/scripts/copyRenderer.js +87 -0
  578. package/scripts/deployBackend.sh +24 -0
  579. package/scripts/launchElectron.js +51 -0
  580. package/src/core/backends/fileBackend.ts +154 -0
  581. package/src/core/backends/memoryBackend.ts +27 -0
  582. package/src/core/backends/systemBackend.ts +66 -0
  583. package/src/core/backends/types.ts +17 -0
  584. package/src/core/keyringManager.ts +208 -0
  585. package/src/utils/abiCache/.gitkeep +0 -0
  586. package/src/utils/abiResolver.ts +200 -0
  587. package/src/utils/runtimeObservability.ts +110 -0
  588. package/src/utils/secretsResolver.ts +144 -0
  589. package/tests/chainlinkService.test.ts +32 -0
  590. package/tests/diagnosticsService.test.ts +68 -0
  591. package/tests/etherscanController.test.ts +99 -0
  592. package/tests/etherscanService.test.ts +116 -0
  593. package/tests/keyringManager.test.ts +135 -0
  594. package/tests/onchainToolkit.test.ts +71 -0
  595. package/tests/robinhoodClient.test.ts +54 -0
  596. package/tests/robinhoodController.test.ts +81 -0
  597. package/tests/robinhoodIntegrationService.test.ts +50 -0
  598. package/tests/safeServicePersistence.test.ts +81 -0
  599. package/tests/test_contract_sandbox/sandbox.test.js +407 -0
  600. package/tests/walletController.test.ts +57 -0
  601. package/touch +14 -0
  602. package/tsconfig.backend.json +7 -0
  603. package/tsconfig.cli.json +7 -0
  604. package/tsconfig.json +18 -0
  605. package/tsconfig.main.json +7 -0
  606. package/webhook-shim.js +50 -0
@@ -0,0 +1,179 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import Database from 'better-sqlite3';
4
+ import type { Statement } from 'better-sqlite3';
5
+
6
+ export interface PersistedWalletRecord {
7
+ address: string;
8
+ alias?: string;
9
+ encryptedSecret: string;
10
+ iv: string;
11
+ salt: string;
12
+ hidden: boolean;
13
+ createdAt: string;
14
+ source: string;
15
+ publicKey?: string;
16
+ mnemonic?: string;
17
+ derivationPath?: string;
18
+ network?: string;
19
+ balance?: string;
20
+ privateKey: string;
21
+ }
22
+
23
+ const storageDir = path.join(process.cwd(), '.gnoman');
24
+ if (!fs.existsSync(storageDir)) {
25
+ fs.mkdirSync(storageDir, { recursive: true, mode: 0o700 });
26
+ }
27
+
28
+ const databasePath = path.join(storageDir, 'wallets.db');
29
+ const db = new Database(databasePath);
30
+
31
+ try {
32
+ db.pragma('journal_mode = WAL');
33
+ db.pragma('synchronous = FULL');
34
+ } catch (error) {
35
+ console.warn('Unable to configure wallet database pragmas', error);
36
+ }
37
+
38
+ db.exec(`
39
+ CREATE TABLE IF NOT EXISTS wallets (
40
+ address TEXT PRIMARY KEY COLLATE NOCASE,
41
+ alias TEXT,
42
+ encryptedSecret TEXT NOT NULL,
43
+ iv TEXT NOT NULL,
44
+ salt TEXT NOT NULL,
45
+ hidden INTEGER NOT NULL DEFAULT 0,
46
+ createdAt TEXT NOT NULL,
47
+ source TEXT NOT NULL,
48
+ publicKey TEXT,
49
+ mnemonic TEXT,
50
+ derivationPath TEXT,
51
+ network TEXT,
52
+ balance TEXT,
53
+ privateKey TEXT NOT NULL
54
+ )
55
+ `);
56
+
57
+ const ensureColumn = (name: string, type: string, defaultValue?: string) => {
58
+ try {
59
+ db.exec(
60
+ `ALTER TABLE wallets ADD COLUMN ${name} ${type}${
61
+ typeof defaultValue !== 'undefined' ? ` DEFAULT ${defaultValue}` : ''
62
+ }`
63
+ );
64
+ } catch (error) {
65
+ if (!(error instanceof Error) || !/duplicate column name/i.test(error.message)) {
66
+ throw error;
67
+ }
68
+ }
69
+ };
70
+
71
+ ensureColumn('publicKey', 'TEXT');
72
+ ensureColumn('mnemonic', 'TEXT');
73
+ ensureColumn('derivationPath', 'TEXT');
74
+ ensureColumn('network', 'TEXT');
75
+ ensureColumn('balance', 'TEXT');
76
+ ensureColumn('privateKey', 'TEXT');
77
+
78
+ const insertStatement: Statement = db.prepare(
79
+ `INSERT INTO wallets (address, alias, encryptedSecret, iv, salt, hidden, createdAt, source, publicKey, mnemonic, derivationPath, network, balance, privateKey)
80
+ VALUES (@address, @alias, @encryptedSecret, @iv, @salt, @hidden, @createdAt, @source, @publicKey, @mnemonic, @derivationPath, @network, @balance, @privateKey)
81
+ ON CONFLICT(address) DO UPDATE SET
82
+ alias = excluded.alias,
83
+ encryptedSecret = excluded.encryptedSecret,
84
+ iv = excluded.iv,
85
+ salt = excluded.salt,
86
+ hidden = excluded.hidden,
87
+ source = excluded.source,
88
+ publicKey = excluded.publicKey,
89
+ mnemonic = excluded.mnemonic,
90
+ derivationPath = excluded.derivationPath,
91
+ network = excluded.network,
92
+ balance = excluded.balance,
93
+ privateKey = excluded.privateKey`
94
+ );
95
+
96
+ const listStatement: Statement = db.prepare(`
97
+ SELECT address, alias, encryptedSecret, iv, salt, hidden, createdAt, source, publicKey, mnemonic, derivationPath, network, balance, privateKey
98
+ FROM wallets
99
+ ORDER BY datetime(createdAt) DESC
100
+ `);
101
+
102
+ const getStatement: Statement = db.prepare(`
103
+ SELECT address, alias, encryptedSecret, iv, salt, hidden, createdAt, source, publicKey, mnemonic, derivationPath, network, balance, privateKey
104
+ FROM wallets
105
+ WHERE address = ?
106
+ `);
107
+
108
+ const deleteStatement: Statement = db.prepare(`
109
+ DELETE FROM wallets
110
+ WHERE address = ?
111
+ `);
112
+
113
+ type WalletRow = {
114
+ address: string;
115
+ alias: string | null;
116
+ encryptedSecret: string;
117
+ iv: string;
118
+ salt: string;
119
+ hidden: number;
120
+ createdAt: string;
121
+ source: string;
122
+ publicKey: string | null;
123
+ mnemonic: string | null;
124
+ derivationPath: string | null;
125
+ network: string | null;
126
+ balance: string | null;
127
+ privateKey: string;
128
+ };
129
+
130
+ const mapRow = (row: WalletRow): PersistedWalletRecord => {
131
+ return {
132
+ address: row.address,
133
+ alias: row.alias ?? undefined,
134
+ encryptedSecret: row.encryptedSecret,
135
+ iv: row.iv,
136
+ salt: row.salt,
137
+ hidden: Boolean(row.hidden),
138
+ createdAt: row.createdAt,
139
+ source: row.source,
140
+ publicKey: row.publicKey ?? undefined,
141
+ mnemonic: row.mnemonic ?? undefined,
142
+ derivationPath: row.derivationPath ?? undefined,
143
+ network: row.network ?? undefined,
144
+ balance: row.balance ?? undefined,
145
+ privateKey: row.privateKey ?? ''
146
+ };
147
+ };
148
+
149
+ export const walletRepository = {
150
+ save(record: PersistedWalletRecord) {
151
+ insertStatement.run({
152
+ ...record,
153
+ alias: record.alias ?? null,
154
+ hidden: record.hidden ? 1 : 0,
155
+ publicKey: record.publicKey ?? null,
156
+ mnemonic: record.mnemonic ?? null,
157
+ derivationPath: record.derivationPath ?? null,
158
+ network: record.network ?? null,
159
+ balance: record.balance ?? null
160
+ });
161
+ },
162
+
163
+ list(): PersistedWalletRecord[] {
164
+ return (listStatement.all() as WalletRow[]).map((row) => mapRow(row));
165
+ },
166
+
167
+ find(address: string): PersistedWalletRecord | undefined {
168
+ const row = getStatement.get(address) as WalletRow | undefined;
169
+ if (!row) {
170
+ return undefined;
171
+ }
172
+ return mapRow(row);
173
+ },
174
+
175
+ delete(address: string): boolean {
176
+ const result = deleteStatement.run(address);
177
+ return result.changes > 0;
178
+ }
179
+ };
@@ -0,0 +1,13 @@
1
+ declare module 'express-async-handler' {
2
+ import type { NextFunction, Request, RequestHandler, Response } from 'express';
3
+
4
+ type AsyncRequestHandler<P = any, ResBody = any, ReqBody = any, ReqQuery = any> = (
5
+ req: Request<P, ResBody, ReqBody, ReqQuery>,
6
+ res: Response<ResBody>,
7
+ next: NextFunction
8
+ ) => Promise<unknown> | unknown;
9
+
10
+ export default function asyncHandler<P = any, ResBody = any, ReqBody = any, ReqQuery = any>(
11
+ handler: AsyncRequestHandler<P, ResBody, ReqBody, ReqQuery>
12
+ ): RequestHandler<P, ResBody, ReqBody, ReqQuery>;
13
+ }
@@ -0,0 +1,19 @@
1
+ declare module 'keyring' {
2
+ interface KeyringApi {
3
+ load(ondone?: () => void): KeyringApi;
4
+ save(ondone?: () => void): KeyringApi;
5
+ store(key: string, value: string): KeyringApi;
6
+ storeEncrypted(key: string, value: string): KeyringApi;
7
+ retrieve(key: string): string | null;
8
+ retrieveEncrypted(key: string): string | null;
9
+ db?: Record<string, unknown>;
10
+ }
11
+
12
+ interface KeyringModule {
13
+ ALGORITHM: string;
14
+ instance(encryptionKey: string, keyringDatabase?: string): KeyringApi;
15
+ }
16
+
17
+ const api: KeyringModule;
18
+ export = api;
19
+ }
@@ -0,0 +1,208 @@
1
+ import crypto from 'crypto';
2
+ import fs from 'fs';
3
+ import path from 'path';
4
+ import { ethers } from 'ethers';
5
+ import { runtimeTelemetry } from '../services/runtimeTelemetryService';
6
+ import { secretsResolver } from './secretsResolver';
7
+
8
+ export type AbiResolveResult = {
9
+ abi: unknown[];
10
+ contractName: string;
11
+ source: 'cache' | 'sourcify' | 'etherscan' | 'manual-registry';
12
+ fetchedAt: string;
13
+ cachePath: string;
14
+ verified: boolean;
15
+ cached: boolean;
16
+ };
17
+
18
+ type IndexEntry = { address: string; contractName: string; source: string; abiHash: string; updatedAt: string };
19
+
20
+ const root = path.join(process.cwd(), 'abi-cache');
21
+ const manualRegistryPath = path.join(process.cwd(), 'abi', 'manual-registry.json');
22
+
23
+ const chainDir = (chainId: number) => path.join(root, `${chainId}`);
24
+ const abiPath = (chainId: number, address: string) => path.join(chainDir(chainId), `${address.toLowerCase()}.json`);
25
+ const indexPath = (chainId: number) => path.join(chainDir(chainId), 'index.json');
26
+
27
+ const readJson = <T>(filePath: string, fallback: T): T => {
28
+ try {
29
+ if (!fs.existsSync(filePath)) return fallback;
30
+ return JSON.parse(fs.readFileSync(filePath, 'utf8')) as T;
31
+ } catch (error) {
32
+ console.error(JSON.stringify({ event: 'CONFIG_EARLY_RETURN', fn: 'AbiResolver.readJson', reason: error instanceof Error ? error.message : String(error), filePath }));
33
+ return fallback;
34
+ }
35
+ };
36
+
37
+ const writeJson = (filePath: string, payload: unknown) => {
38
+ fs.mkdirSync(path.dirname(filePath), { recursive: true });
39
+ fs.writeFileSync(filePath, JSON.stringify(payload, null, 2), 'utf8');
40
+ };
41
+
42
+ const normalizeAddress = (address: string) => ethers.getAddress(address).toLowerCase();
43
+
44
+ export class AbiResolver {
45
+ async resolve(chainId: number, addressInput: string, contractNameHint?: string): Promise<AbiResolveResult> {
46
+ console.info(JSON.stringify({ event: 'ABI_RESOLVE start', address: addressInput, chainId }));
47
+ const address = normalizeAddress(addressInput);
48
+ const cachedPath = abiPath(chainId, address);
49
+ if (fs.existsSync(cachedPath)) {
50
+ const payload = readJson<{ abi: unknown[]; contractName?: string; source?: AbiResolveResult['source']; fetchedAt?: string; verified?: boolean }>(cachedPath, { abi: [] });
51
+ const result: AbiResolveResult = {
52
+ abi: Array.isArray(payload.abi) ? payload.abi : [],
53
+ contractName: payload.contractName ?? contractNameHint ?? address,
54
+ source: 'cache',
55
+ fetchedAt: payload.fetchedAt ?? new Date().toISOString(),
56
+ cachePath: cachedPath,
57
+ verified: payload.verified ?? true,
58
+ cached: true
59
+ };
60
+ this.logResolve(chainId, address, result);
61
+ console.debug(JSON.stringify({ event: 'TRACE', phase: 'exit', fn: 'AbiResolver.resolve', ok: true, source: 'cache' }));
62
+ return result;
63
+ }
64
+
65
+ const sourcify = await this.resolveFromSourcify(chainId, address);
66
+ if (sourcify) {
67
+ const persisted = this.persist(chainId, address, sourcify.abi, sourcify.contractName, 'sourcify', sourcify.verified);
68
+ console.debug(JSON.stringify({ event: 'TRACE', phase: 'exit', fn: 'AbiResolver.resolve', ok: true, source: 'sourcify' }));
69
+ return persisted;
70
+ }
71
+
72
+ const etherscan = await this.resolveFromEtherscan(chainId, address);
73
+ if (etherscan) {
74
+ const persisted = this.persist(chainId, address, etherscan.abi, etherscan.contractName, 'etherscan', true);
75
+ console.debug(JSON.stringify({ event: 'TRACE', phase: 'exit', fn: 'AbiResolver.resolve', ok: true, source: 'etherscan' }));
76
+ return persisted;
77
+ }
78
+
79
+ const manual = this.resolveFromManualRegistry(chainId, address, contractNameHint);
80
+ if (manual) {
81
+ const persisted = this.persist(chainId, address, manual.abi, manual.contractName, 'manual-registry', manual.verified);
82
+ console.debug(JSON.stringify({ event: 'TRACE', phase: 'exit', fn: 'AbiResolver.resolve', ok: true, source: 'manual-registry' }));
83
+ return persisted;
84
+ }
85
+
86
+ console.error(JSON.stringify({ event: 'ABI_RESOLVE_FAILED', reason: 'no-source-hit', chainId, address }));
87
+ console.debug(JSON.stringify({ event: 'TRACE', phase: 'exit', fn: 'AbiResolver.resolve', ok: false }));
88
+ throw new Error(`Unable to resolve ABI for chainId=${chainId} address=${address}`);
89
+ }
90
+
91
+ private async resolveFromSourcify(chainId: number, address: string) {
92
+ const url = `https://repo.sourcify.dev/contracts/full_match/${chainId}/${address}/metadata.json`;
93
+ try {
94
+ const response = await fetch(url);
95
+ if (!response.ok) {
96
+ console.info(JSON.stringify({ event: 'ABI_RESOLVE source=sourcify fail', chainId, address, reason: `http_${response.status}` }));
97
+ return null;
98
+ }
99
+ const payload = (await response.json()) as { output?: { abi?: unknown[] }; contractName?: string };
100
+ if (!Array.isArray(payload.output?.abi)) {
101
+ console.info(JSON.stringify({ event: 'ABI_RESOLVE source=sourcify fail', chainId, address, reason: 'missing_abi' }));
102
+ return null;
103
+ }
104
+ console.info(JSON.stringify({ event: 'ABI_RESOLVE source=sourcify success', chainId, address }));
105
+ return { abi: payload.output.abi, contractName: payload.contractName ?? address, verified: true };
106
+ } catch (error) {
107
+ console.error(JSON.stringify({ event: 'ABI_RESOLVE source=sourcify fail', chainId, address, reason: error instanceof Error ? error.message : String(error) }));
108
+ return null;
109
+ }
110
+ }
111
+
112
+ private async resolveFromEtherscan(chainId: number, address: string) {
113
+ const cacheKey = `${chainId}:${address.toLowerCase()}`;
114
+ const apiKey = await secretsResolver.resolve('ETHERSCAN_API_KEY', { required: true, failClosed: false });
115
+ if (!apiKey) {
116
+ console.error(JSON.stringify({ event: 'ABI_RESOLVE source=etherscan fail', chainId, address, reason: 'missing_key', attemptedNetwork: false, cacheKey }));
117
+ return null;
118
+ }
119
+ const endpoint = process.env.ETHERSCAN_BASE_URL?.trim() || 'https://api.etherscan.io/api';
120
+ const query = new URLSearchParams({ module: 'contract', action: 'getabi', address, chainid: `${chainId}`, apikey: apiKey });
121
+ const response = await fetch(`${endpoint}?${query.toString()}`);
122
+ if (!response.ok) {
123
+ const reason = response.status === 401 || response.status === 403 ? 'key_auth' : response.status === 404 ? 'unverified_or_missing' : response.status === 429 ? 'rate_limit' : `http_${response.status}`;
124
+ console.error(JSON.stringify({ event: 'ABI_RESOLVE source=etherscan fail', chainId, address, reason, attemptedNetwork: true, cacheKey }));
125
+ return null;
126
+ }
127
+ const payload = (await response.json()) as { status?: string; result?: string; message?: string };
128
+ if (payload.status !== '1' || !payload.result) {
129
+ console.error(JSON.stringify({ event: 'ABI_RESOLVE source=etherscan fail', chainId, address, reason: payload.message ?? 'invalid_payload', attemptedNetwork: true, cacheKey }));
130
+ return null;
131
+ }
132
+ try {
133
+ const abi = JSON.parse(payload.result) as unknown[];
134
+ console.info(JSON.stringify({ event: 'ABI_RESOLVE source=etherscan success', chainId, address, attemptedNetwork: true, cacheKey }));
135
+ return { abi, contractName: address };
136
+ } catch (error) {
137
+ console.error(JSON.stringify({ event: 'ABI_RESOLVE source=etherscan fail', chainId, address, reason: 'parse_error', attemptedNetwork: true, cacheKey, detail: error instanceof Error ? error.message : String(error) }));
138
+ return null;
139
+ }
140
+ }
141
+
142
+ private resolveFromManualRegistry(chainId: number, address: string, contractNameHint?: string) {
143
+ const registry = readJson<Record<string, Record<string, { abi: unknown[]; contractName?: string; verified?: boolean }>>>(manualRegistryPath, {});
144
+ const chainKey = `${chainId}`;
145
+ const byChain = registry[chainKey] ?? {};
146
+ const byAddress = byChain[address];
147
+ if (byAddress) {
148
+ return { abi: byAddress.abi, contractName: byAddress.contractName ?? contractNameHint ?? address, verified: Boolean(byAddress.verified) };
149
+ }
150
+ if (contractNameHint) {
151
+ const entries = Object.entries(byChain).filter(([, value]) => value.contractName === contractNameHint);
152
+ if (entries.length === 1) {
153
+ return { abi: entries[0][1].abi, contractName: contractNameHint, verified: Boolean(entries[0][1].verified) };
154
+ }
155
+ if (entries.length > 1) {
156
+ throw new Error(`Contract name ${contractNameHint} resolves to multiple addresses on chain ${chainId}`);
157
+ }
158
+ }
159
+ console.error(JSON.stringify({ event: 'ABI_RESOLVE source=manual-registry fail', chainId, address, reason: 'not_found' }));
160
+ return null;
161
+ }
162
+
163
+ private persist(
164
+ chainId: number,
165
+ address: string,
166
+ abi: unknown[],
167
+ contractName: string,
168
+ source: AbiResolveResult['source'],
169
+ verified: boolean
170
+ ): AbiResolveResult {
171
+ const fetchedAt = new Date().toISOString();
172
+ const cachePath = abiPath(chainId, address);
173
+ const payload = { abi, contractName, source, fetchedAt, verified };
174
+ writeJson(cachePath, payload);
175
+ const index = readJson<Record<string, IndexEntry>>(indexPath(chainId), {});
176
+ index[address] = {
177
+ address,
178
+ contractName,
179
+ source,
180
+ abiHash: crypto.createHash('sha256').update(JSON.stringify(abi)).digest('hex'),
181
+ updatedAt: fetchedAt
182
+ };
183
+ writeJson(indexPath(chainId), index);
184
+
185
+ const result: AbiResolveResult = { abi, contractName, source, fetchedAt, cachePath, verified, cached: false };
186
+ this.logResolve(chainId, address, result);
187
+ return result;
188
+ }
189
+
190
+ private logResolve(chainId: number, address: string, result: AbiResolveResult) {
191
+ const functionsCount = result.abi.filter(
192
+ (entry) => entry && typeof entry === 'object' && (entry as { type?: string }).type === 'function'
193
+ ).length;
194
+ const line = {
195
+ event: 'ABI_RESOLVED',
196
+ chainId,
197
+ address,
198
+ contractName: result.contractName,
199
+ source: result.source,
200
+ cached: result.cached,
201
+ functionsCount
202
+ };
203
+ console.info(JSON.stringify(line));
204
+ runtimeTelemetry.recordAbiResolve({ ...line, verified: result.verified, fetchedAt: result.fetchedAt });
205
+ }
206
+ }
207
+
208
+ export const abiResolver = new AbiResolver();
@@ -0,0 +1,6 @@
1
+ import axios from 'axios';
2
+
3
+ export const http = axios.create({
4
+ baseURL: process.env.ETHERSCAN_BASE_URL?.trim() || 'https://api.etherscan.io/api',
5
+ timeout: 10_000
6
+ });
@@ -0,0 +1,150 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { FileBackend } from '../../src/core/backends/fileBackend';
4
+ import { SystemBackend } from '../../src/core/backends/systemBackend';
5
+ import { runtimeTelemetry, type SecretsTelemetryEntry } from '../services/runtimeTelemetryService';
6
+
7
+ type SourceName = 'env' | 'project-env' | 'encrypted-file' | 'keyring' | 'missing';
8
+
9
+ type ResolveOpts = { required?: boolean; failClosed?: boolean };
10
+
11
+ const projectEnvPath = path.join(process.cwd(), '.env');
12
+
13
+ const parseDotEnv = (raw: string): Record<string, string> => {
14
+ const result: Record<string, string> = {};
15
+ for (const line of raw.split(/\r?\n/)) {
16
+ const trimmed = line.trim();
17
+ if (!trimmed || trimmed.startsWith('#')) continue;
18
+ const idx = trimmed.indexOf('=');
19
+ if (idx <= 0) continue;
20
+ const key = trimmed.slice(0, idx).trim();
21
+ const value = trimmed.slice(idx + 1).trim().replace(/^"|"$/g, '');
22
+ result[key] = value;
23
+ }
24
+ return result;
25
+ };
26
+
27
+ const redact = (value?: string | null) => {
28
+ if (!value) return '****';
29
+ if (value.length <= 4) return '****';
30
+ return `****${value.slice(-4)}`;
31
+ };
32
+
33
+ class SecretsResolver {
34
+ private projectEnv: Record<string, string> = {};
35
+
36
+ private encryptedFileSecrets: Record<string, string> = {};
37
+
38
+ private keyringSecrets: Record<string, string> = {};
39
+
40
+ private initialized = false;
41
+
42
+ private diagnostics = new Map<string, SecretsTelemetryEntry>();
43
+
44
+ private async loadProjectEnv() {
45
+ if (!fs.existsSync(projectEnvPath)) return;
46
+ this.projectEnv = parseDotEnv(fs.readFileSync(projectEnvPath, 'utf8'));
47
+ }
48
+
49
+ private async loadEncryptedFile() {
50
+ try {
51
+ const backend = new FileBackend();
52
+ await backend.initialize();
53
+ this.encryptedFileSecrets = await backend.list();
54
+ } catch (error) {
55
+ console.warn('Encrypted local secrets file unavailable.', error);
56
+ }
57
+ }
58
+
59
+ private hasGuiSession() {
60
+ return Boolean(process.env.DISPLAY && process.env.DBUS_SESSION_BUS_ADDRESS);
61
+ }
62
+
63
+ private async loadSystemKeyring() {
64
+ if (!this.hasGuiSession()) {
65
+ return;
66
+ }
67
+ try {
68
+ const backend = new SystemBackend();
69
+ await backend.initialize();
70
+ this.keyringSecrets = await backend.list();
71
+ } catch (error) {
72
+ console.warn('OS keyring unavailable.', error);
73
+ }
74
+ }
75
+
76
+ async initialize() {
77
+ if (this.initialized) return;
78
+ await this.loadProjectEnv();
79
+ await this.loadEncryptedFile();
80
+ await this.loadSystemKeyring();
81
+ this.initialized = true;
82
+ }
83
+
84
+ private setDiagnostic(key: string, source: SourceName, required: boolean, value?: string | null) {
85
+ const entry: SecretsTelemetryEntry = {
86
+ key,
87
+ required,
88
+ present: Boolean(value && value.trim()),
89
+ source,
90
+ redacted: redact(value)
91
+ };
92
+ this.diagnostics.set(key, entry);
93
+ runtimeTelemetry.setSecretsStatus(Array.from(this.diagnostics.values()));
94
+ }
95
+
96
+ async resolve(key: string, opts: ResolveOpts = {}): Promise<string | null> {
97
+ await this.initialize();
98
+ const required = Boolean(opts.required);
99
+ const envValue = process.env[key]?.trim();
100
+ if (envValue) {
101
+ this.setDiagnostic(key, 'env', required, envValue);
102
+ return envValue;
103
+ }
104
+
105
+ const projectValue = this.projectEnv[key]?.trim();
106
+ if (projectValue) {
107
+ this.setDiagnostic(key, 'project-env', required, projectValue);
108
+ return projectValue;
109
+ }
110
+
111
+ const fileValue = this.encryptedFileSecrets[key]?.trim();
112
+ if (fileValue) {
113
+ this.setDiagnostic(key, 'encrypted-file', required, fileValue);
114
+ return fileValue;
115
+ }
116
+
117
+ const keyringValue = this.keyringSecrets[key]?.trim();
118
+ if (keyringValue) {
119
+ this.setDiagnostic(key, 'keyring', required, keyringValue);
120
+ return keyringValue;
121
+ }
122
+
123
+ const checked = ['env', 'project-env', 'encrypted-file', this.hasGuiSession() ? 'keyring' : 'keyring-skipped'];
124
+ const payload = { event: 'MISSING_SECRET', key, checked };
125
+ console.error(JSON.stringify(payload));
126
+ this.setDiagnostic(key, 'missing', required, null);
127
+
128
+ if (required && opts.failClosed !== false) {
129
+ throw new Error(`Missing required secret: ${key}`);
130
+ }
131
+
132
+ return null;
133
+ }
134
+
135
+ getDiagnostics() {
136
+ return Array.from(this.diagnostics.values());
137
+ }
138
+
139
+ logBootSummary(requiredKeys: string[] = []) {
140
+ const summary = requiredKeys.map((key) => {
141
+ const existing = this.diagnostics.get(key);
142
+ return existing ?? { key, required: true, present: false, source: 'missing', redacted: '****' };
143
+ });
144
+ console.info(
145
+ JSON.stringify({ event: 'SECRETS_BOOT_STATUS', secrets: summary.map((entry) => ({ ...entry, value: entry.present ? entry.redacted : '****' })) })
146
+ );
147
+ }
148
+ }
149
+
150
+ export const secretsResolver = new SecretsResolver();
@@ -0,0 +1,11 @@
1
+ import { JsonRpcProvider, Wallet } from 'ethers';
2
+ import { requireRpcUrl } from '../services/rpcService';
3
+
4
+ export const createBackendSigner = async (preferredRpcUrl?: string) => {
5
+ const privateKey = process.env.PRIVATE_KEY?.trim();
6
+ if (!privateKey) {
7
+ throw new Error('PRIVATE_KEY is missing. Configure PRIVATE_KEY in the backend environment.');
8
+ }
9
+ const rpcUrl = await requireRpcUrl(preferredRpcUrl ?? process.env.RPC_URL);
10
+ return new Wallet(privateKey, new JsonRpcProvider(rpcUrl));
11
+ };
@@ -0,0 +1,76 @@
1
+ import { parentPort, workerData } from 'worker_threads';
2
+ import { ethers } from 'ethers';
3
+
4
+ interface VanityWorkerData {
5
+ prefix?: string;
6
+ suffix?: string;
7
+ regex?: string;
8
+ maxAttempts: number;
9
+ derivationPath: string;
10
+ progressInterval?: number;
11
+ }
12
+
13
+ const { prefix, suffix, regex: regexPattern, maxAttempts, derivationPath, progressInterval } =
14
+ workerData as VanityWorkerData;
15
+ const normalizedPrefix = prefix?.replace(/^0x/i, '').toLowerCase();
16
+ const normalizedSuffix = suffix?.replace(/^0x/i, '').toLowerCase();
17
+ const regex = regexPattern ? new RegExp(regexPattern, 'i') : undefined;
18
+
19
+ let cancelled = false;
20
+ let attempts = 0;
21
+ const startedAt = Date.now();
22
+ const attemptModulo = Math.max(100, progressInterval ?? 5000);
23
+ let lastReport = startedAt;
24
+
25
+ parentPort?.on('message', (message: { type: 'cancel' }) => {
26
+ if (message.type === 'cancel') {
27
+ cancelled = true;
28
+ }
29
+ });
30
+
31
+ const reportProgress = () => {
32
+ parentPort?.postMessage({
33
+ type: 'progress',
34
+ attempts,
35
+ elapsedMs: Date.now() - startedAt
36
+ });
37
+ };
38
+
39
+ const matchesPattern = (address: string) => {
40
+ const normalized = address.toLowerCase().replace(/^0x/, '');
41
+ const prefixMatch = normalizedPrefix ? normalized.startsWith(normalizedPrefix) : true;
42
+ const suffixMatch = normalizedSuffix ? normalized.endsWith(normalizedSuffix) : true;
43
+ const regexMatch = regex ? regex.test(address) : true;
44
+ return prefixMatch && suffixMatch && regexMatch;
45
+ };
46
+
47
+ (async () => {
48
+ while (!cancelled && (maxAttempts <= 0 || attempts < maxAttempts)) {
49
+ const randomWallet = ethers.Wallet.createRandom();
50
+ const mnemonic = randomWallet.mnemonic;
51
+ const wallet = mnemonic ? ethers.HDNodeWallet.fromMnemonic(mnemonic, derivationPath) : randomWallet;
52
+ attempts += 1;
53
+ if (matchesPattern(wallet.address)) {
54
+ parentPort?.postMessage({
55
+ type: 'result',
56
+ address: wallet.address,
57
+ mnemonic: wallet.mnemonic?.phrase,
58
+ derivationPath: wallet.path,
59
+ attempts,
60
+ elapsedMs: Date.now() - startedAt
61
+ });
62
+ return;
63
+ }
64
+ if (attempts % attemptModulo === 0 || Date.now() - lastReport >= 2000) {
65
+ reportProgress();
66
+ lastReport = Date.now();
67
+ }
68
+ }
69
+ parentPort?.postMessage({
70
+ type: 'complete',
71
+ attempts,
72
+ elapsedMs: Date.now() - startedAt,
73
+ cancelled,
74
+ success: false
75
+ });
76
+ })();
@@ -0,0 +1,13 @@
1
+ import type { CapacitorConfig } from '@capacitor/cli';
2
+
3
+ const config: CapacitorConfig = {
4
+ appId: 'com.gnoman.app',
5
+ appName: 'GNOMAN',
6
+ webDir: 'renderer/dist',
7
+ bundledWebRuntime: false,
8
+ server: {
9
+ androidScheme: 'https'
10
+ }
11
+ };
12
+
13
+ export default config;