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.
- package/.eslintrc.cjs +24 -0
- package/.gnoman/contracts.json +4094 -0
- package/.gnoman/exec_package/runtime-debug.jsonl +45 -0
- package/.gnoman/holds.sqlite +0 -0
- package/.gnoman/license.json +7 -0
- package/.gnoman/safes.json +37 -0
- package/.gnoman/vanity-jobs.json +67 -0
- package/.gnoman/wallets.db +0 -0
- package/.prettierrc.json +6 -0
- package/CODex_TASKS.md +124 -0
- package/LICENSE.md +164 -0
- package/README.md +95 -0
- package/assets/GnoLogo.jpg +0 -0
- package/assets/self.png +0 -0
- package/backend/controllers/contractController.ts +49 -0
- package/backend/controllers/devToolsController.ts +76 -0
- package/backend/controllers/etherscanController.ts +59 -0
- package/backend/controllers/historyController.ts +7 -0
- package/backend/controllers/keyringController.ts +134 -0
- package/backend/controllers/robinhoodController.ts +80 -0
- package/backend/controllers/safeController.ts +167 -0
- package/backend/controllers/sandboxController.ts +63 -0
- package/backend/controllers/settingsController.ts +38 -0
- package/backend/controllers/walletController.ts +151 -0
- package/backend/index.ts +133 -0
- package/backend/licenses/license_public.pem +3 -0
- package/backend/licenses/verify_license.py +43 -0
- package/backend/routes/contractRoutes.ts +11 -0
- package/backend/routes/devToolsRoutes.ts +11 -0
- package/backend/routes/etherscanRoutes.ts +11 -0
- package/backend/routes/historyRoutes.ts +8 -0
- package/backend/routes/keyringRoutes.ts +25 -0
- package/backend/routes/licenseRoutes.ts +35 -0
- package/backend/routes/robinhoodRoutes.ts +22 -0
- package/backend/routes/runtimeRoutes.ts +29 -0
- package/backend/routes/safeRoutes.ts +28 -0
- package/backend/routes/sandboxRoutes.ts +17 -0
- package/backend/routes/settingsRoutes.ts +14 -0
- package/backend/routes/walletRoutes.ts +21 -0
- package/backend/services/chainlinkService.ts +65 -0
- package/backend/services/contractRegistryService.ts +205 -0
- package/backend/services/devToolsService.ts +251 -0
- package/backend/services/diagnosticsService.ts +350 -0
- package/backend/services/etherscanService.ts +152 -0
- package/backend/services/historyService.ts +89 -0
- package/backend/services/keyringAccessor.ts +4 -0
- package/backend/services/licenseService.ts +163 -0
- package/backend/services/onchain/abiRegistry.ts +57 -0
- package/backend/services/onchain/chainlinkClient.ts +56 -0
- package/backend/services/onchain/errors.ts +16 -0
- package/backend/services/onchain/etherscanClient.ts +94 -0
- package/backend/services/onchain/index.ts +76 -0
- package/backend/services/onchain/tenderlyRpcClient.ts +74 -0
- package/backend/services/onchain/types.ts +33 -0
- package/backend/services/onchainAutomationService.ts +424 -0
- package/backend/services/robinhood/auth.ts +42 -0
- package/backend/services/robinhood/client.ts +123 -0
- package/backend/services/robinhood/integrationService.ts +140 -0
- package/backend/services/robinhood/provider.ts +22 -0
- package/backend/services/robinhood/unofficialClient.ts +66 -0
- package/backend/services/rpcService.ts +44 -0
- package/backend/services/runtimeTelemetryService.ts +158 -0
- package/backend/services/safeConfigRepository.ts +205 -0
- package/backend/services/safeService.ts +588 -0
- package/backend/services/sandboxService.ts +157 -0
- package/backend/services/secureSettingsService.ts +45 -0
- package/backend/services/transactionHoldService.ts +223 -0
- package/backend/services/vanityService.ts +293 -0
- package/backend/services/walletService.ts +290 -0
- package/backend/services/walletStore.ts +179 -0
- package/backend/types/express-async-handler.d.ts +13 -0
- package/backend/types/keyring.d.ts +19 -0
- package/backend/utils/abiResolver.ts +208 -0
- package/backend/utils/http.ts +6 -0
- package/backend/utils/secretsResolver.ts +150 -0
- package/backend/utils/signer.ts +11 -0
- package/backend/workers/vanityWorker.ts +76 -0
- package/capacitor.config.ts +13 -0
- package/cli/gnoman.ts +424 -0
- package/contracts/OracleConsumer.sol +20 -0
- package/contracts/PriceFeedConsumer.sol +22 -0
- package/dist/backend/backend/controllers/contractController.js +41 -0
- package/dist/backend/backend/controllers/contractController.js.map +1 -0
- package/dist/backend/backend/controllers/devToolsController.js +63 -0
- package/dist/backend/backend/controllers/devToolsController.js.map +1 -0
- package/dist/backend/backend/controllers/etherscanController.js +53 -0
- package/dist/backend/backend/controllers/etherscanController.js.map +1 -0
- package/dist/backend/backend/controllers/historyController.js +12 -0
- package/dist/backend/backend/controllers/historyController.js.map +1 -0
- package/dist/backend/backend/controllers/keyringController.js +126 -0
- package/dist/backend/backend/controllers/keyringController.js.map +1 -0
- package/dist/backend/backend/controllers/robinhoodController.js +69 -0
- package/dist/backend/backend/controllers/robinhoodController.js.map +1 -0
- package/dist/backend/backend/controllers/safeController.js +137 -0
- package/dist/backend/backend/controllers/safeController.js.map +1 -0
- package/dist/backend/backend/controllers/sandboxController.js +48 -0
- package/dist/backend/backend/controllers/sandboxController.js.map +1 -0
- package/dist/backend/backend/controllers/settingsController.js +34 -0
- package/dist/backend/backend/controllers/settingsController.js.map +1 -0
- package/dist/backend/backend/controllers/walletController.js +140 -0
- package/dist/backend/backend/controllers/walletController.js.map +1 -0
- package/dist/backend/backend/index.js +119 -0
- package/dist/backend/backend/index.js.map +1 -0
- package/dist/backend/backend/routes/contractRoutes.js +44 -0
- package/dist/backend/backend/routes/contractRoutes.js.map +1 -0
- package/dist/backend/backend/routes/devToolsRoutes.js +44 -0
- package/dist/backend/backend/routes/devToolsRoutes.js.map +1 -0
- package/dist/backend/backend/routes/etherscanRoutes.js +44 -0
- package/dist/backend/backend/routes/etherscanRoutes.js.map +1 -0
- package/dist/backend/backend/routes/historyRoutes.js +41 -0
- package/dist/backend/backend/routes/historyRoutes.js.map +1 -0
- package/dist/backend/backend/routes/keyringRoutes.js +18 -0
- package/dist/backend/backend/routes/keyringRoutes.js.map +1 -0
- package/dist/backend/backend/routes/licenseRoutes.js +30 -0
- package/dist/backend/backend/routes/licenseRoutes.js.map +1 -0
- package/dist/backend/backend/routes/robinhoodRoutes.js +14 -0
- package/dist/backend/backend/routes/robinhoodRoutes.js.map +1 -0
- package/dist/backend/backend/routes/runtimeRoutes.js +26 -0
- package/dist/backend/backend/routes/runtimeRoutes.js.map +1 -0
- package/dist/backend/backend/routes/safeRoutes.js +61 -0
- package/dist/backend/backend/routes/safeRoutes.js.map +1 -0
- package/dist/backend/backend/routes/sandboxRoutes.js +50 -0
- package/dist/backend/backend/routes/sandboxRoutes.js.map +1 -0
- package/dist/backend/backend/routes/settingsRoutes.js +10 -0
- package/dist/backend/backend/routes/settingsRoutes.js.map +1 -0
- package/dist/backend/backend/routes/walletRoutes.js +54 -0
- package/dist/backend/backend/routes/walletRoutes.js.map +1 -0
- package/dist/backend/backend/services/chainlinkService.js +48 -0
- package/dist/backend/backend/services/chainlinkService.js.map +1 -0
- package/dist/backend/backend/services/contractRegistryService.js +138 -0
- package/dist/backend/backend/services/contractRegistryService.js.map +1 -0
- package/dist/backend/backend/services/devToolsService.js +213 -0
- package/dist/backend/backend/services/devToolsService.js.map +1 -0
- package/dist/backend/backend/services/diagnosticsService.js +286 -0
- package/dist/backend/backend/services/diagnosticsService.js.map +1 -0
- package/dist/backend/backend/services/etherscanService.js +125 -0
- package/dist/backend/backend/services/etherscanService.js.map +1 -0
- package/dist/backend/backend/services/historyService.js +75 -0
- package/dist/backend/backend/services/historyService.js.map +1 -0
- package/dist/backend/backend/services/keyringAccessor.js +40 -0
- package/dist/backend/backend/services/keyringAccessor.js.map +1 -0
- package/dist/backend/backend/services/licenseService.js +130 -0
- package/dist/backend/backend/services/licenseService.js.map +1 -0
- package/dist/backend/backend/services/onchain/abiRegistry.js +47 -0
- package/dist/backend/backend/services/onchain/abiRegistry.js.map +1 -0
- package/dist/backend/backend/services/onchain/chainlinkClient.js +43 -0
- package/dist/backend/backend/services/onchain/chainlinkClient.js.map +1 -0
- package/dist/backend/backend/services/onchain/errors.js +13 -0
- package/dist/backend/backend/services/onchain/errors.js.map +1 -0
- package/dist/backend/backend/services/onchain/etherscanClient.js +82 -0
- package/dist/backend/backend/services/onchain/etherscanClient.js.map +1 -0
- package/dist/backend/backend/services/onchain/index.js +79 -0
- package/dist/backend/backend/services/onchain/index.js.map +1 -0
- package/dist/backend/backend/services/onchain/tenderlyRpcClient.js +60 -0
- package/dist/backend/backend/services/onchain/tenderlyRpcClient.js.map +1 -0
- package/dist/backend/backend/services/onchain/types.js +14 -0
- package/dist/backend/backend/services/onchain/types.js.map +1 -0
- package/dist/backend/backend/services/onchainAutomationService.js +316 -0
- package/dist/backend/backend/services/onchainAutomationService.js.map +1 -0
- package/dist/backend/backend/services/robinhood/auth.js +26 -0
- package/dist/backend/backend/services/robinhood/auth.js.map +1 -0
- package/dist/backend/backend/services/robinhood/client.js +73 -0
- package/dist/backend/backend/services/robinhood/client.js.map +1 -0
- package/dist/backend/backend/services/robinhood/integrationService.js +119 -0
- package/dist/backend/backend/services/robinhood/integrationService.js.map +1 -0
- package/dist/backend/backend/services/robinhood/provider.js +17 -0
- package/dist/backend/backend/services/robinhood/provider.js.map +1 -0
- package/dist/backend/backend/services/robinhood/unofficialClient.js +61 -0
- package/dist/backend/backend/services/robinhood/unofficialClient.js.map +1 -0
- package/dist/backend/backend/services/rpcService.js +48 -0
- package/dist/backend/backend/services/rpcService.js.map +1 -0
- package/dist/backend/backend/services/runtimeTelemetryService.js +96 -0
- package/dist/backend/backend/services/runtimeTelemetryService.js.map +1 -0
- package/dist/backend/backend/services/safeConfigRepository.js +147 -0
- package/dist/backend/backend/services/safeConfigRepository.js.map +1 -0
- package/dist/backend/backend/services/safeService.js +527 -0
- package/dist/backend/backend/services/safeService.js.map +1 -0
- package/dist/backend/backend/services/sandboxService.js +135 -0
- package/dist/backend/backend/services/sandboxService.js.map +1 -0
- package/dist/backend/backend/services/secureSettingsService.js +50 -0
- package/dist/backend/backend/services/secureSettingsService.js.map +1 -0
- package/dist/backend/backend/services/transactionHoldService.js +184 -0
- package/dist/backend/backend/services/transactionHoldService.js.map +1 -0
- package/dist/backend/backend/services/vanityService.js +235 -0
- package/dist/backend/backend/services/vanityService.js.map +1 -0
- package/dist/backend/backend/services/walletService.js +202 -0
- package/dist/backend/backend/services/walletService.js.map +1 -0
- package/dist/backend/backend/services/walletStore.js +132 -0
- package/dist/backend/backend/services/walletStore.js.map +1 -0
- package/dist/backend/backend/utils/abiResolver.js +182 -0
- package/dist/backend/backend/utils/abiResolver.js.map +1 -0
- package/dist/backend/backend/utils/http.js +12 -0
- package/dist/backend/backend/utils/http.js.map +1 -0
- package/dist/backend/backend/utils/secretsResolver.js +137 -0
- package/dist/backend/backend/utils/secretsResolver.js.map +1 -0
- package/dist/backend/backend/utils/signer.js +15 -0
- package/dist/backend/backend/utils/signer.js.map +1 -0
- package/dist/backend/backend/workers/vanityWorker.js +63 -0
- package/dist/backend/backend/workers/vanityWorker.js.map +1 -0
- package/dist/backend/cli/gnoman.js +387 -0
- package/dist/backend/cli/gnoman.js.map +1 -0
- package/dist/backend/modules/sandbox/abiLoader.js +78 -0
- package/dist/backend/modules/sandbox/abiLoader.js.map +1 -0
- package/dist/backend/modules/sandbox/contractSimulator.js +205 -0
- package/dist/backend/modules/sandbox/contractSimulator.js.map +1 -0
- package/dist/backend/modules/sandbox/formBuilder.js +14 -0
- package/dist/backend/modules/sandbox/formBuilder.js.map +1 -0
- package/dist/backend/modules/sandbox/index.js +24 -0
- package/dist/backend/modules/sandbox/index.js.map +1 -0
- package/dist/backend/modules/sandbox/localFork.js +103 -0
- package/dist/backend/modules/sandbox/localFork.js.map +1 -0
- package/dist/backend/modules/sandbox/sandboxManager.js +130 -0
- package/dist/backend/modules/sandbox/sandboxManager.js.map +1 -0
- package/dist/backend/modules/sandbox/types.js +3 -0
- package/dist/backend/modules/sandbox/types.js.map +1 -0
- package/dist/backend/src/core/backends/fileBackend.js +136 -0
- package/dist/backend/src/core/backends/fileBackend.js.map +1 -0
- package/dist/backend/src/core/backends/memoryBackend.js +26 -0
- package/dist/backend/src/core/backends/memoryBackend.js.map +1 -0
- package/dist/backend/src/core/backends/systemBackend.js +86 -0
- package/dist/backend/src/core/backends/systemBackend.js.map +1 -0
- package/dist/backend/src/core/backends/types.js +12 -0
- package/dist/backend/src/core/backends/types.js.map +1 -0
- package/dist/backend/src/core/keyringManager.js +178 -0
- package/dist/backend/src/core/keyringManager.js.map +1 -0
- package/dist/backend/src/utils/abiResolver.js +180 -0
- package/dist/backend/src/utils/abiResolver.js.map +1 -0
- package/dist/backend/src/utils/runtimeObservability.js +78 -0
- package/dist/backend/src/utils/runtimeObservability.js.map +1 -0
- package/dist/backend/src/utils/secretsResolver.js +138 -0
- package/dist/backend/src/utils/secretsResolver.js.map +1 -0
- package/dist/cli/backend/services/diagnosticsService.js +286 -0
- package/dist/cli/backend/services/diagnosticsService.js.map +1 -0
- package/dist/cli/backend/services/keyringAccessor.js +40 -0
- package/dist/cli/backend/services/keyringAccessor.js.map +1 -0
- package/dist/cli/backend/services/rpcService.js +48 -0
- package/dist/cli/backend/services/rpcService.js.map +1 -0
- package/dist/cli/backend/services/runtimeTelemetryService.js +96 -0
- package/dist/cli/backend/services/runtimeTelemetryService.js.map +1 -0
- package/dist/cli/backend/services/walletService.js +202 -0
- package/dist/cli/backend/services/walletService.js.map +1 -0
- package/dist/cli/backend/services/walletStore.js +132 -0
- package/dist/cli/backend/services/walletStore.js.map +1 -0
- package/dist/cli/backend/utils/http.js +12 -0
- package/dist/cli/backend/utils/http.js.map +1 -0
- package/dist/cli/backend/utils/secretsResolver.js +137 -0
- package/dist/cli/backend/utils/secretsResolver.js.map +1 -0
- package/dist/cli/cli/gnoman.js +387 -0
- package/dist/cli/cli/gnoman.js.map +1 -0
- package/dist/cli/src/core/backends/fileBackend.js +136 -0
- package/dist/cli/src/core/backends/fileBackend.js.map +1 -0
- package/dist/cli/src/core/backends/memoryBackend.js +26 -0
- package/dist/cli/src/core/backends/memoryBackend.js.map +1 -0
- package/dist/cli/src/core/backends/systemBackend.js +86 -0
- package/dist/cli/src/core/backends/systemBackend.js.map +1 -0
- package/dist/cli/src/core/backends/types.js +12 -0
- package/dist/cli/src/core/backends/types.js.map +1 -0
- package/dist/cli/src/core/keyringManager.js +178 -0
- package/dist/cli/src/core/keyringManager.js.map +1 -0
- package/dist/cli/src/utils/abiResolver.js +180 -0
- package/dist/cli/src/utils/abiResolver.js.map +1 -0
- package/dist/cli/src/utils/runtimeObservability.js +78 -0
- package/dist/cli/src/utils/runtimeObservability.js.map +1 -0
- package/dist/cli/src/utils/secretsResolver.js +138 -0
- package/dist/cli/src/utils/secretsResolver.js.map +1 -0
- package/dist/main/backend/services/keyringAccessor.js +40 -0
- package/dist/main/backend/services/keyringAccessor.js.map +1 -0
- package/dist/main/backend/utils/http.js +12 -0
- package/dist/main/backend/utils/http.js.map +1 -0
- package/dist/main/main/ipcHandlers/index.js +26 -0
- package/dist/main/main/ipcHandlers/index.js.map +1 -0
- package/dist/main/main/keyring/keyringmanager.js +101 -0
- package/dist/main/main/keyring/keyringmanager.js.map +1 -0
- package/dist/main/main/main.js +224 -0
- package/dist/main/main/main.js.map +1 -0
- package/dist/main/main/preload/index.js +19 -0
- package/dist/main/main/preload/index.js.map +1 -0
- package/dist/main/main/preload/licenseBridge.js +105 -0
- package/dist/main/main/preload/licenseBridge.js.map +1 -0
- package/dist/main/src/core/backends/fileBackend.js +136 -0
- package/dist/main/src/core/backends/fileBackend.js.map +1 -0
- package/dist/main/src/core/backends/memoryBackend.js +26 -0
- package/dist/main/src/core/backends/memoryBackend.js.map +1 -0
- package/dist/main/src/core/backends/systemBackend.js +86 -0
- package/dist/main/src/core/backends/systemBackend.js.map +1 -0
- package/dist/main/src/core/backends/types.js +12 -0
- package/dist/main/src/core/backends/types.js.map +1 -0
- package/dist/main/src/core/keyringManager.js +178 -0
- package/dist/main/src/core/keyringManager.js.map +1 -0
- package/dist/main/src/utils/abiResolver.js +180 -0
- package/dist/main/src/utils/abiResolver.js.map +1 -0
- package/dist/main/src/utils/runtimeObservability.js +78 -0
- package/dist/main/src/utils/runtimeObservability.js.map +1 -0
- package/dist/main/src/utils/secretsResolver.js +138 -0
- package/dist/main/src/utils/secretsResolver.js.map +1 -0
- package/docs/development-guide.md +203 -0
- package/docs/etherscan-chainlink-integration.md +44 -0
- package/docs/gnoman-20-user-manual-STANDARD-PRINT-READY.pdf +0 -0
- package/docs/gnoman-20-user-manual-STANDARD.pdf +0 -0
- package/docs/license-dev-guide.md +106 -0
- package/docs/robinhood-integration.md +30 -0
- package/docs/system-audit-gpt-guide.md +208 -0
- package/docs/system-robustness-audit.md +50 -0
- package/docs/user-guide.md +73 -0
- package/docs/wiki/development-guide.md +203 -0
- package/docs/wiki/license-dev-guide.md +106 -0
- package/docs/wiki/user-guide.md +73 -0
- package/eslint.config.js +85 -0
- package/gnoman2.0/.eslintrc.cjs +24 -0
- package/gnoman2.0/.prettierrc.json +6 -0
- package/gnoman2.0/CODex_TASKS.md +124 -0
- package/gnoman2.0/LICENSE.md +164 -0
- package/gnoman2.0/README.md +95 -0
- package/gnoman2.0/assets/GnoLogo.jpg +0 -0
- package/gnoman2.0/assets/self.png +0 -0
- package/gnoman2.0/backend/controllers/contractController.ts +49 -0
- package/gnoman2.0/backend/controllers/devToolsController.ts +76 -0
- package/gnoman2.0/backend/controllers/etherscanController.ts +59 -0
- package/gnoman2.0/backend/controllers/historyController.ts +7 -0
- package/gnoman2.0/backend/controllers/keyringController.ts +134 -0
- package/gnoman2.0/backend/controllers/robinhoodController.ts +80 -0
- package/gnoman2.0/backend/controllers/safeController.ts +167 -0
- package/gnoman2.0/backend/controllers/sandboxController.ts +63 -0
- package/gnoman2.0/backend/controllers/settingsController.ts +38 -0
- package/gnoman2.0/backend/controllers/walletController.ts +151 -0
- package/gnoman2.0/backend/index.ts +133 -0
- package/gnoman2.0/backend/licenses/license_public.pem +3 -0
- package/gnoman2.0/backend/licenses/verify_license.py +43 -0
- package/gnoman2.0/backend/routes/contractRoutes.ts +11 -0
- package/gnoman2.0/backend/routes/devToolsRoutes.ts +11 -0
- package/gnoman2.0/backend/routes/etherscanRoutes.ts +11 -0
- package/gnoman2.0/backend/routes/historyRoutes.ts +8 -0
- package/gnoman2.0/backend/routes/keyringRoutes.ts +25 -0
- package/gnoman2.0/backend/routes/licenseRoutes.ts +35 -0
- package/gnoman2.0/backend/routes/robinhoodRoutes.ts +22 -0
- package/gnoman2.0/backend/routes/runtimeRoutes.ts +29 -0
- package/gnoman2.0/backend/routes/safeRoutes.ts +28 -0
- package/gnoman2.0/backend/routes/sandboxRoutes.ts +17 -0
- package/gnoman2.0/backend/routes/settingsRoutes.ts +14 -0
- package/gnoman2.0/backend/routes/walletRoutes.ts +21 -0
- package/gnoman2.0/backend/services/chainlinkService.ts +65 -0
- package/gnoman2.0/backend/services/contractRegistryService.ts +205 -0
- package/gnoman2.0/backend/services/devToolsService.ts +251 -0
- package/gnoman2.0/backend/services/diagnosticsService.ts +350 -0
- package/gnoman2.0/backend/services/etherscanService.ts +152 -0
- package/gnoman2.0/backend/services/historyService.ts +89 -0
- package/gnoman2.0/backend/services/keyringAccessor.ts +4 -0
- package/gnoman2.0/backend/services/licenseService.ts +163 -0
- package/gnoman2.0/backend/services/onchain/abiRegistry.ts +57 -0
- package/gnoman2.0/backend/services/onchain/chainlinkClient.ts +56 -0
- package/gnoman2.0/backend/services/onchain/errors.ts +16 -0
- package/gnoman2.0/backend/services/onchain/etherscanClient.ts +94 -0
- package/gnoman2.0/backend/services/onchain/index.ts +76 -0
- package/gnoman2.0/backend/services/onchain/tenderlyRpcClient.ts +74 -0
- package/gnoman2.0/backend/services/onchain/types.ts +33 -0
- package/gnoman2.0/backend/services/onchainAutomationService.ts +424 -0
- package/gnoman2.0/backend/services/robinhood/auth.ts +42 -0
- package/gnoman2.0/backend/services/robinhood/client.ts +123 -0
- package/gnoman2.0/backend/services/robinhood/integrationService.ts +140 -0
- package/gnoman2.0/backend/services/robinhood/provider.ts +22 -0
- package/gnoman2.0/backend/services/robinhood/unofficialClient.ts +66 -0
- package/gnoman2.0/backend/services/rpcService.ts +44 -0
- package/gnoman2.0/backend/services/runtimeTelemetryService.ts +158 -0
- package/gnoman2.0/backend/services/safeConfigRepository.ts +205 -0
- package/gnoman2.0/backend/services/safeService.ts +588 -0
- package/gnoman2.0/backend/services/sandboxService.ts +157 -0
- package/gnoman2.0/backend/services/secureSettingsService.ts +45 -0
- package/gnoman2.0/backend/services/transactionHoldService.ts +223 -0
- package/gnoman2.0/backend/services/vanityService.ts +293 -0
- package/gnoman2.0/backend/services/walletService.ts +290 -0
- package/gnoman2.0/backend/services/walletStore.ts +179 -0
- package/gnoman2.0/backend/types/express-async-handler.d.ts +13 -0
- package/gnoman2.0/backend/types/keyring.d.ts +19 -0
- package/gnoman2.0/backend/utils/abiResolver.ts +208 -0
- package/gnoman2.0/backend/utils/http.ts +6 -0
- package/gnoman2.0/backend/utils/secretsResolver.ts +150 -0
- package/gnoman2.0/backend/utils/signer.ts +11 -0
- package/gnoman2.0/backend/workers/vanityWorker.ts +76 -0
- package/gnoman2.0/capacitor.config.ts +13 -0
- package/gnoman2.0/cli/gnoman.ts +424 -0
- package/gnoman2.0/contracts/OracleConsumer.sol +20 -0
- package/gnoman2.0/contracts/PriceFeedConsumer.sol +22 -0
- package/gnoman2.0/docs/development-guide.md +203 -0
- package/gnoman2.0/docs/etherscan-chainlink-integration.md +44 -0
- package/gnoman2.0/docs/gnoman-20-user-manual-STANDARD-PRINT-READY.pdf +0 -0
- package/gnoman2.0/docs/gnoman-20-user-manual-STANDARD.pdf +0 -0
- package/gnoman2.0/docs/license-dev-guide.md +106 -0
- package/gnoman2.0/docs/robinhood-integration.md +30 -0
- package/gnoman2.0/docs/system-audit-gpt-guide.md +208 -0
- package/gnoman2.0/docs/system-robustness-audit.md +50 -0
- package/gnoman2.0/docs/user-guide.md +73 -0
- package/gnoman2.0/docs/wiki/development-guide.md +203 -0
- package/gnoman2.0/docs/wiki/license-dev-guide.md +106 -0
- package/gnoman2.0/docs/wiki/user-guide.md +73 -0
- package/gnoman2.0/eslint.config.js +85 -0
- package/gnoman2.0/gnomon/__init__.py +0 -0
- package/gnoman2.0/gnomon/api/__init__.py +0 -0
- package/gnoman2.0/gnomon/api/etherscan_tracker.py +72 -0
- package/gnoman2.0/gnomon/core/__init__.py +0 -0
- package/gnoman2.0/gnomon/core/safe_manager.py +111 -0
- package/gnoman2.0/gnomon/tests/test_abi_resolver.py +181 -0
- package/gnoman2.0/gnomon/tests/test_safe_persistence_and_etherscan.py +97 -0
- package/gnoman2.0/gnomon/utils/__init__.py +5 -0
- package/gnoman2.0/gnomon/utils/abi_resolver.py +255 -0
- package/gnoman2.0/ios/ExportOptions.plist +16 -0
- package/gnoman2.0/ios/README.md +33 -0
- package/gnoman2.0/jest.config.ts +18 -0
- package/gnoman2.0/keyring/__init__.py +17 -0
- package/gnoman2.0/licensingServer/package.json +23 -0
- package/gnoman2.0/licensingServer/src/config/keys.ts +84 -0
- package/gnoman2.0/licensingServer/src/index.ts +30 -0
- package/gnoman2.0/licensingServer/src/lib/canonicalize.ts +5 -0
- package/gnoman2.0/licensingServer/src/lib/crypto.ts +25 -0
- package/gnoman2.0/licensingServer/src/lib/validate.ts +62 -0
- package/gnoman2.0/licensingServer/src/middleware/auth.ts +20 -0
- package/gnoman2.0/licensingServer/src/routes/licenses.ts +110 -0
- package/gnoman2.0/licensingServer/tsconfig.json +12 -0
- package/gnoman2.0/main/ipcHandlers/index.ts +23 -0
- package/gnoman2.0/main/keyring/keyringmanager.ts +154 -0
- package/gnoman2.0/main/main.ts +234 -0
- package/gnoman2.0/main/preload/index.ts +31 -0
- package/gnoman2.0/main/preload/licenseBridge.ts +73 -0
- package/gnoman2.0/modules/sandbox/abiLoader.ts +78 -0
- package/gnoman2.0/modules/sandbox/contractSimulator.ts +241 -0
- package/gnoman2.0/modules/sandbox/formBuilder.ts +16 -0
- package/gnoman2.0/modules/sandbox/index.ts +6 -0
- package/gnoman2.0/modules/sandbox/localFork.ts +129 -0
- package/gnoman2.0/modules/sandbox/safe.abi.json +82 -0
- package/gnoman2.0/modules/sandbox/sandboxManager.ts +154 -0
- package/gnoman2.0/modules/sandbox/types.ts +84 -0
- package/gnoman2.0/modules/sandbox/ui/LogViewer.tsx +30 -0
- package/gnoman2.0/modules/sandbox/ui/ParameterForm.tsx +49 -0
- package/gnoman2.0/modules/sandbox/ui/SandboxPanel.tsx +568 -0
- package/gnoman2.0/package-lock.json +10904 -0
- package/gnoman2.0/package.json +82 -0
- package/gnoman2.0/renderer/components/LicenseScreen.tsx +134 -0
- package/gnoman2.0/renderer/index.html +12 -0
- package/gnoman2.0/renderer/package-lock.json +4104 -0
- package/gnoman2.0/renderer/package.json +35 -0
- package/gnoman2.0/renderer/postcss.config.cjs +6 -0
- package/gnoman2.0/renderer/src/App.tsx +229 -0
- package/gnoman2.0/renderer/src/context/KeyringContext.tsx +217 -0
- package/gnoman2.0/renderer/src/context/SafeContext.tsx +49 -0
- package/gnoman2.0/renderer/src/context/ThemeContext.tsx +60 -0
- package/gnoman2.0/renderer/src/context/WalletContext.tsx +50 -0
- package/gnoman2.0/renderer/src/context/main.tsx +18 -0
- package/gnoman2.0/renderer/src/main.tsx +18 -0
- package/gnoman2.0/renderer/src/pages/Contracts.tsx +482 -0
- package/gnoman2.0/renderer/src/pages/Dashboard.tsx +653 -0
- package/gnoman2.0/renderer/src/pages/DeveloperTools.tsx +270 -0
- package/gnoman2.0/renderer/src/pages/History.tsx +149 -0
- package/gnoman2.0/renderer/src/pages/Keyring.tsx +449 -0
- package/gnoman2.0/renderer/src/pages/Safes.tsx +1089 -0
- package/gnoman2.0/renderer/src/pages/Sandbox.tsx +146 -0
- package/gnoman2.0/renderer/src/pages/Settings.tsx +871 -0
- package/gnoman2.0/renderer/src/pages/Wallets.tsx +752 -0
- package/gnoman2.0/renderer/src/pages/WikiGuide.tsx +75 -0
- package/gnoman2.0/renderer/src/styles.css +32 -0
- package/gnoman2.0/renderer/src/types/gnoman.d.ts +9 -0
- package/gnoman2.0/renderer/src/types/license.ts +8 -0
- package/gnoman2.0/renderer/src/types/safevault.d.ts +17 -0
- package/gnoman2.0/renderer/src/utils/backend.ts +88 -0
- package/gnoman2.0/renderer/tailwind.config.cjs +8 -0
- package/gnoman2.0/renderer/tsconfig.json +13 -0
- package/gnoman2.0/renderer/tsconfig.node.json +9 -0
- package/gnoman2.0/renderer/vite.config.ts +19 -0
- package/gnoman2.0/requests/__init__.py +35 -0
- package/gnoman2.0/scripts/build-ios.sh +30 -0
- package/gnoman2.0/scripts/copyBackendAssets.js +24 -0
- package/gnoman2.0/scripts/copyRenderer.js +87 -0
- package/gnoman2.0/scripts/launchElectron.js +51 -0
- package/gnoman2.0/src/core/backends/fileBackend.ts +154 -0
- package/gnoman2.0/src/core/backends/memoryBackend.ts +27 -0
- package/gnoman2.0/src/core/backends/systemBackend.ts +66 -0
- package/gnoman2.0/src/core/backends/types.ts +17 -0
- package/gnoman2.0/src/core/keyringManager.ts +208 -0
- package/gnoman2.0/src/utils/abiCache/.gitkeep +0 -0
- package/gnoman2.0/src/utils/abiResolver.ts +200 -0
- package/gnoman2.0/src/utils/runtimeObservability.ts +110 -0
- package/gnoman2.0/src/utils/secretsResolver.ts +144 -0
- package/gnoman2.0/tests/chainlinkService.test.ts +32 -0
- package/gnoman2.0/tests/diagnosticsService.test.ts +68 -0
- package/gnoman2.0/tests/etherscanController.test.ts +99 -0
- package/gnoman2.0/tests/etherscanService.test.ts +116 -0
- package/gnoman2.0/tests/keyringManager.test.ts +135 -0
- package/gnoman2.0/tests/onchainToolkit.test.ts +71 -0
- package/gnoman2.0/tests/robinhoodClient.test.ts +54 -0
- package/gnoman2.0/tests/robinhoodController.test.ts +81 -0
- package/gnoman2.0/tests/robinhoodIntegrationService.test.ts +50 -0
- package/gnoman2.0/tests/safeServicePersistence.test.ts +81 -0
- package/gnoman2.0/tests/test_contract_sandbox/sandbox.test.js +407 -0
- package/gnoman2.0/tests/walletController.test.ts +57 -0
- package/gnoman2.0/tsconfig.backend.json +7 -0
- package/gnoman2.0/tsconfig.cli.json +7 -0
- package/gnoman2.0/tsconfig.json +18 -0
- package/gnoman2.0/tsconfig.main.json +7 -0
- package/gnomon/__init__.py +0 -0
- package/gnomon/__pycache__/__init__.cpython-310.pyc +0 -0
- package/gnomon/api/__init__.py +0 -0
- package/gnomon/api/__pycache__/__init__.cpython-310.pyc +0 -0
- package/gnomon/api/__pycache__/etherscan_tracker.cpython-310.pyc +0 -0
- package/gnomon/api/etherscan_tracker.py +72 -0
- package/gnomon/core/__init__.py +0 -0
- package/gnomon/core/safe_manager.py +111 -0
- package/gnomon/tests/__pycache__/test_safe_persistence_and_etherscan.cpython-310-pytest-8.3.3.pyc +0 -0
- package/gnomon/tests/test_abi_resolver.py +181 -0
- package/gnomon/tests/test_safe_persistence_and_etherscan.py +97 -0
- package/gnomon/utils/__init__.py +5 -0
- package/gnomon/utils/abi_resolver.py +255 -0
- package/ios/ExportOptions.plist +16 -0
- package/ios/README.md +33 -0
- package/jest.config.ts +18 -0
- package/keyring/__init__.py +17 -0
- package/launcher.sh +57 -0
- package/license.env +2 -0
- package/licensingServer/package.json +23 -0
- package/licensingServer/src/config/keys.ts +84 -0
- package/licensingServer/src/index.ts +30 -0
- package/licensingServer/src/lib/canonicalize.ts +5 -0
- package/licensingServer/src/lib/crypto.ts +25 -0
- package/licensingServer/src/lib/validate.ts +62 -0
- package/licensingServer/src/middleware/auth.ts +20 -0
- package/licensingServer/src/routes/licenses.ts +110 -0
- package/licensingServer/tsconfig.json +12 -0
- package/main/ipcHandlers/index.ts +23 -0
- package/main/keyring/keyringmanager.ts +154 -0
- package/main/main.ts +234 -0
- package/main/preload/index.ts +31 -0
- package/main/preload/licenseBridge.ts +73 -0
- package/modules/sandbox/abiLoader.ts +78 -0
- package/modules/sandbox/contractSimulator.ts +241 -0
- package/modules/sandbox/formBuilder.ts +16 -0
- package/modules/sandbox/index.ts +6 -0
- package/modules/sandbox/localFork.ts +129 -0
- package/modules/sandbox/safe.abi.json +82 -0
- package/modules/sandbox/sandboxManager.ts +154 -0
- package/modules/sandbox/types.ts +84 -0
- package/modules/sandbox/ui/LogViewer.tsx +30 -0
- package/modules/sandbox/ui/ParameterForm.tsx +49 -0
- package/modules/sandbox/ui/SandboxPanel.tsx +568 -0
- package/package.json +82 -0
- package/renderer/components/LicenseScreen.tsx +134 -0
- package/renderer/index.html +12 -0
- package/renderer/package-lock.json +4104 -0
- package/renderer/package.json +35 -0
- package/renderer/postcss.config.cjs +6 -0
- package/renderer/src/App.tsx +229 -0
- package/renderer/src/context/KeyringContext.tsx +217 -0
- package/renderer/src/context/SafeContext.tsx +49 -0
- package/renderer/src/context/ThemeContext.tsx +60 -0
- package/renderer/src/context/WalletContext.tsx +50 -0
- package/renderer/src/context/main.tsx +18 -0
- package/renderer/src/main.tsx +18 -0
- package/renderer/src/pages/Contracts.tsx +482 -0
- package/renderer/src/pages/Dashboard.tsx +653 -0
- package/renderer/src/pages/DeveloperTools.tsx +270 -0
- package/renderer/src/pages/History.tsx +149 -0
- package/renderer/src/pages/Keyring.tsx +449 -0
- package/renderer/src/pages/Safes.tsx +1089 -0
- package/renderer/src/pages/Sandbox.tsx +146 -0
- package/renderer/src/pages/Settings.tsx +871 -0
- package/renderer/src/pages/Wallets.tsx +752 -0
- package/renderer/src/pages/WikiGuide.tsx +75 -0
- package/renderer/src/styles.css +32 -0
- package/renderer/src/types/gnoman.d.ts +9 -0
- package/renderer/src/types/license.ts +8 -0
- package/renderer/src/types/safevault.d.ts +17 -0
- package/renderer/src/utils/backend.ts +88 -0
- package/renderer/tailwind.config.cjs +8 -0
- package/renderer/tsconfig.json +13 -0
- package/renderer/tsconfig.node.json +9 -0
- package/renderer/vite.config.ts +19 -0
- package/requests/__init__.py +35 -0
- package/requests/__pycache__/__init__.cpython-310.pyc +0 -0
- package/scripts/build-ios.sh +30 -0
- package/scripts/copyBackendAssets.js +24 -0
- package/scripts/copyRenderer.js +87 -0
- package/scripts/deployBackend.sh +24 -0
- package/scripts/launchElectron.js +51 -0
- package/src/core/backends/fileBackend.ts +154 -0
- package/src/core/backends/memoryBackend.ts +27 -0
- package/src/core/backends/systemBackend.ts +66 -0
- package/src/core/backends/types.ts +17 -0
- package/src/core/keyringManager.ts +208 -0
- package/src/utils/abiCache/.gitkeep +0 -0
- package/src/utils/abiResolver.ts +200 -0
- package/src/utils/runtimeObservability.ts +110 -0
- package/src/utils/secretsResolver.ts +144 -0
- package/tests/chainlinkService.test.ts +32 -0
- package/tests/diagnosticsService.test.ts +68 -0
- package/tests/etherscanController.test.ts +99 -0
- package/tests/etherscanService.test.ts +116 -0
- package/tests/keyringManager.test.ts +135 -0
- package/tests/onchainToolkit.test.ts +71 -0
- package/tests/robinhoodClient.test.ts +54 -0
- package/tests/robinhoodController.test.ts +81 -0
- package/tests/robinhoodIntegrationService.test.ts +50 -0
- package/tests/safeServicePersistence.test.ts +81 -0
- package/tests/test_contract_sandbox/sandbox.test.js +407 -0
- package/tests/walletController.test.ts +57 -0
- package/touch +14 -0
- package/tsconfig.backend.json +7 -0
- package/tsconfig.cli.json +7 -0
- package/tsconfig.json +18 -0
- package/tsconfig.main.json +7 -0
- 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,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;
|