imcp 0.1.4 → 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 (228) 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 -114
  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.map +1 -0
  75. package/dist/core/installers/requirements/NpmInstaller.js.map +1 -0
  76. package/dist/core/installers/requirements/NugetInstaller.js.map +1 -0
  77. package/dist/core/installers/requirements/PipInstaller.js.map +1 -0
  78. package/dist/core/installers/requirements/RequirementInstaller.js.map +1 -0
  79. package/dist/core/loaders/ConfigurationLoader.js.map +1 -0
  80. package/dist/core/loaders/ConfigurationProvider.js.map +1 -0
  81. package/dist/core/loaders/InstallOperationManager.js.map +1 -0
  82. package/dist/core/loaders/ServerSchemaLoader.js.map +1 -0
  83. package/dist/core/loaders/ServerSchemaProvider.js.map +1 -0
  84. package/dist/core/loaders/SystemSettingsManager.js.map +1 -0
  85. package/dist/core/metadatas/constants.js.map +1 -0
  86. package/dist/core/metadatas/recordingConstants.js.map +1 -0
  87. package/dist/core/metadatas/types.js.map +1 -0
  88. package/dist/core/onboard/FeedOnboardService.js.map +1 -0
  89. package/dist/core/onboard/OnboardProcessor.js.map +1 -0
  90. package/dist/core/onboard/OnboardStatus.js.map +1 -0
  91. package/dist/core/onboard/OnboardStatusManager.js.map +1 -0
  92. package/dist/core/types.d.ts +166 -0
  93. package/dist/core/types.js +16 -0
  94. package/dist/core/types.js.map +1 -0
  95. package/dist/core/validators/FeedValidator.js.map +1 -0
  96. package/dist/core/validators/IServerValidator.js.map +1 -0
  97. package/dist/core/validators/SSEServerValidator.js.map +1 -0
  98. package/dist/core/validators/ServerValidatorFactory.js.map +1 -0
  99. package/dist/core/validators/StdioServerValidator.js.map +1 -0
  100. package/dist/index.js.map +1 -0
  101. package/dist/services/InstallRequestValidator.d.ts +21 -0
  102. package/dist/services/InstallRequestValidator.js +99 -0
  103. package/dist/services/InstallRequestValidator.js.map +1 -0
  104. package/dist/services/InstallationService.js.map +1 -0
  105. package/dist/services/MCPManager.js.map +1 -0
  106. package/dist/services/RequirementService.js.map +1 -0
  107. package/dist/services/ServerService.js.map +1 -0
  108. package/dist/services/TelemetryService.js.map +1 -0
  109. package/dist/utils/UpdateCheckTracker.js.map +1 -0
  110. package/dist/utils/adoUtils.js.map +1 -0
  111. package/dist/utils/clientUtils.js.map +1 -0
  112. package/dist/utils/feedUtils.js.map +1 -0
  113. package/dist/utils/githubAuth.js.map +1 -0
  114. package/dist/utils/githubUtils.js.map +1 -0
  115. package/dist/utils/logger.js.map +1 -0
  116. package/dist/utils/macroExpressionUtils.js.map +1 -0
  117. package/dist/utils/osUtils.js.map +1 -0
  118. package/dist/utils/versionUtils.js.map +1 -0
  119. package/dist/web/contract/serverContract.js.map +1 -0
  120. package/dist/web/public/index.html +1 -1
  121. package/dist/web/public/js/modal/installHandler.js +227 -0
  122. package/dist/web/public/js/modal/loadingUI.js +74 -0
  123. package/dist/web/public/js/modal/messageQueue.js +101 -45
  124. package/dist/web/public/js/modal/modalUI.js +214 -0
  125. package/{src/web/public/js/modal/versionUtils.js → dist/web/public/js/modal/version.js} +1 -1
  126. package/dist/web/public/onboard.html +4 -4
  127. package/dist/web/server.js.map +1 -0
  128. package/package.json +5 -1
  129. package/.github/ISSUE_TEMPLATE/JitAccess.yml +0 -28
  130. package/.github/acl/access.yml +0 -20
  131. package/.github/compliance/inventory.yml +0 -5
  132. package/.github/policies/jit.yml +0 -19
  133. package/.roo/rules-code/rules.md +0 -88
  134. package/dist/core/onboard/InstallOperationManager.d.ts +0 -23
  135. package/dist/core/onboard/InstallOperationManager.js +0 -144
  136. package/docs/ONBOARDING_PAGE_DESIGN.md +0 -260
  137. package/docs/Telemetry.md +0 -136
  138. package/memory-bank/activeContext.md +0 -26
  139. package/memory-bank/decisionLog.md +0 -91
  140. package/memory-bank/productContext.md +0 -41
  141. package/memory-bank/progress.md +0 -35
  142. package/memory-bank/systemPatterns.md +0 -10
  143. package/src/cli/commands/install.ts +0 -139
  144. package/src/cli/commands/list.ts +0 -113
  145. package/src/cli/commands/pull.ts +0 -16
  146. package/src/cli/commands/serve.ts +0 -39
  147. package/src/cli/commands/uninstall.ts +0 -64
  148. package/src/cli/index.ts +0 -82
  149. package/src/core/installers/clients/BaseClientInstaller.ts +0 -341
  150. package/src/core/installers/clients/ClientInstaller.ts +0 -222
  151. package/src/core/installers/clients/ClientInstallerFactory.ts +0 -43
  152. package/src/core/installers/clients/ClineInstaller.ts +0 -35
  153. package/src/core/installers/clients/ExtensionInstaller.ts +0 -165
  154. package/src/core/installers/clients/GithubCopilotInstaller.ts +0 -79
  155. package/src/core/installers/clients/MSRooCodeInstaller.ts +0 -32
  156. package/src/core/installers/index.ts +0 -11
  157. package/src/core/installers/requirements/BaseInstaller.ts +0 -85
  158. package/src/core/installers/requirements/GeneralInstaller.ts +0 -133
  159. package/src/core/installers/requirements/NpmInstaller.ts +0 -271
  160. package/src/core/installers/requirements/NugetInstaller.ts +0 -203
  161. package/src/core/installers/requirements/PipInstaller.ts +0 -207
  162. package/src/core/loaders/InstallOperationManager.ts +0 -367
  163. package/src/core/loaders/ServerSchemaLoader.ts +0 -117
  164. package/src/core/loaders/ServerSchemaProvider.ts +0 -99
  165. package/src/core/loaders/SystemSettingsManager.ts +0 -278
  166. package/src/core/metadatas/constants.ts +0 -122
  167. package/src/core/metadatas/recordingConstants.ts +0 -65
  168. package/src/core/metadatas/types.ts +0 -202
  169. package/src/core/onboard/FeedOnboardService.ts +0 -501
  170. package/src/core/onboard/OnboardProcessor.ts +0 -356
  171. package/src/core/onboard/OnboardStatus.ts +0 -60
  172. package/src/core/onboard/OnboardStatusManager.ts +0 -416
  173. package/src/core/validators/FeedValidator.ts +0 -135
  174. package/src/core/validators/IServerValidator.ts +0 -21
  175. package/src/core/validators/SSEServerValidator.ts +0 -43
  176. package/src/core/validators/ServerValidatorFactory.ts +0 -51
  177. package/src/core/validators/StdioServerValidator.ts +0 -313
  178. package/src/index.ts +0 -44
  179. package/src/services/InstallationService.ts +0 -102
  180. package/src/services/MCPManager.ts +0 -249
  181. package/src/services/RequirementService.ts +0 -627
  182. package/src/services/ServerService.ts +0 -161
  183. package/src/services/TelemetryService.ts +0 -59
  184. package/src/utils/UpdateCheckTracker.ts +0 -86
  185. package/src/utils/adoUtils.ts +0 -293
  186. package/src/utils/clientUtils.ts +0 -72
  187. package/src/utils/feedUtils.ts +0 -31
  188. package/src/utils/githubAuth.ts +0 -212
  189. package/src/utils/githubUtils.ts +0 -164
  190. package/src/utils/logger.ts +0 -195
  191. package/src/utils/macroExpressionUtils.ts +0 -104
  192. package/src/utils/osUtils.ts +0 -700
  193. package/src/utils/versionUtils.ts +0 -114
  194. package/src/web/contract/serverContract.ts +0 -74
  195. package/src/web/public/css/detailsWidget.css +0 -235
  196. package/src/web/public/css/modal.css +0 -757
  197. package/src/web/public/css/notifications.css +0 -101
  198. package/src/web/public/css/onboard.css +0 -107
  199. package/src/web/public/css/serverCategoryList.css +0 -120
  200. package/src/web/public/css/serverDetails.css +0 -139
  201. package/src/web/public/index.html +0 -359
  202. package/src/web/public/js/api.js +0 -132
  203. package/src/web/public/js/detailsWidget.js +0 -264
  204. package/src/web/public/js/flights/flights.js +0 -127
  205. package/src/web/public/js/modal/index.js +0 -52
  206. package/src/web/public/js/modal/installModal.js +0 -162
  207. package/src/web/public/js/modal/installation.js +0 -266
  208. package/src/web/public/js/modal/loadingModal.js +0 -182
  209. package/src/web/public/js/modal/modalSetup.js +0 -595
  210. package/src/web/public/js/modal/modalUtils.js +0 -37
  211. package/src/web/public/js/modal.js +0 -42
  212. package/src/web/public/js/notifications.js +0 -137
  213. package/src/web/public/js/onboard/formProcessor.js +0 -1037
  214. package/src/web/public/js/onboard/index.js +0 -374
  215. package/src/web/public/js/onboard/publishHandler.js +0 -172
  216. package/src/web/public/js/onboard/state.js +0 -76
  217. package/src/web/public/js/onboard/templates.js +0 -342
  218. package/src/web/public/js/onboard/uiHandlers.js +0 -1076
  219. package/src/web/public/js/onboard/validationHandlers.js +0 -493
  220. package/src/web/public/js/serverCategoryDetails.js +0 -364
  221. package/src/web/public/js/serverCategoryList.js +0 -241
  222. package/src/web/public/js/settings.js +0 -314
  223. package/src/web/public/modal.html +0 -84
  224. package/src/web/public/onboard.html +0 -296
  225. package/src/web/public/settings.html +0 -135
  226. package/src/web/public/styles.css +0 -277
  227. package/src/web/server.ts +0 -478
  228. package/tsconfig.json +0 -18
