fix: macOS Agent CLI 호환성 수정 (모델명, Base64 인코딩, 경로)

Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
DDD1542 2026-02-06 13:33:19 +09:00
parent 04565eb480
commit 40057c7d3c
3 changed files with 39 additions and 302 deletions

View File

@ -2,7 +2,7 @@
"mcpServers": { "mcpServers": {
"agent-orchestrator": { "agent-orchestrator": {
"command": "node", "command": "node",
"args": ["C:/Users/defaultuser0/ERP-node/mcp-agent-orchestrator/build/index.js"] "args": ["/Users/gbpark/ERP-node/mcp-agent-orchestrator/build/index.js"]
} }
} }
} }

View File

@ -1,14 +1,13 @@
{ {
"name": "mcp-agent-orchestrator", "name": "mcp-agent-orchestrator",
"version": "1.0.0", "version": "2.0.0",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "mcp-agent-orchestrator", "name": "mcp-agent-orchestrator",
"version": "1.0.0", "version": "2.0.0",
"dependencies": { "dependencies": {
"@anthropic-ai/sdk": "^0.39.0",
"@modelcontextprotocol/sdk": "^1.0.0" "@modelcontextprotocol/sdk": "^1.0.0"
}, },
"devDependencies": { "devDependencies": {
@ -19,36 +18,6 @@
"node": ">=18.0.0" "node": ">=18.0.0"
} }
}, },
"node_modules/@anthropic-ai/sdk": {
"version": "0.39.0",
"resolved": "https://registry.npmjs.org/@anthropic-ai/sdk/-/sdk-0.39.0.tgz",
"integrity": "sha512-eMyDIPRZbt1CCLErRCi3exlAvNkBtRe+kW5vvJyef93PmNr/clstYgHhtvmkxN82nlKgzyGPCyGxrm0JQ1ZIdg==",
"license": "MIT",
"dependencies": {
"@types/node": "^18.11.18",
"@types/node-fetch": "^2.6.4",
"abort-controller": "^3.0.0",
"agentkeepalive": "^4.2.1",
"form-data-encoder": "1.7.2",
"formdata-node": "^4.3.2",
"node-fetch": "^2.6.7"
}
},
"node_modules/@anthropic-ai/sdk/node_modules/@types/node": {
"version": "18.19.130",
"resolved": "https://registry.npmjs.org/@types/node/-/node-18.19.130.tgz",
"integrity": "sha512-GRaXQx6jGfL8sKfaIDD6OupbIHBr9jv7Jnaml9tB7l4v068PAOXqfcujMMo5PhbIs6ggR1XODELqahT2R8v0fg==",
"license": "MIT",
"dependencies": {
"undici-types": "~5.26.4"
}
},
"node_modules/@anthropic-ai/sdk/node_modules/undici-types": {
"version": "5.26.5",
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz",
"integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==",
"license": "MIT"
},
"node_modules/@hono/node-server": { "node_modules/@hono/node-server": {
"version": "1.19.9", "version": "1.19.9",
"resolved": "https://registry.npmjs.org/@hono/node-server/-/node-server-1.19.9.tgz", "resolved": "https://registry.npmjs.org/@hono/node-server/-/node-server-1.19.9.tgz",
@ -105,33 +74,12 @@
"version": "20.19.31", "version": "20.19.31",
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.31.tgz", "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.31.tgz",
"integrity": "sha512-5jsi0wpncvTD33Sh1UCgacK37FFwDn+EG7wCmEvs62fCvBL+n8/76cAYDok21NF6+jaVWIqKwCZyX7Vbu8eB3A==", "integrity": "sha512-5jsi0wpncvTD33Sh1UCgacK37FFwDn+EG7wCmEvs62fCvBL+n8/76cAYDok21NF6+jaVWIqKwCZyX7Vbu8eB3A==",
"dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"undici-types": "~6.21.0" "undici-types": "~6.21.0"
} }
}, },
"node_modules/@types/node-fetch": {
"version": "2.6.13",
"resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.6.13.tgz",
"integrity": "sha512-QGpRVpzSaUs30JBSGPjOg4Uveu384erbHBoT1zeONvyCfwQxIkUshLAOqN/k9EjGviPRmWTTe6aH2qySWKTVSw==",
"license": "MIT",
"dependencies": {
"@types/node": "*",
"form-data": "^4.0.4"
}
},
"node_modules/abort-controller": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz",
"integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==",
"license": "MIT",
"dependencies": {
"event-target-shim": "^5.0.0"
},
"engines": {
"node": ">=6.5"
}
},
"node_modules/accepts": { "node_modules/accepts": {
"version": "2.0.0", "version": "2.0.0",
"resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz", "resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz",
@ -145,18 +93,6 @@
"node": ">= 0.6" "node": ">= 0.6"
} }
}, },
"node_modules/agentkeepalive": {
"version": "4.6.0",
"resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-4.6.0.tgz",
"integrity": "sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ==",
"license": "MIT",
"dependencies": {
"humanize-ms": "^1.2.1"
},
"engines": {
"node": ">= 8.0.0"
}
},
"node_modules/ajv": { "node_modules/ajv": {
"version": "8.17.1", "version": "8.17.1",
"resolved": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz", "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz",
@ -190,12 +126,6 @@
} }
} }
}, },
"node_modules/asynckit": {
"version": "0.4.0",
"resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
"integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
"license": "MIT"
},
"node_modules/body-parser": { "node_modules/body-parser": {
"version": "2.2.2", "version": "2.2.2",
"resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.2.tgz", "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.2.tgz",
@ -258,18 +188,6 @@
"url": "https://github.com/sponsors/ljharb" "url": "https://github.com/sponsors/ljharb"
} }
}, },
"node_modules/combined-stream": {
"version": "1.0.8",
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
"integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
"license": "MIT",
"dependencies": {
"delayed-stream": "~1.0.0"
},
"engines": {
"node": ">= 0.8"
}
},
"node_modules/content-disposition": { "node_modules/content-disposition": {
"version": "1.0.1", "version": "1.0.1",
"resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.0.1.tgz", "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.0.1.tgz",
@ -358,15 +276,6 @@
} }
} }
}, },
"node_modules/delayed-stream": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
"integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
"license": "MIT",
"engines": {
"node": ">=0.4.0"
}
},
"node_modules/depd": { "node_modules/depd": {
"version": "2.0.0", "version": "2.0.0",
"resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
@ -435,21 +344,6 @@
"node": ">= 0.4" "node": ">= 0.4"
} }
}, },
"node_modules/es-set-tostringtag": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz",
"integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==",
"license": "MIT",
"dependencies": {
"es-errors": "^1.3.0",
"get-intrinsic": "^1.2.6",
"has-tostringtag": "^1.0.2",
"hasown": "^2.0.2"
},
"engines": {
"node": ">= 0.4"
}
},
"node_modules/escape-html": { "node_modules/escape-html": {
"version": "1.0.3", "version": "1.0.3",
"resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
@ -465,15 +359,6 @@
"node": ">= 0.6" "node": ">= 0.6"
} }
}, },
"node_modules/event-target-shim": {
"version": "5.0.1",
"resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz",
"integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==",
"license": "MIT",
"engines": {
"node": ">=6"
}
},
"node_modules/eventsource": { "node_modules/eventsource": {
"version": "3.0.7", "version": "3.0.7",
"resolved": "https://registry.npmjs.org/eventsource/-/eventsource-3.0.7.tgz", "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-3.0.7.tgz",
@ -600,62 +485,6 @@
"url": "https://opencollective.com/express" "url": "https://opencollective.com/express"
} }
}, },
"node_modules/form-data": {
"version": "4.0.5",
"resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.5.tgz",
"integrity": "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w==",
"license": "MIT",
"dependencies": {
"asynckit": "^0.4.0",
"combined-stream": "^1.0.8",
"es-set-tostringtag": "^2.1.0",
"hasown": "^2.0.2",
"mime-types": "^2.1.12"
},
"engines": {
"node": ">= 6"
}
},
"node_modules/form-data-encoder": {
"version": "1.7.2",
"resolved": "https://registry.npmjs.org/form-data-encoder/-/form-data-encoder-1.7.2.tgz",
"integrity": "sha512-qfqtYan3rxrnCk1VYaA4H+Ms9xdpPqvLZa6xmMgFvhO32x7/3J/ExcTd6qpxM0vH2GdMI+poehyBZvqfMTto8A==",
"license": "MIT"
},
"node_modules/form-data/node_modules/mime-db": {
"version": "1.52.0",
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
"license": "MIT",
"engines": {
"node": ">= 0.6"
}
},
"node_modules/form-data/node_modules/mime-types": {
"version": "2.1.35",
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
"license": "MIT",
"dependencies": {
"mime-db": "1.52.0"
},
"engines": {
"node": ">= 0.6"
}
},
"node_modules/formdata-node": {
"version": "4.4.1",
"resolved": "https://registry.npmjs.org/formdata-node/-/formdata-node-4.4.1.tgz",
"integrity": "sha512-0iirZp3uVDjVGt9p49aTaqjk84TrglENEDuqfdlZQ1roC9CWlPk6Avf8EEnZNcAqPonwkG35x4n3ww/1THYAeQ==",
"license": "MIT",
"dependencies": {
"node-domexception": "1.0.0",
"web-streams-polyfill": "4.0.0-beta.3"
},
"engines": {
"node": ">= 12.20"
}
},
"node_modules/forwarded": { "node_modules/forwarded": {
"version": "0.2.0", "version": "0.2.0",
"resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
@ -744,21 +573,6 @@
"url": "https://github.com/sponsors/ljharb" "url": "https://github.com/sponsors/ljharb"
} }
}, },
"node_modules/has-tostringtag": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz",
"integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==",
"license": "MIT",
"dependencies": {
"has-symbols": "^1.0.3"
},
"engines": {
"node": ">= 0.4"
},
"funding": {
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/hasown": { "node_modules/hasown": {
"version": "2.0.2", "version": "2.0.2",
"resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
@ -801,15 +615,6 @@
"url": "https://opencollective.com/express" "url": "https://opencollective.com/express"
} }
}, },
"node_modules/humanize-ms": {
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/humanize-ms/-/humanize-ms-1.2.1.tgz",
"integrity": "sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==",
"license": "MIT",
"dependencies": {
"ms": "^2.0.0"
}
},
"node_modules/iconv-lite": { "node_modules/iconv-lite": {
"version": "0.7.2", "version": "0.7.2",
"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.2.tgz", "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.2.tgz",
@ -953,46 +758,6 @@
"node": ">= 0.6" "node": ">= 0.6"
} }
}, },
"node_modules/node-domexception": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz",
"integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==",
"deprecated": "Use your platform's native DOMException instead",
"funding": [
{
"type": "github",
"url": "https://github.com/sponsors/jimmywarting"
},
{
"type": "github",
"url": "https://paypal.me/jimmywarting"
}
],
"license": "MIT",
"engines": {
"node": ">=10.5.0"
}
},
"node_modules/node-fetch": {
"version": "2.7.0",
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz",
"integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==",
"license": "MIT",
"dependencies": {
"whatwg-url": "^5.0.0"
},
"engines": {
"node": "4.x || >=6.0.0"
},
"peerDependencies": {
"encoding": "^0.1.0"
},
"peerDependenciesMeta": {
"encoding": {
"optional": true
}
}
},
"node_modules/object-assign": { "node_modules/object-assign": {
"version": "4.1.1", "version": "4.1.1",
"resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
@ -1317,12 +1082,6 @@
"node": ">=0.6" "node": ">=0.6"
} }
}, },
"node_modules/tr46": {
"version": "0.0.3",
"resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz",
"integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==",
"license": "MIT"
},
"node_modules/type-is": { "node_modules/type-is": {
"version": "2.0.1", "version": "2.0.1",
"resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz", "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz",
@ -1355,6 +1114,7 @@
"version": "6.21.0", "version": "6.21.0",
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz", "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
"integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==", "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
"dev": true,
"license": "MIT" "license": "MIT"
}, },
"node_modules/unpipe": { "node_modules/unpipe": {
@ -1375,31 +1135,6 @@
"node": ">= 0.8" "node": ">= 0.8"
} }
}, },
"node_modules/web-streams-polyfill": {
"version": "4.0.0-beta.3",
"resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-4.0.0-beta.3.tgz",
"integrity": "sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug==",
"license": "MIT",
"engines": {
"node": ">= 14"
}
},
"node_modules/webidl-conversions": {
"version": "3.0.1",
"resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz",
"integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==",
"license": "BSD-2-Clause"
},
"node_modules/whatwg-url": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz",
"integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==",
"license": "MIT",
"dependencies": {
"tr46": "~0.0.3",
"webidl-conversions": "^3.0.0"
}
},
"node_modules/which": { "node_modules/which": {
"version": "2.0.2", "version": "2.0.2",
"resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",

View File

@ -1,6 +1,6 @@
#!/usr/bin/env node #!/usr/bin/env node
/** /**
* Multi-Agent Orchestrator MCP Server * Multi-Agent Orchestrator MCP Server v2.0
* *
* Cursor Agent CLI를 * Cursor Agent CLI를
* - PM (Cursor IDE): * - PM (Cursor IDE):
@ -48,7 +48,7 @@ const server = new Server(
* *
* : * :
* - Windows: cmd /c "echo. | agent ..." (stdin ) * - Windows: cmd /c "echo. | agent ..." (stdin )
* - Mac/Linux: echo "" | agent ... (bash ) * - Mac/Linux: ~/.local/bin/agent
*/ */
async function callAgentCLI( async function callAgentCLI(
agentType: AgentType, agentType: AgentType,
@ -63,46 +63,45 @@ async function callAgentCLI(
logger.info(`Calling ${agentType} agent via CLI`, { model, task: task.substring(0, 100) }); logger.info(`Calling ${agentType} agent via CLI`, { model, task: task.substring(0, 100) });
try { try {
// 프롬프트 구성
const systemPrompt = config.systemPrompt
.replace(/\r?\n/g, ' ') // 줄바꿈을 공백으로
.replace(/"/g, '\\"'); // 쌍따옴표 이스케이프
const userMessage = context const userMessage = context
? `${task} (Background info: ${context})` ? `${task}\n\n배경 정보:\n${context}`
: task; : task;
// 전체 프롬프트 (시스템 + 유저) // 프롬프트를 임시 파일에 저장하여 쉘 이스케이프 문제 회피
const fullPrompt = `SYSTEM INSTRUCTIONS: ${systemPrompt} --- TASK REQUEST: ${userMessage}` const fullPrompt = `${config.systemPrompt}\n\n---\n\n${userMessage}`;
.replace(/\[/g, '(') // 대괄호를 괄호로 변환 (쉘 호환)
.replace(/\]/g, ')') // Base64 인코딩으로 특수문자 문제 해결
.replace(/"/g, '\\"'); // 쌍따옴표 이스케이프 const encodedPrompt = Buffer.from(fullPrompt).toString('base64');
let cmd: string; let cmd: string;
let shell: string; let shell: string;
const agentPath = isWindows ? 'agent' : `${process.env.HOME}/.local/bin/agent`;
if (isWindows) { if (isWindows) {
// Windows: CMD를 통해 echo로 빈 입력 파이프 // Windows: PowerShell을 통해 Base64 디코딩 후 실행
cmd = `cmd /c "echo. | agent -p \\"${fullPrompt}\\" --model ${model} --output-format text"`; cmd = `powershell -Command "$prompt = [System.Text.Encoding]::UTF8.GetString([System.Convert]::FromBase64String('${encodedPrompt}')); echo $prompt | ${agentPath} --model ${model} --print"`;
shell = 'cmd.exe'; shell = 'powershell.exe';
} else { } else {
// Mac/Linux: Bash를 통해 빈 문자열 파이프 // Mac/Linux: echo로 base64 디코딩 후 파이프
// 참고: Mac에서는 agent CLI가 ~/.cursor-agent/bin/agent 경로에 있을 수 있음 cmd = `echo "${encodedPrompt}" | base64 -d | ${agentPath} --model ${model} --print`;
cmd = `echo "" | agent -p "${fullPrompt}" --model ${model} --output-format text`;
shell = '/bin/bash'; shell = '/bin/bash';
} }
logger.debug(`Executing on ${isWindows ? 'Windows' : 'Mac/Linux'}: agent -p "..." --model ${model}`); logger.debug(`Executing: ${agentPath} --model ${model} --print`);
const { stdout, stderr } = await execAsync(cmd, { const { stdout, stderr } = await execAsync(cmd, {
cwd: process.cwd(), cwd: process.cwd(),
maxBuffer: 10 * 1024 * 1024, // 10MB buffer maxBuffer: 10 * 1024 * 1024, // 10MB buffer
timeout: 300000, // 5분 타임아웃 timeout: 300000, // 5분 타임아웃
shell, shell,
env: {
...process.env,
PATH: `${process.env.HOME}/.local/bin:${process.env.PATH}`,
},
}); });
if (stderr && !stderr.includes('warning')) { if (stderr && !stderr.includes('warning') && !stderr.includes('info')) {
logger.warn(`${agentType} agent stderr`, { stderr }); logger.warn(`${agentType} agent stderr`, { stderr: stderr.substring(0, 500) });
} }
logger.info(`${agentType} agent completed via CLI`); logger.info(`${agentType} agent completed via CLI`);
@ -124,7 +123,7 @@ server.setRequestHandler(ListToolsRequestSchema, async () => {
description: description:
"백엔드 전문가에게 질문하거나 작업을 요청합니다. " + "백엔드 전문가에게 질문하거나 작업을 요청합니다. " +
"API 설계, 서비스 로직, 라우팅, 미들웨어 관련 작업에 사용하세요. " + "API 설계, 서비스 로직, 라우팅, 미들웨어 관련 작업에 사용하세요. " +
"담당 폴더: backend-node/src/ (Cursor Team Plan 사용, sonnet-4.5 모델)", "담당 폴더: backend-node/src/ (Cursor Agent CLI, sonnet-4.5 모델)",
inputSchema: { inputSchema: {
type: "object" as const, type: "object" as const,
properties: { properties: {
@ -145,7 +144,7 @@ server.setRequestHandler(ListToolsRequestSchema, async () => {
description: description:
"DB 전문가에게 질문하거나 작업을 요청합니다. " + "DB 전문가에게 질문하거나 작업을 요청합니다. " +
"스키마 설계, SQL 쿼리, MyBatis 매퍼, 마이그레이션 관련 작업에 사용하세요. " + "스키마 설계, SQL 쿼리, MyBatis 매퍼, 마이그레이션 관련 작업에 사용하세요. " +
"담당 폴더: src/com/pms/mapper/, db/ (Cursor Team Plan 사용, sonnet-4.5 모델)", "담당 폴더: src/com/pms/mapper/, db/ (Cursor Agent CLI, sonnet-4.5 모델)",
inputSchema: { inputSchema: {
type: "object" as const, type: "object" as const,
properties: { properties: {
@ -166,7 +165,7 @@ server.setRequestHandler(ListToolsRequestSchema, async () => {
description: description:
"프론트엔드 전문가에게 질문하거나 작업을 요청합니다. " + "프론트엔드 전문가에게 질문하거나 작업을 요청합니다. " +
"React 컴포넌트, 페이지, 스타일링, 상태관리 관련 작업에 사용하세요. " + "React 컴포넌트, 페이지, 스타일링, 상태관리 관련 작업에 사용하세요. " +
"담당 폴더: frontend/ (Cursor Team Plan 사용, sonnet-4.5 모델)", "담당 폴더: frontend/ (Cursor Agent CLI, sonnet-4.5 모델)",
inputSchema: { inputSchema: {
type: "object" as const, type: "object" as const,
properties: { properties: {
@ -187,7 +186,7 @@ server.setRequestHandler(ListToolsRequestSchema, async () => {
description: description:
"여러 전문가에게 동시에 질문합니다 (진짜 병렬 실행!). " + "여러 전문가에게 동시에 질문합니다 (진짜 병렬 실행!). " +
"정보 수집 단계에서 모든 영역의 현황을 빠르게 파악할 때 유용합니다. " + "정보 수집 단계에서 모든 영역의 현황을 빠르게 파악할 때 유용합니다. " +
"모든 에이전트가 동시에 실행되어 시간 절약! (Cursor Team Plan 사용)", "모든 에이전트가 Cursor Agent CLI를 통해 동시에 실행되어 시간 절약!",
inputSchema: { inputSchema: {
type: "object" as const, type: "object" as const,
properties: { properties: {
@ -316,7 +315,8 @@ server.setRequestHandler(CallToolRequestSchema, async (request) => {
system: "Multi-Agent Orchestrator v2.0", system: "Multi-Agent Orchestrator v2.0",
version: "2.0.0", version: "2.0.0",
backend: "Cursor Agent CLI (Team Plan)", backend: "Cursor Agent CLI (Team Plan)",
apiKey: "NOT REQUIRED! Using Cursor subscription", cliPath: `${process.env.HOME}/.local/bin/agent`,
apiKey: "NOT REQUIRED! Using Cursor Team Plan credits",
agents: { agents: {
pm: { pm: {
role: "Project Manager", role: "Project Manager",
@ -325,19 +325,19 @@ server.setRequestHandler(CallToolRequestSchema, async (request) => {
}, },
backend: { backend: {
role: "Backend Specialist", role: "Backend Specialist",
model: "sonnet-4.5 (via agent CLI)", model: "sonnet-4.5 (via Agent CLI)",
description: "API, 서비스 로직, 라우팅 담당", description: "API, 서비스 로직, 라우팅 담당",
folder: "backend-node/src/", folder: "backend-node/src/",
}, },
db: { db: {
role: "Database Specialist", role: "Database Specialist",
model: "sonnet-4.5 (via agent CLI)", model: "sonnet-4.5 (via Agent CLI)",
description: "스키마, 쿼리, 마이그레이션 담당", description: "스키마, 쿼리, 마이그레이션 담당",
folder: "src/com/pms/mapper/, db/", folder: "src/com/pms/mapper/, db/",
}, },
frontend: { frontend: {
role: "Frontend Specialist", role: "Frontend Specialist",
model: "sonnet-4.5 (via agent CLI)", model: "sonnet-4.5 (via Agent CLI)",
description: "컴포넌트, 페이지, 스타일링 담당", description: "컴포넌트, 페이지, 스타일링 담당",
folder: "frontend/", folder: "frontend/",
}, },
@ -345,8 +345,9 @@ server.setRequestHandler(CallToolRequestSchema, async (request) => {
features: { features: {
parallel_execution: true, parallel_execution: true,
cursor_team_plan: true, cursor_team_plan: true,
cursor_agent_cli: true,
separate_api_key: false, separate_api_key: false,
real_multi_session: true, cross_platform: true,
}, },
usage: { usage: {
single_agent: "ask_backend_agent, ask_db_agent, ask_frontend_agent", single_agent: "ask_backend_agent, ask_db_agent, ask_frontend_agent",
@ -387,7 +388,8 @@ server.setRequestHandler(CallToolRequestSchema, async (request) => {
*/ */
async function main() { async function main() {
logger.info("Starting Multi-Agent Orchestrator MCP Server v2.0..."); logger.info("Starting Multi-Agent Orchestrator MCP Server v2.0...");
logger.info("Backend: Cursor Agent CLI (Team Plan - No API Key Required!)"); logger.info(`Backend: Cursor Agent CLI (${process.env.HOME}/.local/bin/agent)`);
logger.info("Credits: Cursor Team Plan - No API Key Required!");
const transport = new StdioServerTransport(); const transport = new StdioServerTransport();
await server.connect(transport); await server.connect(transport);