@opensumi/ide-ai-native 3.8.1-next-1740478950.0 → 3.8.1-next-1740556231.0
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.
- package/lib/browser/ai-core.contribution.d.ts +4 -1
- package/lib/browser/ai-core.contribution.d.ts.map +1 -1
- package/lib/browser/ai-core.contribution.js +23 -11
- package/lib/browser/ai-core.contribution.js.map +1 -1
- package/lib/browser/chat/chat-manager.service.d.ts +1 -5
- package/lib/browser/chat/chat-manager.service.d.ts.map +1 -1
- package/lib/browser/chat/chat-manager.service.js +13 -17
- package/lib/browser/chat/chat-manager.service.js.map +1 -1
- package/lib/browser/chat/chat-model.d.ts +0 -2
- package/lib/browser/chat/chat-model.d.ts.map +1 -1
- package/lib/browser/chat/chat-model.js +2 -8
- package/lib/browser/chat/chat-model.js.map +1 -1
- package/lib/browser/chat/chat.internal.service.d.ts +1 -0
- package/lib/browser/chat/chat.internal.service.d.ts.map +1 -1
- package/lib/browser/chat/chat.internal.service.js +3 -0
- package/lib/browser/chat/chat.internal.service.js.map +1 -1
- package/lib/browser/chat/chat.module.less +2 -1
- package/lib/browser/chat/chat.view.d.ts.map +1 -1
- package/lib/browser/chat/chat.view.js +9 -6
- package/lib/browser/chat/chat.view.js.map +1 -1
- package/lib/browser/components/ChatContext/index.js +2 -2
- package/lib/browser/components/ChatContext/index.js.map +1 -1
- package/lib/browser/components/ChatHistory.d.ts +0 -1
- package/lib/browser/components/ChatHistory.d.ts.map +1 -1
- package/lib/browser/components/ChatHistory.js +14 -14
- package/lib/browser/components/ChatHistory.js.map +1 -1
- package/lib/browser/components/ChatInput.d.ts.map +1 -1
- package/lib/browser/components/ChatInput.js +1 -25
- package/lib/browser/components/ChatInput.js.map +1 -1
- package/lib/browser/components/ChatToolRender.d.ts.map +1 -1
- package/lib/browser/components/ChatToolRender.js +3 -2
- package/lib/browser/components/ChatToolRender.js.map +1 -1
- package/lib/browser/components/{chat-history.css → chat-history.module.less} +1 -1
- package/lib/browser/components/components.module.less +0 -20
- package/lib/browser/context/llm-context.service.d.ts +5 -16
- package/lib/browser/context/llm-context.service.d.ts.map +1 -1
- package/lib/browser/context/llm-context.service.js +47 -78
- package/lib/browser/context/llm-context.service.js.map +1 -1
- package/lib/browser/index.d.ts.map +1 -1
- package/lib/browser/index.js +0 -4
- package/lib/browser/index.js.map +1 -1
- package/lib/browser/layout/layout.module.less +4 -4
- package/lib/browser/mcp/base-apply.service.d.ts +31 -40
- package/lib/browser/mcp/base-apply.service.d.ts.map +1 -1
- package/lib/browser/mcp/base-apply.service.js +233 -167
- package/lib/browser/mcp/base-apply.service.js.map +1 -1
- package/lib/browser/mcp/mcp-server-proxy.service.d.ts +0 -6
- package/lib/browser/mcp/mcp-server-proxy.service.d.ts.map +1 -1
- package/lib/browser/mcp/mcp-server-proxy.service.js +1 -10
- package/lib/browser/mcp/mcp-server-proxy.service.js.map +1 -1
- package/lib/browser/mcp/mcp-server.feature.registry.d.ts.map +1 -1
- package/lib/browser/mcp/mcp-server.feature.registry.js +2 -3
- package/lib/browser/mcp/mcp-server.feature.registry.js.map +1 -1
- package/lib/browser/mcp/tools/components/EditFile.d.ts.map +1 -1
- package/lib/browser/mcp/tools/components/EditFile.js +55 -41
- package/lib/browser/mcp/tools/components/EditFile.js.map +1 -1
- package/lib/browser/mcp/tools/components/index.module.less +22 -4
- package/lib/browser/mcp/tools/createNewFileWithText.d.ts.map +1 -1
- package/lib/browser/mcp/tools/createNewFileWithText.js +0 -1
- package/lib/browser/mcp/tools/createNewFileWithText.js.map +1 -1
- package/lib/browser/mcp/tools/editFile.js +1 -1
- package/lib/browser/mcp/tools/editFile.js.map +1 -1
- package/lib/browser/mcp/tools/getDiagnosticsByPath.d.ts.map +1 -1
- package/lib/browser/mcp/tools/getDiagnosticsByPath.js +0 -1
- package/lib/browser/mcp/tools/getDiagnosticsByPath.js.map +1 -1
- package/lib/browser/mcp/tools/handlers/EditFile.d.ts +5 -1
- package/lib/browser/mcp/tools/handlers/EditFile.d.ts.map +1 -1
- package/lib/browser/mcp/tools/handlers/EditFile.js +4 -4
- package/lib/browser/mcp/tools/handlers/EditFile.js.map +1 -1
- package/lib/browser/mcp/tools/handlers/RunCommand.d.ts.map +1 -1
- package/lib/browser/mcp/tools/handlers/RunCommand.js +0 -2
- package/lib/browser/mcp/tools/handlers/RunCommand.js.map +1 -1
- package/lib/browser/mcp/tools/runTerminalCmd.d.ts.map +1 -1
- package/lib/browser/mcp/tools/runTerminalCmd.js +0 -1
- package/lib/browser/mcp/tools/runTerminalCmd.js.map +1 -1
- package/lib/browser/model/msg-history-manager.d.ts +1 -0
- package/lib/browser/model/msg-history-manager.d.ts.map +1 -1
- package/lib/browser/model/msg-history-manager.js +12 -2
- package/lib/browser/model/msg-history-manager.js.map +1 -1
- package/lib/browser/preferences/schema.d.ts.map +1 -1
- package/lib/browser/preferences/schema.js +0 -16
- package/lib/browser/preferences/schema.js.map +1 -1
- package/lib/browser/types.d.ts +1 -1
- package/lib/browser/types.d.ts.map +1 -1
- package/lib/browser/widget/inline-diff/inline-diff-manager.d.ts +6 -0
- package/lib/browser/widget/inline-diff/inline-diff-manager.d.ts.map +1 -0
- package/lib/browser/widget/inline-diff/inline-diff-manager.js +27 -0
- package/lib/browser/widget/inline-diff/inline-diff-manager.js.map +1 -0
- package/lib/browser/widget/inline-diff/inline-diff-widget.module.less +12 -0
- package/lib/browser/widget/inline-stream-diff/inline-stream-diff.handler.d.ts +2 -0
- package/lib/browser/widget/inline-stream-diff/inline-stream-diff.handler.d.ts.map +1 -1
- package/lib/browser/widget/inline-stream-diff/inline-stream-diff.handler.js +11 -4
- package/lib/browser/widget/inline-stream-diff/inline-stream-diff.handler.js.map +1 -1
- package/lib/common/index.d.ts +1 -8
- package/lib/common/index.d.ts.map +1 -1
- package/lib/common/index.js +1 -3
- package/lib/common/index.js.map +1 -1
- package/lib/common/llm-context.d.ts +9 -12
- package/lib/common/llm-context.d.ts.map +1 -1
- package/lib/common/llm-context.js.map +1 -1
- package/lib/common/mcp-server-manager.d.ts +1 -17
- package/lib/common/mcp-server-manager.d.ts.map +1 -1
- package/lib/common/mcp-server-manager.js.map +1 -1
- package/lib/common/prompts/context-prompt-provider.d.ts +3 -2
- package/lib/common/prompts/context-prompt-provider.d.ts.map +1 -1
- package/lib/common/prompts/context-prompt-provider.js +22 -21
- package/lib/common/prompts/context-prompt-provider.js.map +1 -1
- package/lib/common/tool-invocation-registry.d.ts +2 -2
- package/lib/common/tool-invocation-registry.d.ts.map +1 -1
- package/lib/common/tool-invocation-registry.js +1 -1
- package/lib/common/tool-invocation-registry.js.map +1 -1
- package/lib/common/types.d.ts +17 -6
- package/lib/common/types.d.ts.map +1 -1
- package/lib/common/types.js.map +1 -1
- package/lib/common/utils.d.ts.map +1 -1
- package/lib/common/utils.js +1 -2
- package/lib/common/utils.js.map +1 -1
- package/lib/node/base-language-model.d.ts +1 -1
- package/lib/node/base-language-model.d.ts.map +1 -1
- package/lib/node/base-language-model.js +54 -3
- package/lib/node/base-language-model.js.map +1 -1
- package/lib/node/mcp/sumi-mcp-server.d.ts +3 -17
- package/lib/node/mcp/sumi-mcp-server.d.ts.map +1 -1
- package/lib/node/mcp/sumi-mcp-server.js +6 -59
- package/lib/node/mcp/sumi-mcp-server.js.map +1 -1
- package/lib/node/mcp-server-manager-impl.d.ts +3 -4
- package/lib/node/mcp-server-manager-impl.d.ts.map +1 -1
- package/lib/node/mcp-server-manager-impl.js +6 -26
- package/lib/node/mcp-server-manager-impl.js.map +1 -1
- package/lib/node/mcp-server.d.ts +16 -5
- package/lib/node/mcp-server.d.ts.map +1 -1
- package/lib/node/mcp-server.js +6 -12
- package/lib/node/mcp-server.js.map +1 -1
- package/lib/node/openai/openai-language-model.d.ts +3 -4
- package/lib/node/openai/openai-language-model.d.ts.map +1 -1
- package/lib/node/openai/openai-language-model.js +2 -3
- package/lib/node/openai/openai-language-model.js.map +1 -1
- package/package.json +27 -27
- package/src/browser/ai-core.contribution.ts +29 -14
- package/src/browser/chat/chat-manager.service.ts +13 -17
- package/src/browser/chat/chat-model.ts +3 -18
- package/src/browser/chat/chat.internal.service.ts +4 -0
- package/src/browser/chat/chat.module.less +2 -1
- package/src/browser/chat/chat.view.tsx +26 -10
- package/src/browser/components/ChatContext/index.tsx +2 -2
- package/src/browser/components/ChatHistory.tsx +21 -15
- package/src/browser/components/ChatInput.tsx +4 -67
- package/src/browser/components/ChatToolRender.tsx +2 -1
- package/src/browser/components/{chat-history.css → chat-history.module.less} +1 -1
- package/src/browser/components/components.module.less +0 -20
- package/src/browser/context/llm-context.service.ts +54 -90
- package/src/browser/index.ts +0 -4
- package/src/browser/layout/layout.module.less +4 -4
- package/src/browser/mcp/base-apply.service.ts +266 -213
- package/src/browser/mcp/mcp-server-proxy.service.ts +2 -14
- package/src/browser/mcp/mcp-server.feature.registry.ts +2 -3
- package/src/browser/mcp/tools/components/EditFile.tsx +82 -60
- package/src/browser/mcp/tools/components/index.module.less +22 -4
- package/src/browser/mcp/tools/createNewFileWithText.ts +0 -1
- package/src/browser/mcp/tools/editFile.ts +2 -2
- package/src/browser/mcp/tools/getDiagnosticsByPath.ts +0 -1
- package/src/browser/mcp/tools/handlers/EditFile.ts +4 -4
- package/src/browser/mcp/tools/handlers/RunCommand.ts +0 -2
- package/src/browser/mcp/tools/runTerminalCmd.ts +0 -1
- package/src/browser/model/msg-history-manager.ts +12 -2
- package/src/browser/preferences/schema.ts +0 -16
- package/src/browser/types.ts +1 -1
- package/src/browser/widget/inline-diff/inline-diff-manager.tsx +38 -0
- package/src/browser/widget/inline-diff/inline-diff-widget.module.less +12 -0
- package/src/browser/widget/inline-stream-diff/inline-stream-diff.handler.tsx +13 -4
- package/src/common/index.ts +1 -7
- package/src/common/llm-context.ts +4 -10
- package/src/common/mcp-server-manager.ts +1 -17
- package/src/common/prompts/context-prompt-provider.ts +28 -26
- package/src/common/tool-invocation-registry.ts +2 -2
- package/src/common/types.ts +20 -6
- package/src/common/utils.ts +1 -3
- package/src/node/base-language-model.ts +63 -1
- package/src/node/mcp/sumi-mcp-server.ts +9 -67
- package/src/node/mcp-server-manager-impl.ts +9 -30
- package/src/node/mcp-server.ts +14 -11
- package/src/node/openai/openai-language-model.ts +4 -7
- package/lib/browser/mcp/config/components/mcp-config.module.less +0 -178
- package/lib/browser/mcp/config/components/mcp-config.view.d.ts +0 -3
- package/lib/browser/mcp/config/components/mcp-config.view.d.ts.map +0 -1
- package/lib/browser/mcp/config/components/mcp-config.view.js +0 -150
- package/lib/browser/mcp/config/components/mcp-config.view.js.map +0 -1
- package/lib/browser/mcp/config/components/mcp-server-form.d.ts +0 -16
- package/lib/browser/mcp/config/components/mcp-server-form.d.ts.map +0 -1
- package/lib/browser/mcp/config/components/mcp-server-form.js +0 -84
- package/lib/browser/mcp/config/components/mcp-server-form.js.map +0 -1
- package/lib/browser/mcp/config/components/mcp-server-form.module.less +0 -78
- package/lib/browser/mcp/config/mcp-config.commands.d.ts +0 -10
- package/lib/browser/mcp/config/mcp-config.commands.d.ts.map +0 -1
- package/lib/browser/mcp/config/mcp-config.commands.js +0 -35
- package/lib/browser/mcp/config/mcp-config.commands.js.map +0 -1
- package/lib/browser/mcp/config/mcp-config.contribution.d.ts +0 -16
- package/lib/browser/mcp/config/mcp-config.contribution.d.ts.map +0 -1
- package/lib/browser/mcp/config/mcp-config.contribution.js +0 -62
- package/lib/browser/mcp/config/mcp-config.contribution.js.map +0 -1
- package/src/browser/mcp/config/components/mcp-config.module.less +0 -178
- package/src/browser/mcp/config/components/mcp-config.view.tsx +0 -215
- package/src/browser/mcp/config/components/mcp-server-form.module.less +0 -78
- package/src/browser/mcp/config/components/mcp-server-form.tsx +0 -144
- package/src/browser/mcp/config/mcp-config.commands.ts +0 -29
- package/src/browser/mcp/config/mcp-config.contribution.ts +0 -65
|
@@ -1,62 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.MCPConfigContribution = exports.MCP_CONFIG_COMPONENTS_SCHEME_ID = void 0;
|
|
4
|
-
const tslib_1 = require("tslib");
|
|
5
|
-
const di_1 = require("@opensumi/di");
|
|
6
|
-
const services_1 = require("@opensumi/ide-core-browser/lib/services");
|
|
7
|
-
const ide_core_common_1 = require("@opensumi/ide-core-common");
|
|
8
|
-
const types_1 = require("@opensumi/ide-editor/lib/browser/types");
|
|
9
|
-
const browser_1 = require("@opensumi/ide-theme/lib/browser");
|
|
10
|
-
const common_1 = require("@opensumi/ide-workspace/lib/common");
|
|
11
|
-
const mcp_config_view_1 = require("./components/mcp-config.view");
|
|
12
|
-
const COMPONENTS_ID = 'opensumi-mcp-config-viewer';
|
|
13
|
-
exports.MCP_CONFIG_COMPONENTS_SCHEME_ID = 'mcp-config';
|
|
14
|
-
let MCPConfigContribution = class MCPConfigContribution {
|
|
15
|
-
registerEditorComponent(registry) {
|
|
16
|
-
registry.registerEditorComponent({
|
|
17
|
-
uid: COMPONENTS_ID,
|
|
18
|
-
scheme: exports.MCP_CONFIG_COMPONENTS_SCHEME_ID,
|
|
19
|
-
component: mcp_config_view_1.MCPConfigView,
|
|
20
|
-
renderMode: types_1.EditorComponentRenderMode.ONE_PER_WORKBENCH,
|
|
21
|
-
});
|
|
22
|
-
registry.registerEditorComponentResolver(exports.MCP_CONFIG_COMPONENTS_SCHEME_ID, (resource, results) => {
|
|
23
|
-
results.push({
|
|
24
|
-
type: 'component',
|
|
25
|
-
componentId: COMPONENTS_ID,
|
|
26
|
-
});
|
|
27
|
-
});
|
|
28
|
-
}
|
|
29
|
-
registerResource(service) {
|
|
30
|
-
service.registerResourceProvider({
|
|
31
|
-
scheme: exports.MCP_CONFIG_COMPONENTS_SCHEME_ID,
|
|
32
|
-
provideResource: async (uri) => {
|
|
33
|
-
const { configType } = uri.getParsedQuery();
|
|
34
|
-
return {
|
|
35
|
-
uri,
|
|
36
|
-
name: 'MCP Configuration',
|
|
37
|
-
icon: 'settings',
|
|
38
|
-
metadata: {
|
|
39
|
-
configType,
|
|
40
|
-
},
|
|
41
|
-
};
|
|
42
|
-
},
|
|
43
|
-
});
|
|
44
|
-
}
|
|
45
|
-
};
|
|
46
|
-
exports.MCPConfigContribution = MCPConfigContribution;
|
|
47
|
-
tslib_1.__decorate([
|
|
48
|
-
(0, di_1.Autowired)(common_1.IWorkspaceService),
|
|
49
|
-
tslib_1.__metadata("design:type", Object)
|
|
50
|
-
], MCPConfigContribution.prototype, "workspaceService", void 0);
|
|
51
|
-
tslib_1.__decorate([
|
|
52
|
-
(0, di_1.Autowired)(browser_1.IconService),
|
|
53
|
-
tslib_1.__metadata("design:type", browser_1.IconService)
|
|
54
|
-
], MCPConfigContribution.prototype, "iconService", void 0);
|
|
55
|
-
tslib_1.__decorate([
|
|
56
|
-
(0, di_1.Autowired)(),
|
|
57
|
-
tslib_1.__metadata("design:type", services_1.LabelService)
|
|
58
|
-
], MCPConfigContribution.prototype, "labelService", void 0);
|
|
59
|
-
exports.MCPConfigContribution = MCPConfigContribution = tslib_1.__decorate([
|
|
60
|
-
(0, ide_core_common_1.Domain)(types_1.BrowserEditorContribution)
|
|
61
|
-
], MCPConfigContribution);
|
|
62
|
-
//# sourceMappingURL=mcp-config.contribution.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"mcp-config.contribution.js","sourceRoot":"","sources":["../../../../src/browser/mcp/config/mcp-config.contribution.ts"],"names":[],"mappings":";;;;AAAA,qCAAyC;AACzC,sEAAuE;AACvE,+DAAiE;AACjE,kEAMgD;AAChD,6DAA8D;AAC9D,+DAAuE;AAEvE,kEAA6D;AAE7D,MAAM,aAAa,GAAG,4BAA4B,CAAC;AACtC,QAAA,+BAA+B,GAAG,YAAY,CAAC;AAKrD,IAAM,qBAAqB,GAA3B,MAAM,qBAAqB;IAUhC,uBAAuB,CAAC,QAAiC;QACvD,QAAQ,CAAC,uBAAuB,CAAC;YAC/B,GAAG,EAAE,aAAa;YAClB,MAAM,EAAE,uCAA+B;YACvC,SAAS,EAAE,+BAAa;YACxB,UAAU,EAAE,iCAAyB,CAAC,iBAAiB;SACxD,CAAC,CAAC;QAEH,QAAQ,CAAC,+BAA+B,CAAC,uCAA+B,EAAE,CAAC,QAAQ,EAAE,OAAO,EAAE,EAAE;YAC9F,OAAO,CAAC,IAAI,CAAC;gBACX,IAAI,EAAE,WAAW;gBACjB,WAAW,EAAE,aAAa;aAC3B,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED,gBAAgB,CAAC,OAAwB;QACvC,OAAO,CAAC,wBAAwB,CAAC;YAC/B,MAAM,EAAE,uCAA+B;YACvC,eAAe,EAAE,KAAK,EAAE,GAAQ,EAA+B,EAAE;gBAC/D,MAAM,EAAE,UAAU,EAAE,GAAG,GAAG,CAAC,cAAc,EAAE,CAAC;gBAE5C,OAAO;oBACL,GAAG;oBACH,IAAI,EAAE,mBAAmB;oBACzB,IAAI,EAAE,UAAU;oBAChB,QAAQ,EAAE;wBACR,UAAU;qBACX;iBACF,CAAC;YACJ,CAAC;SACF,CAAC,CAAC;IACL,CAAC;CACF,CAAA;AA3CY,sDAAqB;AAEb;IADlB,IAAA,cAAS,EAAC,0BAAiB,CAAC;;+DAC0B;AAGpC;IADlB,IAAA,cAAS,EAAC,qBAAW,CAAC;sCACS,qBAAW;0DAAC;AAG5C;IADC,IAAA,cAAS,GAAE;sCACE,uBAAY;2DAAC;gCARhB,qBAAqB;IADjC,IAAA,wBAAM,EAAC,iCAAyB,CAAC;GACrB,qBAAqB,CA2CjC"}
|
|
@@ -1,178 +0,0 @@
|
|
|
1
|
-
.container {
|
|
2
|
-
padding: 20px;
|
|
3
|
-
color: var(--foreground);
|
|
4
|
-
}
|
|
5
|
-
|
|
6
|
-
.header {
|
|
7
|
-
display: flex;
|
|
8
|
-
justify-content: space-between;
|
|
9
|
-
align-items: flex-start;
|
|
10
|
-
margin-bottom: 24px;
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
.title {
|
|
14
|
-
margin: 0 0 8px 0;
|
|
15
|
-
font-size: 24px;
|
|
16
|
-
font-weight: 600;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
.description {
|
|
20
|
-
margin: 0;
|
|
21
|
-
color: var(--descriptionForeground);
|
|
22
|
-
font-size: 14px;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
.addButton {
|
|
26
|
-
padding: 8px 16px;
|
|
27
|
-
border-radius: 4px;
|
|
28
|
-
background-color: var(--button-primary-background);
|
|
29
|
-
color: var(--button-primary-foreground);
|
|
30
|
-
border: none;
|
|
31
|
-
cursor: pointer;
|
|
32
|
-
font-size: 13px;
|
|
33
|
-
|
|
34
|
-
&:hover {
|
|
35
|
-
background-color: var(--button-primary-hover-background);
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
.serversList {
|
|
40
|
-
display: flex;
|
|
41
|
-
flex-direction: column;
|
|
42
|
-
gap: 12px;
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
.serverItem {
|
|
46
|
-
padding: 16px;
|
|
47
|
-
border-radius: 8px;
|
|
48
|
-
background-color: var(--editorWidget-background);
|
|
49
|
-
border: 1px solid var(--border-color);
|
|
50
|
-
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.05);
|
|
51
|
-
transition: all 0.2s ease;
|
|
52
|
-
|
|
53
|
-
&:hover {
|
|
54
|
-
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
|
|
55
|
-
border-color: var(--focusBorder);
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
.serverHeader {
|
|
60
|
-
display: flex;
|
|
61
|
-
justify-content: space-between;
|
|
62
|
-
align-items: center;
|
|
63
|
-
margin-bottom: 12px;
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
.serverTitleRow {
|
|
67
|
-
display: flex;
|
|
68
|
-
align-items: center;
|
|
69
|
-
gap: 8px;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
.serverName {
|
|
73
|
-
margin: 0;
|
|
74
|
-
font-size: 14px;
|
|
75
|
-
font-weight: 500;
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
.serverStatus,
|
|
79
|
-
.serverType {
|
|
80
|
-
font-size: 12px;
|
|
81
|
-
padding: 2px 8px;
|
|
82
|
-
border-radius: 12px;
|
|
83
|
-
font-weight: 500;
|
|
84
|
-
display: inline-flex;
|
|
85
|
-
align-items: center;
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
.serverType {
|
|
89
|
-
background-color: var(--editor-background);
|
|
90
|
-
color: var(--descriptionForeground);
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
.running {
|
|
94
|
-
background-color: var(--notification-info-background);
|
|
95
|
-
color: var(--notification-info-foreground);
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
.stopped {
|
|
99
|
-
background-color: var(--notification-error-background);
|
|
100
|
-
color: var(--notification-error-foreground);
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
.serverActions {
|
|
104
|
-
display: flex;
|
|
105
|
-
gap: 4px;
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
.iconButton {
|
|
109
|
-
padding: 4px;
|
|
110
|
-
border: none;
|
|
111
|
-
background: none;
|
|
112
|
-
color: var(--icon-foreground);
|
|
113
|
-
cursor: pointer;
|
|
114
|
-
border-radius: 4px;
|
|
115
|
-
|
|
116
|
-
&:hover {
|
|
117
|
-
background-color: var(--list-hover-background);
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
.serverDetail {
|
|
122
|
-
margin-top: 12px;
|
|
123
|
-
padding: 12px;
|
|
124
|
-
border-radius: 6px;
|
|
125
|
-
background-color: var(--editor-background);
|
|
126
|
-
font-size: 13px;
|
|
127
|
-
|
|
128
|
-
&:not(:last-child) {
|
|
129
|
-
margin-bottom: 8px;
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
.detailRow {
|
|
134
|
-
display: flex;
|
|
135
|
-
gap: 8px;
|
|
136
|
-
margin-bottom: 4px;
|
|
137
|
-
align-items: center;
|
|
138
|
-
|
|
139
|
-
&:last-child {
|
|
140
|
-
margin-bottom: 0;
|
|
141
|
-
}
|
|
142
|
-
}
|
|
143
|
-
|
|
144
|
-
.detailLabel {
|
|
145
|
-
color: var(--descriptionForeground);
|
|
146
|
-
min-width: 70px;
|
|
147
|
-
height: 24px;
|
|
148
|
-
display: flex;
|
|
149
|
-
align-items: center;
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
.detailContent {
|
|
153
|
-
color: var(--foreground);
|
|
154
|
-
word-break: break-all;
|
|
155
|
-
flex: 1;
|
|
156
|
-
background-color: var(--editor-background);
|
|
157
|
-
padding: 4px 8px;
|
|
158
|
-
border-radius: 4px;
|
|
159
|
-
font-family: var(--monaco-monospace-font);
|
|
160
|
-
min-height: 24px;
|
|
161
|
-
display: flex;
|
|
162
|
-
align-items: center;
|
|
163
|
-
gap: 8px;
|
|
164
|
-
flex-wrap: wrap;
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
.toolTag {
|
|
168
|
-
display: inline-flex;
|
|
169
|
-
align-items: center;
|
|
170
|
-
padding: 2px 8px;
|
|
171
|
-
background-color: var(--editorWidget-background);
|
|
172
|
-
color: var(--button-secondary-foreground);
|
|
173
|
-
border-radius: 12px;
|
|
174
|
-
font-size: 12px;
|
|
175
|
-
font-weight: 500;
|
|
176
|
-
transition: all 0.2s ease;
|
|
177
|
-
border: 1px solid var(--border-color);
|
|
178
|
-
}
|
|
@@ -1,215 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
|
|
3
|
-
import { AINativeSettingSectionsId, ILogger, useInjectable } from '@opensumi/ide-core-browser';
|
|
4
|
-
import { PreferenceService } from '@opensumi/ide-core-browser/lib/preferences';
|
|
5
|
-
|
|
6
|
-
import { BUILTIN_MCP_SERVER_NAME } from '../../../../common';
|
|
7
|
-
import { MCPServerDescription } from '../../../../common/mcp-server-manager';
|
|
8
|
-
import { MCPServerProxyService } from '../../mcp-server-proxy.service';
|
|
9
|
-
|
|
10
|
-
import styles from './mcp-config.module.less';
|
|
11
|
-
import { MCPServerForm, MCPServerFormData } from './mcp-server-form';
|
|
12
|
-
|
|
13
|
-
interface MCPServer {
|
|
14
|
-
name: string;
|
|
15
|
-
isStarted: boolean;
|
|
16
|
-
tools?: string[];
|
|
17
|
-
command?: string;
|
|
18
|
-
type?: string;
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
export const MCPConfigView: React.FC = () => {
|
|
22
|
-
const mcpServerProxyService = useInjectable<MCPServerProxyService>(MCPServerProxyService);
|
|
23
|
-
const preferenceService = useInjectable<PreferenceService>(PreferenceService);
|
|
24
|
-
const logger = useInjectable<ILogger>(ILogger);
|
|
25
|
-
const [servers, setServers] = React.useState<MCPServer[]>([]);
|
|
26
|
-
const [formVisible, setFormVisible] = React.useState(false);
|
|
27
|
-
const [editingServer, setEditingServer] = React.useState<MCPServerFormData | undefined>();
|
|
28
|
-
|
|
29
|
-
const loadServers = React.useCallback(async () => {
|
|
30
|
-
const allServers = await mcpServerProxyService.$getServers();
|
|
31
|
-
setServers(allServers);
|
|
32
|
-
}, [mcpServerProxyService]);
|
|
33
|
-
|
|
34
|
-
React.useEffect(() => {
|
|
35
|
-
loadServers();
|
|
36
|
-
const disposer = mcpServerProxyService.onChangeMCPServers(() => {
|
|
37
|
-
loadServers();
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
return () => {
|
|
41
|
-
disposer.dispose();
|
|
42
|
-
};
|
|
43
|
-
}, [mcpServerProxyService, loadServers]);
|
|
44
|
-
|
|
45
|
-
const handleServerControl = async (serverName: string, start: boolean) => {
|
|
46
|
-
try {
|
|
47
|
-
if (start) {
|
|
48
|
-
await mcpServerProxyService.$startServer(serverName);
|
|
49
|
-
} else {
|
|
50
|
-
await mcpServerProxyService.$stopServer(serverName);
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
// Update enabled state in preferences
|
|
54
|
-
const servers = preferenceService.get<MCPServerDescription[]>(AINativeSettingSectionsId.MCPServers, []);
|
|
55
|
-
let updatedServers = servers;
|
|
56
|
-
|
|
57
|
-
// 处理内置服务器的特殊情况
|
|
58
|
-
if (serverName === BUILTIN_MCP_SERVER_NAME) {
|
|
59
|
-
const builtinServerExists = servers.some((server) => server.name === BUILTIN_MCP_SERVER_NAME);
|
|
60
|
-
if (!builtinServerExists && !start) {
|
|
61
|
-
// 如果是停止内置服务器且之前没有配置,添加一个新的配置项
|
|
62
|
-
// 内置服务器不需要 command,因为它是直接集成在 IDE 中的
|
|
63
|
-
updatedServers = [
|
|
64
|
-
...servers,
|
|
65
|
-
{
|
|
66
|
-
name: BUILTIN_MCP_SERVER_NAME,
|
|
67
|
-
enabled: false,
|
|
68
|
-
command: '', // 内置服务器的 command 为空字符串
|
|
69
|
-
},
|
|
70
|
-
];
|
|
71
|
-
} else {
|
|
72
|
-
// 如果已经存在配置,更新 enabled 状态
|
|
73
|
-
updatedServers = servers.map((server) => {
|
|
74
|
-
if (server.name === BUILTIN_MCP_SERVER_NAME) {
|
|
75
|
-
return { ...server, enabled: start };
|
|
76
|
-
}
|
|
77
|
-
return server;
|
|
78
|
-
});
|
|
79
|
-
}
|
|
80
|
-
} else {
|
|
81
|
-
// 处理其他外部服务器
|
|
82
|
-
updatedServers = servers.map((server) => {
|
|
83
|
-
if (server.name === serverName) {
|
|
84
|
-
return { ...server, enabled: start };
|
|
85
|
-
}
|
|
86
|
-
return server;
|
|
87
|
-
});
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
await preferenceService.set(AINativeSettingSectionsId.MCPServers, updatedServers);
|
|
91
|
-
await loadServers();
|
|
92
|
-
} catch (error) {
|
|
93
|
-
logger.error(`Failed to ${start ? 'start' : 'stop'} server ${serverName}:`, error);
|
|
94
|
-
}
|
|
95
|
-
};
|
|
96
|
-
|
|
97
|
-
const handleAddServer = () => {
|
|
98
|
-
setEditingServer(undefined);
|
|
99
|
-
setFormVisible(true);
|
|
100
|
-
};
|
|
101
|
-
|
|
102
|
-
const handleEditServer = (server: MCPServer) => {
|
|
103
|
-
const servers = preferenceService.get<MCPServerFormData[]>(AINativeSettingSectionsId.MCPServers, []);
|
|
104
|
-
const serverConfig = servers.find((s) => s.name === server.name);
|
|
105
|
-
|
|
106
|
-
if (serverConfig) {
|
|
107
|
-
setEditingServer(serverConfig);
|
|
108
|
-
setFormVisible(true);
|
|
109
|
-
}
|
|
110
|
-
};
|
|
111
|
-
|
|
112
|
-
const handleDeleteServer = async (serverName: string) => {
|
|
113
|
-
const servers = preferenceService.get<MCPServerFormData[]>(AINativeSettingSectionsId.MCPServers, []);
|
|
114
|
-
const updatedServers = servers.filter((s) => s.name !== serverName);
|
|
115
|
-
await preferenceService.set(AINativeSettingSectionsId.MCPServers, updatedServers);
|
|
116
|
-
await loadServers();
|
|
117
|
-
};
|
|
118
|
-
|
|
119
|
-
const handleSaveServer = async (data: MCPServerFormData) => {
|
|
120
|
-
const servers = preferenceService.get<MCPServerFormData[]>(AINativeSettingSectionsId.MCPServers, []);
|
|
121
|
-
const existingIndex = servers.findIndex((s) => s.name === data.name);
|
|
122
|
-
|
|
123
|
-
if (existingIndex >= 0) {
|
|
124
|
-
servers[existingIndex] = data;
|
|
125
|
-
} else {
|
|
126
|
-
servers.push(data);
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
await preferenceService.set(AINativeSettingSectionsId.MCPServers, servers);
|
|
130
|
-
setFormVisible(false);
|
|
131
|
-
await loadServers();
|
|
132
|
-
};
|
|
133
|
-
|
|
134
|
-
return (
|
|
135
|
-
<div className={styles.container}>
|
|
136
|
-
<div className={styles.header}>
|
|
137
|
-
<div>
|
|
138
|
-
<h2 className={styles.title}>MCP Servers</h2>
|
|
139
|
-
<p className={styles.description}>Manage your MCP server connections.</p>
|
|
140
|
-
</div>
|
|
141
|
-
<button className={styles.addButton} onClick={handleAddServer}>
|
|
142
|
-
+ Add new MCP server
|
|
143
|
-
</button>
|
|
144
|
-
</div>
|
|
145
|
-
<div className={styles.serversList}>
|
|
146
|
-
{servers.map((server) => (
|
|
147
|
-
<div key={server.name} className={styles.serverItem}>
|
|
148
|
-
<div className={styles.serverHeader}>
|
|
149
|
-
<div className={styles.serverTitleRow}>
|
|
150
|
-
<h3 className={styles.serverName}>{server.name}</h3>
|
|
151
|
-
</div>
|
|
152
|
-
<div className={styles.serverActions}>
|
|
153
|
-
<button className={styles.iconButton} title='Edit' onClick={() => handleEditServer(server)}>
|
|
154
|
-
<i className='codicon codicon-edit' />
|
|
155
|
-
</button>
|
|
156
|
-
<button
|
|
157
|
-
className={styles.iconButton}
|
|
158
|
-
title={server.isStarted ? 'Stop' : 'Start'}
|
|
159
|
-
onClick={() => handleServerControl(server.name, !server.isStarted)}
|
|
160
|
-
>
|
|
161
|
-
<i className={`codicon ${server.isStarted ? 'codicon-debug-stop' : 'codicon-debug-start'}`} />
|
|
162
|
-
</button>
|
|
163
|
-
<button className={styles.iconButton} title='Delete' onClick={() => handleDeleteServer(server.name)}>
|
|
164
|
-
<i className='codicon codicon-trash' />
|
|
165
|
-
</button>
|
|
166
|
-
</div>
|
|
167
|
-
</div>
|
|
168
|
-
<div className={styles.serverDetail}>
|
|
169
|
-
<div className={styles.detailRow}>
|
|
170
|
-
<span className={styles.detailLabel}>Status:</span>
|
|
171
|
-
<span className={`${styles.serverStatus} ${server.isStarted ? styles.running : styles.stopped}`}>
|
|
172
|
-
{server.isStarted ? 'Running' : 'Stopped'}
|
|
173
|
-
</span>
|
|
174
|
-
</div>
|
|
175
|
-
{server.type && (
|
|
176
|
-
<div className={styles.detailRow}>
|
|
177
|
-
<span className={styles.detailLabel}>Type:</span>
|
|
178
|
-
<span className={styles.serverType}>{server.type}</span>
|
|
179
|
-
</div>
|
|
180
|
-
)}
|
|
181
|
-
</div>
|
|
182
|
-
{server.tools && server.tools.length > 0 && (
|
|
183
|
-
<div className={styles.serverDetail}>
|
|
184
|
-
<div className={styles.detailRow}>
|
|
185
|
-
<span className={styles.detailLabel}>Tools:</span>
|
|
186
|
-
<span className={styles.detailContent}>
|
|
187
|
-
{server.tools.map((tool, index) => (
|
|
188
|
-
<span key={index} className={styles.toolTag}>
|
|
189
|
-
{tool}
|
|
190
|
-
</span>
|
|
191
|
-
))}
|
|
192
|
-
</span>
|
|
193
|
-
</div>
|
|
194
|
-
</div>
|
|
195
|
-
)}
|
|
196
|
-
{server.command && (
|
|
197
|
-
<div className={styles.serverDetail}>
|
|
198
|
-
<div className={styles.detailRow}>
|
|
199
|
-
<span className={styles.detailLabel}>Command:</span>
|
|
200
|
-
<span className={styles.detailContent}>{server.command}</span>
|
|
201
|
-
</div>
|
|
202
|
-
</div>
|
|
203
|
-
)}
|
|
204
|
-
</div>
|
|
205
|
-
))}
|
|
206
|
-
</div>
|
|
207
|
-
<MCPServerForm
|
|
208
|
-
visible={formVisible}
|
|
209
|
-
initialData={editingServer}
|
|
210
|
-
onSave={handleSaveServer}
|
|
211
|
-
onCancel={() => setFormVisible(false)}
|
|
212
|
-
/>
|
|
213
|
-
</div>
|
|
214
|
-
);
|
|
215
|
-
};
|
|
@@ -1,78 +0,0 @@
|
|
|
1
|
-
.form {
|
|
2
|
-
padding: 16px;
|
|
3
|
-
}
|
|
4
|
-
|
|
5
|
-
.formItem {
|
|
6
|
-
margin-bottom: 16px;
|
|
7
|
-
|
|
8
|
-
label {
|
|
9
|
-
display: block;
|
|
10
|
-
margin-bottom: 8px;
|
|
11
|
-
color: var(--foreground);
|
|
12
|
-
font-size: 13px;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
input,
|
|
16
|
-
textarea {
|
|
17
|
-
width: 100%;
|
|
18
|
-
padding: 8px;
|
|
19
|
-
border: 1px solid var(--input-border);
|
|
20
|
-
border-radius: 4px;
|
|
21
|
-
background-color: var(--input-background);
|
|
22
|
-
color: var(--input-foreground);
|
|
23
|
-
font-size: 13px;
|
|
24
|
-
font-family: var(--monaco-monospace-font);
|
|
25
|
-
|
|
26
|
-
&:focus {
|
|
27
|
-
border-color: var(--focusBorder);
|
|
28
|
-
outline: none;
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
&::placeholder {
|
|
32
|
-
color: var(--descriptionForeground);
|
|
33
|
-
opacity: 0.6;
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
textarea {
|
|
38
|
-
resize: vertical;
|
|
39
|
-
min-height: 60px;
|
|
40
|
-
}
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
.formActions {
|
|
44
|
-
display: flex;
|
|
45
|
-
justify-content: flex-end;
|
|
46
|
-
gap: 8px;
|
|
47
|
-
margin-top: 24px;
|
|
48
|
-
padding-top: 16px;
|
|
49
|
-
border-top: 1px solid var(--border-color);
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
.cancelButton {
|
|
53
|
-
padding: 6px 12px;
|
|
54
|
-
border: 1px solid var(--button-border);
|
|
55
|
-
border-radius: 4px;
|
|
56
|
-
background-color: var(--button-secondary-background);
|
|
57
|
-
color: var(--button-secondary-foreground);
|
|
58
|
-
cursor: pointer;
|
|
59
|
-
font-size: 13px;
|
|
60
|
-
|
|
61
|
-
&:hover {
|
|
62
|
-
background-color: var(--button-secondary-hover-background);
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
.submitButton {
|
|
67
|
-
padding: 6px 12px;
|
|
68
|
-
border: none;
|
|
69
|
-
border-radius: 4px;
|
|
70
|
-
background-color: var(--button-primary-background);
|
|
71
|
-
color: var(--button-primary-foreground);
|
|
72
|
-
cursor: pointer;
|
|
73
|
-
font-size: 13px;
|
|
74
|
-
|
|
75
|
-
&:hover {
|
|
76
|
-
background-color: var(--button-primary-hover-background);
|
|
77
|
-
}
|
|
78
|
-
}
|
|
@@ -1,144 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
|
|
3
|
-
import { Button } from '@opensumi/ide-components/lib/button';
|
|
4
|
-
import { Modal } from '@opensumi/ide-components/lib/modal';
|
|
5
|
-
|
|
6
|
-
import styles from './mcp-server-form.module.less';
|
|
7
|
-
|
|
8
|
-
export interface MCPServerFormData {
|
|
9
|
-
name: string;
|
|
10
|
-
command: string;
|
|
11
|
-
args: string[];
|
|
12
|
-
env?: Record<string, string>;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
interface Props {
|
|
16
|
-
visible: boolean;
|
|
17
|
-
initialData?: MCPServerFormData;
|
|
18
|
-
onSave: (data: MCPServerFormData) => void;
|
|
19
|
-
onCancel: () => void;
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
export const MCPServerForm: React.FC<Props> = ({ visible, initialData, onSave, onCancel }) => {
|
|
23
|
-
const [formData, setFormData] = React.useState<MCPServerFormData>(() => ({
|
|
24
|
-
name: '',
|
|
25
|
-
command: '',
|
|
26
|
-
args: [],
|
|
27
|
-
env: {},
|
|
28
|
-
type: 'stdio', // TODO: 支持 SSE
|
|
29
|
-
...initialData,
|
|
30
|
-
}));
|
|
31
|
-
|
|
32
|
-
const [argsText, setArgsText] = React.useState(() => initialData?.args?.join(' ') || '');
|
|
33
|
-
const [envText, setEnvText] = React.useState(() => {
|
|
34
|
-
if (!initialData?.env) {
|
|
35
|
-
return '';
|
|
36
|
-
}
|
|
37
|
-
return Object.entries(initialData.env)
|
|
38
|
-
.map(([key, value]) => `${key}=${value}`)
|
|
39
|
-
.join('\n');
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
// Update form data when initialData changes
|
|
43
|
-
React.useEffect(() => {
|
|
44
|
-
setFormData({
|
|
45
|
-
name: '',
|
|
46
|
-
command: '',
|
|
47
|
-
args: [],
|
|
48
|
-
env: {},
|
|
49
|
-
...initialData,
|
|
50
|
-
});
|
|
51
|
-
setArgsText(initialData?.args?.join(' ') || '');
|
|
52
|
-
setEnvText(
|
|
53
|
-
initialData?.env
|
|
54
|
-
? Object.entries(initialData.env)
|
|
55
|
-
.map(([key, value]) => `${key}=${value}`)
|
|
56
|
-
.join('\n')
|
|
57
|
-
: '',
|
|
58
|
-
);
|
|
59
|
-
}, [initialData]);
|
|
60
|
-
|
|
61
|
-
const handleSubmit = (e: React.FormEvent) => {
|
|
62
|
-
e.preventDefault();
|
|
63
|
-
const args = argsText.split(' ').filter(Boolean);
|
|
64
|
-
const env = envText
|
|
65
|
-
.split('\n')
|
|
66
|
-
.filter(Boolean)
|
|
67
|
-
.reduce((acc, line) => {
|
|
68
|
-
const [key, value] = line.split('=');
|
|
69
|
-
if (key && value) {
|
|
70
|
-
acc[key.trim()] = value.trim();
|
|
71
|
-
}
|
|
72
|
-
return acc;
|
|
73
|
-
}, {} as Record<string, string>);
|
|
74
|
-
|
|
75
|
-
onSave({
|
|
76
|
-
...formData,
|
|
77
|
-
args,
|
|
78
|
-
env,
|
|
79
|
-
});
|
|
80
|
-
};
|
|
81
|
-
|
|
82
|
-
return (
|
|
83
|
-
<Modal
|
|
84
|
-
title={initialData ? 'Edit MCP Server' : 'Add New MCP Server'}
|
|
85
|
-
visible={visible}
|
|
86
|
-
onCancel={onCancel}
|
|
87
|
-
centered
|
|
88
|
-
width={600}
|
|
89
|
-
footer={null}
|
|
90
|
-
style={{
|
|
91
|
-
background: 'var(--editor-background)',
|
|
92
|
-
}}
|
|
93
|
-
>
|
|
94
|
-
<form className={styles.form} onSubmit={(e) => e.preventDefault()}>
|
|
95
|
-
<div className={styles.formItem}>
|
|
96
|
-
<label>Name:</label>
|
|
97
|
-
<input
|
|
98
|
-
type='text'
|
|
99
|
-
value={formData.name}
|
|
100
|
-
onChange={(e) => setFormData({ ...formData, name: e.target.value })}
|
|
101
|
-
placeholder='Enter server name'
|
|
102
|
-
required
|
|
103
|
-
/>
|
|
104
|
-
</div>
|
|
105
|
-
<div className={styles.formItem}>
|
|
106
|
-
<label>Command:</label>
|
|
107
|
-
<input
|
|
108
|
-
type='text'
|
|
109
|
-
value={formData.command}
|
|
110
|
-
onChange={(e) => setFormData({ ...formData, command: e.target.value })}
|
|
111
|
-
placeholder='Enter command (e.g., npx)'
|
|
112
|
-
required
|
|
113
|
-
/>
|
|
114
|
-
</div>
|
|
115
|
-
<div className={styles.formItem}>
|
|
116
|
-
<label>Arguments:</label>
|
|
117
|
-
<textarea
|
|
118
|
-
value={argsText}
|
|
119
|
-
onChange={(e) => setArgsText(e.target.value)}
|
|
120
|
-
placeholder='Enter arguments separated by space'
|
|
121
|
-
rows={3}
|
|
122
|
-
/>
|
|
123
|
-
</div>
|
|
124
|
-
<div className={styles.formItem}>
|
|
125
|
-
<label>Environment Variables:</label>
|
|
126
|
-
<textarea
|
|
127
|
-
value={envText}
|
|
128
|
-
onChange={(e) => setEnvText(e.target.value)}
|
|
129
|
-
placeholder='KEY=value (one per line)'
|
|
130
|
-
rows={3}
|
|
131
|
-
/>
|
|
132
|
-
</div>
|
|
133
|
-
<div className={styles.formActions}>
|
|
134
|
-
<Button onClick={onCancel} type='secondary'>
|
|
135
|
-
Cancel
|
|
136
|
-
</Button>
|
|
137
|
-
<Button onClick={handleSubmit} type='primary'>
|
|
138
|
-
Save
|
|
139
|
-
</Button>
|
|
140
|
-
</div>
|
|
141
|
-
</form>
|
|
142
|
-
</Modal>
|
|
143
|
-
);
|
|
144
|
-
};
|