@hashgraph/solo
Version:
An opinionated CLI tool to deploy and manage private Hedera Networks.
110 lines (102 loc) • 3.91 kB
text/typescript
// SPDX-License-Identifier: Apache-2.0
import {inject, injectable} from 'tsyringe-neo';
import {type SoloLogger} from './logging/solo-logger.js';
import {patchInject} from './dependency-injection/container-helper.js';
import {SoloError} from './errors/solo-error.js';
import {type Lock} from './lock/lock.js';
import * as constants from './constants.js';
import fs from 'node:fs';
import {type ConfigManager} from './config-manager.js';
import {type OneShotState} from './one-shot-state.js';
import {InjectTokens} from './dependency-injection/inject-tokens.js';
import {type AccountManager} from './account-manager.js';
import {type TaskList} from './task-list/task-list.js';
import {ListrContext, ListrRendererValue} from 'listr2';
import {SoloListr} from '../types/index.js';
import {AnyListrContext} from '../types/aliases.js';
()
export class CommandHandler {
protected readonly _configMaps: Map<string, any> = new Map();
public constructor(
(InjectTokens.SoloLogger) public readonly logger?: SoloLogger,
(InjectTokens.ConfigManager) private readonly configManager?: ConfigManager,
(InjectTokens.AccountManager) private readonly accountManager?: AccountManager,
(InjectTokens.TaskList)
private readonly taskList?: TaskList<ListrContext, ListrRendererValue, ListrRendererValue>,
(InjectTokens.OneShotState) private readonly oneShotState?: OneShotState,
) {
this.logger = patchInject(logger, InjectTokens.SoloLogger, this.constructor.name);
this.configManager = patchInject(configManager, InjectTokens.ConfigManager, this.constructor.name);
this.accountManager = patchInject(accountManager, InjectTokens.AccountManager, this.constructor.name);
this.taskList = patchInject(taskList, InjectTokens.TaskList, this.constructor.name);
this.oneShotState = patchInject(oneShotState, InjectTokens.OneShotState, this.constructor.name);
}
public async commandAction(
argv: any,
actionTasks: any,
options: any,
errorString: string,
lease?: Lock,
commandName?: string,
): Promise<void> {
if (!commandName) {
commandName = argv._.slice(0, 3).join(' ');
}
const tasks: SoloListr<AnyListrContext> = this.taskList.newTaskList(
[...actionTasks],
options,
undefined,
commandName,
);
if (tasks.isRoot()) {
try {
await tasks.run();
} catch (error) {
throw new SoloError(`${errorString}: ${error.message}`, error);
} finally {
const promises: Promise<void>[] = [];
if (!this.oneShotState.isActive() && lease) {
promises.push(lease?.release());
}
promises.push(this.accountManager.close());
await Promise.all(promises);
}
} else {
this.taskList.registerCloseFunction(async (): Promise<void> => {
const promises: Promise<void>[] = [];
if (!this.oneShotState.isActive() && lease) {
promises.push(lease?.release());
}
promises.push(this.accountManager.close());
await Promise.all(promises);
});
}
}
/**
* Setup home directories
* @param directories
*/
public setupHomeDirectory(
directories: string[] = [
constants.SOLO_HOME_DIR,
constants.SOLO_LOGS_DIR,
constants.SOLO_CACHE_DIR,
constants.SOLO_VALUES_DIR,
],
): string[] {
try {
for (const directoryPath of directories) {
if (!fs.existsSync(directoryPath)) {
fs.mkdirSync(directoryPath, {recursive: true});
}
this.logger.debug(`OK: setup directory: ${directoryPath}`);
}
} catch (error) {
throw new SoloError(`failed to create directory: ${error.message}`, error);
}
return directories;
}
public getUnusedConfigs(configName: string): string[] {
return this._configMaps.get(configName).getUnusedConfigs();
}
}