imcp 0.1.5 → 0.1.6

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 (186) hide show
  1. package/.github/ISSUE_TEMPLATE/JitAccess.yml +28 -0
  2. package/.github/acl/access.yml +20 -0
  3. package/.github/compliance/inventory.yml +5 -0
  4. package/.github/policies/jit.yml +19 -0
  5. package/.github/workflows/build.yml +28 -0
  6. package/.roo/rules-code/rules.md +88 -0
  7. package/docs/ONBOARDING_PAGE_DESIGN.md +260 -0
  8. package/docs/Telemetry.md +136 -0
  9. package/memory-bank/activeContext.md +26 -0
  10. package/memory-bank/decisionLog.md +91 -0
  11. package/memory-bank/productContext.md +41 -0
  12. package/memory-bank/progress.md +35 -0
  13. package/memory-bank/systemPatterns.md +10 -0
  14. package/package.json +1 -5
  15. package/src/cli/commands/install.ts +139 -0
  16. package/src/cli/commands/list.ts +113 -0
  17. package/src/cli/commands/pull.ts +16 -0
  18. package/src/cli/commands/serve.ts +39 -0
  19. package/src/cli/commands/uninstall.ts +64 -0
  20. package/src/cli/index.ts +82 -0
  21. package/src/core/installers/clients/BaseClientInstaller.ts +341 -0
  22. package/src/core/installers/clients/ClientInstaller.ts +222 -0
  23. package/src/core/installers/clients/ClientInstallerFactory.ts +43 -0
  24. package/src/core/installers/clients/ClineInstaller.ts +35 -0
  25. package/src/core/installers/clients/ExtensionInstaller.ts +165 -0
  26. package/src/core/installers/clients/GithubCopilotInstaller.ts +79 -0
  27. package/src/core/installers/clients/MSRooCodeInstaller.ts +32 -0
  28. package/src/core/installers/index.ts +11 -0
  29. package/src/core/installers/requirements/BaseInstaller.ts +85 -0
  30. package/src/core/installers/requirements/CommandInstaller.ts +231 -0
  31. package/src/core/installers/requirements/GeneralInstaller.ts +133 -0
  32. package/src/core/installers/requirements/InstallerFactory.ts +114 -0
  33. package/src/core/installers/requirements/NpmInstaller.ts +271 -0
  34. package/src/core/installers/requirements/NugetInstaller.ts +203 -0
  35. package/src/core/installers/requirements/PipInstaller.ts +207 -0
  36. package/src/core/installers/requirements/RequirementInstaller.ts +42 -0
  37. package/src/core/loaders/ConfigurationLoader.ts +298 -0
  38. package/src/core/loaders/ConfigurationProvider.ts +462 -0
  39. package/src/core/loaders/InstallOperationManager.ts +367 -0
  40. package/src/core/loaders/ServerSchemaLoader.ts +117 -0
  41. package/src/core/loaders/ServerSchemaProvider.ts +99 -0
  42. package/src/core/loaders/SystemSettingsManager.ts +278 -0
  43. package/src/core/metadatas/constants.ts +122 -0
  44. package/src/core/metadatas/recordingConstants.ts +65 -0
  45. package/src/core/metadatas/types.ts +202 -0
  46. package/src/core/onboard/FeedOnboardService.ts +501 -0
  47. package/src/core/onboard/OnboardProcessor.ts +356 -0
  48. package/src/core/onboard/OnboardStatus.ts +60 -0
  49. package/src/core/onboard/OnboardStatusManager.ts +416 -0
  50. package/src/core/validators/FeedValidator.ts +135 -0
  51. package/src/core/validators/IServerValidator.ts +21 -0
  52. package/src/core/validators/SSEServerValidator.ts +43 -0
  53. package/src/core/validators/ServerValidatorFactory.ts +51 -0
  54. package/src/core/validators/StdioServerValidator.ts +313 -0
  55. package/src/index.ts +44 -0
  56. package/src/services/InstallationService.ts +102 -0
  57. package/src/services/MCPManager.ts +249 -0
  58. package/src/services/RequirementService.ts +627 -0
  59. package/src/services/ServerService.ts +161 -0
  60. package/src/services/TelemetryService.ts +59 -0
  61. package/src/utils/UpdateCheckTracker.ts +86 -0
  62. package/src/utils/adoUtils.ts +293 -0
  63. package/src/utils/clientUtils.ts +72 -0
  64. package/src/utils/feedUtils.ts +31 -0
  65. package/src/utils/githubAuth.ts +212 -0
  66. package/src/utils/githubUtils.ts +164 -0
  67. package/src/utils/logger.ts +195 -0
  68. package/src/utils/macroExpressionUtils.ts +104 -0
  69. package/src/utils/osUtils.ts +700 -0
  70. package/src/utils/versionUtils.ts +114 -0
  71. package/src/web/contract/serverContract.ts +74 -0
  72. package/src/web/public/css/detailsWidget.css +235 -0
  73. package/src/web/public/css/modal.css +757 -0
  74. package/src/web/public/css/notifications.css +101 -0
  75. package/src/web/public/css/onboard.css +107 -0
  76. package/src/web/public/css/serverCategoryList.css +120 -0
  77. package/src/web/public/css/serverDetails.css +139 -0
  78. package/src/web/public/index.html +359 -0
  79. package/src/web/public/js/api.js +132 -0
  80. package/src/web/public/js/detailsWidget.js +264 -0
  81. package/src/web/public/js/flights/flights.js +127 -0
  82. package/src/web/public/js/modal/index.js +52 -0
  83. package/src/web/public/js/modal/installModal.js +162 -0
  84. package/src/web/public/js/modal/installation.js +266 -0
  85. package/src/web/public/js/modal/loadingModal.js +182 -0
  86. package/src/web/public/js/modal/modalSetup.js +595 -0
  87. package/src/web/public/js/modal/modalUtils.js +37 -0
  88. package/src/web/public/js/modal/versionUtils.js +20 -0
  89. package/src/web/public/js/modal.js +42 -0
  90. package/src/web/public/js/notifications.js +137 -0
  91. package/src/web/public/js/onboard/formProcessor.js +1037 -0
  92. package/src/web/public/js/onboard/index.js +374 -0
  93. package/src/web/public/js/onboard/publishHandler.js +172 -0
  94. package/src/web/public/js/onboard/state.js +76 -0
  95. package/src/web/public/js/onboard/templates.js +342 -0
  96. package/src/web/public/js/onboard/uiHandlers.js +1076 -0
  97. package/src/web/public/js/onboard/validationHandlers.js +493 -0
  98. package/src/web/public/js/serverCategoryDetails.js +364 -0
  99. package/src/web/public/js/serverCategoryList.js +241 -0
  100. package/src/web/public/js/settings.js +314 -0
  101. package/src/web/public/modal.html +84 -0
  102. package/src/web/public/onboard.html +296 -0
  103. package/src/web/public/settings.html +135 -0
  104. package/src/web/public/styles.css +277 -0
  105. package/src/web/server.ts +478 -0
  106. package/tsconfig.json +18 -0
  107. package/wiki/Installation.md +3 -0
  108. package/wiki/Publish.md +3 -0
  109. package/dist/cli/commands/install.js.map +0 -1
  110. package/dist/cli/commands/list.js.map +0 -1
  111. package/dist/cli/commands/pull.js.map +0 -1
  112. package/dist/cli/commands/serve.js.map +0 -1
  113. package/dist/cli/commands/start.js.map +0 -1
  114. package/dist/cli/commands/sync.js.map +0 -1
  115. package/dist/cli/commands/uninstall.js.map +0 -1
  116. package/dist/cli/index.js.map +0 -1
  117. package/dist/core/ConfigurationLoader.js.map +0 -1
  118. package/dist/core/ConfigurationProvider.js.map +0 -1
  119. package/dist/core/InstallationService.js.map +0 -1
  120. package/dist/core/MCPManager.js.map +0 -1
  121. package/dist/core/RequirementService.js.map +0 -1
  122. package/dist/core/ServerSchemaLoader.js.map +0 -1
  123. package/dist/core/ServerSchemaProvider.js.map +0 -1
  124. package/dist/core/constants.js.map +0 -1
  125. package/dist/core/installers/BaseInstaller.js.map +0 -1
  126. package/dist/core/installers/ClientInstaller.js.map +0 -1
  127. package/dist/core/installers/CommandInstaller.js.map +0 -1
  128. package/dist/core/installers/GeneralInstaller.js.map +0 -1
  129. package/dist/core/installers/InstallerFactory.js.map +0 -1
  130. package/dist/core/installers/NpmInstaller.js.map +0 -1
  131. package/dist/core/installers/PipInstaller.js.map +0 -1
  132. package/dist/core/installers/RequirementInstaller.js.map +0 -1
  133. package/dist/core/installers/clients/BaseClientInstaller.js.map +0 -1
  134. package/dist/core/installers/clients/ClientInstaller.js.map +0 -1
  135. package/dist/core/installers/clients/ClientInstallerFactory.js.map +0 -1
  136. package/dist/core/installers/clients/ClineInstaller.js.map +0 -1
  137. package/dist/core/installers/clients/ExtensionInstaller.js.map +0 -1
  138. package/dist/core/installers/clients/GithubCopilotInstaller.js.map +0 -1
  139. package/dist/core/installers/clients/MSRooCodeInstaller.js.map +0 -1
  140. package/dist/core/installers/index.js.map +0 -1
  141. package/dist/core/installers/requirements/BaseInstaller.js.map +0 -1
  142. package/dist/core/installers/requirements/CommandInstaller.js.map +0 -1
  143. package/dist/core/installers/requirements/GeneralInstaller.js.map +0 -1
  144. package/dist/core/installers/requirements/InstallerFactory.js.map +0 -1
  145. package/dist/core/installers/requirements/NpmInstaller.js.map +0 -1
  146. package/dist/core/installers/requirements/NugetInstaller.js.map +0 -1
  147. package/dist/core/installers/requirements/PipInstaller.js.map +0 -1
  148. package/dist/core/installers/requirements/RequirementInstaller.js.map +0 -1
  149. package/dist/core/loaders/ConfigurationLoader.js.map +0 -1
  150. package/dist/core/loaders/ConfigurationProvider.js.map +0 -1
  151. package/dist/core/loaders/InstallOperationManager.js.map +0 -1
  152. package/dist/core/loaders/ServerSchemaLoader.js.map +0 -1
  153. package/dist/core/loaders/ServerSchemaProvider.js.map +0 -1
  154. package/dist/core/loaders/SystemSettingsManager.js.map +0 -1
  155. package/dist/core/metadatas/constants.js.map +0 -1
  156. package/dist/core/metadatas/recordingConstants.js.map +0 -1
  157. package/dist/core/metadatas/types.js.map +0 -1
  158. package/dist/core/onboard/FeedOnboardService.js.map +0 -1
  159. package/dist/core/onboard/OnboardProcessor.js.map +0 -1
  160. package/dist/core/onboard/OnboardStatus.js.map +0 -1
  161. package/dist/core/onboard/OnboardStatusManager.js.map +0 -1
  162. package/dist/core/types.js.map +0 -1
  163. package/dist/core/validators/FeedValidator.js.map +0 -1
  164. package/dist/core/validators/IServerValidator.js.map +0 -1
  165. package/dist/core/validators/SSEServerValidator.js.map +0 -1
  166. package/dist/core/validators/ServerValidatorFactory.js.map +0 -1
  167. package/dist/core/validators/StdioServerValidator.js.map +0 -1
  168. package/dist/index.js.map +0 -1
  169. package/dist/services/InstallRequestValidator.js.map +0 -1
  170. package/dist/services/InstallationService.js.map +0 -1
  171. package/dist/services/MCPManager.js.map +0 -1
  172. package/dist/services/RequirementService.js.map +0 -1
  173. package/dist/services/ServerService.js.map +0 -1
  174. package/dist/services/TelemetryService.js.map +0 -1
  175. package/dist/utils/UpdateCheckTracker.js.map +0 -1
  176. package/dist/utils/adoUtils.js.map +0 -1
  177. package/dist/utils/clientUtils.js.map +0 -1
  178. package/dist/utils/feedUtils.js.map +0 -1
  179. package/dist/utils/githubAuth.js.map +0 -1
  180. package/dist/utils/githubUtils.js.map +0 -1
  181. package/dist/utils/logger.js.map +0 -1
  182. package/dist/utils/macroExpressionUtils.js.map +0 -1
  183. package/dist/utils/osUtils.js.map +0 -1
  184. package/dist/utils/versionUtils.js.map +0 -1
  185. package/dist/web/contract/serverContract.js.map +0 -1
  186. package/dist/web/server.js.map +0 -1
