UNPKG

@hashgraph/solo

Version:

An opinionated CLI tool to deploy and manage private Hedera Networks.

165 lines (142 loc) 6.07 kB
// SPDX-License-Identifier: Apache-2.0 import * as constants from '../constants.js'; import * as version from '../../../version.js'; import {inject, injectable} from 'tsyringe-neo'; import {patchInject} from '../dependency-injection/container-helper.js'; import {InjectTokens} from '../dependency-injection/inject-tokens.js'; import {BaseDependencyManager} from './base-dependency-manager.js'; import {PackageDownloader} from '../package-downloader.js'; import util from 'node:util'; import {SoloError} from '../errors/solo-error.js'; import {GitHubRelease, GitHubReleaseAsset, ReleaseInfo} from '../../types/index.js'; import {OperatingSystem} from '../../business/utils/operating-system.js'; const VFKIT_RELEASES_LIST_URL: string = 'https://api.github.com/repos/crc-org/vfkit/releases'; @injectable() export class VfkitDependencyManager extends BaseDependencyManager { protected checksum: string; protected releaseBaseUrl: string; protected artifactFileName: string; protected artifactVersion: string; public constructor( @inject(InjectTokens.PackageDownloader) downloader: PackageDownloader, @inject(InjectTokens.PodmanDependenciesInstallationDirectory) installationDirectory: string, @inject(InjectTokens.OsArch) osArch: string, @inject(InjectTokens.VfkitVersion) vfkitVersion: string, ) { super( patchInject(downloader, InjectTokens.PackageDownloader, VfkitDependencyManager.name), patchInject( installationDirectory, InjectTokens.PodmanDependenciesInstallationDirectory, VfkitDependencyManager.name, ), patchInject(osArch, InjectTokens.OsArch, VfkitDependencyManager.name), patchInject(vfkitVersion, InjectTokens.VfkitVersion, VfkitDependencyManager.name) || version.VFKIT_VERSION, constants.VFKIT, '', ); } public override getVerifyChecksum(): boolean { return false; } /** * Get the Vfkit artifact name based on version, OS, and architecture */ protected getArtifactName(): string { return util.format( this.artifactFileName, this.getRequiredVersion(), OperatingSystem.getFormattedPlatform(), this.osArch, ); } public async getVersion(executableWithPath: string): Promise<string> { // The retry logic is to handle potential transient issues with the command execution // The command `vfkit --version` was sometimes observed to return an empty output in the CI environment const maxAttempts: number = 3; for (let attempt: number = 1; attempt <= maxAttempts; attempt++) { try { const output: string[] = await this.run(`${executableWithPath} --version`); if (output.length > 0) { const match: RegExpMatchArray | null = output[0].trim().match(/(\d+\.\d+\.\d+)/); return match[1]; } } catch (error: any) { throw new SoloError('Failed to check vfkit version', error); } } throw new SoloError('Failed to check vfkit version'); } /** * Fetches the latest release information from GitHub API * @returns Promise with the release base URL, asset name, digest, and version */ private async fetchReleaseInfo(tagName: string): Promise<ReleaseInfo> { try { // Make a GET request to GitHub API using fetch const response = await fetch(VFKIT_RELEASES_LIST_URL, { method: 'GET', // Changed from HEAD to GET to retrieve the body headers: { 'User-Agent': constants.SOLO_USER_AGENT_HEADER, Accept: 'application/vnd.github.v3+json', // Explicitly request GitHub API v3 format }, }); if (!response.ok) { throw new SoloError(`GitHub API request failed with status ${response.status}`); } // Parse the JSON response const releases: GitHubRelease[] = await response.json(); if (!releases || releases.length === 0) { throw new SoloError('No releases found'); } // Get the latest release const release: GitHubRelease = releases.find(release => release.tag_name === tagName); const version: string = release.tag_name.replace(/^v/, ''); // Remove 'v' prefix if present // Normalize platform/arch for asset matching const arch: string = this.getArch(); const assetName: string = 'vfkit'; const matchingAsset: GitHubReleaseAsset = release.assets.find(asset => asset.name.includes(assetName)); if (!matchingAsset) { throw new SoloError(`No matching asset found for ${OperatingSystem.getFormattedPlatform()}-${arch}`); } const checksum: string = matchingAsset.digest || '0000000000000000000000000000000000000000000000000000000000000000'; // Construct the release base URL (removing the filename from the download URL) const downloadUrl: string = matchingAsset.browser_download_url.slice( 0, Math.max(0, matchingAsset.browser_download_url.lastIndexOf('/')), ); return { downloadUrl, assetName: matchingAsset.name, checksum, version, }; } catch (error) { if (error instanceof SoloError) { throw error; } throw new SoloError('Failed to parse GitHub API response', error); } } protected override async preInstall(): Promise<void> { const releaseInfo: ReleaseInfo = await this.fetchReleaseInfo(version.VFKIT_VERSION); this.checksum = releaseInfo.checksum; this.releaseBaseUrl = releaseInfo.downloadUrl; this.artifactFileName = releaseInfo.assetName; this.artifactVersion = releaseInfo.version; } protected getDownloadURL(): string { return `${this.releaseBaseUrl}/${this.artifactFileName}`; } /** * Handle any post-download processing before copying to destination * Child classes can override this for custom extraction or processing */ protected async processDownloadedPackage(packageFilePath: string, _temporaryDirectory: string): Promise<string[]> { return [packageFilePath]; } protected getChecksumURL(): string { return this.checksum; } }