@@ -0,0 +1,227 @@
1
+ import { showLoadingModal, appendLoadingMessage } from './loadingUI.js';
2
+ import { compareVersions } from './version.js';
3
+
4
+ /**
5
+ * Handle bulk client installations
6
+ * @param {string} categoryName Category name
7
+ * @param {string} serverName Server name
8
+ * @param {string[]} targets Target clients
9
+ * @param {Object} options Installation options
10
+ */
11
+ export async function handleInstallation(categoryName, serverName, targets, options = {}) {
12
+ const { envVars = {}, installingMessage = "Starting installation...", serverData = null } = options;
13
+
14
+ hideInstallModal();
15
+ initializeInstallation(installingMessage, serverData, serverName, targets);
16
+
17
+ try {
18
+ await executeInstallation(categoryName, serverName, targets, options);
19
+ } catch (error) {
20
+ handleInstallationError(error);
21
+ }
22
+ }
23
+
24
+ /**
25
+ * Handle uninstallation of tools
26
+ * @param {string} categoryName Category name
27
+ * @param {Object|Array} serverList Servers to uninstall
28
+ * @param {string[]} targets Target clients
29
+ */
30
+ export async function handleUninstallation(categoryName, serverList, targets) {
31
+ const selectedTargets = getSelectedTargets(targets);
32
+ validateTargets(selectedTargets);
33
+
34
+ try {
35
+ await executeUninstallation(categoryName, serverList, selectedTargets);
36
+ } catch (error) {
37
+ handleUninstallationError(error);
38
+ throw error;
39
+ }
40
+ }
41
+
42
+ // Private helper functions
43
+
44
+ function hideInstallModal() {
45
+ const installModal = document.getElementById('installModal');
46
+ if (installModal) {
47
+ installModal.style.display = "none";
48
+ }
49
+ }
50
+
51
+ function initializeInstallation(installingMessage, serverData, serverName, targets) {
52
+ showLoadingModal();
53
+
54
+ if (hasExistingStatus(serverData, serverName, targets)) {
55
+ const existingMessage = getExistingMessage(serverData, serverName, targets[0]);
56
+ if (existingMessage && existingMessage !== "Starting installation...") {
57
+ appendLoadingMessage(existingMessage);
58
+ }
59
+ }
60
+
61
+ appendLoadingMessage(installingMessage);
62
+ }
63
+
64
+ function hasExistingStatus(serverData, serverName, targets) {
65
+ return serverData?.data?.installationStatus?.serversStatus &&
66
+ targets?.length > 0;
67
+ }
68
+
69
+ function getExistingMessage(serverData, serverName, target) {
70
+ const serverStatuses = serverData.data.installationStatus.serversStatus;
71
+ const serverStatus = serverStatuses[serverName] || { installedStatus: {} };
72
+ return serverStatus.installedStatus?.[target]?.message;
73
+ }
74
+
75
+ async function executeInstallation(categoryName, serverName, targets, options) {
76
+ appendLoadingMessage("Installing, please wait...");
77
+
78
+ const requestBody = buildInstallRequestBody(serverName, targets, options);
79
+ const response = await fetchWithErrorHandling(
80
+ `api/categories/${categoryName}/install`,
81
+ requestBody
82
+ );
83
+
84
+ if (response.success) {
85
+ startStatusPolling(categoryName, serverName, targets, options);
86
+ }
87
+ }
88
+
89
+ function buildInstallRequestBody(serverName, targets, options) {
90
+ const { envVars = {}, args = [], pythonEnv, requirements = [] } = options;
91
+
92
+ const serverOptions = {
93
+ targetClients: targets,
94
+ env: envVars,
95
+ args: args,
96
+ settings: pythonEnv ? { pythonEnv } : undefined
97
+ };
98
+
99
+ if (requirements.length > 0) {
100
+ serverOptions.requirements = requirements;
101
+ }
102
+
103
+ return {
104
+ serverList: {
105
+ [serverName]: serverOptions
106
+ }
107
+ };
108
+ }
109
+
110
+ async function executeUninstallation(categoryName, serverList, selectedTargets) {
111
+ appendLoadingMessage('Starting uninstallation...');
112
+
113
+ const formattedServerList = formatServerList(serverList);
114
+ await fetchWithErrorHandling(
115
+ `api/categories/${categoryName}/uninstall`,
116
+ {
117
+ serverList: formattedServerList,
118
+ options: {
119
+ targets: selectedTargets,
120
+ removeData: true
121
+ }
122
+ }
123
+ );
124
+
125
+ appendLoadingMessage(`Successfully uninstalled from ${selectedTargets.join(', ')}`, true);
126
+ window.selectedClients = [];
127
+ }
128
+
129
+ function formatServerList(serverList) {
130
+ if (Array.isArray(serverList)) {
131
+ return serverList.reduce((acc, server) => {
132
+ acc[server] = { removeData: true };
133
+ return acc;
134
+ }, {});
135
+ }
136
+ return serverList;
137
+ }
138
+
139
+ async function fetchWithErrorHandling(endpoint, body) {
140
+ const response = await fetch(endpoint, {
141
+ method: 'POST',
142
+ headers: {
143
+ 'Content-Type': 'application/json',
144
+ 'Accept': 'application/json'
145
+ },
146
+ body: JSON.stringify(body)
147
+ });
148
+
149
+ if (!response.ok) {
150
+ const errorData = await response.text();
151
+ throw new Error(`Operation failed: ${errorData || response.statusText}`);
152
+ }
153
+
154
+ const result = await response.json();
155
+ if (!result.success) {
156
+ throw new Error(result.error || 'Operation failed');
157
+ }
158
+
159
+ return result;
160
+ }
161
+
162
+ function handleInstallationError(error) {
163
+ console.error('[LoadingModal] Error:', error);
164
+ appendLoadingMessage(`<span style="color:red;">Error: ${error.message}</span>`);
165
+ }
166
+
167
+ function handleUninstallationError(error) {
168
+ console.error('Error uninstalling tools:', error);
169
+ appendLoadingMessage(`Error: ${error.message}`, true);
170
+ }
171
+
172
+ function getSelectedTargets(targets) {
173
+ return window.selectedClients || (Array.isArray(targets) ? targets : [targets]);
174
+ }
175
+
176
+ function validateTargets(selectedTargets) {
177
+ if (!selectedTargets || selectedTargets.length === 0) {
178
+ throw new Error('Please select at least one client to uninstall.');
179
+ }
180
+ }
181
+
182
+ // Status polling functionality
183
+ async function startStatusPolling(categoryName, serverName, targets, options) {
184
+ const { requirements = [] } = options;
185
+ let lastMessages = {};
186
+ let lastRequirementMessages = {};
187
+ let lastRequirementErrorMessages = {};
188
+ let completionMessageSent = false;
189
+
190
+ const startTime = Date.now();
191
+ const maxTimeout = 10 * 60 * 1000; // 10 minutes
192
+
193
+ while (Date.now() - startTime < maxTimeout) {
194
+ try {
195
+ const status = await pollStatus(categoryName, serverName, targets, requirements);
196
+
197
+ if (status.isComplete) {
198
+ sendCompletionMessage(status, targets);
199
+ return;
200
+ }
201
+
202
+ updateMessages(status, lastMessages, lastRequirementMessages, lastRequirementErrorMessages);
203
+ } catch (error) {
204
+ console.error('[LoadingModal] Error polling status:', error);
205
+ }
206
+
207
+ await new Promise(resolve => setTimeout(resolve, 2000));
208
+ }
209
+
210
+ handlePollingTimeout(completionMessageSent);
211
+ }
212
+
213
+ async function pollStatus(categoryName, serverName, targets, requirements) {
214
+ const response = await fetch(`api/categories/${categoryName}`);
215
+ if (!response.ok) {
216
+ throw new Error('Failed to fetch status');
217
+ }
218
+
219
+ const data = await response.json();
220
+ return processStatusData(data, serverName, targets, requirements);
221
+ }
222
+
223
+ function processStatusData(data, serverName, targets, requirements) {
224
+ // Implementation details for processing status data
225
+ // This would include checking requirements status and target status
226
+ // Returns an object with status information
227
+ }
@@ -0,0 +1,74 @@
1
+ import { queueMessage } from './messageQueue.js';
2
+
3
+ /**
4
+ * Display the installation loading modal and prepare it for messages
5
+ * @param {string} operation Operation name to display
6
+ */
7
+ export function showLoadingModal(operation = 'Installing') {
8
+ const elements = getLoadingElements();
9
+ if (!elements) {
10
+ console.error('[LoadingModal] Required elements not found');
11
+ return;
12
+ }
13
+
14
+ elements.modal.style.display = 'block';
15
+ elements.messageContainer.innerHTML = '';
16
+ elements.title.textContent = `${operation}...`;
17
+ }
18
+
19
+ /**
20
+ * Hide the installation loading modal
21
+ */
22
+ export function hideLoadingModal() {
23
+ const modal = document.getElementById('installLoadingModal');
24
+ if (!modal) {
25
+ console.error('[LoadingModal] Loading modal not found');
26
+ return;
27
+ }
28
+
29
+ modal.style.display = 'none';
30
+ refreshPageWithCategory();
31
+ }
32
+
33
+ /**
34
+ * Append a message to the loading modal
35
+ * @param {string} message Message to display
36
+ * @param {boolean} isCompletion Whether this is a completion message
37
+ */
38
+ export function appendLoadingMessage(message, isCompletion = false) {
39
+ queueMessage(message, isCompletion);
40
+ }
41
+
42
+ /**
43
+ * Get the loading modal elements
44
+ * @private
45
+ * @returns {Object|null} Object containing modal elements or null if not found
46
+ */
47
+ function getLoadingElements() {
48
+ const modal = document.getElementById('installLoadingModal');
49
+ const messageContainer = document.getElementById('installLoadingMessage');
50
+ const title = document.querySelector('.loading-title');
51
+
52
+ if (!modal || !messageContainer || !title) {
53
+ return null;
54
+ }
55
+
56
+ return { modal, messageContainer, title };
57
+ }
58
+
59
+ /**
60
+ * Refresh the page maintaining category selection
61
+ * @private
62
+ */
63
+ function refreshPageWithCategory() {
64
+ const lastSelected = localStorage.getItem('lastSelectedCategory');
65
+
66
+ setTimeout(() => {
67
+ if (lastSelected) {
68
+ window.location.href = window.location.pathname +
69
+ '?category=' + encodeURIComponent(lastSelected);
70
+ } else {
71
+ location.reload();
72
+ }
73
+ }, 100);
74
+ }
@@ -1,56 +1,112 @@
1
- // This file previously handled a message queue for the loading modal.
2
- // With the new design in loadingModal.js, which directly manipulates its content
3
- // for overall status and step details, this queueing mechanism for individual
4
- // step messages is no longer the primary way messages are displayed.
5
- // The loadingModal.js now has `updateOverallInstallStatus` and `addInstallationStep`.
6
-
7
- // The `updateCompletionUI` function might still be relevant if we want to
8
- // change the overall modal's appearance (e.g., title, main icon) upon final completion,
9
- // separate from the step-by-step updates.
1
+ /** @type {boolean} Flag indicating if a message is currently being appended */
2
+ let isAppending = false;
10
3
 
