imcp 0.1.3 → 0.1.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +21 -4
- package/dist/cli/commands/install.js.map +1 -0
- package/dist/cli/commands/list.js.map +1 -0
- package/dist/cli/commands/pull.js.map +1 -0
- package/dist/cli/commands/serve.js.map +1 -0
- package/dist/cli/commands/start.d.ts +2 -0
- package/dist/cli/commands/start.js +32 -0
- package/dist/cli/commands/start.js.map +1 -0
- package/dist/cli/commands/sync.d.ts +2 -0
- package/dist/cli/commands/sync.js +17 -0
- package/dist/cli/commands/sync.js.map +1 -0
- package/dist/cli/commands/uninstall.js.map +1 -0
- package/dist/cli/index.js +0 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/core/ConfigurationLoader.d.ts +32 -0
- package/{src/core/loaders/ConfigurationLoader.ts → dist/core/ConfigurationLoader.js} +236 -298
- package/dist/core/ConfigurationLoader.js.map +1 -0
- package/dist/core/ConfigurationProvider.d.ts +35 -0
- package/{src/core/loaders/ConfigurationProvider.ts → dist/core/ConfigurationProvider.js} +375 -462
- package/dist/core/ConfigurationProvider.js.map +1 -0
- package/dist/core/InstallationService.d.ts +50 -0
- package/dist/core/InstallationService.js +350 -0
- package/dist/core/InstallationService.js.map +1 -0
- package/dist/core/MCPManager.d.ts +28 -0
- package/dist/core/MCPManager.js +188 -0
- package/dist/core/MCPManager.js.map +1 -0
- package/dist/core/RequirementService.d.ts +40 -0
- package/dist/core/RequirementService.js +110 -0
- package/dist/core/RequirementService.js.map +1 -0
- package/dist/core/ServerSchemaLoader.d.ts +11 -0
- package/dist/core/ServerSchemaLoader.js +43 -0
- package/dist/core/ServerSchemaLoader.js.map +1 -0
- package/dist/core/ServerSchemaProvider.d.ts +17 -0
- package/dist/core/ServerSchemaProvider.js +120 -0
- package/dist/core/ServerSchemaProvider.js.map +1 -0
- package/dist/core/constants.d.ts +47 -0
- package/dist/core/constants.js +94 -0
- package/dist/core/constants.js.map +1 -0
- package/dist/core/installers/BaseInstaller.d.ts +74 -0
- package/dist/core/installers/BaseInstaller.js +253 -0
- package/dist/core/installers/BaseInstaller.js.map +1 -0
- package/dist/core/installers/ClientInstaller.d.ts +23 -0
- package/dist/core/installers/ClientInstaller.js +564 -0
- package/dist/core/installers/ClientInstaller.js.map +1 -0
- package/dist/core/installers/CommandInstaller.d.ts +37 -0
- package/{src/core/installers/requirements/CommandInstaller.ts → dist/core/installers/CommandInstaller.js} +173 -231
- package/dist/core/installers/CommandInstaller.js.map +1 -0
- package/dist/core/installers/GeneralInstaller.d.ts +33 -0
- package/dist/core/installers/GeneralInstaller.js +85 -0
- package/dist/core/installers/GeneralInstaller.js.map +1 -0
- package/dist/core/installers/InstallerFactory.d.ts +54 -0
- package/{src/core/installers/requirements/InstallerFactory.ts → dist/core/installers/InstallerFactory.js} +97 -112
- package/dist/core/installers/InstallerFactory.js.map +1 -0
- package/dist/core/installers/NpmInstaller.d.ts +26 -0
- package/dist/core/installers/NpmInstaller.js +127 -0
- package/dist/core/installers/NpmInstaller.js.map +1 -0
- package/dist/core/installers/PipInstaller.d.ts +28 -0
- package/dist/core/installers/PipInstaller.js +127 -0
- package/dist/core/installers/PipInstaller.js.map +1 -0
- package/{src/core/installers/requirements/RequirementInstaller.ts → dist/core/installers/RequirementInstaller.d.ts} +33 -42
- package/dist/core/installers/RequirementInstaller.js +3 -0
- package/dist/core/installers/RequirementInstaller.js.map +1 -0
- package/dist/core/installers/clients/BaseClientInstaller.js.map +1 -0
- package/dist/core/installers/clients/ClientInstaller.js.map +1 -0
- package/dist/core/installers/clients/ClientInstallerFactory.js.map +1 -0
- package/dist/core/installers/clients/ClineInstaller.js.map +1 -0
- package/dist/core/installers/clients/ExtensionInstaller.js.map +1 -0
- package/dist/core/installers/clients/GithubCopilotInstaller.js.map +1 -0
- package/dist/core/installers/clients/MSRooCodeInstaller.js.map +1 -0
- package/dist/core/installers/index.js.map +1 -0
- package/dist/core/installers/requirements/BaseInstaller.js.map +1 -0
- package/dist/core/installers/requirements/CommandInstaller.js.map +1 -0
- package/dist/core/installers/requirements/GeneralInstaller.js.map +1 -0
- package/dist/core/installers/requirements/InstallerFactory.js +2 -0
- package/dist/core/installers/requirements/InstallerFactory.js.map +1 -0
- package/dist/core/installers/requirements/NpmInstaller.js.map +1 -0
- package/dist/core/installers/requirements/NugetInstaller.d.ts +37 -0
- package/dist/core/installers/requirements/NugetInstaller.js +189 -0
- package/dist/core/installers/requirements/NugetInstaller.js.map +1 -0
- package/dist/core/installers/requirements/PipInstaller.js.map +1 -0
- package/dist/core/installers/requirements/RequirementInstaller.js.map +1 -0
- package/dist/core/loaders/ConfigurationLoader.js.map +1 -0
- package/dist/core/loaders/ConfigurationProvider.js.map +1 -0
- package/dist/core/loaders/InstallOperationManager.js.map +1 -0
- package/dist/core/loaders/ServerSchemaLoader.js.map +1 -0
- package/dist/core/loaders/ServerSchemaProvider.js.map +1 -0
- package/dist/core/loaders/SystemSettingsManager.js.map +1 -0
- package/dist/core/metadatas/constants.js.map +1 -0
- package/dist/core/metadatas/recordingConstants.d.ts +2 -0
- package/dist/core/metadatas/recordingConstants.js +2 -0
- package/dist/core/metadatas/recordingConstants.js.map +1 -0
- package/dist/core/metadatas/types.d.ts +1 -1
- package/dist/core/metadatas/types.js.map +1 -0
- package/dist/core/onboard/FeedOnboardService.js +1 -22
- package/dist/core/onboard/FeedOnboardService.js.map +1 -0
- package/dist/core/onboard/OnboardProcessor.js.map +1 -0
- package/dist/core/onboard/OnboardStatus.js.map +1 -0
- package/dist/core/onboard/OnboardStatusManager.js.map +1 -0
- package/dist/core/types.d.ts +166 -0
- package/dist/core/types.js +16 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/validators/FeedValidator.js.map +1 -0
- package/dist/core/validators/IServerValidator.js.map +1 -0
- package/dist/core/validators/SSEServerValidator.js.map +1 -0
- package/dist/core/validators/ServerValidatorFactory.js.map +1 -0
- package/dist/core/validators/StdioServerValidator.js +5 -5
- package/dist/core/validators/StdioServerValidator.js.map +1 -0
- package/dist/index.js.map +1 -0
- package/dist/services/InstallRequestValidator.d.ts +21 -0
- package/dist/services/InstallRequestValidator.js +99 -0
- package/dist/services/InstallRequestValidator.js.map +1 -0
- package/dist/services/InstallationService.js.map +1 -0
- package/dist/services/MCPManager.js.map +1 -0
- package/dist/services/RequirementService.js.map +1 -0
- package/dist/services/ServerService.js.map +1 -0
- package/dist/services/TelemetryService.js.map +1 -0
- package/dist/utils/UpdateCheckTracker.js.map +1 -0
- package/dist/utils/adoUtils.js.map +1 -0
- package/dist/utils/clientUtils.js.map +1 -0
- package/dist/utils/feedUtils.js.map +1 -0
- package/dist/utils/githubAuth.js.map +1 -0
- package/dist/utils/githubUtils.js.map +1 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/macroExpressionUtils.js.map +1 -0
- package/dist/utils/osUtils.d.ts +11 -0
- package/dist/utils/osUtils.js +100 -0
- package/dist/utils/osUtils.js.map +1 -0
- package/dist/utils/versionUtils.js.map +1 -0
- package/dist/web/contract/serverContract.js.map +1 -0
- package/dist/web/public/index.html +1 -1
- package/dist/web/public/js/modal/installHandler.js +227 -0
- package/dist/web/public/js/modal/loadingUI.js +74 -0
- package/dist/web/public/js/modal/messageQueue.js +101 -45
- package/dist/web/public/js/modal/modalUI.js +214 -0
- package/{src/web/public/js/modal/versionUtils.js → dist/web/public/js/modal/version.js} +1 -1
- package/dist/web/public/js/onboard/templates.js +1 -0
- package/dist/web/public/js/serverCategoryList.js +3 -3
- package/dist/web/public/onboard.html +4 -4
- package/dist/web/server.js.map +1 -0
- package/package.json +5 -1
- package/.github/ISSUE_TEMPLATE/JitAccess.yml +0 -28
- package/.github/acl/access.yml +0 -20
- package/.github/compliance/inventory.yml +0 -5
- package/.github/policies/jit.yml +0 -19
- package/.roo/rules-code/rules.md +0 -88
- package/dist/core/onboard/InstallOperationManager.d.ts +0 -23
- package/dist/core/onboard/InstallOperationManager.js +0 -144
- package/docs/ONBOARDING_PAGE_DESIGN.md +0 -260
- package/docs/Telemetry.md +0 -136
- package/memory-bank/activeContext.md +0 -26
- package/memory-bank/decisionLog.md +0 -91
- package/memory-bank/productContext.md +0 -41
- package/memory-bank/progress.md +0 -35
- package/memory-bank/systemPatterns.md +0 -10
- package/src/cli/commands/install.ts +0 -139
- package/src/cli/commands/list.ts +0 -113
- package/src/cli/commands/pull.ts +0 -16
- package/src/cli/commands/serve.ts +0 -39
- package/src/cli/commands/uninstall.ts +0 -64
- package/src/cli/index.ts +0 -82
- package/src/core/installers/clients/BaseClientInstaller.ts +0 -341
- package/src/core/installers/clients/ClientInstaller.ts +0 -222
- package/src/core/installers/clients/ClientInstallerFactory.ts +0 -43
- package/src/core/installers/clients/ClineInstaller.ts +0 -35
- package/src/core/installers/clients/ExtensionInstaller.ts +0 -165
- package/src/core/installers/clients/GithubCopilotInstaller.ts +0 -79
- package/src/core/installers/clients/MSRooCodeInstaller.ts +0 -32
- package/src/core/installers/index.ts +0 -11
- package/src/core/installers/requirements/BaseInstaller.ts +0 -85
- package/src/core/installers/requirements/GeneralInstaller.ts +0 -133
- package/src/core/installers/requirements/NpmInstaller.ts +0 -271
- package/src/core/installers/requirements/PipInstaller.ts +0 -207
- package/src/core/loaders/InstallOperationManager.ts +0 -367
- package/src/core/loaders/ServerSchemaLoader.ts +0 -117
- package/src/core/loaders/ServerSchemaProvider.ts +0 -99
- package/src/core/loaders/SystemSettingsManager.ts +0 -278
- package/src/core/metadatas/constants.ts +0 -122
- package/src/core/metadatas/recordingConstants.ts +0 -62
- package/src/core/metadatas/types.ts +0 -202
- package/src/core/onboard/FeedOnboardService.ts +0 -524
- package/src/core/onboard/OnboardProcessor.ts +0 -356
- package/src/core/onboard/OnboardStatus.ts +0 -60
- package/src/core/onboard/OnboardStatusManager.ts +0 -416
- package/src/core/validators/FeedValidator.ts +0 -135
- package/src/core/validators/IServerValidator.ts +0 -21
- package/src/core/validators/SSEServerValidator.ts +0 -43
- package/src/core/validators/ServerValidatorFactory.ts +0 -51
- package/src/core/validators/StdioServerValidator.ts +0 -312
- package/src/index.ts +0 -44
- package/src/services/InstallationService.ts +0 -102
- package/src/services/MCPManager.ts +0 -249
- package/src/services/RequirementService.ts +0 -627
- package/src/services/ServerService.ts +0 -161
- package/src/services/TelemetryService.ts +0 -59
- package/src/utils/UpdateCheckTracker.ts +0 -86
- package/src/utils/adoUtils.ts +0 -293
- package/src/utils/clientUtils.ts +0 -72
- package/src/utils/feedUtils.ts +0 -31
- package/src/utils/githubAuth.ts +0 -212
- package/src/utils/githubUtils.ts +0 -164
- package/src/utils/logger.ts +0 -195
- package/src/utils/macroExpressionUtils.ts +0 -104
- package/src/utils/osUtils.ts +0 -597
- package/src/utils/versionUtils.ts +0 -114
- package/src/web/contract/serverContract.ts +0 -74
- package/src/web/public/css/detailsWidget.css +0 -235
- package/src/web/public/css/modal.css +0 -757
- package/src/web/public/css/notifications.css +0 -101
- package/src/web/public/css/onboard.css +0 -107
- package/src/web/public/css/serverCategoryList.css +0 -120
- package/src/web/public/css/serverDetails.css +0 -139
- package/src/web/public/index.html +0 -359
- package/src/web/public/js/api.js +0 -132
- package/src/web/public/js/detailsWidget.js +0 -264
- package/src/web/public/js/flights/flights.js +0 -127
- package/src/web/public/js/modal/index.js +0 -52
- package/src/web/public/js/modal/installModal.js +0 -162
- package/src/web/public/js/modal/installation.js +0 -266
- package/src/web/public/js/modal/loadingModal.js +0 -182
- package/src/web/public/js/modal/modalSetup.js +0 -595
- package/src/web/public/js/modal/modalUtils.js +0 -37
- package/src/web/public/js/modal.js +0 -42
- package/src/web/public/js/notifications.js +0 -137
- package/src/web/public/js/onboard/formProcessor.js +0 -1037
- package/src/web/public/js/onboard/index.js +0 -374
- package/src/web/public/js/onboard/publishHandler.js +0 -172
- package/src/web/public/js/onboard/state.js +0 -76
- package/src/web/public/js/onboard/templates.js +0 -341
- package/src/web/public/js/onboard/uiHandlers.js +0 -1076
- package/src/web/public/js/onboard/validationHandlers.js +0 -493
- package/src/web/public/js/serverCategoryDetails.js +0 -364
- package/src/web/public/js/serverCategoryList.js +0 -241
- package/src/web/public/js/settings.js +0 -314
- package/src/web/public/modal.html +0 -84
- package/src/web/public/onboard.html +0 -296
- package/src/web/public/settings.html +0 -135
- package/src/web/public/styles.css +0 -277
- package/src/web/server.ts +0 -478
- package/tsconfig.json +0 -18
|
@@ -1,271 +0,0 @@
|
|
|
1
|
-
import { RequirementConfig, RequirementStatus, ServerInstallOptions } from '../../metadatas/types.js';
|
|
2
|
-
import { BaseInstaller } from './BaseInstaller.js';
|
|
3
|
-
import { compareVersions } from '../../../utils/versionUtils.js';
|
|
4
|
-
import { handleGitHubRelease, getGitHubLatestVersion } from '../../../utils/githubUtils.js';
|
|
5
|
-
// Assuming getArtifactLatestVersion will be available in adoUtils.ts
|
|
6
|
-
import { handleArtifact as handleAdoArtifact, AdoArtifactResult, getArtifactLatestVersion } from '../../../utils/adoUtils.js';
|
|
7
|
-
import path from 'path';
|
|
8
|
-
import fs from 'fs/promises';
|
|
9
|
-
import { SETTINGS_DIR } from '../../metadatas/constants.js'; // Corrected path
|
|
10
|
-
import { Logger } from '../../../utils/logger.js';
|
|
11
|
-
import { InstallOperationManager } from '../../loaders/InstallOperationManager.js';
|
|
12
|
-
import * as RecordingConstants from '../../metadatas/recordingConstants.js';
|
|
13
|
-
|
|
14
|
-
/**
|
|
15
|
-
* Installer implementation for NPM packages
|
|
16
|
-
*/
|
|
17
|
-
export class NpmInstaller extends BaseInstaller {
|
|
18
|
-
/**
|
|
19
|
-
* Check if this installer can handle the given requirement type
|
|
20
|
-
* @param requirement The requirement to check
|
|
21
|
-
* @returns True if this installer can handle the requirement
|
|
22
|
-
*/
|
|
23
|
-
canHandle(requirement: RequirementConfig): boolean {
|
|
24
|
-
return requirement.type === 'npm';
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
supportCheckUpdates(): boolean {
|
|
28
|
-
return true;
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* Get the latest version available for the NPM package.
|
|
33
|
-
* @param requirement The requirement to check.
|
|
34
|
-
* @param options Optional server install options.
|
|
35
|
-
* @returns The latest version string, or undefined if not found or not applicable.
|
|
36
|
-
*/
|
|
37
|
-
async getLatestVersion(requirement: RequirementConfig, options?: ServerInstallOptions): Promise<string | undefined> {
|
|
38
|
-
if (requirement.registry) {
|
|
39
|
-
if (requirement.registry.githubRelease) {
|
|
40
|
-
return getGitHubLatestVersion(this.execPromise, requirement.registry.githubRelease.repository);
|
|
41
|
-
} else if (requirement.registry.artifacts) {
|
|
42
|
-
// Assuming getArtifactLatestVersion exists and has a compatible signature
|
|
43
|
-
// This might need adjustment based on the actual implementation of getArtifactLatestVersion
|
|
44
|
-
const targetDir = options?.settings?.folderName || this._getRequirementFolderPath(requirement);
|
|
45
|
-
return getArtifactLatestVersion(requirement, requirement.registry.artifacts, options, targetDir);
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
// Default: get common latest version from npm registry
|
|
49
|
-
const { stdout } = await this.execPromise(`npm view ${requirement.name} version`);
|
|
50
|
-
return stdout.trim();
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
/**
|
|
54
|
-
* Generates a dedicated folder path for a requirement.
|
|
55
|
-
* @param requirement The requirement configuration.
|
|
56
|
-
* @returns The path to the requirement's dedicated folder.
|
|
57
|
-
* @private
|
|
58
|
-
*/
|
|
59
|
-
private _getRequirementFolderPath(requirement: RequirementConfig): string {
|
|
60
|
-
return path.join(
|
|
61
|
-
SETTINGS_DIR,
|
|
62
|
-
'npm_requirements',
|
|
63
|
-
requirement.name,
|
|
64
|
-
requirement.version.includes('latest') ? 'latest' : requirement.version);
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* Check if the NPM package is already installed in its dedicated folder.
|
|
69
|
-
* @param requirement The requirement to check
|
|
70
|
-
* @param options Installation options, may contain folderName.
|
|
71
|
-
* @returns The status of the requirement
|
|
72
|
-
*/
|
|
73
|
-
async checkInstallation(requirement: RequirementConfig, options?: ServerInstallOptions): Promise<RequirementStatus> {
|
|
74
|
-
const requirementDir = options?.settings?.folderName || this._getRequirementFolderPath(requirement);
|
|
75
|
-
const requirementName = this._getRequirementName(requirement);
|
|
76
|
-
Logger.debug(`Checking installation for ${requirementName} in ${requirementDir}`);
|
|
77
|
-
try {
|
|
78
|
-
const installedVersion = await this._getInstalledVersion(requirementName, requirementDir);
|
|
79
|
-
|
|
80
|
-
if (installedVersion) {
|
|
81
|
-
return {
|
|
82
|
-
name: requirement.name,
|
|
83
|
-
type: 'npm',
|
|
84
|
-
installed: true,
|
|
85
|
-
version: installedVersion,
|
|
86
|
-
npmPath: requirementDir,
|
|
87
|
-
inProgress: false,
|
|
88
|
-
};
|
|
89
|
-
}
|
|
90
|
-
// If package not found in dependencies or version is missing
|
|
91
|
-
return {
|
|
92
|
-
name: requirement.name,
|
|
93
|
-
type: 'npm',
|
|
94
|
-
installed: false,
|
|
95
|
-
inProgress: false,
|
|
96
|
-
error: `Package ${requirement.name} not found or version missing in npm list output in ${requirementDir}.`,
|
|
97
|
-
};
|
|
98
|
-
} catch (error) {
|
|
99
|
-
// npm list command likely failed (e.g., package not installed, or requirementDir not an npm project)
|
|
100
|
-
Logger.debug(`Error checking installation for ${requirement.name} in ${requirementDir}: ${error instanceof Error ? error.message : String(error)}`);
|
|
101
|
-
return {
|
|
102
|
-
name: requirement.name,
|
|
103
|
-
type: 'npm',
|
|
104
|
-
installed: false,
|
|
105
|
-
inProgress: false,
|
|
106
|
-
error: error instanceof Error ? error.message : String(error),
|
|
107
|
-
};
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
/**
|
|
112
|
-
* Retrieves the installed version of an NPM package from a given directory.
|
|
113
|
-
* @param requirementName The name of the NPM package.
|
|
114
|
-
* @param directory The directory to check for the package.
|
|
115
|
-
* @returns The installed version string, or undefined if not found or an error occurs.
|
|
116
|
-
* @private
|
|
117
|
-
*/
|
|
118
|
-
private async _getInstalledVersion(requirementName: string, directory: string): Promise<string | undefined> {
|
|
119
|
-
try {
|
|
120
|
-
const command = `npm list ${requirementName} --depth=0 --json --prefix "${directory}"`;
|
|
121
|
-
Logger.debug(`Getting installed version for ${requirementName} in ${directory} with command: ${command}`);
|
|
122
|
-
const { stdout } = await this.execPromise(command);
|
|
123
|
-
const listOutput = JSON.parse(stdout);
|
|
124
|
-
|
|
125
|
-
if (listOutput.dependencies && listOutput.dependencies[requirementName] && listOutput.dependencies[requirementName].version) {
|
|
126
|
-
return listOutput.dependencies[requirementName].version;
|
|
127
|
-
}
|
|
128
|
-
Logger.debug(`Package ${requirementName} not found in npm list output in ${directory}.`);
|
|
129
|
-
return undefined;
|
|
130
|
-
} catch (error) {
|
|
131
|
-
Logger.debug(`Error getting installed version for ${requirementName} in ${directory}: ${error instanceof Error ? error.message : String(error)}`);
|
|
132
|
-
return undefined; // Return undefined on error to indicate version couldn't be retrieved
|
|
133
|
-
}
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
/**
|
|
138
|
-
* Get the name of the requirement, including registry information if applicable.
|
|
139
|
-
* @param requirement The requirement configuration.
|
|
140
|
-
* @returns The formatted requirement name.
|
|
141
|
-
* @private
|
|
142
|
-
*/
|
|
143
|
-
private _getRequirementName(requirement: RequirementConfig): string {
|
|
144
|
-
return requirement.registry?.artifacts?.registryName ? `@${requirement.registry.artifacts.registryName}/${requirement.name}` : requirement.name;
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
/**
|
|
149
|
-
* Installs an NPM package into a dedicated local folder.
|
|
150
|
-
* @param requirement The requirement to install.
|
|
151
|
-
* @param packageSource This can be name@version, path to tgz, or path to folder.
|
|
152
|
-
* @param targetDir Target directory for installation.
|
|
153
|
-
* @returns The installed version of the package.
|
|
154
|
-
*/
|
|
155
|
-
private async _installPackage(
|
|
156
|
-
requirement: RequirementConfig,
|
|
157
|
-
packageSource: string,
|
|
158
|
-
targetDir: string,
|
|
159
|
-
recorder: InstallOperationManager
|
|
160
|
-
): Promise<{ version: string }> {
|
|
161
|
-
Logger.debug(`Installing NPM package from "${packageSource}" into "${targetDir}"`);
|
|
162
|
-
await fs.mkdir(targetDir, { recursive: true });
|
|
163
|
-
|
|
164
|
-
const installCommand = `npm install ${packageSource} --prefix "${targetDir}"`;
|
|
165
|
-
Logger.debug(`Executing install command: ${installCommand}`);
|
|
166
|
-
const requirementName = this._getRequirementName(requirement);
|
|
167
|
-
|
|
168
|
-
return await recorder.recording(
|
|
169
|
-
async () => {
|
|
170
|
-
const { stdout: installStdout, stderr: installStderr } = await this.execPromise(installCommand);
|
|
171
|
-
Logger.debug(`NPM install stdout for ${packageSource} in ${targetDir}: ${installStdout}`);
|
|
172
|
-
if (installStderr && !installStderr.toLowerCase().includes('added') && !installStderr.toLowerCase().includes('updated') && !installStderr.toLowerCase().includes('found 0 vulnerabilities')) {
|
|
173
|
-
// Log stderr if it's not just typical success noise
|
|
174
|
-
Logger.log(`NPM install stderr for ${packageSource} in ${targetDir}: ${installStderr}`);
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
const installedVersion = await this._getInstalledVersion(requirementName, targetDir);
|
|
179
|
-
|
|
180
|
-
if (installedVersion) {
|
|
181
|
-
Logger.log(`Successfully installed and verified ${requirementName}@${installedVersion} into ${targetDir}`);
|
|
182
|
-
return { version: installedVersion };
|
|
183
|
-
} else {
|
|
184
|
-
throw new Error(`Successfully ran npm install for ${packageSource}, but ${requirement.name} version could not be determined via npm list in ${targetDir}, stderr: ${installStderr}`);
|
|
185
|
-
}
|
|
186
|
-
},
|
|
187
|
-
{
|
|
188
|
-
stepName: RecordingConstants.STEP_INSTALLATION_COMMAND_EXECUTION,
|
|
189
|
-
inProgressMessage: `Running: ${installCommand}`,
|
|
190
|
-
onError: (error) => {
|
|
191
|
-
Logger.error(`Error during NPM installation: ${error instanceof Error ? error.message : String(error)}`);
|
|
192
|
-
throw error;
|
|
193
|
-
}
|
|
194
|
-
}
|
|
195
|
-
)
|
|
196
|
-
}
|
|
197
|
-
|
|
198
|
-
/**
|
|
199
|
-
* Install the NPM package.
|
|
200
|
-
* @param requirement The requirement to install.
|
|
201
|
-
* @param recorder Optional InstallOperationManager for recording steps.
|
|
202
|
-
* @param options Installation options.
|
|
203
|
-
* @returns The status of the installation.
|
|
204
|
-
*/
|
|
205
|
-
async install(requirement: RequirementConfig, recorder: InstallOperationManager, options?: ServerInstallOptions): Promise<RequirementStatus> {
|
|
206
|
-
const targetDir = options?.settings?.folderName || this._getRequirementFolderPath(requirement);
|
|
207
|
-
await fs.mkdir(targetDir, { recursive: true });
|
|
208
|
-
return recorder.recording(
|
|
209
|
-
async (): Promise<RequirementStatus> => {
|
|
210
|
-
const status = await this.checkInstallation(requirement, { settings: { folderName: targetDir } });
|
|
211
|
-
if (status.installed && status.version && compareVersions(status.version, requirement.version) === 0 && !requirement.version.toLowerCase().includes('latest')) {
|
|
212
|
-
Logger.log(`${requirement.name}@${status.version} already installed in ${targetDir}.`);
|
|
213
|
-
return status;
|
|
214
|
-
}
|
|
215
|
-
|
|
216
|
-
let resolvedVersion = requirement.version;
|
|
217
|
-
let packageToInstall: string = `${requirement.name}@${requirement.version}`;
|
|
218
|
-
|
|
219
|
-
if (requirement.registry) {
|
|
220
|
-
if (requirement.registry.githubRelease) {
|
|
221
|
-
const result = await handleGitHubRelease(requirement, requirement.registry.githubRelease);
|
|
222
|
-
packageToInstall = result.resolvedPath;
|
|
223
|
-
resolvedVersion = result.resolvedVersion;
|
|
224
|
-
} else if (requirement.registry.artifacts) {
|
|
225
|
-
const adoResult: AdoArtifactResult = await handleAdoArtifact(
|
|
226
|
-
requirement,
|
|
227
|
-
requirement.registry.artifacts,
|
|
228
|
-
options?.settings?.pythonCommand,
|
|
229
|
-
targetDir
|
|
230
|
-
);
|
|
231
|
-
packageToInstall = `${adoResult.package}@${adoResult.version}`;
|
|
232
|
-
resolvedVersion = adoResult.version;
|
|
233
|
-
} else {
|
|
234
|
-
if (recorder) await recorder.recordStep('NpmInstaller:RegistryConfig', 'failed', 'Invalid registry configuration for npm.');
|
|
235
|
-
throw new Error('Invalid registry configuration for npm.');
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
const finalInstallResult = await this._installPackage(requirement, packageToInstall, targetDir, recorder);
|
|
239
|
-
resolvedVersion = finalInstallResult.version;
|
|
240
|
-
|
|
241
|
-
return {
|
|
242
|
-
name: requirement.name,
|
|
243
|
-
type: 'npm',
|
|
244
|
-
installed: true,
|
|
245
|
-
version: resolvedVersion,
|
|
246
|
-
inProgress: false,
|
|
247
|
-
npmPath: targetDir
|
|
248
|
-
};
|
|
249
|
-
},
|
|
250
|
-
{
|
|
251
|
-
stepName: RecordingConstants.STEP_NPM_INSTALLER_INSTALL,
|
|
252
|
-
inProgressMessage: `Installing npm package: ${requirement.name}`,
|
|
253
|
-
endMessage: (result) =>
|
|
254
|
-
result.installed
|
|
255
|
-
? `Install completed for ${requirement.name}`
|
|
256
|
-
: `Install failed for ${requirement.name}`,
|
|
257
|
-
onError: (error) => {
|
|
258
|
-
return {
|
|
259
|
-
result: {
|
|
260
|
-
name: requirement.name,
|
|
261
|
-
type: 'npm',
|
|
262
|
-
installed: false,
|
|
263
|
-
error: error instanceof Error ? error.message : String(error),
|
|
264
|
-
inProgress: false
|
|
265
|
-
},
|
|
266
|
-
message: error instanceof Error ? error.message : String(error),
|
|
267
|
-
};
|
|
268
|
-
},
|
|
269
|
-
});
|
|
270
|
-
}
|
|
271
|
-
}
|
|
@@ -1,207 +0,0 @@
|
|
|
1
|
-
import { RequirementConfig, RequirementStatus, ServerInstallOptions } from '../../metadatas/types.js';
|
|
2
|
-
import { BaseInstaller } from './BaseInstaller.js';
|
|
3
|
-
import { handleGitHubRelease, getGitHubLatestVersion } from '../../../utils/githubUtils.js';
|
|
4
|
-
import { handleArtifact as handleAdoArtifact, getArtifactLatestVersion } from '../../../utils/adoUtils.js';
|
|
5
|
-
import { compareVersions } from '../../../utils/versionUtils.js';
|
|
6
|
-
import { Logger } from '../../../utils/logger.js';
|
|
7
|
-
import { InstallOperationManager } from '../../loaders/InstallOperationManager.js';
|
|
8
|
-
import * as RecordingConstants from '../../metadatas/recordingConstants.js';
|
|
9
|
-
/**
|
|
10
|
-
* Installer implementation for Python packages using pip
|
|
11
|
-
*/
|
|
12
|
-
export class PipInstaller extends BaseInstaller {
|
|
13
|
-
private getPythonCommand(options?: ServerInstallOptions): string {
|
|
14
|
-
return options?.settings?.pythonEnv as string || 'python';
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
private getPipCommand(options?: ServerInstallOptions): string {
|
|
18
|
-
const pythonCmd = this.getPythonCommand(options);
|
|
19
|
-
return `${pythonCmd} -m pip`;
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
/**
|
|
23
|
-
* Check if this installer can handle the given requirement type
|
|
24
|
-
* @param requirement The requirement to check
|
|
25
|
-
* @returns True if this installer can handle the requirement
|
|
26
|
-
*/
|
|
27
|
-
canHandle(requirement: RequirementConfig): boolean {
|
|
28
|
-
return requirement.type === 'pip';
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
supportCheckUpdates(): boolean {
|
|
32
|
-
return true;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
/**
|
|
36
|
-
* Get the latest version available for the pip package.
|
|
37
|
-
* @param requirement The requirement to check.
|
|
38
|
-
* @param options Optional server install options.
|
|
39
|
-
* @returns The latest version string, or undefined if not found or not applicable.
|
|
40
|
-
*/
|
|
41
|
-
async getLatestVersion(requirement: RequirementConfig, options?: ServerInstallOptions): Promise<string | undefined> {
|
|
42
|
-
if (requirement.registry) {
|
|
43
|
-
if (requirement.registry.githubRelease) {
|
|
44
|
-
return getGitHubLatestVersion(this.execPromise, requirement.registry.githubRelease.repository);
|
|
45
|
-
} else if (requirement.registry.artifacts) {
|
|
46
|
-
// Assuming getArtifactLatestVersion exists and has a compatible signature
|
|
47
|
-
return getArtifactLatestVersion(requirement, requirement.registry.artifacts, options);
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
// Default: get common latest version from pip index
|
|
51
|
-
const pipCmd = this.getPipCommand(options);
|
|
52
|
-
const { stdout } = await this.execPromise(`${pipCmd} index versions ${requirement.name} --pre=0`);
|
|
53
|
-
// Parse output to find the latest version. Example output:
|
|
54
|
-
// mypackage (1.0.0)
|
|
55
|
-
// Available versions: 1.0.0, 0.9.0
|
|
56
|
-
// LATEST: 1.0.0
|
|
57
|
-
// Or for some packages:
|
|
58
|
-
// mypackage
|
|
59
|
-
// VERSIONS: 1.0.0, 0.9.0
|
|
60
|
-
// Latest: 1.0.0
|
|
61
|
-
const latestMatch = stdout.match(/(?:LATEST|Latest):\s*([^\s]+)/);
|
|
62
|
-
if (latestMatch && latestMatch[1]) {
|
|
63
|
-
return latestMatch[1];
|
|
64
|
-
}
|
|
65
|
-
// Fallback if LATEST line is not found, try to get the first version from "Available versions" or "VERSIONS"
|
|
66
|
-
const versionsMatch = stdout.match(/(?:Available versions|VERSIONS):\s*([^\n]+)/);
|
|
67
|
-
if (versionsMatch && versionsMatch[1]) {
|
|
68
|
-
const versions = versionsMatch[1].split(',').map(v => v.trim());
|
|
69
|
-
if (versions.length > 0) {
|
|
70
|
-
// Assuming versions are listed in a somewhat reasonable order,
|
|
71
|
-
// or we might need more sophisticated version sorting here.
|
|
72
|
-
return versions[0];
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
return undefined; // Or throw an error if version cannot be determined
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
/**
|
|
79
|
-
* Check if the Python package is already installed
|
|
80
|
-
* @param requirement The requirement to check
|
|
81
|
-
* @param options Optional server install options
|
|
82
|
-
* @returns The status of the requirement
|
|
83
|
-
*/
|
|
84
|
-
async checkInstallation(requirement: RequirementConfig, options?: ServerInstallOptions): Promise<RequirementStatus> {
|
|
85
|
-
try {
|
|
86
|
-
const pipCmd = this.getPipCommand(options);
|
|
87
|
-
const { stdout, stderr } = await this.execPromise(`${pipCmd} show ${requirement.name}`);
|
|
88
|
-
|
|
89
|
-
const installed = stdout.includes(requirement.name.toLowerCase());
|
|
90
|
-
const versionMatch = stdout.match(/Version: (.+)/);
|
|
91
|
-
const installedVersion = versionMatch ? versionMatch[1] : undefined;
|
|
92
|
-
|
|
93
|
-
return {
|
|
94
|
-
name: requirement.name,
|
|
95
|
-
type: 'pip',
|
|
96
|
-
installed: installed,
|
|
97
|
-
version: installedVersion,
|
|
98
|
-
inProgress: false,
|
|
99
|
-
pythonEnv: this.getPythonCommand(options)
|
|
100
|
-
};
|
|
101
|
-
} catch (error) {
|
|
102
|
-
return {
|
|
103
|
-
name: requirement.name,
|
|
104
|
-
type: 'pip',
|
|
105
|
-
installed: false,
|
|
106
|
-
error: error instanceof Error ? error.message : String(error),
|
|
107
|
-
inProgress: false,
|
|
108
|
-
pythonEnv: this.getPythonCommand(options)
|
|
109
|
-
};
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
/**
|
|
114
|
-
* Install the Python package
|
|
115
|
-
* @param requirement The requirement to install
|
|
116
|
-
* @param recorder Optional InstallOperationManager for recording steps
|
|
117
|
-
* @param options Optional server install options
|
|
118
|
-
* @returns The status of the installation
|
|
119
|
-
*/
|
|
120
|
-
async install(requirement: RequirementConfig, recorder: InstallOperationManager, options?: ServerInstallOptions): Promise<RequirementStatus> {
|
|
121
|
-
|
|
122
|
-
return await recorder.recording(
|
|
123
|
-
async (): Promise<RequirementStatus> => {
|
|
124
|
-
const status = await this.checkInstallation(requirement, options);
|
|
125
|
-
if (status.installed && status.version && compareVersions(status.version, requirement.version) === 0 && !requirement.version.toLowerCase().includes('latest')) {
|
|
126
|
-
Logger.log(`${requirement.name}==${status.version} already installed for ${this.getPythonCommand}.`);
|
|
127
|
-
return status;
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
const pipCmd = this.getPipCommand(options);
|
|
131
|
-
let command: string
|
|
132
|
-
|
|
133
|
-
if (!requirement.registry) {
|
|
134
|
-
if (requirement.version && !requirement.version.includes('latest')) {
|
|
135
|
-
command = `${pipCmd} install ${requirement.name}==${requirement.version}`;
|
|
136
|
-
} else {
|
|
137
|
-
command = `${pipCmd} install --upgrade ${requirement.name}`;
|
|
138
|
-
}
|
|
139
|
-
} else {
|
|
140
|
-
let packageSource: string;
|
|
141
|
-
if (requirement.registry.githubRelease) {
|
|
142
|
-
const result = await handleGitHubRelease(requirement, requirement.registry.githubRelease);
|
|
143
|
-
packageSource = result.resolvedPath;
|
|
144
|
-
command = `${pipCmd} install "${packageSource}"`
|
|
145
|
-
} else if (requirement.registry.artifacts) {
|
|
146
|
-
const pythonCmd = this.getPythonCommand(options);
|
|
147
|
-
const adoArtifactResult = await handleAdoArtifact(requirement, requirement.registry.artifacts, pythonCmd);
|
|
148
|
-
|
|
149
|
-
command = `${pipCmd} install ${adoArtifactResult.package} --extra-index-url ${adoArtifactResult.registryUrl}`;
|
|
150
|
-
} else {
|
|
151
|
-
await recorder.recordStep('PipInstaller:RegistryConfig', 'failed', 'Invalid registry configuration');
|
|
152
|
-
throw new Error('Invalid registry configuration');
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
return await recorder.recording(
|
|
156
|
-
async () => {
|
|
157
|
-
const { stderr } = await this.execPromise(command);
|
|
158
|
-
if (stderr && stderr.toLowerCase().includes('error')) {
|
|
159
|
-
Logger.debug(`Pip installation error: ${stderr}`);
|
|
160
|
-
|
|
161
|
-
// wait for 5 seconds as python pip would be little delayed
|
|
162
|
-
await new Promise(resolve => setTimeout(resolve, 5000));
|
|
163
|
-
const checkStatus = await this.checkInstallation(requirement, options);
|
|
164
|
-
if (!checkStatus.installed) {
|
|
165
|
-
Logger.error(`Package not found after the command, ${stderr}`);
|
|
166
|
-
throw new Error(`Pip installation failed with: ${stderr}`);
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
return {
|
|
171
|
-
name: requirement.name,
|
|
172
|
-
type: 'pip',
|
|
173
|
-
installed: true,
|
|
174
|
-
version: requirement.version, // This might need to be updated to actual installed version
|
|
175
|
-
inProgress: false,
|
|
176
|
-
pythonEnv: this.getPythonCommand(options)
|
|
177
|
-
};
|
|
178
|
-
},
|
|
179
|
-
{
|
|
180
|
-
stepName: `${RecordingConstants.STEP_INSTALL_COMMAND_PREFIX}: ${requirement.name} : ${requirement.version}`,
|
|
181
|
-
inProgressMessage: `Running: ${command}`,
|
|
182
|
-
endMessage: (result) => result.installed ? `Succeeded: ${command}` : `Failed: ${command}`,
|
|
183
|
-
}
|
|
184
|
-
);
|
|
185
|
-
},
|
|
186
|
-
{
|
|
187
|
-
stepName: RecordingConstants.STEP_PIP_INSTALLER_INSTALL,
|
|
188
|
-
inProgressMessage: `Installing pip package: ${requirement.name}`,
|
|
189
|
-
endMessage: (result) => result.installed
|
|
190
|
-
? `Install completed for ${requirement.name} with version ${result.version}`
|
|
191
|
-
: `Install failed for ${requirement.name}`,
|
|
192
|
-
onError: (error) => {
|
|
193
|
-
return {
|
|
194
|
-
result: {
|
|
195
|
-
name: requirement.name,
|
|
196
|
-
type: 'pip',
|
|
197
|
-
installed: false,
|
|
198
|
-
error: error instanceof Error ? error.message : String(error),
|
|
199
|
-
inProgress: false,
|
|
200
|
-
pythonEnv: this.getPythonCommand(options)
|
|
201
|
-
},
|
|
202
|
-
message: error instanceof Error ? error.message : String(error),
|
|
203
|
-
};
|
|
204
|
-
},
|
|
205
|
-
});
|
|
206
|
-
}
|
|
207
|
-
}
|