@hashgraph/solo
Version:
An opinionated CLI tool to deploy and manage private Hedera Networks.
394 lines (363 loc) • 17.9 kB
text/typescript
// SPDX-License-Identifier: Apache-2.0
import {inject, injectable} from 'tsyringe-neo';
import {ShellRunner} from './shell-runner.js';
import {InjectTokens} from './dependency-injection/inject-tokens.js';
import {BrewPackageManager} from './package-managers/brew-package-manager.js';
import {OsPackageManager} from './package-managers/os-package-manager.js';
import {patchInject} from './dependency-injection/container-helper.js';
import {PodmanMode, SoloListrTask, type SoloListrTaskWrapper} from '../types/index.js';
import {InitContext} from '../commands/init/init-context.js';
import {AptGetPackageManager} from './package-managers/apt-get-package-manager.js';
import {SoloError} from './errors/solo-error.js';
import * as constants from './constants.js';
import {getTemporaryDirectory} from './helpers.js';
import fs from 'node:fs';
import * as yaml from 'yaml';
import {type AnyObject} from '../types/aliases.js';
import path from 'node:path';
import {KindClient} from '../integration/kind/kind-client.js';
import {ClusterCreateResponse} from '../integration/kind/model/create-cluster/cluster-create-response.js';
import {type ClusterCreateOptions} from '../integration/kind/model/create-cluster/cluster-create-options.js';
import {ClusterCreateOptionsBuilder} from '../integration/kind/model/create-cluster/create-cluster-options-builder.js';
import {type DefaultKindClientBuilder} from '../integration/kind/impl/default-kind-client-builder.js';
import {type DependencyManager, KindDependencyManager, PodmanDependencyManager} from './dependency-managers/index.js';
import {K8} from '../integration/kube/k8.js';
import {MissingActiveContextError} from '../integration/kube/errors/missing-active-context-error.js';
import {MissingActiveClusterError} from '../integration/kube/errors/missing-active-cluster-error.js';
import {type K8Factory} from '../integration/kube/k8-factory.js';
import {type GitClient} from '../integration/git/git-client.js';
import {ImageCacheHandler} from '../integration/cache/impl/image-cache-handler.js';
import {KindNodeImageTargetProvider} from '../integration/cache/target-providers/kind-image-target-provider.js';
import {ImageCacheHandlerBuilder} from '../integration/cache/impl/image-cache-handler-builder.js';
import {type ContainerEngineClient} from '../integration/container-engine/container-engine-client.js';
()
export class ClusterTaskManager extends ShellRunner {
public constructor(
(InjectTokens.BrewPackageManager) protected readonly brewPackageManager: BrewPackageManager,
(InjectTokens.OsPackageManager) protected readonly osPackageManager: OsPackageManager,
(InjectTokens.KindBuilder) protected readonly kindBuilder: DefaultKindClientBuilder,
(InjectTokens.PodmanDependencyManager) protected readonly podmanDependencyManager: PodmanDependencyManager,
(InjectTokens.KindDependencyManager) protected readonly kindDependencyManager: KindDependencyManager,
(InjectTokens.PodmanInstallationDirectory) protected readonly podmanInstallationDirectory: string,
(InjectTokens.K8Factory) protected readonly k8Factory: K8Factory,
(InjectTokens.DependencyManager) protected readonly depManager: DependencyManager,
(InjectTokens.KindInstallationDirectory) protected readonly kindInstallationDirectory: string,
(InjectTokens.GitClient) protected readonly gitClient: GitClient,
(InjectTokens.ContainerEngineClient) protected readonly containerEngineClient: ContainerEngineClient,
) {
super();
this.brewPackageManager = patchInject(brewPackageManager, InjectTokens.BrewPackageManager, ClusterTaskManager.name);
this.osPackageManager = patchInject(osPackageManager, InjectTokens.OsPackageManager, ClusterTaskManager.name);
this.kindBuilder = patchInject(kindBuilder, InjectTokens.KindBuilder, ClusterTaskManager.name);
this.podmanDependencyManager = patchInject(
podmanDependencyManager,
InjectTokens.KindBuilder,
ClusterTaskManager.name,
);
this.kindDependencyManager = patchInject(kindDependencyManager, InjectTokens.KindBuilder, ClusterTaskManager.name);
this.podmanInstallationDirectory = patchInject(
podmanInstallationDirectory,
InjectTokens.PodmanInstallationDirectory,
ClusterTaskManager.name,
);
this.k8Factory = patchInject(k8Factory, InjectTokens.K8Factory, ClusterTaskManager.name);
this.depManager = patchInject(depManager, InjectTokens.DependencyManager, ClusterTaskManager.name);
this.kindInstallationDirectory = patchInject(
kindInstallationDirectory,
InjectTokens.KindInstallationDirectory,
ClusterTaskManager.name,
);
this.gitClient = patchInject(gitClient, InjectTokens.GitClient, ClusterTaskManager.name);
this.containerEngineClient = patchInject(
containerEngineClient,
InjectTokens.ContainerEngineClient,
ClusterTaskManager.name,
);
}
private sudoCallbacks(task: SoloListrTaskWrapper<InitContext>): {
onSudoRequested: (message: string) => void;
onSudoGranted: (message: string) => void;
} {
const originalTitle: string = task.title;
const onSudoRequested: (message: string) => void = (message: string): void => {
task.title = message;
};
const onSudoGranted: (message: string) => void = (message: string): void => {
void message;
task.title = originalTitle;
};
return {onSudoGranted, onSudoRequested};
}
public rootfullInstallTasks(parentTask: SoloListrTaskWrapper<InitContext>): SoloListrTask<InitContext>[] {
return [
{
title: 'Install git, iptables...',
task: async (): Promise<void> => {
try {
await this.gitClient.version();
} catch {
this.logger.info('Git not found, installing git...');
const {onSudoGranted, onSudoRequested} = this.sudoCallbacks(parentTask);
const osPackageManager: AptGetPackageManager =
this.osPackageManager.getPackageManager() as AptGetPackageManager;
osPackageManager.setOnSudoGranted(onSudoGranted);
osPackageManager.setOnSudoRequested(onSudoRequested);
await osPackageManager.update();
await osPackageManager.installPackages(['git', 'iptables']);
}
},
},
{
title: 'Install brew...',
task: async (): Promise<void> => {
const brewInstalled: boolean = await this.brewPackageManager.isAvailable();
if (!brewInstalled) {
this.logger.info('Homebrew not found, installing Homebrew...');
if (!(await this.brewPackageManager.install())) {
throw new SoloError('Failed to install Homebrew');
}
}
},
},
{
title: 'Install podman...',
task: async (): Promise<void> => {
try {
const podmanVersion: string[] = await this.run('podman --version');
this.logger.info(`Podman already installed: ${podmanVersion}`);
} catch {
this.logger.info('Podman not found, installing Podman...');
await this.brewPackageManager.installPackages(['podman']);
const brewBin: string[] = await this.run('which podman');
process.env.PATH = `${process.env.PATH}:${brewBin.join('').replace('/podman', '')}`;
}
},
} as SoloListrTask<InitContext>,
{
title: 'Creating local cluster...',
task: async (_context: InitContext, task: SoloListrTaskWrapper<InitContext>): Promise<void> => {
void _context;
const whichPodman: string[] = await this.run('which podman');
const podmanPath: string = whichPodman.join('').replace('/podman', '');
const sudoRunOptions: [string[], boolean?, boolean?, Record<string, string>?] = [
[],
undefined,
undefined,
{
PATH:
`${this.podmanInstallationDirectory}${path.delimiter}` +
`${this.kindInstallationDirectory}${path.delimiter}${process.env.PATH}`,
},
];
const {onSudoGranted, onSudoRequested} = this.sudoCallbacks(task);
await this.sudoRun(
onSudoRequested,
onSudoGranted,
`KIND_EXPERIMENTAL_PROVIDER=podman PATH="$PATH:${podmanPath}" kind create cluster --image "${constants.KIND_NODE_IMAGE}" --config "${constants.KIND_CLUSTER_CONFIG_FILE}"`,
...sudoRunOptions,
);
// Merge kubeconfig data from root user into normal user's kubeconfig
const user: string[] = await this.run('whoami');
const temporaryDirectory: string = getTemporaryDirectory();
await this.sudoRun(
onSudoRequested,
onSudoGranted,
`cp /root/.kube/config ${temporaryDirectory}/kube-config-root`,
...sudoRunOptions,
);
await this.sudoRun(
onSudoRequested,
onSudoGranted,
`chown ${user} ${temporaryDirectory}/kube-config-root`,
...sudoRunOptions,
);
await this.sudoRun(
onSudoRequested,
onSudoGranted,
`chmod 755 ${temporaryDirectory}/kube-config-root`,
...sudoRunOptions,
);
const rootYamlData: string = fs.readFileSync(`${temporaryDirectory}/kube-config-root`, 'utf8');
const rootConfig: Record<string, AnyObject> = yaml.parse(rootYamlData) as Record<string, AnyObject>;
let userConfig: Record<string, AnyObject>;
const clusterName: string = 'kind-kind';
try {
const userYamlData: string = fs.readFileSync(`/home/${user}/.kube/config`, 'utf8');
userConfig = yaml.parse(userYamlData) as Record<string, AnyObject>;
if (!userConfig.clusters) {
userConfig.clusters = [];
}
userConfig.clusters.push(rootConfig.clusters.find((c: AnyObject): boolean => c.name === clusterName));
if (!userConfig.contexts) {
userConfig.contexts = [];
}
userConfig.contexts.push(rootConfig.contexts.find((c: AnyObject): boolean => c.name === clusterName));
if (!userConfig.users) {
userConfig.users = [];
}
userConfig.users.push(rootConfig.users.find((c: AnyObject): boolean => c.name === clusterName));
userConfig['current-context'] = rootConfig['current-context'];
} catch (error) {
if (error.code === 'ENOENT') {
const kubeConfigDirectory: string = `/home/${user}/.kube/`;
if (!fs.existsSync(kubeConfigDirectory)) {
fs.mkdirSync(kubeConfigDirectory, {recursive: true});
}
userConfig = rootConfig;
userConfig.clusters = userConfig.clusters.filter((c: AnyObject): boolean => c.name === clusterName);
userConfig.contexts = userConfig.contexts.filter((c: AnyObject): boolean => c.name === clusterName);
userConfig.users = userConfig.users.filter((c: AnyObject): boolean => c.name === clusterName);
} else {
throw error;
}
}
fs.writeFileSync(`/home/${user}/.kube/config`, yaml.stringify(userConfig), 'utf8');
fs.rmSync(`${temporaryDirectory}/kube-config-root`);
},
} as SoloListrTask<InitContext>,
];
}
public async installationTasks(parentTask: SoloListrTaskWrapper<InitContext>): Promise<SoloListrTask<InitContext>[]> {
const skipPodmanTasks: boolean = !(await this.podmanDependencyManager.shouldInstall());
if (this.podmanDependencyManager.mode === PodmanMode.ROOTFUL) {
{
return skipPodmanTasks ? [this.defaultCreateClusterTask(parentTask)] : this.rootfullInstallTasks(parentTask);
}
} else if (this.podmanDependencyManager.mode === PodmanMode.VIRTUAL_MACHINE) {
{
return [
{
title: 'Create Podman machine...',
task: async (): Promise<void> => {
const podmanRunOptions: [string[], boolean?, boolean?, Record<string, string>?] = [
[],
undefined,
undefined,
{
PATH: `${this.podmanInstallationDirectory}${path.delimiter}${process.env.PATH}`,
},
];
await this.podmanDependencyManager.setupConfig();
const podmanExecutable: string = await this.podmanDependencyManager.getExecutable();
try {
await this.run(
`${podmanExecutable} machine inspect ${constants.PODMAN_MACHINE_NAME}`,
...podmanRunOptions,
);
} catch (error) {
if (error.message.includes('VM does not exist')) {
await this.run(
`${podmanExecutable} machine init ${constants.PODMAN_MACHINE_NAME} --memory=16384`, // 16GB
...podmanRunOptions,
);
await this.run(
`${podmanExecutable} machine start ${constants.PODMAN_MACHINE_NAME}`,
...podmanRunOptions,
);
} else {
throw new SoloError(`Failed to inspect Podman machine: ${error.message}`);
}
}
},
skip: (): boolean => skipPodmanTasks,
} as SoloListrTask<InitContext>,
{
title: 'Configure kind to use podman...',
task: async (): Promise<void> => {
process.env.KIND_EXPERIMENTAL_PROVIDER = 'podman';
},
skip: (): boolean => skipPodmanTasks,
} as SoloListrTask<InitContext>,
this.defaultCreateClusterTask(parentTask),
];
}
}
return [];
}
private defaultCreateClusterTask(parentTask: SoloListrTaskWrapper<InitContext>): SoloListrTask<InitContext> {
return {
title: 'Creating local cluster...',
task: async (): Promise<void> => {
const kindExecutable: string = await this.kindDependencyManager.getExecutable();
const kindClient: KindClient = await this.kindBuilder.executable(kindExecutable).build();
if (constants.CONFIG.ENABLE_IMAGE_CACHE) {
const kindImageCacheHandler: ImageCacheHandler = new ImageCacheHandlerBuilder()
.provider(new KindNodeImageTargetProvider())
.engine(this.containerEngineClient)
.build();
await kindImageCacheHandler.pullKindNodeImageIfMissing();
await kindImageCacheHandler.loadKindNodeImageIntoEngine();
}
const clusterCreateOptions: ClusterCreateOptions = ClusterCreateOptionsBuilder.builder()
.image(constants.KIND_NODE_IMAGE)
.config(constants.KIND_CLUSTER_CONFIG_FILE)
.build();
const clusterResponse: ClusterCreateResponse = await kindClient.createCluster(
constants.DEFAULT_CLUSTER,
clusterCreateOptions,
);
parentTask.title = `Created local cluster '${clusterResponse.name}'; connect with context '${clusterResponse.context}'`;
},
} as SoloListrTask<InitContext>;
}
public setupLocalClusterTasks(): SoloListrTask<InitContext>[] {
return [
{
title: 'Install Kind',
task: async (_context: InitContext, task: SoloListrTaskWrapper<InitContext>): Promise<unknown> => {
void _context;
const podmanDependency: PodmanDependencyManager = this.podmanDependencyManager;
const shouldInstallPodman: boolean = await podmanDependency.shouldInstall();
const podmanDependencies: string[] =
shouldInstallPodman && podmanDependency.mode === PodmanMode.VIRTUAL_MACHINE
? [constants.PODMAN, constants.VFKIT, constants.GVPROXY]
: [];
const deps: string[] = [...podmanDependencies, constants.KIND];
const subTasks: SoloListrTask<InitContext>[] = this.depManager.taskCheckDependencies<InitContext>(deps);
// set up the sub-tasks
return task.newListr(subTasks, {
concurrent: true,
rendererOptions: {
collapseSubtasks: false,
},
});
},
skip: this.skipKindSetup.bind(this),
},
{
title: 'Create default cluster',
task: async (_context: InitContext, task: SoloListrTaskWrapper<InitContext>): Promise<unknown> => {
void _context;
const subTasks: SoloListrTask<InitContext>[] = await this.installationTasks(task);
return task.newListr(subTasks, {
concurrent: false, // should not use concurrent as cluster creation may be called before dependencies are finished installing
rendererOptions: {
collapseSubtasks: false,
},
});
},
skip: this.skipKindSetup.bind(this),
},
];
}
private async skipKindSetup(): Promise<boolean> {
try {
const k8: K8 = this.k8Factory.default();
const contextName: string = k8.contexts().readCurrent();
if (!contextName) {
return false;
}
// Try to verify the cluster is actually accessible by making a simple API call
try {
await k8.namespaces().list();
return true;
} catch {
// If we can't connect to the cluster, don't skip cluster creation
// This handles cases where contexts exist but clusters are not running
return false;
}
} catch (error) {
return !(error instanceof MissingActiveContextError || error instanceof MissingActiveClusterError);
}
}
}