beeline-cli
Version:
A terminal wallet for the Hive blockchain - type, sign, rule the chain
204 lines • 10 kB
JavaScript
;
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