@hashgraph/solo
Version:
An opinionated CLI tool to deploy and manage private Hedera Networks.
170 lines • 8.47 kB
JavaScript
// SPDX-License-Identifier: Apache-2.0
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var __metadata = (this && this.__metadata) || function (k, v) {
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
};
var __param = (this && this.__param) || function (paramIndex, decorator) {
return function (target, key) { decorator(target, key, paramIndex); }
};
var GvproxyDependencyManager_1;
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 fs from 'node:fs';
import { OperatingSystem } from '../../business/utils/operating-system.js';
import { PathEx } from '../../business/utils/path-ex.js';
const GVPROXY_RELEASES_LIST_URL = 'https://api.github.com/repos/containers/gvisor-tap-vsock/releases';
let GvproxyDependencyManager = GvproxyDependencyManager_1 = class GvproxyDependencyManager extends BaseDependencyManager {
checksum;
releaseBaseUrl;
artifactFileName;
artifactVersion;
constructor(downloader, installationDirectory, osArch, gvproxyVersion) {
super(patchInject(downloader, InjectTokens.PackageDownloader, GvproxyDependencyManager_1.name), patchInject(installationDirectory, InjectTokens.PodmanDependenciesInstallationDirectory, GvproxyDependencyManager_1.name), patchInject(osArch, InjectTokens.OsArch, GvproxyDependencyManager_1.name), patchInject(gvproxyVersion, InjectTokens.GvproxyVersion, GvproxyDependencyManager_1.name) ||
version.GVPROXY_VERSION, constants.GVPROXY, '');
}
/**
* Get the Gvproxy artifact name based on version, OS, and architecture
*/
getArtifactName() {
return util.format(this.artifactFileName, this.getRequiredVersion(), OperatingSystem.getFormattedPlatform(), this.osArch);
}
async getVersion(executableWithPath) {
// The retry logic is to handle potential transient issues with the command execution
// The command `gvproxy --version` was sometimes observed to return an empty output in the CI environment
const maxAttempts = 3;
for (let attempt = 1; attempt <= maxAttempts; attempt++) {
try {
const output = await this.run(`"${executableWithPath}" --version`);
if (output.length > 0) {
const match = output[0].trim().match(/(\d+\.\d+\.\d+)/);
return match[1];
}
}
catch (error) {
throw new SoloError('Failed to check gvproxy version', error);
}
}
throw new SoloError('Failed to check gvproxy version');
}
/**
* Determine the appropriate asset name for the current platform and architecture
* based on the naming conventions used in gvproxy GitHub releases
*/
getAssetName() {
// Normalize platform/arch for asset matching
const arch = this.getArch();
// Select the appropriate asset name based on platform and architecture
let assetName;
if (OperatingSystem.isWin32()) {
// For Windows, use the regular exe (not the GUI version)
assetName = arch === 'arm64' ? 'gvproxy-windows-arm64.exe' : 'gvproxy-windows.exe';
}
else if (OperatingSystem.isDarwin()) {
assetName = 'gvproxy-darwin';
}
else if (OperatingSystem.isLinux()) {
assetName = `gvproxy-linux-${arch}`;
}
else {
throw new SoloError(`Unsupported platform: ${OperatingSystem.getPlatform()}`);
}
return assetName;
}
/**
* Fetches the latest release information from GitHub API
* @returns Promise with the release base URL, asset name, digest, and version
*/
async fetchReleaseInfo(tagName) {
try {
// Make a GET request to GitHub API using fetch
const response = await fetch(GVPROXY_RELEASES_LIST_URL, {
method: 'GET',
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 = await response.json();
if (!releases || releases.length === 0) {
throw new SoloError('No releases found');
}
// Get the latest release
const release = releases.find(release => release.tag_name === tagName);
const version = release.tag_name.replace(/^v/, ''); // Remove 'v' prefix if present
const assetName = this.getAssetName();
const matchingAsset = release.assets.find(asset => asset.name === assetName);
if (!matchingAsset) {
throw new SoloError(`No matching asset found (${assetName})`);
}
// Get the digest from the shasums file
const checksum = matchingAsset.digest
? matchingAsset.digest.replace('sha256:', '')
: '0000000000000000000000000000000000000000000000000000000000000000';
// Construct the release base URL (removing the filename from the download URL)
const downloadUrl = 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);
}
}
async preInstall() {
const latestReleaseInfo = await this.fetchReleaseInfo(version.GVPROXY_VERSION);
this.checksum = latestReleaseInfo.checksum;
this.releaseBaseUrl = latestReleaseInfo.downloadUrl;
this.artifactFileName = latestReleaseInfo.assetName;
this.artifactVersion = latestReleaseInfo.version;
}
getDownloadURL() {
return `${this.releaseBaseUrl}/${this.artifactFileName}`;
}
/**
* Handle any post-download processing before copying to destination
* Child classes can override this for custom extraction or processing
*/
async processDownloadedPackage(packageFilePath, temporaryDirectory) {
// Determine the target filename based on the platform
const targetFileName = OperatingSystem.isWin32() ? 'gvproxy.exe' : 'gvproxy';
const targetPath = PathEx.join(temporaryDirectory, targetFileName);
// Rename the downloaded file
fs.renameSync(packageFilePath, targetPath);
return [targetPath];
}
getChecksumURL() {
return this.checksum;
}
};
GvproxyDependencyManager = GvproxyDependencyManager_1 = __decorate([
injectable(),
__param(0, inject(InjectTokens.PackageDownloader)),
__param(1, inject(InjectTokens.PodmanDependenciesInstallationDirectory)),
__param(2, inject(InjectTokens.OsArch)),
__param(3, inject(InjectTokens.GvproxyVersion)),
__metadata("design:paramtypes", [PackageDownloader, String, String, String])
], GvproxyDependencyManager);
export { GvproxyDependencyManager };
//# sourceMappingURL=gvproxy-dependency-manager.js.map