@@ -0,0 +1,203 @@
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 { compareVersions } from '../../../utils/versionUtils.js';
5
+ import { Logger } from '../../../utils/logger.js';
6
+ import { InstallOperationManager } from '../../loaders/InstallOperationManager.js';
7
+ import * as RecordingConstants from '../../metadatas/recordingConstants.js';
8
+ import { ensureDotnetToolsInPath } from '../../../utils/osUtils.js';
9
+ import * as fs from 'fs';
10
+ import * as path from 'path';
11
+
12
+ /**
13
+ * Installer implementation for .NET packages using NuGet
14
+ */
15
+ export class NugetInstaller extends BaseInstaller {
16
+ /**
17
+ * Check if this installer can handle the given requirement type
18
+ * @param requirement The requirement to check
19
+ * @returns True if this installer can handle the requirement
20
+ */
21
+ canHandle(requirement: RequirementConfig): boolean {
22
+ return requirement.type === 'nuget';
23
+ }
24
+
25
+ supportCheckUpdates(): boolean {
26
+ return true;
27
+ }
28
+
29
+ /**
30
+ * Get the latest version available for the NuGet package.
31
+ * @param requirement The requirement to check.
32
+ * @param _options Optional server install options (not used for NuGet).
33
+ * @returns The latest version string, or undefined if not found or not applicable.
34
+ */
35
+ async getLatestVersion(requirement: RequirementConfig, _options?: ServerInstallOptions): Promise<string | undefined> {
36
+ if (requirement.registry && requirement.registry.githubRelease) {
37
+ return getGitHubLatestVersion(this.execPromise, requirement.registry.githubRelease.repository);
38
+ }
39
+ // Artifacts registry is not supported for nuget tools
40
+ if (requirement.registry && requirement.registry.artifacts) {
41
+ Logger.warn(`Artifacts registry is not supported for NuGet tool '${requirement.name}'.`);
42
+ return undefined;
43
+ }
44
+ // Default behavior: Nuget tools are often specific versions from specific sources,
45
+ // or global tools might not have a central "latest version" query like pip/npm.
46
+ // Returning current version if specified, otherwise undefined.
47
+ Logger.warn(`Direct latest version check for NuGet tool '${requirement.name}' without a GitHub release registry is not supported. Please specify a version or use a GitHub release.`);
48
+ return requirement.version || undefined;
49
+ }
50
+
51
+ /**
52
+ * Check if the .NET tool is already installed
53
+ * @param requirement The requirement to check
54
+ * @param _options Optional server install options (not used for NuGet)
55
+ * @returns The status of the requirement
56
+ */
57
+ async checkInstallation(requirement: RequirementConfig, _options?: ServerInstallOptions): Promise<RequirementStatus> {
58
+ try {
59
+ // Command: dotnet tool list -g
60
+ // Output:
61
+ // Package Id Version Commands
62
+ // -----------------------------------------
63
+ // jarvistools 1.0.0 jarvistools
64
+ const { stdout } = await this.execPromise(`dotnet tool list -g`);
65
+ const lines = stdout.split('\n');
66
+ let installedVersion: string | undefined;
67
+ let isInstalled = false;
68
+
69
+ for (const line of lines) {
70
+ const parts = line.trim().split(/\s+/);
71
+ if (parts.length >= 2 && parts[0].toLowerCase() === requirement.name.toLowerCase()) {
72
+ installedVersion = parts[1];
73
+ isInstalled = true;
74
+ break;
75
+ }
76
+ }
77
+
78
+ return {
79
+ name: requirement.name,
80
+ type: 'nuget',
81
+ installed: isInstalled,
82
+ version: installedVersion,
83
+ inProgress: false,
84
+ };
85
+ } catch (error) {
86
+ // If 'dotnet tool list -g' fails, it might mean dotnet CLI is not properly installed or configured.
87
+ // Or it could mean no tools are installed, which in some dotnet versions might return non-zero exit code.
88
+ // We'll assume not installed in case of error, but log it.
89
+ Logger.debug(`Error checking NuGet tool installation for ${requirement.name}: ${error instanceof Error ? error.message : String(error)}`);
90
+ return {
91
+ name: requirement.name,
92
+ type: 'nuget',
93
+ installed: false,
94
+ error: `Failed to check installation: ${error instanceof Error ? error.message : String(error)}`,
95
+ inProgress: false,
96
+ };
97
+ }
98
+ }
99
+
100
+ /**
101
+ * Install the .NET tool
102
+ * @param requirement The requirement to install
103
+ * @param recorder Optional InstallOperationManager for recording steps
104
+ * @param _options Optional server install options (not used for NuGet)
105
+ * @returns The status of the installation
106
+ */
107
+ async install(requirement: RequirementConfig, recorder: InstallOperationManager, _options?: ServerInstallOptions): Promise<RequirementStatus> {
108
+ return await recorder.recording(
109
+ async (): Promise<RequirementStatus> => {
110
+ const status = await this.checkInstallation(requirement, _options);
111
+ if (status.installed && status.version && requirement.version &&
112
+ compareVersions(status.version, requirement.version) === 0 &&
113
+ !requirement.version.toLowerCase().includes('latest')) {
114
+ Logger.log(`NuGet tool ${requirement.name}==${status.version} already installed.`);
115
+ return status;
116
+ }
117
+
118
+ let command: string;
119
+
120
+ if (requirement.registry && requirement.registry.githubRelease) {
121
+ const result = await handleGitHubRelease(requirement, requirement.registry.githubRelease);
122
+ // Nuget package name might be different from the requirement name if alias is used.
123
+ // However, dotnet tool install uses the package ID from the nupkg.
124
+ // We assume requirement.name is the package ID.
125
+ const packageId = requirement.name;
126
+ const resolvedDir = fs.existsSync(result.resolvedPath) && fs.lstatSync(result.resolvedPath).isDirectory() ? result.resolvedPath : path.dirname(result.resolvedPath);
127
+
128
+ if (requirement.version && !requirement.version.toLowerCase().includes('latest')) {
129
+ command = `dotnet tool install --global --add-source "${resolvedDir}" ${packageId} --version ${requirement.version}`;
130
+ } else {
131
+ // Install latest from the source
132
+ command = `dotnet tool install --global --add-source "${resolvedDir}" ${packageId}`;
133
+ }
134
+ } else if (requirement.registry && requirement.registry.artifacts) {
135
+ const errorMessage = `Artifacts registry is not supported for NuGet tool yet'${requirement.name}'. Only GitHubRelease is supported.`;
136
+ Logger.error(errorMessage);
137
+ await recorder.recordStep('NugetInstaller:RegistryConfig', 'failed', errorMessage);
138
+ throw new Error(errorMessage);
139
+ } else {
140
+ // Default installation from nuget.org or configured feeds
141
+ if (requirement.version && !requirement.version.toLowerCase().includes('latest')) {
142
+ command = `dotnet tool install --global ${requirement.name} --version ${requirement.version}`;
143
+ } else {
144
+ command = `dotnet tool install --global ${requirement.name}`;
145
+ }
146
+ }
147
+
148
+ return await recorder.recording(
149
+ async () => {
150
+ const { stdout, stderr } = await this.execPromise(command);
151
+ if (stderr && !stdout.toLowerCase().includes('already installed')) { // Some warnings might go to stderr
152
+ Logger.debug(`NuGet tool installation stderr for ${requirement.name}: ${stderr}`);
153
+ // Check if it was actually an error or just a warning
154
+ const checkStatus = await this.checkInstallation(requirement, _options);
155
+ if (!checkStatus.installed) {
156
+ Logger.error(`NuGet tool ${requirement.name} not found after install command, stderr: ${stderr}`);
157
+ throw new Error(`NuGet tool installation failed with: ${stderr}`);
158
+ }
159
+ }
160
+
161
+ const finalStatus = await this.checkInstallation(requirement, _options);
162
+ if (!finalStatus.installed) {
163
+ throw new Error(`NuGet tool ${requirement.name} failed to install. Please check logs.`);
164
+ }
165
+ // After successful installation, ensure .NET tools path is in system PATH
166
+ await ensureDotnetToolsInPath();
167
+ return {
168
+ name: requirement.name,
169
+ type: 'nuget',
170
+ installed: true,
171
+ version: finalStatus.version || requirement.version, // Use checked version if available
172
+ inProgress: false,
173
+ };
174
+ },
175
+ {
176
+ stepName: `${RecordingConstants.STEP_INSTALL_COMMAND_PREFIX}: ${requirement.name} : ${requirement.version || 'latest'}`,
177
+ inProgressMessage: `Running: ${command}`,
178
+ endMessage: (result) => result.installed ? `Succeeded: ${command}` : `Failed: ${command}`,
179
+ }
180
+ );
181
+ },
182
+ {
183
+ stepName: RecordingConstants.STEP_NUGET_INSTALLER_INSTALL,
184
+ inProgressMessage: `Installing NuGet tool: ${requirement.name}`,
185
+ endMessage: (result) => result.installed
186
+ ? `Install completed for ${requirement.name} with version ${result.version}`
187
+ : `Install failed for ${requirement.name}`,
188
+ onError: (error) => {
189
+ return {
190
+ result: {
191
+ name: requirement.name,
192
+ type: 'nuget',
193
+ installed: false,
194
+ error: error instanceof Error ? error.message : String(error),
195
+ inProgress: false,
196
+ },
197
+ message: error instanceof Error ? error.message : String(error),
198
+ };
199
+ },
200
+ }
201
+ );
202
+ }
203
+ }
@@ -0,0 +1,207 @@
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
+ }
@@ -0,0 +1,42 @@
1
+ import { RequirementConfig, RequirementStatus, ServerInstallOptions } from '../../metadatas/types.js';
2
+ import { InstallOperationManager } from '../../loaders/InstallOperationManager.js';
3
+
4
+ /**
5
+ * Interface for requirement installers.
6
+ * Implementations should handle specific requirement types.
7
+ */
8
+ export interface RequirementInstaller {
9
+ /**
10
+ * Check if this installer can handle the given requirement type
11
+ * @param requirement The requirement to check
12
+ * @returns True if this installer can handle the requirement
13
+ */
14
+ canHandle(requirement: RequirementConfig): boolean;
15
+
16
+ supportCheckUpdates(): boolean;
17
+ /**
18
+ * Install the requirement
19
+ * @param requirement The requirement to install
20
+ * @param options Optional install options
21
+ * @param recorder Optional InstallOperationManager for recording steps
22
+ * @returns The status of the installation
23
+ */
24
+ install(requirement: RequirementConfig, recorder: InstallOperationManager, options?: ServerInstallOptions): Promise<RequirementStatus>;
25
+
26
+ /**
27
+ * Check if the requirement is already installed
28
+ * @param requirement The requirement to check
29
+ * @returns The status of the requirement
30
+ */
31
+ checkInstallation(requirement: RequirementConfig, options?: ServerInstallOptions): Promise<RequirementStatus>;
32
+
33
+ /**
34
+ * Check if updates are available for the requirement
35
+ * @param requirement The requirement to check
36
+ * @param currentStatus The current status of the requirement
37
+ * @returns The status of the requirement with update information
38
+ */
39
+ checkForUpdates(requirement: RequirementConfig, currentStatus: RequirementStatus): Promise<RequirementStatus>;
40
+ }
41
+
42
+ // Note: Do not re-export implementations from here to avoid circular dependencies