imcp 0.0.13 → 0.0.14

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 (44) hide show
  1. package/dist/core/installers/clients/ClientInstaller.js +58 -40
  2. package/dist/core/onboard/FeedOnboardService.d.ts +35 -0
  3. package/dist/core/onboard/FeedOnboardService.js +137 -0
  4. package/dist/core/types.d.ts +2 -1
  5. package/dist/core/validators/FeedValidator.d.ts +13 -0
  6. package/dist/core/validators/FeedValidator.js +27 -0
  7. package/dist/web/contract/serverContract.d.ts +64 -0
  8. package/dist/web/contract/serverContract.js +2 -0
  9. package/dist/web/public/css/onboard.css +44 -0
  10. package/dist/web/public/index.html +17 -13
  11. package/dist/web/public/js/modal/index.js +58 -0
  12. package/dist/web/public/js/modal/installHandler.js +227 -0
  13. package/dist/web/public/js/modal/installModal.js +163 -0
  14. package/dist/web/public/js/modal/installation.js +281 -0
  15. package/dist/web/public/js/modal/loadingModal.js +52 -0
  16. package/dist/web/public/js/modal/loadingUI.js +74 -0
  17. package/dist/web/public/js/modal/messageQueue.js +112 -0
  18. package/dist/web/public/js/modal/modalSetup.js +512 -0
  19. package/dist/web/public/js/modal/modalUI.js +214 -0
  20. package/dist/web/public/js/modal/modalUtils.js +49 -0
  21. package/dist/web/public/js/modal/version.js +20 -0
  22. package/dist/web/public/js/modal/versionUtils.js +20 -0
  23. package/dist/web/public/js/modal.js +25 -1041
  24. package/dist/web/public/js/onboard/formProcessor.js +309 -0
  25. package/dist/web/public/js/onboard/index.js +131 -0
  26. package/dist/web/public/js/onboard/state.js +32 -0
  27. package/dist/web/public/js/onboard/templates.js +375 -0
  28. package/dist/web/public/js/onboard/uiHandlers.js +196 -0
  29. package/dist/web/public/js/serverCategoryDetails.js +43 -17
  30. package/dist/web/public/onboard.html +150 -0
  31. package/package.json +1 -1
  32. package/src/core/installers/clients/ClientInstaller.ts +66 -49
  33. package/src/core/types.ts +2 -1
  34. package/src/web/public/index.html +17 -13
  35. package/src/web/public/js/modal/index.js +58 -0
  36. package/src/web/public/js/modal/installModal.js +163 -0
  37. package/src/web/public/js/modal/installation.js +281 -0
  38. package/src/web/public/js/modal/loadingModal.js +52 -0
  39. package/src/web/public/js/modal/messageQueue.js +112 -0
  40. package/src/web/public/js/modal/modalSetup.js +512 -0
  41. package/src/web/public/js/modal/modalUtils.js +49 -0
  42. package/src/web/public/js/modal/versionUtils.js +20 -0
  43. package/src/web/public/js/modal.js +25 -1041
  44. package/src/web/public/js/serverCategoryDetails.js +43 -17
