UNPKG

beeline-cli

Version:

A terminal wallet for the Hive blockchain - type, sign, rule the chain

204 lines 10 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const core_1 = require("@oclif/core"); const neon_js_1 = require("../utils/neon.js"); const crypto_js_1 = require("../utils/crypto.js"); const hive_js_1 = require("../utils/hive.js"); const inquirer_1 = __importDefault(require("inquirer")); class Transfer extends core_1.Command { async run() { const { args, flags } = await this.parse(Transfer); const keyManager = new crypto_js_1.KeyManager(); await keyManager.initialize(); let fromAccount = flags.from; let toAccount = args.to; // Clean @ prefix if provided if (fromAccount?.startsWith('@')) { fromAccount = fromAccount.substring(1); } if (toAccount.startsWith('@')) { toAccount = toAccount.substring(1); } // Use default account if no from account specified if (!fromAccount) { fromAccount = keyManager.getDefaultAccount(); if (!fromAccount) { console.log(neon_js_1.neonChalk.warning(`${neon_js_1.neonSymbols.cross} No sender account specified and no default account set`)); console.log(neon_js_1.neonChalk.info('Import a key first with: ') + neon_js_1.neonChalk.highlight('beeline keys import <account> active')); return; } } // Validate amount format const amount = parseFloat(args.amount); if (isNaN(amount) || amount <= 0) { console.log(neon_js_1.neonChalk.error(`${neon_js_1.neonSymbols.cross} Invalid amount: ${args.amount}`)); return; } const currency = args.currency; const memo = args.memo || ''; console.log(neon_js_1.neonChalk.glow(`${neon_js_1.neonSymbols.diamond} Preparing transfer...`)); console.log(''); // Display transfer details const transferDetails = [ `${neon_js_1.neonChalk.cyan('FROM')} ${neon_js_1.neonSymbols.arrow} ${neon_js_1.neonChalk.highlight('@' + fromAccount)}`, `${neon_js_1.neonChalk.magenta('TO')} ${neon_js_1.neonSymbols.arrow} ${neon_js_1.neonChalk.highlight('@' + toAccount)}`, `${neon_js_1.neonChalk.electric('AMOUNT')} ${neon_js_1.neonSymbols.arrow} ${neon_js_1.neonChalk.white(amount.toFixed(3))} ${neon_js_1.neonChalk.yellow(currency)}`, memo ? `${neon_js_1.neonChalk.orange('MEMO')} ${neon_js_1.neonSymbols.arrow} ${neon_js_1.neonChalk.white('"' + memo + '"')}` : '', ``, `${neon_js_1.neonChalk.darkCyan('Transaction will be signed with your active key')}` ].filter(Boolean).join('\n'); console.log((0, neon_js_1.createNeonBox)(transferDetails, `${neon_js_1.neonSymbols.star} TRANSFER PREVIEW ${neon_js_1.neonSymbols.star}`)); console.log(''); if (flags.mock) { console.log(neon_js_1.neonChalk.warning(`${neon_js_1.neonSymbols.star} Mock mode - transaction will NOT be broadcast`)); console.log(''); } // Confirmation prompt if (!flags.confirm) { const confirmPrompt = await inquirer_1.default.prompt([{ type: 'confirm', name: 'confirm', message: flags.mock ? neon_js_1.neonChalk.cyan('Simulate this transfer?') : neon_js_1.neonChalk.warning('Execute this transfer? This action cannot be undone.'), default: false }]); if (!confirmPrompt.confirm) { console.log(neon_js_1.neonChalk.info('Transfer cancelled')); return; } } if (flags.mock) { return this.simulateTransfer(fromAccount, toAccount, amount, currency, memo); } // Get PIN for key decryption const keys = await keyManager.listKeys(fromAccount); const activeKey = keys.find(k => k.role === 'active'); if (!activeKey) { console.log(neon_js_1.neonChalk.error(`${neon_js_1.neonSymbols.cross} Active key not found for account @${fromAccount}`)); console.log(neon_js_1.neonChalk.info('Import active key with: ') + neon_js_1.neonChalk.highlight(`beeline keys import ${fromAccount} active`)); return; } let pin; if (activeKey.encrypted) { const pinPrompt = await inquirer_1.default.prompt([{ type: 'password', name: 'pin', message: neon_js_1.neonChalk.cyan('Enter PIN to unlock active key:'), validate: (input) => input.length > 0 || 'PIN required' }]); pin = pinPrompt.pin; } const spinner = (0, neon_js_1.neonSpinner)('Broadcasting to Hive blockchain'); try { const hiveClient = new hive_js_1.HiveClient(keyManager, flags.node); // Execute transfer const txId = await hiveClient.transfer(fromAccount, toAccount, amount.toFixed(3), currency, memo, pin); clearInterval(spinner); process.stdout.write('\r' + ' '.repeat(80) + '\r'); console.log(neon_js_1.neonChalk.success(`${neon_js_1.neonSymbols.check} Transfer successful!`)); console.log(''); const successMessage = [ `${neon_js_1.neonChalk.glow('Transaction broadcast successfully')}`, ``, `${neon_js_1.neonChalk.cyan('Transaction ID:')} ${neon_js_1.neonChalk.highlight(txId)}`, `${neon_js_1.neonChalk.magenta('From:')} @${fromAccount}`, `${neon_js_1.neonChalk.electric('To:')} @${toAccount}`, `${neon_js_1.neonChalk.orange('Amount:')} ${amount.toFixed(3)} ${currency}`, memo ? `${neon_js_1.neonChalk.pink('Memo:')} "${memo}"` : '', ``, `${neon_js_1.neonChalk.info('Transaction will be confirmed in ~3 seconds')}` ].filter(Boolean).join('\n'); console.log((0, neon_js_1.createNeonBox)(successMessage, `${neon_js_1.neonSymbols.star} TRANSFER COMPLETE ${neon_js_1.neonSymbols.star}`)); // Memory scrubbing if (pin) keyManager.scrubMemory(pin); } catch (error) { clearInterval(spinner); process.stdout.write('\r' + ' '.repeat(80) + '\r'); console.log(neon_js_1.neonChalk.error(`${neon_js_1.neonSymbols.cross} Transfer failed: ${error instanceof Error ? error.message : 'Unknown error'}`)); console.log(''); console.log(neon_js_1.neonChalk.info('Possible causes:')); console.log(neon_js_1.neonChalk.darkCyan('• Insufficient balance')); console.log(neon_js_1.neonChalk.darkCyan('• Invalid recipient account')); console.log(neon_js_1.neonChalk.darkCyan('• Network connectivity issues')); console.log(neon_js_1.neonChalk.darkCyan('• Incorrect PIN')); // Memory scrubbing on error too if (pin) keyManager.scrubMemory(pin); } } simulateTransfer(from, to, amount, currency, memo) { console.log(neon_js_1.neonChalk.glow(`${neon_js_1.neonSymbols.diamond} Simulating transfer...`)); console.log(''); // Simulate some processing time setTimeout(() => { const mockTxId = '0x' + Math.random().toString(16).substring(2, 18); console.log(neon_js_1.neonChalk.success(`${neon_js_1.neonSymbols.check} Transfer simulation complete!`)); console.log(''); const simulationMessage = [ `${neon_js_1.neonChalk.warning('SIMULATION ONLY - NO REAL TRANSFER')}`, ``, `${neon_js_1.neonChalk.cyan('Mock Transaction ID:')} ${neon_js_1.neonChalk.highlight(mockTxId)}`, `${neon_js_1.neonChalk.magenta('From:')} @${from}`, `${neon_js_1.neonChalk.electric('To:')} @${to}`, `${neon_js_1.neonChalk.orange('Amount:')} ${amount.toFixed(3)} ${currency}`, memo ? `${neon_js_1.neonChalk.pink('Memo:')} "${memo}"` : '', ``, `${neon_js_1.neonChalk.info('Remove --mock flag to execute real transfer')}` ].filter(Boolean).join('\n'); console.log((0, neon_js_1.createNeonBox)(simulationMessage, `${neon_js_1.neonSymbols.star} SIMULATION RESULT ${neon_js_1.neonSymbols.star}`)); }, 1500); } } Transfer.description = 'Transfer HIVE or HBD with cyberpunk style'; Transfer.examples = [ `$ beeline transfer @alice 10 HIVE "Hello!"`, `$ beeline transfer @bob 5.000 HBD`, `$ beeline transfer @charlie 1.000 HIVE --from @alice` ]; Transfer.flags = { from: core_1.Flags.string({ char: 'f', description: 'account to send from (defaults to default account)' }), node: core_1.Flags.string({ char: 'n', description: 'RPC node to use' }), confirm: core_1.Flags.boolean({ char: 'y', description: 'skip confirmation prompt', default: false }), mock: core_1.Flags.boolean({ char: 'm', description: 'simulate transfer without broadcasting', default: false }) }; Transfer.args = { to: core_1.Args.string({ description: 'recipient account name', required: true }), amount: core_1.Args.string({ description: 'amount to transfer', required: true }), currency: core_1.Args.string({ description: 'currency (HIVE or HBD)', required: true, options: ['HIVE', 'HBD'] }), memo: core_1.Args.string({ description: 'transfer memo', required: false }) }; exports.default = Transfer; //# sourceMappingURL=transfer.js.map