GitLab MCP Server

#!/usr/bin/env node import { Server } from "@modelcontextprotocol/sdk/server/index.js"; import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js"; import { CallToolRequestSchema, ListToolsRequestSchema, } from "@modelcontextprotocol/sdk/types.js"; import fetch from "node-fetch"; import { z } from "zod"; import { zodToJsonSchema } from "zod-to-json-schema"; import { GitLabForkSchema, GitLabReferenceSchema, GitLabRepositorySchema, GitLabIssueSchema, GitLabMergeRequestSchema, GitLabContentSchema, GitLabCreateUpdateFileResponseSchema, GitLabSearchResponseSchema, GitLabTreeSchema, GitLabCommitSchema, CreateOrUpdateFileSchema, SearchRepositoriesSchema, CreateRepositorySchema, GetFileContentsSchema, PushFilesSchema, CreateIssueSchema, CreateMergeRequestSchema, ForkRepositorySchema, CreateBranchSchema, GitLabMergeRequestDiffSchema, GetMergeRequestSchema, GetMergeRequestDiffsSchema, UpdateMergeRequestSchema, } from "./schemas.js"; const server = new Server({ name: "gitlab-mcp-server", version: "0.0.1", }, { capabilities: { tools: {}, }, }); const GITLAB_PERSONAL_ACCESS_TOKEN = process.env.GITLAB_PERSONAL_ACCESS_TOKEN; const GITLAB_API_URL = process.env.GITLAB_API_URL || "https://gitlab.com"; if (!GITLAB_PERSONAL_ACCESS_TOKEN) { console.error("GITLAB_PERSONAL_ACCESS_TOKEN environment variable is not set"); process.exit(1); } // Remove the DEFAULT_HEADERS constant and create a function to get the URL with token function getGitLabUrl(path) { const url = new URL(`${GITLAB_API_URL}${path}`); url.searchParams.append("access_token", GITLAB_PERSONAL_ACCESS_TOKEN); return url; } // API 에러 처리를 위한 유틸리티 함수 async function handleGitLabError(response) { if (!response.ok) { const errorBody = await response.text(); throw new Error(`GitLab API error: ${response.status} ${response.statusText}\n${errorBody}`); } } // 프로젝트 포크 생성 async function forkProject(projectId, namespace) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/fork`); if (namespace) { url.searchParams.append("namespace", namespace); } const response = await fetch(url.toString(), { method: "POST", headers: { Accept: "application/json", "Content-Type": "application/json", }, }); // 이미 존재하는 프로젝트인 경우 처리 if (response.status === 409) { throw new Error("Project already exists in the target namespace"); } await handleGitLabError(response); const data = await response.json(); return GitLabForkSchema.parse(data); } // 새로운 브랜치 생성 async function createBranch(projectId, options) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/repository/branches`); const response = await fetch(url.toString(), { method: "POST", headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify({ branch: options.name, ref: options.ref, }), }); await handleGitLabError(response); return GitLabReferenceSchema.parse(await response.json()); } // 프로젝트의 기본 브랜치 조회 async function getDefaultBranchRef(projectId) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}`); const response = await fetch(url.toString(), { headers: { Accept: "application/json", }, }); await handleGitLabError(response); const project = GitLabRepositorySchema.parse(await response.json()); return project.default_branch ?? "main"; } // 파일 내용 조회 async function getFileContents(projectId, filePath, ref) { const encodedPath = encodeURIComponent(filePath); if (!ref) { ref = await getDefaultBranchRef(projectId); } const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/repository/files/${encodedPath}`); url.searchParams.append("ref", ref); const response = await fetch(url.toString(), { headers: { Accept: "application/json", }, }); // 파일을 찾을 수 없는 경우 처리 if (response.status === 404) { throw new Error(`File not found: ${filePath}`); } await handleGitLabError(response); const data = await response.json(); const parsedData = GitLabContentSchema.parse(data); // Base64로 인코딩된 파일 내용을 UTF-8로 디코딩 if (!Array.isArray(parsedData) && parsedData.content) { parsedData.content = Buffer.from(parsedData.content, "base64").toString("utf8"); parsedData.encoding = "utf8"; } return parsedData; } // 이슈 생성 async function createIssue(projectId, options) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/issues`); const response = await fetch(url.toString(), { method: "POST", headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify({ title: options.title, description: options.description, assignee_ids: options.assignee_ids, milestone_id: options.milestone_id, labels: options.labels?.join(","), }), }); // 잘못된 요청 처리 if (response.status === 400) { const errorBody = await response.text(); throw new Error(`Invalid request: ${errorBody}`); } await handleGitLabError(response); const data = await response.json(); return GitLabIssueSchema.parse(data); } async function createMergeRequest(projectId, options) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/merge_requests`); const response = await fetch(url.toString(), { method: "POST", headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify({ title: options.title, description: options.description, source_branch: options.source_branch, target_branch: options.target_branch, allow_collaboration: options.allow_collaboration, draft: options.draft, }), }); if (response.status === 400) { const errorBody = await response.text(); throw new Error(`Invalid request: ${errorBody}`); } if (!response.ok) { const errorBody = await response.text(); throw new Error(`GitLab API error: ${response.status} ${response.statusText}\n${errorBody}`); } const data = await response.json(); return GitLabMergeRequestSchema.parse(data); } async function createOrUpdateFile(projectId, filePath, content, commitMessage, branch, previousPath) { const encodedPath = encodeURIComponent(filePath); const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/repository/files/${encodedPath}`); const body = { branch, content, commit_message: commitMessage, encoding: "text", ...(previousPath ? { previous_path: previousPath } : {}), }; // Check if file exists let method = "POST"; try { await getFileContents(projectId, filePath, branch); method = "PUT"; } catch (error) { if (!(error instanceof Error && error.message.includes("File not found"))) { throw error; } // File doesn't exist, use POST } const response = await fetch(url.toString(), { method, headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify(body), }); if (!response.ok) { const errorBody = await response.text(); throw new Error(`GitLab API error: ${response.status} ${response.statusText}\n${errorBody}`); } const data = await response.json(); return GitLabCreateUpdateFileResponseSchema.parse(data); } async function createTree(projectId, files, ref) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/repository/tree`); if (ref) { url.searchParams.append("ref", ref); } const response = await fetch(url.toString(), { method: "POST", headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify({ files: files.map((file) => ({ file_path: file.path, content: file.content, encoding: "text", })), }), }); if (response.status === 400) { const errorBody = await response.text(); throw new Error(`Invalid request: ${errorBody}`); } if (!response.ok) { const errorBody = await response.text(); throw new Error(`GitLab API error: ${response.status} ${response.statusText}\n${errorBody}`); } const data = await response.json(); return GitLabTreeSchema.parse(data); } async function createCommit(projectId, message, branch, actions) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/repository/commits`); const response = await fetch(url.toString(), { method: "POST", headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify({ branch, commit_message: message, actions: actions.map((action) => ({ action: "create", file_path: action.path, content: action.content, encoding: "text", })), }), }); if (response.status === 400) { const errorBody = await response.text(); throw new Error(`Invalid request: ${errorBody}`); } if (!response.ok) { const errorBody = await response.text(); throw new Error(`GitLab API error: ${response.status} ${response.statusText}\n${errorBody}`); } const data = await response.json(); return GitLabCommitSchema.parse(data); } async function searchProjects(query, page = 1, perPage = 20) { const url = getGitLabUrl(`/api/v4/projects`); url.searchParams.append("search", query); url.searchParams.append("page", page.toString()); url.searchParams.append("per_page", perPage.toString()); url.searchParams.append("order_by", "id"); url.searchParams.append("sort", "desc"); const response = await fetch(url.toString(), { headers: { Accept: "application/json", "Content-Type": "application/json", }, }); if (!response.ok) { const errorBody = await response.text(); throw new Error(`GitLab API error: ${response.status} ${response.statusText}\n${errorBody}`); } const projects = (await response.json()); const totalCount = response.headers.get("x-total"); const totalPages = response.headers.get("x-total-pages"); // GitLab API doesn't return these headers for results > 10,000 const count = totalCount ? parseInt(totalCount) : projects.length; return GitLabSearchResponseSchema.parse({ count, total_pages: totalPages ? parseInt(totalPages) : Math.ceil(count / perPage), current_page: page, items: projects, }); } async function createRepository(options) { const response = await fetch(getGitLabUrl("/api/v4/projects"), { method: "POST", headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify({ name: options.name, description: options.description, visibility: options.visibility, initialize_with_readme: options.initialize_with_readme, default_branch: "main", path: options.name.toLowerCase().replace(/\s+/g, "-"), }), }); if (!response.ok) { const errorBody = await response.text(); throw new Error(`GitLab API error: ${response.status} ${response.statusText}\n${errorBody}`); } const data = await response.json(); return GitLabRepositorySchema.parse(data); } // MR 조회 함수 async function getMergeRequest(projectId, mergeRequestIid) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/merge_requests/${mergeRequestIid}`); const response = await fetch(url.toString(), { headers: { Accept: "application/json", }, }); await handleGitLabError(response); return GitLabMergeRequestSchema.parse(await response.json()); } // MR 변경사항 조회 함수 async function getMergeRequestDiffs(projectId, mergeRequestIid, view) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/merge_requests/${mergeRequestIid}/changes`); if (view) { url.searchParams.append("view", view); } const response = await fetch(url.toString(), { headers: { Accept: "application/json", }, }); await handleGitLabError(response); const data = (await response.json()); return z.array(GitLabMergeRequestDiffSchema).parse(data.changes); } // MR 업데이트 함수 async function updateMergeRequest(projectId, mergeRequestIid, options) { const url = getGitLabUrl(`/api/v4/projects/${encodeURIComponent(projectId)}/merge_requests/${mergeRequestIid}`); const response = await fetch(url.toString(), { method: "PUT", headers: { Accept: "application/json", "Content-Type": "application/json", }, body: JSON.stringify(options), }); await handleGitLabError(response); return GitLabMergeRequestSchema.parse(await response.json()); } server.setRequestHandler(ListToolsRequestSchema, async () => { return { tools: [ { name: "create_or_update_file", description: "Create or update a single file in a GitLab project", inputSchema: zodToJsonSchema(CreateOrUpdateFileSchema), }, { name: "search_repositories", description: "Search for GitLab projects", inputSchema: zodToJsonSchema(SearchRepositoriesSchema), }, { name: "create_repository", description: "Create a new GitLab project", inputSchema: zodToJsonSchema(CreateRepositorySchema), }, { name: "get_file_contents", description: "Get the contents of a file or directory from a GitLab project", inputSchema: zodToJsonSchema(GetFileContentsSchema), }, { name: "push_files", description: "Push multiple files to a GitLab project in a single commit", inputSchema: zodToJsonSchema(PushFilesSchema), }, { name: "create_issue", description: "Create a new issue in a GitLab project", inputSchema: zodToJsonSchema(CreateIssueSchema), }, { name: "create_merge_request", description: "Create a new merge request in a GitLab project", inputSchema: zodToJsonSchema(CreateMergeRequestSchema), }, { name: "fork_repository", description: "Fork a GitLab project to your account or specified namespace", inputSchema: zodToJsonSchema(ForkRepositorySchema), }, { name: "create_branch", description: "Create a new branch in a GitLab project", inputSchema: zodToJsonSchema(CreateBranchSchema), }, { name: "get_merge_request", description: "Get details of a merge request", inputSchema: zodToJsonSchema(GetMergeRequestSchema), }, { name: "get_merge_request_diffs", description: "Get the changes/diffs of a merge request", inputSchema: zodToJsonSchema(GetMergeRequestDiffsSchema), }, { name: "update_merge_request", description: "Update a merge request", inputSchema: zodToJsonSchema(UpdateMergeRequestSchema), }, ], }; }); server.setRequestHandler(CallToolRequestSchema, async (request) => { try { if (!request.params.arguments) { throw new Error("Arguments are required"); } switch (request.params.name) { case "fork_repository": { const args = ForkRepositorySchema.parse(request.params.arguments); const fork = await forkProject(args.project_id, args.namespace); return { content: [{ type: "text", text: JSON.stringify(fork, null, 2) }], }; } case "create_branch": { const args = CreateBranchSchema.parse(request.params.arguments); let ref = args.ref; if (!ref) { ref = await getDefaultBranchRef(args.project_id); } const branch = await createBranch(args.project_id, { name: args.branch, ref, }); return { content: [{ type: "text", text: JSON.stringify(branch, null, 2) }], }; } case "search_repositories": { const args = SearchRepositoriesSchema.parse(request.params.arguments); const results = await searchProjects(args.search, args.page, args.per_page); return { content: [{ type: "text", text: JSON.stringify(results, null, 2) }], }; } case "create_repository": { const args = CreateRepositorySchema.parse(request.params.arguments); const repository = await createRepository(args); return { content: [ { type: "text", text: JSON.stringify(repository, null, 2) }, ], }; } case "get_file_contents": { const args = GetFileContentsSchema.parse(request.params.arguments); const contents = await getFileContents(args.project_id, args.file_path, args.ref); return { content: [{ type: "text", text: JSON.stringify(contents, null, 2) }], }; } case "create_or_update_file": { const args = CreateOrUpdateFileSchema.parse(request.params.arguments); const result = await createOrUpdateFile(args.project_id, args.file_path, args.content, args.commit_message, args.branch, args.previous_path); return { content: [{ type: "text", text: JSON.stringify(result, null, 2) }], }; } case "push_files": { const args = PushFilesSchema.parse(request.params.arguments); const result = await createCommit(args.project_id, args.commit_message, args.branch, args.files.map((f) => ({ path: f.file_path, content: f.content }))); return { content: [{ type: "text", text: JSON.stringify(result, null, 2) }], }; } case "create_issue": { const args = CreateIssueSchema.parse(request.params.arguments); const { project_id, ...options } = args; const issue = await createIssue(project_id, options); return { content: [{ type: "text", text: JSON.stringify(issue, null, 2) }], }; } case "create_merge_request": { const args = CreateMergeRequestSchema.parse(request.params.arguments); const { project_id, ...options } = args; const mergeRequest = await createMergeRequest(project_id, options); return { content: [ { type: "text", text: JSON.stringify(mergeRequest, null, 2) }, ], }; } case "get_merge_request": { const args = GetMergeRequestSchema.parse(request.params.arguments); const mergeRequest = await getMergeRequest(args.project_id, args.merge_request_iid); return { content: [ { type: "text", text: JSON.stringify(mergeRequest, null, 2) }, ], }; } case "get_merge_request_diffs": { const args = GetMergeRequestDiffsSchema.parse(request.params.arguments); const diffs = await getMergeRequestDiffs(args.project_id, args.merge_request_iid, args.view); return { content: [{ type: "text", text: JSON.stringify(diffs, null, 2) }], }; } case "update_merge_request": { const args = UpdateMergeRequestSchema.parse(request.params.arguments); const { project_id, merge_request_iid, ...options } = args; const mergeRequest = await updateMergeRequest(project_id, merge_request_iid, options); return { content: [ { type: "text", text: JSON.stringify(mergeRequest, null, 2) }, ], }; } default: throw new Error(`Unknown tool: ${request.params.name}`); } } catch (error) { if (error instanceof z.ZodError) { throw new Error(`Invalid arguments: ${error.errors .map((e) => `${e.path.join(".")}: ${e.message}`) .join(", ")}`); } throw error; } }); async function runServer() { const transport = new StdioServerTransport(); await server.connect(transport); // package.json version console.log("GitLab MCP Server running on stdio"); } runServer().catch((error) => { console.error("Fatal error in main():", error); process.exit(1); });