@@ -0,0 +1,375 @@
1
+ // HTML templates for onboarding form
2
+ export const requirementTemplate = (index) => `
3
+ <div class="requirement-item p-3 border border-gray-200 rounded-lg mb-3" data-index="${index}">
4
+ <div class="grid grid-cols-2 gap-3">
5
+ <div>
6
+ <label class="block text-sm font-medium text-gray-700">Name*</label>
7
+ <input type="text" name="requirements[${index}].name" required
8
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
9
+ placeholder="Package or command name">
10
+ </div>
11
+ <div>
12
+ <label class="block text-sm font-medium text-gray-700">Type*</label>
13
+ <select name="requirements[${index}].type" required onchange="toggleAliasField(${index})"
14
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500">
15
+ <option value="">Select Type</option>
16
+ <option value="npm">NPM Package</option>
17
+ <option value="pip">PIP Package</option>
18
+ <option value="command">Command</option>
19
+ </select>
20
+ </div>
21
+ <div>
22
+ <label class="block text-sm font-medium text-gray-700">Version*</label>
23
+ <input type="text" name="requirements[${index}].version" required
24
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
25
+ placeholder="e.g., latest, v\${latest}, 1.0.0">
26
+ </div>
27
+ <div>
28
+ <label class="block text-sm font-medium text-gray-700">Order</label>
29
+ <input type="number" name="requirements[${index}].order" min="0"
30
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
31
+ placeholder="Installation order">
32
+ </div>
33
+ <div id="alias-field-${index}" class="hidden">
34
+ <label class="block text-sm font-medium text-gray-700">Alias</label>
35
+ <input type="text" name="requirements[${index}].alias"
36
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
37
+ placeholder="Command alias">
38
+ </div>
39
+ <div class="col-span-2">
40
+ <label class="block text-sm font-medium text-gray-700">Registry*</label>
41
+ <select name="requirements[${index}].registryType" onchange="toggleRegistryConfig(${index})"
42
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500" required>
43
+ <option value="">Select Registry Type</option>
44
+ <option value="public">Public Registry</option>
45
+ <option value="github">GitHub Release</option>
46
+ <option value="artifacts">Private Artifacts</option>
47
+ <option value="local">Local Registry</option>
48
+ </select>
49
+ </div>
50
+ </div>
51
+
52
+ <div id="registry-config-${index}" class="registry-configs">
53
+ <!-- GitHub Registry Config -->
54
+ <div id="github-config-${index}" class="registry-config mt-3 p-2 bg-gray-50 rounded-lg hidden">
55
+ <h3 class="text-sm font-medium text-gray-900 mb-2">GitHub Registry Details</h3>
56
+ <div class="grid grid-cols-1 gap-3">
57
+ <div>
58
+ <label class="block text-sm font-medium text-gray-700">Repository*</label>
59
+ <input type="text" name="requirements[${index}].registry.githubRelease.repository"
60
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
61
+ placeholder="org/repo">
62
+ </div>
63
+ <div>
64
+ <label class="block text-sm font-medium text-gray-700">Assets Name Pattern</label>
65
+ <input type="text" name="requirements[${index}].registry.githubRelease.assetsName"
66
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
67
+ placeholder="package-\${latest}.zip">
68
+ </div>
69
+ <div>
70
+ <label class="block text-sm font-medium text-gray-700">Asset Name Pattern*</label>
71
+ <input type="text" name="requirements[${index}].registry.githubRelease.assetName"
72
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
73
+ placeholder="package-\${latest}.tgz">
74
+ </div>
75
+ </div>
76
+ </div>
77
+
78
+ <!-- Artifacts Registry Config -->
79
+ <div id="artifacts-config-${index}" class="registry-config mt-3 p-2 bg-gray-50 rounded-lg hidden">
80
+ <h3 class="text-sm font-medium text-gray-900 mb-2">Private Artifacts Registry Details</h3>
81
+ <div class="grid grid-cols-1 gap-3">
82
+ <div>
83
+ <label class="block text-sm font-medium text-gray-700">Registry URL*</label>
84
+ <input type="text" name="requirements[${index}].registry.artifacts.registryUrl"
85
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
86
+ placeholder="https://artifacts.example.com">
87
+ </div>
88
+ <div>
89
+ <label class="block text-sm font-medium text-gray-700">Asset Name Pattern</label>
90
+ <input type="text" name="requirements[${index}].registry.artifacts.assetName"
91
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
92
+ placeholder="package-\${latest}.tgz">
93
+ </div>
94
+ </div>
95
+ </div>
96
+
97
+ <!-- Local Registry Config -->
98
+ <div id="local-config-${index}" class="registry-config mt-3 p-2 bg-gray-50 rounded-lg hidden">
99
+ <h3 class="text-sm font-medium text-gray-900 mb-2">Local Registry Details</h3>
100
+ <div class="grid grid-cols-1 gap-3">
101
+ <div>
102
+ <label class="block text-sm font-medium text-gray-700">Local Path*</label>
103
+ <input type="text" name="requirements[${index}].registry.local.localPath"
104
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
105
+ placeholder="/path/to/local/registry">
106
+ </div>
107
+ <div>
108
+ <label class="block text-sm font-medium text-gray-700">Asset Name Pattern</label>
109
+ <input type="text" name="requirements[${index}].registry.local.assetName"
110
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
111
+ placeholder="package-\${latest}.tgz">
112
+ </div>
113
+ </div>
114
+ </div>
115
+ </div>
116
+
117
+ <button type="button" onclick="removeRequirement(${index})"
118
+ class="mt-2 px-2 py-1 text-red-600 hover:text-red-800 flex items-center text-sm">
119
+ <i class='bx bx-trash mr-1'></i>
120
+ Remove
121
+ </button>
122
+ </div>
123
+ `;
124
+
125
+ export const serverTemplate = (index) => `
126
+ <div class="server-item p-3 border border-gray-200 rounded-lg mb-3" data-index="${index}">
127
+ <div class="grid grid-cols-2 gap-3">
128
+ <div>
129
+ <label class="block text-sm font-medium text-gray-700">Server Name*</label>
130
+ <input type="text" name="servers[${index}].name" required
131
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
132
+ placeholder="e.g., ado-tools">
133
+ </div>
134
+ <div>
135
+ <label class="block text-sm font-medium text-gray-700">Mode*</label>
136
+ <select name="servers[${index}].mode" required
137
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500">
138
+ <option value="stdio">stdio</option>
139
+ <option value="sse">sse</option>
140
+ </select>
141
+ </div>
142
+ <div class="col-span-2">
143
+ <label class="block text-sm font-medium text-gray-700">Description*</label>
144
+ <textarea name="servers[${index}].description" required rows="2"
145
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
146
+ placeholder="Server description"></textarea>
147
+ </div>
148
+ <div class="col-span-2">
149
+ <label class="block text-sm font-medium text-gray-700 mb-2">Requirements</label>
150
+ <div class="space-y-2" id="server-requirements-${index}">
151
+ <!-- Will be populated dynamically -->
152
+ </div>
153
+ <div class="mt-2">
154
+ <button type="button" onclick="addServerRequirement(${index})"
155
+ class="px-2 py-1 text-sm border border-gray-300 rounded-lg text-gray-600 hover:bg-gray-100 flex items-center">
156
+ <i class='bx bx-plus mr-1'></i>
157
+ Add Requirement
158
+ </button>
159
+ </div>
160
+ </div>
161
+ <div class="col-span-2 flex gap-2">
162
+ <div class="flex-grow">
163
+ <label class="block text-sm font-medium text-gray-700">Schema File</label>
164
+ <input type="text" name="servers[${index}].schemas" id="schema-path-${index}"
165
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
166
+ placeholder="e.g., server_tools.json">
167
+ </div>
168
+ <div class="flex items-end gap-2">
169
+ <button type="button" onclick="browseLocalSchema(${index})"
170
+ class="px-3 py-1.5 text-sm bg-gray-100 hover:bg-gray-200 rounded-lg">
171
+ Browse Local
172
+ </button>
173
+ <button type="button" onclick="browseGithubSchema(${index})"
174
+ class="px-3 py-1.5 text-sm bg-gray-100 hover:bg-gray-200 rounded-lg">
175
+ Browse GitHub
176
+ </button>
177
+ </div>
178
+ </div>
179
+ </div>
180
+
181
+ <div class="installation-config mt-3 p-2 bg-gray-50 rounded-lg">
182
+ <h3 class="text-sm font-medium text-gray-900 mb-2">Installation Configuration</h3>
183
+ <div class="grid grid-cols-2 gap-3">
184
+ <div>
185
+ <label class="block text-sm font-medium text-gray-700">Command*</label>
186
+ <input type="text" name="servers[${index}].installation.command" required
187
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
188
+ placeholder="e.g., node, python">
189
+ </div>
190
+ <div>
191
+ <label class="block text-sm font-medium text-gray-700">Arguments</label>
192
+ <input type="text" name="servers[${index}].installation.args"
193
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
194
+ placeholder="Comma-separated arguments">
195
+ </div>
196
+ </div>
197
+ </div>
198
+
199
+ <div class="env-variables mt-3 p-2 bg-gray-50 rounded-lg">
200
+ <h3 class="text-sm font-medium text-gray-900 mb-2">Environment Variables</h3>
201
+ <div id="envVars_${index}" class="space-y-2">
202
+ <!-- Environment variables will be added here -->
203
+ </div>
204
+ <button type="button" onclick="addEnvVariable(${index})"
205
+ class="mt-2 px-2 py-1 text-sm border border-gray-300 rounded-lg text-gray-600 hover:bg-gray-100 flex items-center">
206
+ <i class='bx bx-plus mr-1'></i>
207
+ Add Variable
208
+ </button>
209
+ </div>
210
+
211
+ <button type="button" onclick="removeServer(${index})"
212
+ class="mt-2 px-2 py-1 text-red-600 hover:text-red-800 flex items-center text-sm">
213
+ <i class='bx bx-trash mr-1'></i>
214
+ Remove Server
215
+ </button>
216
+ </div>
217
+ `;
218
+
219
+ export const envVariableTemplate = (serverIndex, envIndex) => `
220
+ <div class="env-var-item grid grid-cols-1 gap-2" data-env-index="${envIndex}">
221
+ <div class="grid grid-cols-3 gap-2 items-end">
222
+ <div>
223
+ <label class="block text-sm font-medium text-gray-700">Name*</label>
224
+ <input type="text" name="servers[${serverIndex}].installation.env[${envIndex}].name" required
225
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
226
+ placeholder="ENV_VAR_NAME">
227
+ </div>
228
+ <div>
229
+ <label class="block text-sm font-medium text-gray-700">Default Value</label>
230
+ <input type="text" name="servers[${serverIndex}].installation.env[${envIndex}].default"
231
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
232
+ placeholder="Default value">
233
+ </div>
234
+ <div class="flex items-center gap-2">
235
+ <label class="inline-flex items-center text-sm">
236
+ <input type="checkbox" name="servers[${serverIndex}].installation.env[${envIndex}].required"
237
+ class="text-blue-600 focus:ring-blue-500">
238
+ <span class="ml-1">Required</span>
239
+ </label>
240
+ <button type="button" onclick="removeEnvVariable(${serverIndex}, ${envIndex})"
241
+ class="px-2 py-1 text-red-600 hover:text-red-800">
242
+ <i class='bx bx-trash'></i>
243
+ </button>
244
+ </div>
245
+ </div>
246
+ <div>
247
+ <label class="block text-sm font-medium text-gray-700">Description</label>
248
+ <textarea name="servers[${serverIndex}].installation.env[${envIndex}].description" rows="2"
249
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
250
+ placeholder="Environment variable description"></textarea>
251
+ </div>
252
+ </div>
253
+ `;
254
+
255
+ export const serverRequirementTemplate = (serverIndex, reqIndex) => `
256
+ <div class="server-requirement-item p-3 border border-gray-200 rounded-lg mb-3" data-req-index="${reqIndex}">
257
+ <div class="grid grid-cols-2 gap-3">
258
+ <div>
259
+ <label class="block text-sm font-medium text-gray-700">Name*</label>
260
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].name" required
261
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
262
+ placeholder="Package or command name">
263
+ </div>
264
+ <div>
265
+ <label class="block text-sm font-medium text-gray-700">Type*</label>
266
+ <select name="servers[${serverIndex}].requirements[${reqIndex}].type" required onchange="toggleServerAliasField(${serverIndex}, ${reqIndex})"
267
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500">
268
+ <option value="">Select Type</option>
269
+ <option value="npm">NPM Package</option>
270
+ <option value="pip">PIP Package</option>
271
+ <option value="command">Command</option>
272
+ </select>
273
+ </div>
274
+ <div>
275
+ <label class="block text-sm font-medium text-gray-700">Version*</label>
276
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].version" required
277
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
278
+ placeholder="e.g., latest, v\${latest}, 1.0.0">
279
+ </div>
280
+ <div>
281
+ <label class="block text-sm font-medium text-gray-700">Order</label>
282
+ <input type="number" name="servers[${serverIndex}].requirements[${reqIndex}].order" min="0"
283
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
284
+ placeholder="Installation order">
285
+ </div>
286
+ <div id="server-alias-field-${serverIndex}-${reqIndex}" class="hidden">
287
+ <label class="block text-sm font-medium text-gray-700">Alias</label>
288
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].alias"
289
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
290
+ placeholder="Command alias">
291
+ </div>
292
+ <div class="col-span-2">
293
+ <label class="block text-sm font-medium text-gray-700">Registry*</label>
294
+ <select name="servers[${serverIndex}].requirements[${reqIndex}].registryType" onchange="toggleServerRegistryConfig(${serverIndex}, ${reqIndex})"
295
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500" required>
296
+ <option value="">Select Registry Type</option>
297
+ <option value="public">Public Registry</option>
298
+ <option value="github">GitHub Release</option>
299
+ <option value="artifacts">Private Artifacts</option>
300
+ <option value="local">Local Registry</option>
301
+ </select>
302
+ </div>
303
+
304
+ <div id="server-registry-config-${serverIndex}-${reqIndex}" class="col-span-2 registry-configs">
305
+ <!-- GitHub Registry Config -->
306
+ <div id="server-github-config-${serverIndex}-${reqIndex}" class="registry-config mt-3 p-2 bg-gray-50 rounded-lg hidden">
307
+ <h3 class="text-sm font-medium text-gray-900 mb-2">GitHub Registry Details</h3>
308
+ <div class="grid grid-cols-1 gap-3">
309
+ <div>
310
+ <label class="block text-sm font-medium text-gray-700">Repository*</label>
311
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].registry.githubRelease.repository"
312
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
313
+ placeholder="org/repo">
314
+ </div>
315
+ <div>
316
+ <label class="block text-sm font-medium text-gray-700">Assets Name Pattern</label>
317
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].registry.githubRelease.assetsName"
318
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
319
+ placeholder="package-\${latest}.zip">
320
+ </div>
321
+ <div>
322
+ <label class="block text-sm font-medium text-gray-700">Asset Name Pattern*</label>
323
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].registry.githubRelease.assetName"
324
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
325
+ placeholder="package-\${latest}.tgz">
326
+ </div>
327
+ </div>
328
+ </div>
329
+
330
+ <!-- Artifacts Registry Config -->
331
+ <div id="server-artifacts-config-${serverIndex}-${reqIndex}" class="registry-config mt-3 p-2 bg-gray-50 rounded-lg hidden">
332
+ <h3 class="text-sm font-medium text-gray-900 mb-2">Private Artifacts Registry Details</h3>
333
+ <div class="grid grid-cols-1 gap-3">
334
+ <div>
335
+ <label class="block text-sm font-medium text-gray-700">Registry URL*</label>
336
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].registry.artifacts.registryUrl"
337
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
338
+ placeholder="https://artifacts.example.com">
339
+ </div>
340
+ <div>
341
+ <label class="block text-sm font-medium text-gray-700">Asset Name Pattern</label>
342
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].registry.artifacts.assetName"
343
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
344
+ placeholder="package-\${latest}.tgz">
345
+ </div>
346
+ </div>
347
+ </div>
348
+
349
+ <!-- Local Registry Config -->
350
+ <div id="server-local-config-${serverIndex}-${reqIndex}" class="registry-config mt-3 p-2 bg-gray-50 rounded-lg hidden">
351
+ <h3 class="text-sm font-medium text-gray-900 mb-2">Local Registry Details</h3>
352
+ <div class="grid grid-cols-1 gap-3">
353
+ <div>
354
+ <label class="block text-sm font-medium text-gray-700">Local Path*</label>
355
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].registry.local.localPath"
356
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
357
+ placeholder="/path/to/local/registry">
358
+ </div>
359
+ <div>
360
+ <label class="block text-sm font-medium text-gray-700">Asset Name Pattern</label>
361
+ <input type="text" name="servers[${serverIndex}].requirements[${reqIndex}].registry.local.assetName"
362
+ class="w-full px-2 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
363
+ placeholder="package-\${latest}.tgz">
364
+ </div>
365
+ </div>
366
+ </div>
367
+ </div>
368
+ </div>
369
+ <button type="button" onclick="removeServerRequirement(${serverIndex}, ${reqIndex})"
370
+ class="mt-2 px-2 py-1 text-red-600 hover:text-red-800 flex items-center text-sm">
371
+ <i class='bx bx-trash mr-1'></i>
372
+ Remove
373
+ </button>
374
+ </div>
375
+ `;
@@ -0,0 +1,196 @@
1
+ import {
2
+ requirementTemplate,
3
+ serverTemplate,
4
+ envVariableTemplate,
5
+ serverRequirementTemplate
6
+ } from './templates.js';
7
+
8
+ import {
9
+ state,
10
+ incrementRequirementCounter,
11
+ incrementServerCounter,
12
+ setEnvCounter,
13
+ getEnvCounter,
14
+ deleteEnvCounter,
15
+ setServerRequirementCounter,
16
+ getServerRequirementCounter,
17
+ deleteServerRequirementCounter
18
+ } from './state.js';
19
+
20
+ // UI Event Handlers
21
+ export function addRequirement() {
22
+ const container = document.getElementById('requirementsList');
23
+ container.insertAdjacentHTML('beforeend', requirementTemplate(state.requirementCounter));
24
+ incrementRequirementCounter();
25
+ }
26
+
27
+ export function removeRequirement(index) {
28
+ const item = document.querySelector(`.requirement-item[data-index="${index}"]`);
29
+ if (item) {
30
+ item.remove();
31
+ }
32
+ }
33
+
34
+ export function addServer() {
35
+ const container = document.getElementById('serversList');
36
+ container.insertAdjacentHTML('beforeend', serverTemplate(state.serverCounter));
37
+ setEnvCounter(state.serverCounter, 0);
38
+ setServerRequirementCounter(state.serverCounter, 0);
39
+ incrementServerCounter();
40
+ }
41
+
42
+ export function removeServer(index) {
43
+ const item = document.querySelector(`.server-item[data-index="${index}"]`);
44
+ if (item) {
45
+ item.remove();
46
+ deleteEnvCounter(index);
47
+ deleteServerRequirementCounter(index);
48
+ }
49
+ }
50
+
51
+ export function addEnvVariable(serverIndex) {
52
+ const container = document.getElementById(`envVars_${serverIndex}`);
53
+ const envIndex = getEnvCounter(serverIndex);
54
+ container.insertAdjacentHTML('beforeend', envVariableTemplate(serverIndex, envIndex));
55
+ setEnvCounter(serverIndex, envIndex + 1);
56
+ }
57
+
58
+ export function removeEnvVariable(serverIndex, envIndex) {
59
+ const item = document.querySelector(`#envVars_${serverIndex} .env-var-item[data-env-index="${envIndex}"]`);
60
+ if (item) item.remove();
61
+ }
62
+
63
+ export function addServerRequirement(serverIndex) {
64
+ const container = document.getElementById(`server-requirements-${serverIndex}`);
65
+ const reqIndex = getServerRequirementCounter(serverIndex);
66
+ container.insertAdjacentHTML('beforeend', serverRequirementTemplate(serverIndex, reqIndex));
67
+ setServerRequirementCounter(serverIndex, reqIndex + 1);
68
+ }
69
+
70
+ export function removeServerRequirement(serverIndex, reqIndex) {
71
+ const item = document.querySelector(`#server-requirements-${serverIndex} .server-requirement-item[data-req-index="${reqIndex}"]`);
72
+ if (item) item.remove();
73
+ }
74
+
75
+ export function toggleAliasField(index) {
76
+ const typeSelect = document.querySelector(`[name="requirements[${index}].type"]`);
77
+ const aliasField = document.getElementById(`alias-field-${index}`);
78
+
79
+ if (typeSelect && aliasField) {
80
+ aliasField.classList.toggle('hidden', typeSelect.value !== 'command');
81
+ }
82
+ }
83
+
84
+ export function toggleServerAliasField(serverIndex, reqIndex) {
85
+ const typeSelect = document.querySelector(`[name="servers[${serverIndex}].requirements[${reqIndex}].type"]`);
86
+ const aliasField = document.getElementById(`server-alias-field-${serverIndex}-${reqIndex}`);
87
+
88
+ if (typeSelect && aliasField) {
89
+ aliasField.classList.toggle('hidden', typeSelect.value !== 'command');
90
+ }
91
+ }
92
+
93
+ export function toggleRegistryConfig(index) {
94
+ const select = document.querySelector(`[name="requirements[${index}].registryType"]`);
95
+ const githubConfig = document.getElementById(`github-config-${index}`);
96
+ const artifactsConfig = document.getElementById(`artifacts-config-${index}`);
97
+ const localConfig = document.getElementById(`local-config-${index}`);
98
+
99
+ if (select) {
100
+ // Hide all configurations first
101
+ [githubConfig, artifactsConfig, localConfig].forEach(config => {
102
+ if (config) config.classList.add('hidden');
103
+ });
104
+
105
+ // Show the selected configuration
106
+ switch (select.value) {
107
+ case 'github':
108
+ githubConfig?.classList.remove('hidden');
109
+ break;
110
+ case 'artifacts':
111
+ artifactsConfig?.classList.remove('hidden');
112
+ break;
113
+ case 'local':
114
+ localConfig?.classList.remove('hidden');
115
+ break;
116
+ }
117
+ }
118
+ }
119
+
120
+ export function toggleServerRegistryConfig(serverIndex, reqIndex) {
121
+ const select = document.querySelector(`[name="servers[${serverIndex}].requirements[${reqIndex}].registryType"]`);
122
+ const githubConfig = document.getElementById(`server-github-config-${serverIndex}-${reqIndex}`);
123
+ const artifactsConfig = document.getElementById(`server-artifacts-config-${serverIndex}-${reqIndex}`);
124
+ const localConfig = document.getElementById(`server-local-config-${serverIndex}-${reqIndex}`);
125
+
126
+ if (select) {
127
+ // Hide all configurations first
128
+ [githubConfig, artifactsConfig, localConfig].forEach(config => {
129
+ if (config) config.classList.add('hidden');
130
+ });
131
+
132
+ // Show the selected configuration
133
+ switch (select.value) {
134
+ case 'github':
135
+ githubConfig?.classList.remove('hidden');
136
+ break;
137
+ case 'artifacts':
138
+ artifactsConfig?.classList.remove('hidden');
139
+ break;
140
+ case 'local':
141
+ localConfig?.classList.remove('hidden');
142
+ break;
143
+ }
144
+ }
145
+ }
146
+
147
+ export async function browseLocalSchema(index) {
148
+ const schemaPath = document.getElementById(`schema-path-${index}`);
149
+
150
+ try {
151
+ if ('showOpenFilePicker' in window) {
152
+ const [fileHandle] = await window.showOpenFilePicker({
153
+ types: [{
154
+ description: 'JSON Files',
155
+ accept: {
156
+ 'application/json': ['.json']
157
+ }
158
+ }]
159
+ });
160
+ const file = await fileHandle.getFile();
161
+ schemaPath.value = file.name;
162
+ } else {
163
+ // Fallback for browsers that don't support the File System Access API
164
+ const input = document.createElement('input');
165
+ input.type = 'file';
166
+ input.accept = '.json';
167
+ input.onchange = (e) => {
168
+ if (e.target.files.length > 0) {
169
+ schemaPath.value = e.target.files[0].name;
170
+ }
171
+ };
172
+ input.click();
173
+ }
174
+ } catch (err) {
175
+ console.error('Error browsing for schema file:', err);
176
+ }
177
+ }
178
+
179
+ // Attach handlers to window for use in HTML
180
+ Object.entries({
181
+ addRequirement,
182
+ removeRequirement,
183
+ addServer,
184
+ removeServer,
185
+ addEnvVariable,
186
+ removeEnvVariable,
187
+ addServerRequirement,
188
+ removeServerRequirement,
189
+ toggleAliasField,
190
+ toggleServerAliasField,
191
+ toggleRegistryConfig,
192
+ toggleServerRegistryConfig,
193
+ browseLocalSchema
194
+ }).forEach(([name, fn]) => {
195
+ window[name] = fn;
196
+ });