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.
Files changed (239) hide show
  1. package/README.md +21 -4
  2. package/dist/cli/commands/install.js.map +1 -0
  3. package/dist/cli/commands/list.js.map +1 -0
  4. package/dist/cli/commands/pull.js.map +1 -0
  5. package/dist/cli/commands/serve.js.map +1 -0
  6. package/dist/cli/commands/start.d.ts +2 -0
  7. package/dist/cli/commands/start.js +32 -0
  8. package/dist/cli/commands/start.js.map +1 -0
  9. package/dist/cli/commands/sync.d.ts +2 -0
  10. package/dist/cli/commands/sync.js +17 -0
  11. package/dist/cli/commands/sync.js.map +1 -0
  12. package/dist/cli/commands/uninstall.js.map +1 -0
  13. package/dist/cli/index.js +0 -0
  14. package/dist/cli/index.js.map +1 -0
  15. package/dist/core/ConfigurationLoader.d.ts +32 -0
  16. package/{src/core/loaders/ConfigurationLoader.ts → dist/core/ConfigurationLoader.js} +236 -298
  17. package/dist/core/ConfigurationLoader.js.map +1 -0
  18. package/dist/core/ConfigurationProvider.d.ts +35 -0
  19. package/{src/core/loaders/ConfigurationProvider.ts → dist/core/ConfigurationProvider.js} +375 -462
  20. package/dist/core/ConfigurationProvider.js.map +1 -0
  21. package/dist/core/InstallationService.d.ts +50 -0
  22. package/dist/core/InstallationService.js +350 -0
  23. package/dist/core/InstallationService.js.map +1 -0
  24. package/dist/core/MCPManager.d.ts +28 -0
  25. package/dist/core/MCPManager.js +188 -0
  26. package/dist/core/MCPManager.js.map +1 -0
  27. package/dist/core/RequirementService.d.ts +40 -0
  28. package/dist/core/RequirementService.js +110 -0
  29. package/dist/core/RequirementService.js.map +1 -0
  30. package/dist/core/ServerSchemaLoader.d.ts +11 -0
  31. package/dist/core/ServerSchemaLoader.js +43 -0
  32. package/dist/core/ServerSchemaLoader.js.map +1 -0
  33. package/dist/core/ServerSchemaProvider.d.ts +17 -0
  34. package/dist/core/ServerSchemaProvider.js +120 -0
  35. package/dist/core/ServerSchemaProvider.js.map +1 -0
  36. package/dist/core/constants.d.ts +47 -0
  37. package/dist/core/constants.js +94 -0
  38. package/dist/core/constants.js.map +1 -0
  39. package/dist/core/installers/BaseInstaller.d.ts +74 -0
  40. package/dist/core/installers/BaseInstaller.js +253 -0
  41. package/dist/core/installers/BaseInstaller.js.map +1 -0
  42. package/dist/core/installers/ClientInstaller.d.ts +23 -0
  43. package/dist/core/installers/ClientInstaller.js +564 -0
  44. package/dist/core/installers/ClientInstaller.js.map +1 -0
  45. package/dist/core/installers/CommandInstaller.d.ts +37 -0
  46. package/{src/core/installers/requirements/CommandInstaller.ts → dist/core/installers/CommandInstaller.js} +173 -231
  47. package/dist/core/installers/CommandInstaller.js.map +1 -0
  48. package/dist/core/installers/GeneralInstaller.d.ts +33 -0
  49. package/dist/core/installers/GeneralInstaller.js +85 -0
  50. package/dist/core/installers/GeneralInstaller.js.map +1 -0
  51. package/dist/core/installers/InstallerFactory.d.ts +54 -0
  52. package/{src/core/installers/requirements/InstallerFactory.ts → dist/core/installers/InstallerFactory.js} +97 -112
  53. package/dist/core/installers/InstallerFactory.js.map +1 -0
  54. package/dist/core/installers/NpmInstaller.d.ts +26 -0
  55. package/dist/core/installers/NpmInstaller.js +127 -0
  56. package/dist/core/installers/NpmInstaller.js.map +1 -0
  57. package/dist/core/installers/PipInstaller.d.ts +28 -0
  58. package/dist/core/installers/PipInstaller.js +127 -0
  59. package/dist/core/installers/PipInstaller.js.map +1 -0
  60. package/{src/core/installers/requirements/RequirementInstaller.ts → dist/core/installers/RequirementInstaller.d.ts} +33 -42
  61. package/dist/core/installers/RequirementInstaller.js +3 -0
  62. package/dist/core/installers/RequirementInstaller.js.map +1 -0
  63. package/dist/core/installers/clients/BaseClientInstaller.js.map +1 -0
  64. package/dist/core/installers/clients/ClientInstaller.js.map +1 -0
  65. package/dist/core/installers/clients/ClientInstallerFactory.js.map +1 -0
  66. package/dist/core/installers/clients/ClineInstaller.js.map +1 -0
  67. package/dist/core/installers/clients/ExtensionInstaller.js.map +1 -0
  68. package/dist/core/installers/clients/GithubCopilotInstaller.js.map +1 -0
  69. package/dist/core/installers/clients/MSRooCodeInstaller.js.map +1 -0
  70. package/dist/core/installers/index.js.map +1 -0
  71. package/dist/core/installers/requirements/BaseInstaller.js.map +1 -0
  72. package/dist/core/installers/requirements/CommandInstaller.js.map +1 -0
  73. package/dist/core/installers/requirements/GeneralInstaller.js.map +1 -0
  74. package/dist/core/installers/requirements/InstallerFactory.js +2 -0
  75. package/dist/core/installers/requirements/InstallerFactory.js.map +1 -0
  76. package/dist/core/installers/requirements/NpmInstaller.js.map +1 -0
  77. package/dist/core/installers/requirements/NugetInstaller.d.ts +37 -0
  78. package/dist/core/installers/requirements/NugetInstaller.js +189 -0
  79. package/dist/core/installers/requirements/NugetInstaller.js.map +1 -0
  80. package/dist/core/installers/requirements/PipInstaller.js.map +1 -0
  81. package/dist/core/installers/requirements/RequirementInstaller.js.map +1 -0
  82. package/dist/core/loaders/ConfigurationLoader.js.map +1 -0
  83. package/dist/core/loaders/ConfigurationProvider.js.map +1 -0
  84. package/dist/core/loaders/InstallOperationManager.js.map +1 -0
  85. package/dist/core/loaders/ServerSchemaLoader.js.map +1 -0
  86. package/dist/core/loaders/ServerSchemaProvider.js.map +1 -0
  87. package/dist/core/loaders/SystemSettingsManager.js.map +1 -0
  88. package/dist/core/metadatas/constants.js.map +1 -0
  89. package/dist/core/metadatas/recordingConstants.d.ts +2 -0
  90. package/dist/core/metadatas/recordingConstants.js +2 -0
  91. package/dist/core/metadatas/recordingConstants.js.map +1 -0
  92. package/dist/core/metadatas/types.d.ts +1 -1
  93. package/dist/core/metadatas/types.js.map +1 -0
  94. package/dist/core/onboard/FeedOnboardService.js +1 -22
  95. package/dist/core/onboard/FeedOnboardService.js.map +1 -0
  96. package/dist/core/onboard/OnboardProcessor.js.map +1 -0
  97. package/dist/core/onboard/OnboardStatus.js.map +1 -0
  98. package/dist/core/onboard/OnboardStatusManager.js.map +1 -0
  99. package/dist/core/types.d.ts +166 -0
  100. package/dist/core/types.js +16 -0
  101. package/dist/core/types.js.map +1 -0
  102. package/dist/core/validators/FeedValidator.js.map +1 -0
  103. package/dist/core/validators/IServerValidator.js.map +1 -0
  104. package/dist/core/validators/SSEServerValidator.js.map +1 -0
  105. package/dist/core/validators/ServerValidatorFactory.js.map +1 -0
  106. package/dist/core/validators/StdioServerValidator.js +5 -5
  107. package/dist/core/validators/StdioServerValidator.js.map +1 -0
  108. package/dist/index.js.map +1 -0
  109. package/dist/services/InstallRequestValidator.d.ts +21 -0
  110. package/dist/services/InstallRequestValidator.js +99 -0
  111. package/dist/services/InstallRequestValidator.js.map +1 -0
  112. package/dist/services/InstallationService.js.map +1 -0
  113. package/dist/services/MCPManager.js.map +1 -0
  114. package/dist/services/RequirementService.js.map +1 -0
  115. package/dist/services/ServerService.js.map +1 -0
  116. package/dist/services/TelemetryService.js.map +1 -0
  117. package/dist/utils/UpdateCheckTracker.js.map +1 -0
  118. package/dist/utils/adoUtils.js.map +1 -0
  119. package/dist/utils/clientUtils.js.map +1 -0
  120. package/dist/utils/feedUtils.js.map +1 -0
  121. package/dist/utils/githubAuth.js.map +1 -0
  122. package/dist/utils/githubUtils.js.map +1 -0
  123. package/dist/utils/logger.js.map +1 -0
  124. package/dist/utils/macroExpressionUtils.js.map +1 -0
  125. package/dist/utils/osUtils.d.ts +11 -0
  126. package/dist/utils/osUtils.js +100 -0
  127. package/dist/utils/osUtils.js.map +1 -0
  128. package/dist/utils/versionUtils.js.map +1 -0
  129. package/dist/web/contract/serverContract.js.map +1 -0
  130. package/dist/web/public/index.html +1 -1
  131. package/dist/web/public/js/modal/installHandler.js +227 -0
  132. package/dist/web/public/js/modal/loadingUI.js +74 -0
  133. package/dist/web/public/js/modal/messageQueue.js +101 -45
  134. package/dist/web/public/js/modal/modalUI.js +214 -0
  135. package/{src/web/public/js/modal/versionUtils.js → dist/web/public/js/modal/version.js} +1 -1
  136. package/dist/web/public/js/onboard/templates.js +1 -0
  137. package/dist/web/public/js/serverCategoryList.js +3 -3
  138. package/dist/web/public/onboard.html +4 -4
  139. package/dist/web/server.js.map +1 -0
  140. package/package.json +5 -1
  141. package/.github/ISSUE_TEMPLATE/JitAccess.yml +0 -28
  142. package/.github/acl/access.yml +0 -20
  143. package/.github/compliance/inventory.yml +0 -5
  144. package/.github/policies/jit.yml +0 -19
  145. package/.roo/rules-code/rules.md +0 -88
  146. package/dist/core/onboard/InstallOperationManager.d.ts +0 -23
  147. package/dist/core/onboard/InstallOperationManager.js +0 -144
  148. package/docs/ONBOARDING_PAGE_DESIGN.md +0 -260
  149. package/docs/Telemetry.md +0 -136
  150. package/memory-bank/activeContext.md +0 -26
  151. package/memory-bank/decisionLog.md +0 -91
  152. package/memory-bank/productContext.md +0 -41
  153. package/memory-bank/progress.md +0 -35
  154. package/memory-bank/systemPatterns.md +0 -10
  155. package/src/cli/commands/install.ts +0 -139
  156. package/src/cli/commands/list.ts +0 -113
  157. package/src/cli/commands/pull.ts +0 -16
  158. package/src/cli/commands/serve.ts +0 -39
  159. package/src/cli/commands/uninstall.ts +0 -64
  160. package/src/cli/index.ts +0 -82
  161. package/src/core/installers/clients/BaseClientInstaller.ts +0 -341
  162. package/src/core/installers/clients/ClientInstaller.ts +0 -222
  163. package/src/core/installers/clients/ClientInstallerFactory.ts +0 -43
  164. package/src/core/installers/clients/ClineInstaller.ts +0 -35
  165. package/src/core/installers/clients/ExtensionInstaller.ts +0 -165
  166. package/src/core/installers/clients/GithubCopilotInstaller.ts +0 -79
  167. package/src/core/installers/clients/MSRooCodeInstaller.ts +0 -32
  168. package/src/core/installers/index.ts +0 -11
  169. package/src/core/installers/requirements/BaseInstaller.ts +0 -85
  170. package/src/core/installers/requirements/GeneralInstaller.ts +0 -133
  171. package/src/core/installers/requirements/NpmInstaller.ts +0 -271
  172. package/src/core/installers/requirements/PipInstaller.ts +0 -207
  173. package/src/core/loaders/InstallOperationManager.ts +0 -367
  174. package/src/core/loaders/ServerSchemaLoader.ts +0 -117
  175. package/src/core/loaders/ServerSchemaProvider.ts +0 -99
  176. package/src/core/loaders/SystemSettingsManager.ts +0 -278
  177. package/src/core/metadatas/constants.ts +0 -122
  178. package/src/core/metadatas/recordingConstants.ts +0 -62
  179. package/src/core/metadatas/types.ts +0 -202
  180. package/src/core/onboard/FeedOnboardService.ts +0 -524
  181. package/src/core/onboard/OnboardProcessor.ts +0 -356
  182. package/src/core/onboard/OnboardStatus.ts +0 -60
  183. package/src/core/onboard/OnboardStatusManager.ts +0 -416
  184. package/src/core/validators/FeedValidator.ts +0 -135
  185. package/src/core/validators/IServerValidator.ts +0 -21
  186. package/src/core/validators/SSEServerValidator.ts +0 -43
  187. package/src/core/validators/ServerValidatorFactory.ts +0 -51
  188. package/src/core/validators/StdioServerValidator.ts +0 -312
  189. package/src/index.ts +0 -44
  190. package/src/services/InstallationService.ts +0 -102
  191. package/src/services/MCPManager.ts +0 -249
  192. package/src/services/RequirementService.ts +0 -627
  193. package/src/services/ServerService.ts +0 -161
  194. package/src/services/TelemetryService.ts +0 -59
  195. package/src/utils/UpdateCheckTracker.ts +0 -86
  196. package/src/utils/adoUtils.ts +0 -293
  197. package/src/utils/clientUtils.ts +0 -72
  198. package/src/utils/feedUtils.ts +0 -31
  199. package/src/utils/githubAuth.ts +0 -212
  200. package/src/utils/githubUtils.ts +0 -164
  201. package/src/utils/logger.ts +0 -195
  202. package/src/utils/macroExpressionUtils.ts +0 -104
  203. package/src/utils/osUtils.ts +0 -597
  204. package/src/utils/versionUtils.ts +0 -114
  205. package/src/web/contract/serverContract.ts +0 -74
  206. package/src/web/public/css/detailsWidget.css +0 -235
  207. package/src/web/public/css/modal.css +0 -757
  208. package/src/web/public/css/notifications.css +0 -101
  209. package/src/web/public/css/onboard.css +0 -107
  210. package/src/web/public/css/serverCategoryList.css +0 -120
  211. package/src/web/public/css/serverDetails.css +0 -139
  212. package/src/web/public/index.html +0 -359
  213. package/src/web/public/js/api.js +0 -132
  214. package/src/web/public/js/detailsWidget.js +0 -264
  215. package/src/web/public/js/flights/flights.js +0 -127
  216. package/src/web/public/js/modal/index.js +0 -52
  217. package/src/web/public/js/modal/installModal.js +0 -162
  218. package/src/web/public/js/modal/installation.js +0 -266
  219. package/src/web/public/js/modal/loadingModal.js +0 -182
  220. package/src/web/public/js/modal/modalSetup.js +0 -595
  221. package/src/web/public/js/modal/modalUtils.js +0 -37
  222. package/src/web/public/js/modal.js +0 -42
  223. package/src/web/public/js/notifications.js +0 -137
  224. package/src/web/public/js/onboard/formProcessor.js +0 -1037
  225. package/src/web/public/js/onboard/index.js +0 -374
  226. package/src/web/public/js/onboard/publishHandler.js +0 -172
  227. package/src/web/public/js/onboard/state.js +0 -76
  228. package/src/web/public/js/onboard/templates.js +0 -341
  229. package/src/web/public/js/onboard/uiHandlers.js +0 -1076
  230. package/src/web/public/js/onboard/validationHandlers.js +0 -493
  231. package/src/web/public/js/serverCategoryDetails.js +0 -364
  232. package/src/web/public/js/serverCategoryList.js +0 -241
  233. package/src/web/public/js/settings.js +0 -314
  234. package/src/web/public/modal.html +0 -84
  235. package/src/web/public/onboard.html +0 -296
  236. package/src/web/public/settings.html +0 -135
  237. package/src/web/public/styles.css +0 -277
  238. package/src/web/server.ts +0 -478
  239. 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
- }