11
4
  /** @type {boolean} Flag indicating if completion UI update is pending */
12
- let completionPending = false; // This might still be useful for a final "Completed" state change of the modal itself.
5
+ let completionPending = false;
13
6
 
14
7
  /**
15
- * This function could be called by `installation.js` once polling determines
16
- * the absolute final state (e.g., after all messages are processed by `loadingModal.js`
17
- * and a small delay).
18
- * @param {'completed' | 'failed'} finalState
8
+ * Message queue for delayed loading modal updates.
9
+ * @type {Array<{msg: string, isCompletion: boolean}>}
19
10
  */
20
- export function triggerFinalModalStateUpdate(finalState) {
21
- // This function is a placeholder for now.
22
- // The actual update of the overall status icon and text is now handled by
23
- // `updateOverallInstallStatus` in `loadingModal.js`.
24
- // If there's a need for a *separate* final transformation of the modal
25
- // (e.g. changing the main title, showing a global success/fail icon not tied to overallStatusTextEl),
26
- // that logic would go here or be triggered from here.
27
-
28
- // For now, let's assume `updateOverallInstallStatus` covers the visual needs.
29
- // If `updateCompletionUI` was doing something unique beyond what `updateOverallInstallStatus` does,
30
- // we could call it here.
31
- console.log(`[MessageQueue] Triggering final modal state: ${finalState}`);
32
- // if (finalState === 'completed') {
33
- // updateCompletionUI(); // If this has distinct visual changes for the modal wrapper itself.
34
- // }
11
+ let messageQueue = [];
12
+
13
+ /**
14
+ * Process messages in the queue with a delay between each message.
15
+ * When the queue is empty and completion is pending, triggers the completion UI.
16
+ * @private
17
+ */
18
+ function processMessageQueue() {
19
+ if (isAppending || messageQueue.length === 0) {
20
+ // Only update completion UI if all messages have been displayed
21
+ if (messageQueue.length === 0 && completionPending) {
22
+ // Add small delay before showing completion to ensure last message is visible
23
+ setTimeout(() => {
24
+ updateCompletionUI();
25
+ completionPending = false;
26
+ }, 500);
27
+ }
28
+ return;
29
+ }
30
+
31
+ isAppending = true;
32
+ const { msg, isCompletion } = messageQueue.shift();
33
+
34
+ if (isCompletion) {
35
+ completionPending = true;
36
+ }
37
+
38
+ appendInstallLoadingMessage(msg);
39
+
40
+ setTimeout(() => {
41
+ isAppending = false;
42
+ processMessageQueue();
43
+ }, 1000);
35
44
  }
36
45
 
46
+ /**
47
+ * Queue a message to be displayed with a delay.
48
+ * Messages are displayed sequentially with a 1-second delay between each.
49
+ * @param {string} msg The message to display
50
+ * @param {boolean} [isCompletion=false] If true, triggers completion UI after message display
51
+ */
52
+ export function delayedAppendInstallLoadingMessage(msg, isCompletion = false) {
53
+ messageQueue.push({ msg, isCompletion });
54
+ processMessageQueue();
55
+ }
56
+
57
+ function updateCompletionUI() {
58
+ const loadingTitle = document.querySelector('.loading-title');
59
+ const loadingIcon = document.querySelector('.loading-icon');
60
+
61
+ if (loadingTitle) {
62
+ loadingTitle.textContent = 'Completed';
63
+ loadingTitle.classList.add('completed');
64
+ }
65
+
66
+ if (loadingIcon) {
67
+ loadingIcon.innerHTML = `
68
+ <svg width="48" height="48" viewBox="0 0 48 48" fill="none">
69
+ <circle cx="24" cy="24" r="20" stroke="#059669" stroke-width="4"/>
70
+ <path d="M16 24l6 6 12-12" stroke="#059669" stroke-width="4" stroke-linecap="round" stroke-linejoin="round"/>
71
+ </svg>
72
+ `;
73
+ loadingIcon.classList.add('completed');
74
+ }
75
+ }
76
+
77
+ /**
78
+ * Internal function to append a message to the loading modal with formatting.
79
+ * @private
80
+ * @param {string} message - The message to append
81
+ */
82
+ function appendInstallLoadingMessage(message) {
83
+ const loadingMsg = document.getElementById('installLoadingMessage');
84
+ if (loadingMsg) {
85
+ // Split message on newlines
86
+ const lines = message.split('\n');
87
+
88
+ lines.forEach(line => {
89
+ // Check if line contains an error
90
+ const isError = line.toLowerCase().includes('error') ||
91
+ line.toLowerCase().includes('failed') ||
92
+ line.toLowerCase().includes('timeout');
37
93
 
38
- // function updateCompletionUI() {
39
- // // This function's original content might be merged into updateOverallInstallStatus
40
- // // or called if a distinct "modal completed" visual state is needed beyond the text/icon.
41
- // const loadingTitle = document.querySelector('.loading-title'); // This element might not exist with new structure
42
- // const loadingIcon = document.querySelector('.loading-icon'); // This element might not exist
94
+ const formattedMessage = line;
43
95
 
44
- // if (loadingTitle) {
45
- // loadingTitle.textContent = 'Completed';
46
- // loadingTitle.classList.add('completed');
47
- // }
96
+ // Create message container
97
+ const messageDiv = document.createElement('div');
98
+ messageDiv.className = 'message-line';
48
99
 
49
- // if (loadingIcon) {
50
- // // The main overall status icon is now handled by updateOverallInstallStatus
51
- // }
52
- // }
100
+ // Apply yellow color for error messages
101
+ if (isError) {
102
+ messageDiv.style.color = '#f59e0b';
103
+ }
104
+ messageDiv.innerHTML = formattedMessage;
53
105
 
54
- // The old delayedAppendInstallLoadingMessage and appendInstallLoadingMessage are removed
55
- // as step messages are now directly added by loadingModal.js->addInstallationStep
56
- // and overall status by loadingModal.js->updateOverallInstallStatus.
106
+ loadingMsg.appendChild(messageDiv);
107
+ loadingMsg.scrollTop = loadingMsg.scrollHeight;
108
+ });
109
+ } else {
110
+ console.error('[LoadingModal] Element not found: installLoadingMessage');
111
+ }
112
+ }
@@ -0,0 +1,214 @@
1
+ import { showConfirm } from '../notifications.js';
2
+ import { handleInstallation, handleUninstallation } from './installHandler.js';
3
+ import { showLoadingModal } from './loadingUI.js';
4
+ import { compareVersions } from './version.js';
5
+
6
+ /**
7
+ * Show installation modal for MCP tools
8
+ * @param {string} categoryName Category name
9
+ * @param {string} serverName Server name
10
+ */
11
+ export async function showInstallModal(categoryName, serverName) {
12
+ console.log("Showing install modal for:", serverName);
13
+ await new Promise(resolve => setTimeout(resolve, 100));
14
+
15
+ const elements = getModalElements();
16
+ if (!elements) return;
17
+
18
+ initializeModal(elements, serverName);
19
+
20
+ try {
21
+ const { targetData, serverData } = await fetchModalData(categoryName);
22
+ const mcpServer = findMcpServer(serverData, serverName);
23
+
24
+ renderModalContent(elements, {
25
+ targetData,
26
+ serverData,
27
+ mcpServer,
28
+ categoryName,
29
+ serverName
30
+ });
31
+
32
+ setupFormHandler(elements, categoryName, serverName, serverData);
33
+ elements.modal.style.display = "block";
34
+ } catch (error) {
35
+ handleModalError(error, elements.targetDiv);
36
+ }
37
+ }
38
+
39
+ /**
40
+ * Close modal and refresh with selected category
41
+ */
42
+ export function closeModal() {
43
+ const modal = document.getElementById('installModal');
44
+ if (modal) {
45
+ modal.style.display = "none";
46
+ refreshWithCategory();
47
+ }
48
+ }
49
+
50
+ /**
51
+ * Setup click outside modal handler
52
+ */
53
+ export function setupModalOutsideClick() {
54
+ window.onclick = function (event) {
55
+ const modal = document.getElementById('installModal');
56
+ if (event.target === modal) {
57
+ closeModal();
58
+ }
59
+ };
60
+ }
61
+
62
+ // Private helper functions
63
+
64
+ function getModalElements() {
65
+ const modal = document.getElementById('installModal');
66
+ if (!modal) {
67
+ console.error('Modal container not found');
68
+ return null;
69
+ }
70
+
71
+ const elements = {
72
+ modal,
73
+ title: modal.querySelector('#modalTitle'),
74
+ envInputs: modal.querySelector('#modalEnvInputs'),
75
+ targetDiv: modal.querySelector('#modalTargets'),
76
+ requirements: modal.querySelector('#modalRequirements'),
77
+ arguments: modal.querySelector('#modalArguments')
78
+ };
79
+
80
+ if (!Object.values(elements).every(Boolean)) {
81
+ console.error('Required modal elements not found');
82
+ return null;
83
+ }
84
+
85
+ return elements;
86
+ }
87
+
88
+ function initializeModal(elements, serverName) {
89
+ window.selectedClients = [];
90
+ elements.title.textContent = `Install ${serverName}`;
91
+ clearModalSections(elements);
92
+ }
93
+
94
+ function clearModalSections(elements) {
95
+ ['envInputs', 'targetDiv', 'requirements', 'arguments'].forEach(section => {
96
+ elements[section].innerHTML = '';
97
+ });
98
+ }
99
+
100
+ async function fetchModalData(categoryName) {
101
+ const [targetResponse, serverResponse] = await Promise.all([
102
+ fetch('/api/targets'),
103
+ fetch(`/api/categories/${categoryName}`)
104
+ ]);
105
+
106
+ if (!targetResponse.ok || !serverResponse.ok) {
107
+ throw new Error('Failed to fetch required data');
108
+ }
109
+
110
+ const [targetData, serverData] = await Promise.all([
111
+ targetResponse.json(),
112
+ serverResponse.json()
113
+ ]);
114
+
115
+ if (!targetData.success || !serverData.success) {
116
+ throw new Error('Invalid data received');
117
+ }
118
+
119
+ return { targetData, serverData };
120
+ }
121
+
122
+ function findMcpServer(serverData, serverName) {
123
+ const mcpServer = serverData.data.feedConfiguration.mcpServers
124
+ .find(server => server.name === serverName);
125
+
126
+ if (!mcpServer) {
127
+ throw new Error('Server configuration not found');
128
+ }
129
+
130
+ return mcpServer;
131
+ }
132
+
133
+ function renderModalContent(elements, data) {
134
+ const { targetData, serverData, mcpServer, categoryName, serverName } = data;
135
+ const installationStatus = serverData.data.installationStatus || {};
136
+ const serverStatuses = installationStatus.serversStatus || {};
137
+ const serverStatus = serverStatuses[serverName] || { installedStatus: {} };
138
+
139
+ renderClientSection(elements.targetDiv, targetData.data, serverStatus, serverName, categoryName);
140
+ renderEnvironmentSection(elements.envInputs, mcpServer, targetData.clientMcpSettings, serverName);
141
+ renderArgumentsSection(elements.arguments, mcpServer);
142
+ renderRequirementsSection(elements.requirements, mcpServer, serverData, categoryName, serverName);
143
+ }
144
+
145
+ function setupFormHandler(elements, categoryName, serverName, serverData) {
146
+ const form = document.getElementById('installForm');
147
+ form.onsubmit = (e) => handleFormSubmit(e, {
148
+ elements,
149
+ categoryName,
150
+ serverName,
151
+ serverData
152
+ });
153
+ }
154
+
155
+ function handleFormSubmit(event, options) {
156
+ event.preventDefault();
157
+ const { elements, categoryName, serverName, serverData } = options;
158
+
159
+ const formData = collectFormData(elements);
160
+ if (!validateFormData(formData)) return;
161
+
162
+ const installOptions = buildInstallOptions(formData, serverData, serverName);
163
+ handleInstallation(categoryName, serverName, formData.targets, installOptions);
164
+ }
165
+
166
+ function collectFormData(elements) {
167
+ return {
168
+ envVars: getEnvironmentVariables(elements.envInputs),
169
+ args: getArguments(elements.arguments),
170
+ pythonEnv: getPythonEnvironment(),
171
+ targets: getSelectedTargets(),
172
+ requirements: getSelectedRequirements()
173
+ };
174
+ }
175
+
176
+ function validateFormData(formData) {
177
+ const { targets, requirements } = formData;
178
+ if (targets.length === 0 && requirements.length === 0) {
179
+ showError('Please select at least one client or requirement to update.');
180
+ return false;
181
+ }
182
+ return true;
183
+ }
184
+
185
+ function handleModalError(error, targetDiv) {
186
+ console.error("Error loading data:", error);
187
+ targetDiv.innerHTML = `<p class="text-red-500">Error: ${error.message}</p>`;
188
+ }
189
+
190
+ function refreshWithCategory() {
191
+ const lastSelected = localStorage.getItem('lastSelectedCategory');
192
+ setTimeout(() => {
193
+ if (lastSelected) {
194
+ window.location.href = window.location.pathname + '?category=' + encodeURIComponent(lastSelected);
195
+ } else {
196
+ location.reload();
197
+ }
198
+ }, 100);
199
+ }
200
+
201
+ // Add CSS styles for the toggle switch
202
+ const styleElement = document.createElement('style');
203
+ styleElement.textContent = `
204
+ .toggle-bg.bg-blue-500 {
205
+ background-color: #3b82f6;
206
+ }
207
+ .toggle-bg {
208
+ transition: background-color 0.3s;
209
+ }
210
+ .toggle-bg div {
211
+ transition: transform 0.3s;
212
+ }
213
+ `;
214
+ document.head.appendChild(styleElement);
@@ -17,4 +17,4 @@ export function compareVersions(v1, v2) {
17
17
  }
18
18
 
19
19
  return 0;
20
- }
20
+ }