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,359 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+
4
+ <head>
5
+ <meta charset="UTF-8">
6
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
7
+ <title>IMCP Server Manager</title>
8
+ <script src="https://cdn.tailwindcss.com"></script>
9
+ <link href="https://unpkg.com/boxicons@2.1.4/css/boxicons.min.css" rel="stylesheet">
10
+ <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
11
+ <link href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.7.2/font/bootstrap-icons.css" rel="stylesheet">
12
+ <link rel="stylesheet" href="styles.css">
13
+ <link rel="stylesheet" href="css/modal.css">
14
+ <link rel="stylesheet" href="css/notifications.css">
15
+ <link rel="stylesheet" href="css/serverDetails.css">
16
+ <link rel="stylesheet" href="css/detailsWidget.css">
17
+ <link rel="stylesheet" href="css/serverCategoryList.css">
18
+ <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script>
19
+
20
+ <!-- Alert container for notifications -->
21
+ <div class="alert-container position-fixed top-0 end-0 p-3">
22
+ </div>
23
+ </head>
24
+
25
+ <body class="bg-gray-50 min-h-screen">
26
+ <div class="container mx-auto px-4 py-6">
27
+ <div class="flex items-center justify-between mb-8">
28
+ <h1 class="text-3xl font-bold text-gray-900 flex items-center">
29
+ <a href="index.html" class="flex items-center text-gray-900 hover:text-blue-600 transition-colors">
30
+ <i class='bx bx-server mr-3 text-blue-600'></i>
31
+ IMCP Server Manager
32
+ </a>
33
+ <button id="settingsButton"
34
+ class="text-gray-700 hover:text-blue-600 transition-colors flex items-center ml-3 mt-1">
35
+ <i class='bx bx-cog text-3xl'></i>
36
+ </button>
37
+ </h1>
38
+ <div class="lg:w-2/3 flex justify-end items-center gap-4">
39
+ <div class="relative w-48 focus-within:w-96 transition-all duration-300 ease-in-out">
40
+ <input type="text" id="searchBox" placeholder="Search..."
41
+ data-expanded-placeholder="Search servers or tools..." class="w-full px-10 py-2.5 bg-gray-50 border border-gray-200 rounded-lg
42
+ text-gray-700 placeholder-gray-400
43
+ focus:border-blue-400 focus:ring-2 focus:ring-blue-100
44
+ transition-all duration-300 ease-in-out">
45
+ <i class='bx bx-search absolute left-3.5 top-1/2 transform -translate-y-1/2 text-gray-400'></i>
46
+ </div>
47
+ <button id="onboardButton" style="display: none;"
48
+ class="px-4 py-2 bg-blue-600 text-white rounded-lg hover:bg-blue-700 transition-colors flex items-center">
49
+ <i class='bx bx-plus-circle mr-2'></i>
50
+ Publish Server
51
+ </button>
52
+ </div>
53
+ <script>
54
+ document.addEventListener('DOMContentLoaded', () => {
55
+ const searchBox = document.getElementById('searchBox');
56
+ const expandedPlaceholder = searchBox.dataset.expandedPlaceholder;
57
+ searchBox.addEventListener('focus', () => {
58
+ setTimeout(() => searchBox.placeholder = expandedPlaceholder, 150);
59
+ });
60
+ searchBox.addEventListener('blur', () => {
61
+ if (!searchBox.value) {
62
+ searchBox.placeholder = 'Search...';
63
+ }
64
+ });
65
+ });
66
+ </script>
67
+ </div>
68
+
69
+ <div class="flex flex-col lg:flex-row gap-6">
70
+ <!-- Server List Sidebar -->
71
+ <div class="lg:w-1/3">
72
+ <div class="bg-white rounded-xl shadow-sm border border-gray-100 p-6">
73
+ <h2 class="text-xl font-semibold mb-4 text-gray-900 flex items-center">
74
+ <i class='bx bx-list-ul mr-2 text-blue-600'></i>
75
+ MCP Server Categories
76
+ </h2>
77
+ <div class="server-list space-y-3" id="serverCategoryList">
78
+ <div class="animate-pulse">
79
+ <p class="text-gray-500">Loading MCP server categories...</p>
80
+ </div>
81
+ </div>
82
+ </div>
83
+ </div>
84
+
85
+ <!-- Main Content Area -->
86
+ <div class="lg:w-2/3">
87
+ <div class="bg-white rounded-xl shadow-sm border border-gray-100 p-6">
88
+ <h2 class="text-xl font-semibold mb-4 text-gray-900 flex items-center">
89
+ <i class='bx bx-detail mr-2 text-blue-600'></i>
90
+ Server Category Details
91
+ </h2>
92
+ <div id="serverCategoryDetails" class="text-gray-600">
93
+ <div class="flex items-center justify-center h-32 text-gray-400">
94
+ <div class="text-center">
95
+ <i class='bx bx-selection text-4xl mb-2'></i>
96
+ <p>Select a server category from the list to see details</p>
97
+ </div>
98
+ </div>
99
+ </div>
100
+ <div id="toolMcpsList" class="mt-6 grid gap-4">
101
+ <!-- Tool cards will be dynamically inserted here -->
102
+ </div>
103
+ </div>
104
+ </div>
105
+ </div>
106
+ </div>
107
+
108
+ <!-- Installation Modal -->
109
+ <div id="installModal" class="modal">
110
+ <div class="modal-content">
111
+ <div class="flex justify-between items-center mb-4">
112
+ <h3 id="modalTitle" class="text-xl font-semibold text-gray-900">Install MCP Tool</h3>
113
+ <span class="close" onclick="closeModal()">&times;</span>
114
+ </div>
115
+ <form id="installForm">
116
+ <!-- Requirements Section -->
117
+ <div id="modalRequirements" class="mb-6">
118
+ <!-- Requirements will be injected here -->
119
+ </div>
120
+
121
+ <!-- Target Selection Section -->
122
+ <div id="modalTargets" class="bg-gray-50 rounded-lg">
123
+ <!-- Target options will be injected here -->
124
+ </div>
125
+
126
+ <!-- Environment Variables Section -->
127
+ <div id="modalEnvInputs" class="space-y-4">
128
+ <!-- Environment variable inputs will be injected here -->
129
+ </div>
130
+
131
+ <!-- Arguments Section -->
132
+ <div id="modalArguments" class="space-y-4 mb-6">
133
+ <!-- Arguments will be injected here -->
134
+ </div>
135
+
136
+ <div class="flex justify-end space-x-4 mt-6">
137
+ <button type="button" onclick="closeModal()"
138
+ class="px-4 py-2 text-gray-600 hover:text-gray-800 font-medium rounded-md hover:bg-gray-100 transition-colors">
139
+ Cancel
140
+ </button>
141
+ <button type="submit" class="submit-button">
142
+ <i class='bx bx-download mr-2'></i>
143
+ Install
144
+ </button>
145
+ </div>
146
+ </form>
147
+ </div>
148
+ </div>
149
+ <!-- Loading Modal -->
150
+ <div id="installLoadingModal" class="modal" style="display:none; z-index:2000;">
151
+ <div class="modal-content" style="text-align:center; pointer-events:auto; position:relative;">
152
+ <button class="modal-close-button" onclick="hideInstallLoadingModal()" aria-label="Close">&times;</button>
153
+ <div style="margin-top:40px;">
154
+ <div class="loading-icon" style="margin-bottom:16px;">
155
+ <svg width="48" height="48" viewBox="0 0 48 48" fill="none">
156
+ <circle cx="24" cy="24" r="20" stroke="#888" stroke-width="4" opacity="0.2" />
157
+ <circle cx="24" cy="24" r="20" stroke="#3498db" stroke-width="4" stroke-linecap="round"
158
+ stroke-dasharray="100" stroke-dashoffset="60">
159
+ <animateTransform attributeName="transform" type="rotate" from="0 24 24" to="360 24 24"
160
+ dur="1s" repeatCount="indefinite" />
161
+ </circle>
162
+ </svg>
163
+ </div>
164
+ <div class="loading-title" style="font-size:1.5rem; font-weight:bold; margin-bottom:8px;">Installing...
165
+ </div>
166
+ <div id="installLoadingMessage"
167
+ style="min-height:48px; max-height:160px; overflow:auto; background:#f8f8f8; border-radius:6px; padding:12px; font-size:1rem; color:#444; text-align:left;">
168
+ </div>
169
+ </div>
170
+ </div>
171
+ </div>
172
+
173
+ <!-- Module scripts -->
174
+ <script type="module">
175
+ import { fetchServerCategories, handleInstallServer, uninstallTool } from './js/api.js';
176
+ import { setupSearch } from './js/serverCategoryList.js';
177
+ import { showServerDetails } from './js/serverCategoryDetails.js';
178
+ import { showInstallModal, closeModal } from './js/modal.js';
179
+ // import { showOnboardModal as navigateToOnboard } from './js/onboard/index.js'; // We are directly navigating
180
+ import flights, { getFlightQueryParameters, buildUrlWithFlights } from './js/flights/flights.js';
181
+
182
+ // Make necessary functions available globally
183
+ window.showServerDetails = showServerDetails;
184
+ window.showInstallModal = showInstallModal;
185
+ // window.showOnboardModal = showOnboardModal; // This might conflict if showOnboardModal is also in onboard/index.js
186
+ // We'll rely on the event listener above for the main page's onboard button.
187
+ // If other parts of the app call window.showOnboardModal, they'll need updating too.
188
+ window.closeModal = closeModal;
189
+ window.handleInstallServer = handleInstallServer;
190
+ window.uninstallTool = uninstallTool;
191
+
192
+ // Initialize
193
+ document.addEventListener('DOMContentLoaded', async () => {
194
+ setupSearch();
195
+ const onboardButton = document.getElementById('onboardButton');
196
+ if (flights.enableOnboard) {
197
+ onboardButton.style.display = 'flex'; // Make it visible
198
+ onboardButton.addEventListener('click', () => {
199
+ const flightQueryString = getFlightQueryParameters();
200
+ window.location.href = buildUrlWithFlights('onboard.html');
201
+ });
202
+ }
203
+
204
+ const settingsButton = document.getElementById('settingsButton');
205
+ settingsButton.addEventListener('click', () => {
206
+ window.location.href = buildUrlWithFlights('settings.html');
207
+ });
208
+
209
+ // No 'else' block needed as it's hidden by default via inline style
210
+
211
+ try {
212
+ // First fetch categories and wait for completion
213
+ // fetchServerCategories will call renderServerCategoryList, which might call loadLastSelectedCategory.
214
+ // loadLastSelectedCategory calls showServerDetails.
215
+ // The logic for setting initial category URL is now more centralized.
216
+
217
+ await fetchServerCategories(); // This populates allServerCategoriesData
218
+
219
+ const urlParams = new URLSearchParams(window.location.search);
220
+ const categoryParam = urlParams.get('category');
221
+ const lastSelectedCategoryFromStorage = localStorage.getItem('lastSelectedCategory');
222
+
223
+ if (categoryParam) {
224
+ // Category is in URL, load it
225
+ await window.showServerDetails(categoryParam);
226
+ } else if (lastSelectedCategoryFromStorage) {
227
+ // Category not in URL, but in localStorage. Redirect to URL with this category and flights.
228
+ window.location.href = buildUrlWithFlights('index.html', { category: lastSelectedCategoryFromStorage });
229
+ return; // Stop further execution, page will reload
230
+ } else if (allServerCategoriesData && allServerCategoriesData.length > 0) {
231
+ // No category in URL or localStorage, pick first from list and redirect.
232
+ window.location.href = buildUrlWithFlights('index.html', { category: allServerCategoriesData[0].name });
233
+ return; // Stop further execution, page will reload
234
+ } else {
235
+ }
236
+
237
+ } catch (error) {
238
+ console.error('Error during initialization:', error);
239
+ // Potentially show a user-friendly error message in the UI
240
+ const serverCategoryDetailsDiv = document.getElementById('serverCategoryDetails');
241
+ if (serverCategoryDetailsDiv) {
242
+ serverCategoryDetailsDiv.innerHTML = `<p class="text-red-500">Error initializing page: ${error.message}</p>`;
243
+ }
244
+ const serverCategoryListDiv = document.getElementById('serverCategoryList');
245
+ if (serverCategoryListDiv) {
246
+ serverCategoryListDiv.innerHTML = `<p class="text-red-500">Could not load categories.</p>`;
247
+ }
248
+ }
249
+
250
+ // Fetch and display application version
251
+ try {
252
+ const versionResponse = await fetch('/api/version');
253
+ const appVersionSpan = document.getElementById('appVersion');
254
+ const versionInfoContainer = document.getElementById('versionInfoContainer');
255
+ const versionIcon = versionInfoContainer.querySelector('i');
256
+
257
+ // Create tooltip element
258
+ const tooltip = document.createElement('div');
259
+ tooltip.className = 'version-tooltip';
260
+ versionInfoContainer.appendChild(tooltip);
261
+
262
+ let tooltipMessage = "Loading version information...";
263
+ let isUpdateAvailable = false;
264
+
265
+ if (versionResponse.ok) {
266
+ const versionData = await versionResponse.json();
267
+ if (versionData.success && versionData.data) {
268
+ const currentVersion = versionData.data.version || 'N/A';
269
+ appVersionSpan.textContent = currentVersion;
270
+
271
+ if (versionData.data.availableUpdate) {
272
+ isUpdateAvailable = true;
273
+ tooltipMessage = versionData.data.availableUpdate.message;
274
+ } else {
275
+ tooltipMessage = "You're already up to date!";
276
+ }
277
+ } else {
278
+ appVersionSpan.textContent = 'N/A';
279
+ tooltipMessage = "Could not retrieve version information.";
280
+ }
281
+ } else {
282
+ console.error('Failed to fetch version:', versionResponse.status);
283
+ appVersionSpan.textContent = 'N/A';
284
+ tooltipMessage = "Failed to fetch version information.";
285
+ }
286
+
287
+ tooltip.textContent = tooltipMessage;
288
+
289
+ if (isUpdateAvailable) {
290
+ versionInfoContainer.classList.add('text-orange-500');
291
+ versionIcon.classList.add('text-orange-500');
292
+ appVersionSpan.classList.add('text-orange-500');
293
+ }
294
+
295
+ versionInfoContainer.addEventListener('mouseenter', () => {
296
+ tooltip.classList.add('visible');
297
+ });
298
+
299
+ versionInfoContainer.addEventListener('mouseleave', () => {
300
+ // Only hide if the mouse isn't over the tooltip itself
301
+ setTimeout(() => { // Add a small delay to allow moving to the tooltip
302
+ if (!tooltip.matches(':hover')) {
303
+ tooltip.classList.remove('visible');
304
+ }
305
+ }, 100);
306
+ });
307
+
308
+ tooltip.addEventListener('mouseleave', () => {
309
+ tooltip.classList.remove('visible');
310
+ });
311
+
312
+ // Keep tooltip visible if clicked, and allow clicking outside to close
313
+ versionInfoContainer.addEventListener('click', (event) => {
314
+ event.stopPropagation();
315
+ tooltip.classList.add('visible'); // Ensure it's visible on click
316
+ });
317
+
318
+ document.addEventListener('click', (event) => {
319
+ if (!versionInfoContainer.contains(event.target) && !tooltip.contains(event.target)) {
320
+ tooltip.classList.remove('visible');
321
+ }
322
+ });
323
+
324
+
325
+ } catch (error) {
326
+ console.error('Error fetching version:', error);
327
+ const appVersionSpan = document.getElementById('appVersion');
328
+ if (appVersionSpan) {
329
+ appVersionSpan.textContent = 'N/A';
330
+ const versionInfoContainer = document.getElementById('versionInfoContainer');
331
+ const tooltip = versionInfoContainer.querySelector('.version-tooltip');
332
+ if (tooltip) {
333
+ tooltip.textContent = "Error fetching version information.";
334
+ }
335
+ }
336
+ }
337
+ });
338
+ </script>
339
+ <script src="js/modal.js" type="module"></script>
340
+
341
+ <footer class="text-sm text-gray-600 py-6 mt-10 border-t border-gray-200">
342
+ <div class="container mx-auto px-4">
343
+ <div class="flex flex-col sm:flex-row justify-start items-start sm:items-center space-y-2 sm:space-y-0 sm:space-x-6 px-6">
344
+ <a href="https://github.com/ai-microsoft/imcp" target="_blank" rel="noopener noreferrer"
345
+ class="flex items-center text-blue-600 hover:text-blue-700 hover:underline transition-colors"
346
+ title="Welcome for any feedback.">
347
+ <i class='bx bxl-github mr-1.5'></i> GitHub
348
+ </a>
349
+ <span class="hidden sm:inline text-gray-300">|</span>
350
+ <span id="versionInfoContainer" class="version-info-container flex items-center text-gray-700">
351
+ <i class='bx bx-info-circle mr-1.5 text-gray-500'></i>
352
+ Version: <span id="appVersion" class="ml-1 font-medium">Loading...</span>
353
+ </span>
354
+ </div>
355
+ </div>
356
+ </footer>
357
+ </body>
358
+
359
+ </html>
@@ -0,0 +1,132 @@
1
+ import { renderServerCategoryList } from './serverCategoryList.js';
2
+ import { showToast } from './notifications.js';
3
+
4
+ // Global store for server data
5
+ let allServerCategoriesData = [];
6
+ // Make it available globally for legacy code
7
+ window.allServerCategoriesData = allServerCategoriesData;
8
+
9
+ // Fetch and display servers
10
+ async function fetchServerCategories() {
11
+ try {
12
+ // Fetch all servers (without filters)
13
+ const response = await fetch('/api/categories');
14
+ if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`);
15
+ const responseData = await response.json();
16
+ if (!responseData.success || !responseData.data) {
17
+ throw new Error(`API Error fetching servers: ${responseData.error || 'Invalid data format'}`);
18
+ }
19
+ allServerCategoriesData = window.allServerCategoriesData = responseData.data; // Update both local and global references
20
+
21
+ // Use the renderServerCategoryList function to display servers
22
+ renderServerCategoryList(allServerCategoriesData);
23
+
24
+ // Check if search is active
25
+ const searchTerm = document.getElementById('searchBox').value.toLowerCase();
26
+ if (searchTerm) {
27
+ const filteredServerCategories = allServerCategoriesData.filter(server =>
28
+ (server.displayName || server.name).toLowerCase().includes(searchTerm) ||
29
+ (server.description || '').toLowerCase().includes(searchTerm)
30
+ );
31
+ renderServerCategoryList(filteredServerCategories);
32
+ }
33
+ } catch (error) {
34
+ console.error('Error fetching servers:', error);
35
+ document.getElementById('serverCategoryList').innerHTML =
36
+ '<p class="text-red-600">Error loading servers. Please check console and try again.</p>';
37
+ document.getElementById('serverCategoryDetails').innerHTML = ''; // Clear details on error
38
+ }
39
+ }
40
+
41
+ // Handle MCP tool installation
42
+ async function handleInstallServer(event, categoryName, serverName, callback, options = {}) {
43
+ event.preventDefault(); // Prevent page reload
44
+ console.log("Handling install for:", categoryName, serverName);
45
+
46
+ // Get selected targets from the form if not provided in options
47
+ const selectedTargets = options.targets || [...document.querySelectorAll('input[name="targets"]:checked')]
48
+ .map(cb => cb.value);
49
+
50
+ if (selectedTargets.length === 0) {
51
+ showToast('Please select at least one target', 'error');
52
+ return;
53
+ }
54
+
55
+ // Get environment variables from the form
56
+ const form = event.target;
57
+ const formData = new FormData(form);
58
+ const envValues = {};
59
+ formData.forEach((value, key) => {
60
+ if (!key.startsWith('targets') && value) { // Skip target checkboxes
61
+ envValues[key] = value;
62
+ }
63
+ });
64
+
65
+ try {
66
+ const serverList = {};
67
+ serverList[serverName] = {
68
+ env: envValues,
69
+ targetClients: selectedTargets
70
+ }; // Use serverName as key
71
+ const response = await fetch(`/api/categories/${categoryName}/install`, {
72
+ method: 'POST',
73
+ headers: { 'Content-Type': 'application/json' },
74
+ body: JSON.stringify({ serverList: serverList })
75
+ });
76
+
77
+ if (!response.ok) {
78
+ const errorData = await response.text();
79
+ throw new Error(`Installation failed: ${errorData || response.statusText}`);
80
+ }
81
+
82
+ showToast('MCP server "' + serverName + '" for ' + categoryName + ' installation initiated successfully!', 'success');
83
+ window.closeModal();
84
+ fetchServerCategories(); // Refresh the main server list
85
+ window.showServerDetails(categoryName); // Refresh the details pane
86
+
87
+ } catch (error) {
88
+ console.error('Error installing MCP server:', error);
89
+ showToast('Error installing MCP server: ' + error.message + '. Please check console.', 'error');
90
+ }
91
+ }
92
+
93
+
94
+ async function uninstallTool(name) {
95
+ const serverTitle = document.querySelector('#serverCategoryDetails h3')?.textContent;
96
+ if (!serverTitle) {
97
+ showToast('No server selected.', 'error');
98
+ return;
99
+ }
100
+
101
+ const uninstallConfirmed = await showConfirm(`Do you want to uninstall tool '${name}' from ${serverTitle}?`);
102
+ if (!uninstallConfirmed) {
103
+ return;
104
+ }
105
+
106
+ try {
107
+ const response = await fetch(`/api/categories/${serverTitle}/uninstall-tool`, {
108
+ method: 'POST',
109
+ headers: { 'Content-Type': 'application/json' },
110
+ body: JSON.stringify({ toolName: name })
111
+ });
112
+
113
+ if (!response.ok) {
114
+ const errorData = await response.text();
115
+ throw new Error(`Uninstall failed: ${errorData || response.statusText}`);
116
+ }
117
+
118
+ showToast(`Tool '${name}' uninstall initiated successfully!`, 'success');
119
+ fetchServerCategories();
120
+ window.showServerDetails(serverTitle);
121
+ } catch (error) {
122
+ console.error('Error uninstalling tool:', error);
123
+ showToast(`Error uninstalling tool: ${error.message}`, 'error');
124
+ }
125
+ }
126
+
127
+ export {
128
+ allServerCategoriesData,
129
+ fetchServerCategories,
130
+ handleInstallServer,
131
+ uninstallTool
132
+ };