Compare commits
36 Commits
Author | SHA1 | Date | |
---|---|---|---|
2a80988a02 | |||
5762b32a69 | |||
55e7ca3100 | |||
953f748e0d | |||
0b876ebff6 | |||
fd1c8b9704 | |||
a2c2ac185a | |||
2462168697 | |||
88af65fcd0 | |||
0b55cc3cee | |||
40e39d7b36 | |||
cc847772f1 | |||
ab571d211d | |||
f8b1444afd | |||
06f9437329 | |||
dc99f864ca | |||
8ba33986f3 | |||
64a936446e | |||
8ab0ac7145 | |||
ea06c21f29 | |||
140620397b | |||
3d7aa8035d | |||
25be1947b9 | |||
864ee77ae6 | |||
dc6cc59434 | |||
5924fd3ed4 | |||
f4b265bf2e | |||
b326f4c3c3 | |||
1350a024b5 | |||
4c57c37888 | |||
e4a28a9a47 | |||
9f1e7b5bca | |||
f37e210ee8 | |||
6f789692be | |||
1bb70dccb9 | |||
676bbcd4dd |
82
CHANGELOG.md
82
CHANGELOG.md
@ -1,3 +1,85 @@
|
|||||||
|
## [1.0.48] - 2025-05-29
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- 🎯 **Milestone Management Tools**: Added comprehensive milestone management functionality
|
||||||
|
- `create_milestone`: Create new milestones for GitLab projects
|
||||||
|
- `update_milestone`: Update existing milestone properties (title, description, dates, state)
|
||||||
|
- `delete_milestone`: Delete milestones from projects
|
||||||
|
- `list_milestones`: List and filter project milestones
|
||||||
|
- `get_milestone`: Get detailed information about specific milestones
|
||||||
|
- See: [PR #59](https://github.com/zereight/gitlab-mcp/pull/59)
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- 🐳 **Docker Image Push Script**: Added automated Docker image push script for easier deployment
|
||||||
|
- Simplifies the Docker image build and push process
|
||||||
|
- See: [PR #60](https://github.com/zereight/gitlab-mcp/pull/60)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## [1.0.47] - 2025-05-29
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- 🔄 **List Merge Requests Tool**: Added functionality to list and filter merge requests in GitLab projects
|
||||||
|
- `list_merge_requests`: List merge requests with comprehensive filtering options
|
||||||
|
- Supports filtering by state, scope, author, assignee, reviewer, labels, and more
|
||||||
|
- Includes pagination support for large result sets
|
||||||
|
- See: [PR #56](https://github.com/zereight/gitlab-mcp/pull/56)
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Fixed issue where GitLab users without profile pictures would cause JSON-RPC errors
|
||||||
|
- Changed `avatar_url` field to be nullable in GitLabUserSchema
|
||||||
|
- This allows proper handling of users without avatars in GitLab API responses
|
||||||
|
- See: [PR #55](https://github.com/zereight/gitlab-mcp/pull/55)
|
||||||
|
|
||||||
|
- Fixed issue where GitLab pipelines without illustrations would cause JSON-RPC errors
|
||||||
|
- Changed `illustration` field to be nullable in GitLabPipelineSchema
|
||||||
|
- This allows proper handling of pipelines without illustrations
|
||||||
|
- See: [PR #58](https://github.com/zereight/gitlab-mcp/pull/58), [Issue #57](https://github.com/zereight/gitlab-mcp/issues/57)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## [1.0.46] - 2025-05-27
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Fixed issue where GitLab issues and milestones with null descriptions would cause JSON-RPC errors
|
||||||
|
- Changed `description` field to be nullable with default empty string in schemas
|
||||||
|
- This allows proper handling of GitLab issues/milestones without descriptions
|
||||||
|
- See: [PR #53](https://github.com/zereight/gitlab-mcp/pull/53), [Issue #51](https://github.com/zereight/gitlab-mcp/issues/51)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## [1.0.45] - 2025-05-24
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- 🔄 **Pipeline Management Tools**: Added GitLab pipeline status monitoring and management functionality
|
||||||
|
- `list_pipelines`: List project pipelines with various filtering options
|
||||||
|
- `get_pipeline`: Get detailed information about a specific pipeline
|
||||||
|
- `list_pipeline_jobs`: List all jobs in a specific pipeline
|
||||||
|
- `get_pipeline_job`: Get detailed information about a specific pipeline job
|
||||||
|
- `get_pipeline_job_output`: Get execution logs/output from pipeline jobs
|
||||||
|
- 📊 Pipeline status summary and analysis support
|
||||||
|
- Example: "How many of the last N pipelines are successful?"
|
||||||
|
- Example: "Can you make a summary of the output in the last pipeline?"
|
||||||
|
- See: [PR #52](https://github.com/zereight/gitlab-mcp/pull/52)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## [1.0.42] - 2025-05-22
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Added support for creating and updating issue notes (comments) in GitLab.
|
||||||
|
- You can now add or edit comments on issues.
|
||||||
|
- See: [PR #47](https://github.com/zereight/gitlab-mcp/pull/47)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## [1.0.38] - 2025-05-17
|
## [1.0.38] - 2025-05-17
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
24
Dockerfile
Normal file
24
Dockerfile
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
FROM node:22.15-alpine AS builder
|
||||||
|
|
||||||
|
COPY . /app
|
||||||
|
COPY tsconfig.json /tsconfig.json
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
RUN --mount=type=cache,target=/root/.npm npm install
|
||||||
|
|
||||||
|
RUN --mount=type=cache,target=/root/.npm-production npm ci --ignore-scripts --omit-dev
|
||||||
|
|
||||||
|
FROM node:22.12-alpine AS release
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
COPY --from=builder /app/build /app/build
|
||||||
|
COPY --from=builder /app/package.json /app/package.json
|
||||||
|
COPY --from=builder /app/package-lock.json /app/package-lock.json
|
||||||
|
|
||||||
|
ENV NODE_ENV=production
|
||||||
|
|
||||||
|
RUN npm ci --ignore-scripts --omit-dev
|
||||||
|
|
||||||
|
ENTRYPOINT ["node", "build/index.js"]
|
111
README.md
111
README.md
@ -14,6 +14,8 @@ GitLab MCP(Model Context Protocol) Server. **Includes bug fixes and improvements
|
|||||||
|
|
||||||
When using with the Claude App, you need to set up your API key and URLs directly.
|
When using with the Claude App, you need to set up your API key and URLs directly.
|
||||||
|
|
||||||
|
#### npx
|
||||||
|
|
||||||
```json
|
```json
|
||||||
{
|
{
|
||||||
"mcpServers": {
|
"mcpServers": {
|
||||||
@ -24,15 +26,52 @@ When using with the Claude App, you need to set up your API key and URLs directl
|
|||||||
"GITLAB_PERSONAL_ACCESS_TOKEN": "your_gitlab_token",
|
"GITLAB_PERSONAL_ACCESS_TOKEN": "your_gitlab_token",
|
||||||
"GITLAB_API_URL": "your_gitlab_api_url",
|
"GITLAB_API_URL": "your_gitlab_api_url",
|
||||||
"GITLAB_READ_ONLY_MODE": "false",
|
"GITLAB_READ_ONLY_MODE": "false",
|
||||||
"USE_GITLAB_WIKI":"true"
|
"USE_GITLAB_WIKI": "true"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
### Environment Variables
|
#### Docker
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"mcpServers": {
|
||||||
|
"GitLab communication server": {
|
||||||
|
"command": "docker",
|
||||||
|
"args": [
|
||||||
|
"run",
|
||||||
|
"-i",
|
||||||
|
"--rm",
|
||||||
|
"-e",
|
||||||
|
"GITLAB_PERSONAL_ACCESS_TOKEN",
|
||||||
|
"-e",
|
||||||
|
"GITLAB_API_URL",
|
||||||
|
"-e",
|
||||||
|
"GITLAB_READ_ONLY_MODE",
|
||||||
|
"-e",
|
||||||
|
"USE_GITLAB_WIKI",
|
||||||
|
"iwakitakuma/gitlab-mcp"
|
||||||
|
],
|
||||||
|
"env": {
|
||||||
|
"GITLAB_PERSONAL_ACCESS_TOKEN": "your_gitlab_token",
|
||||||
|
"GITLAB_API_URL": "https://gitlab.com/api/v4", // Optional, for self-hosted GitLab
|
||||||
|
"GITLAB_READ_ONLY_MODE": "false",
|
||||||
|
"USE_GITLAB_WIKI": "true"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Docker Image Push
|
||||||
|
|
||||||
|
```shell
|
||||||
|
$ sh scripts/image_push.sh docker_user_name
|
||||||
|
```
|
||||||
|
|
||||||
|
### Environment Variables
|
||||||
|
|
||||||
- `GITLAB_PERSONAL_ACCESS_TOKEN`: Your GitLab personal access token.
|
- `GITLAB_PERSONAL_ACCESS_TOKEN`: Your GitLab personal access token.
|
||||||
- `GITLAB_API_URL`: Your GitLab API URL. (Default: `https://gitlab.com/api/v4`)
|
- `GITLAB_API_URL`: Your GitLab API URL. (Default: `https://gitlab.com/api/v4`)
|
||||||
@ -59,29 +98,47 @@ When using with the Claude App, you need to set up your API key and URLs directl
|
|||||||
15. `mr_discussions` - List discussion items for a merge request
|
15. `mr_discussions` - List discussion items for a merge request
|
||||||
16. `update_merge_request_note` - Modify an existing merge request thread note
|
16. `update_merge_request_note` - Modify an existing merge request thread note
|
||||||
17. `create_merge_request_note` - Add a new note to an existing merge request thread
|
17. `create_merge_request_note` - Add a new note to an existing merge request thread
|
||||||
18. `list_issues` - List issues in a GitLab project with filtering options
|
18. `update_issue_note` - Modify an existing issue thread note
|
||||||
19. `get_issue` - Get details of a specific issue in a GitLab project
|
19. `create_issue_note` - Add a new note to an existing issue thread
|
||||||
20. `update_issue` - Update an issue in a GitLab project
|
20. `list_issues` - List issues in a GitLab project with filtering options
|
||||||
21. `delete_issue` - Delete an issue from a GitLab project
|
21. `get_issue` - Get details of a specific issue in a GitLab project
|
||||||
22. `list_issue_links` - List all issue links for a specific issue
|
22. `update_issue` - Update an issue in a GitLab project
|
||||||
23. `get_issue_link` - Get a specific issue link
|
23. `delete_issue` - Delete an issue from a GitLab project
|
||||||
24. `create_issue_link` - Create an issue link between two issues
|
24. `list_issue_links` - List all issue links for a specific issue
|
||||||
25. `delete_issue_link` - Delete an issue link
|
25. `list_issue_discussions` - List discussions for an issue in a GitLab project
|
||||||
26. `list_namespaces` - List all namespaces available to the current user
|
26. `get_issue_link` - Get a specific issue link
|
||||||
27. `get_namespace` - Get details of a namespace by ID or path
|
27. `create_issue_link` - Create an issue link between two issues
|
||||||
28. `verify_namespace` - Verify if a namespace path exists
|
28. `delete_issue_link` - Delete an issue link
|
||||||
29. `get_project` - Get details of a specific project
|
29. `list_namespaces` - List all namespaces available to the current user
|
||||||
30. `list_projects` - List projects accessible by the current user
|
30. `get_namespace` - Get details of a namespace by ID or path
|
||||||
31. `list_labels` - List labels for a project
|
31. `verify_namespace` - Verify if a namespace path exists
|
||||||
32. `get_label` - Get a single label from a project
|
32. `get_project` - Get details of a specific project
|
||||||
33. `create_label` - Create a new label in a project
|
33. `list_projects` - List projects accessible by the current user
|
||||||
34. `update_label` - Update an existing label in a project
|
34. `list_labels` - List labels for a project
|
||||||
35. `delete_label` - Delete a label from a project
|
35. `get_label` - Get a single label from a project
|
||||||
36. `list_group_projects` - List projects in a GitLab group with filtering options
|
36. `create_label` - Create a new label in a project
|
||||||
37. `list_wiki_pages` - List wiki pages in a GitLab project
|
37. `update_label` - Update an existing label in a project
|
||||||
38. `get_wiki_page` - Get details of a specific wiki page
|
38. `delete_label` - Delete a label from a project
|
||||||
39. `create_wiki_page` - Create a new wiki page in a GitLab project
|
39. `list_group_projects` - List projects in a GitLab group with filtering options
|
||||||
40. `update_wiki_page` - Update an existing wiki page in a GitLab project
|
40. `list_wiki_pages` - List wiki pages in a GitLab project
|
||||||
41. `delete_wiki_page` - Delete a wiki page from a GitLab project
|
41. `get_wiki_page` - Get details of a specific wiki page
|
||||||
42. `get_repository_tree` - Get the repository tree for a GitLab project (list files and directories)
|
42. `create_wiki_page` - Create a new wiki page in a GitLab project
|
||||||
|
43. `update_wiki_page` - Update an existing wiki page in a GitLab project
|
||||||
|
44. `delete_wiki_page` - Delete a wiki page from a GitLab project
|
||||||
|
45. `get_repository_tree` - Get the repository tree for a GitLab project (list files and directories)
|
||||||
|
46. `list_pipelines` - List pipelines in a GitLab project with filtering options
|
||||||
|
47. `get_pipeline` - Get details of a specific pipeline in a GitLab project
|
||||||
|
48. `list_pipeline_jobs` - List all jobs in a specific pipeline
|
||||||
|
49. `get_pipeline_job` - Get details of a GitLab pipeline job number
|
||||||
|
50. `get_pipeline_job_output` - Get the output/trace of a GitLab pipeline job number
|
||||||
|
51. `list_merge_requests` - List merge requests in a GitLab project with filtering options
|
||||||
|
52. `list_milestones` - List milestones in a GitLab project with filtering options
|
||||||
|
53. `get_milestone` - Get details of a specific milestone
|
||||||
|
54. `create_milestone` - Create a new milestone in a GitLab project
|
||||||
|
55. `edit_milestone ` - Edit an existing milestone in a GitLab project
|
||||||
|
56. `delete_milestone` - Delete a milestone from a GitLab project
|
||||||
|
57. `get_milestone_issue` - Get issues associated with a specific milestone
|
||||||
|
58. `get_milestone_merge_requests` - Get merge requests associated with a specific milestone
|
||||||
|
59. `promote_milestone` - Promote a milestone to the next stage
|
||||||
|
60. `get_milestone_burndown_events` - Get burndown events for a specific milestone
|
||||||
<!-- TOOLS-END -->
|
<!-- TOOLS-END -->
|
||||||
|
4
package-lock.json
generated
4
package-lock.json
generated
@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "@zereight/mcp-gitlab",
|
"name": "@zereight/mcp-gitlab",
|
||||||
"version": "1.0.36",
|
"version": "1.0.46",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "@zereight/mcp-gitlab",
|
"name": "@zereight/mcp-gitlab",
|
||||||
"version": "1.0.36",
|
"version": "1.0.46",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@modelcontextprotocol/sdk": "1.8.0",
|
"@modelcontextprotocol/sdk": "1.8.0",
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@zereight/mcp-gitlab",
|
"name": "@zereight/mcp-gitlab",
|
||||||
"version": "1.0.38",
|
"version": "1.0.48",
|
||||||
"description": "MCP server for using the GitLab API",
|
"description": "MCP server for using the GitLab API",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"author": "zereight",
|
"author": "zereight",
|
||||||
|
5
release-notes.md
Normal file
5
release-notes.md
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
### 1.0.40 (2025-05-21)
|
||||||
|
|
||||||
|
- Added support for listing discussions (comments/notes) on GitLab issues.
|
||||||
|
- Example: You can now easily fetch all conversations (comments) attached to an issue via the API.
|
||||||
|
- Related PR: [#44](https://github.com/zereight/gitlab-mcp/pull/44)
|
319
schemas.ts
319
schemas.ts
@ -7,6 +7,119 @@ export const GitLabAuthorSchema = z.object({
|
|||||||
date: z.string(),
|
date: z.string(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Pipeline related schemas
|
||||||
|
export const GitLabPipelineSchema = z.object({
|
||||||
|
id: z.number(),
|
||||||
|
project_id: z.number(),
|
||||||
|
sha: z.string(),
|
||||||
|
ref: z.string(),
|
||||||
|
status: z.string(),
|
||||||
|
source: z.string().optional(),
|
||||||
|
created_at: z.string(),
|
||||||
|
updated_at: z.string(),
|
||||||
|
web_url: z.string(),
|
||||||
|
duration: z.number().nullable().optional(),
|
||||||
|
started_at: z.string().nullable().optional(),
|
||||||
|
finished_at: z.string().nullable().optional(),
|
||||||
|
coverage: z.number().nullable().optional(),
|
||||||
|
user: z.object({
|
||||||
|
id: z.number(),
|
||||||
|
name: z.string(),
|
||||||
|
username: z.string(),
|
||||||
|
avatar_url: z.string().nullable().optional(),
|
||||||
|
}).optional(),
|
||||||
|
detailed_status: z.object({
|
||||||
|
icon: z.string().optional(),
|
||||||
|
text: z.string().optional(),
|
||||||
|
label: z.string().optional(),
|
||||||
|
group: z.string().optional(),
|
||||||
|
tooltip: z.string().optional(),
|
||||||
|
has_details: z.boolean().optional(),
|
||||||
|
details_path: z.string().optional(),
|
||||||
|
illustration: z.object({
|
||||||
|
image: z.string().optional(),
|
||||||
|
size: z.string().optional(),
|
||||||
|
title: z.string().optional(),
|
||||||
|
}).nullable().optional(),
|
||||||
|
favicon: z.string().optional(),
|
||||||
|
}).optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Pipeline job related schemas
|
||||||
|
export const GitLabPipelineJobSchema = z.object({
|
||||||
|
id: z.number(),
|
||||||
|
status: z.string(),
|
||||||
|
stage: z.string(),
|
||||||
|
name: z.string(),
|
||||||
|
ref: z.string(),
|
||||||
|
tag: z.boolean(),
|
||||||
|
coverage: z.number().nullable().optional(),
|
||||||
|
created_at: z.string(),
|
||||||
|
started_at: z.string().nullable().optional(),
|
||||||
|
finished_at: z.string().nullable().optional(),
|
||||||
|
duration: z.number().nullable().optional(),
|
||||||
|
user: z.object({
|
||||||
|
id: z.number(),
|
||||||
|
name: z.string(),
|
||||||
|
username: z.string(),
|
||||||
|
avatar_url: z.string().nullable().optional(),
|
||||||
|
}).optional(),
|
||||||
|
commit: z.object({
|
||||||
|
id: z.string(),
|
||||||
|
short_id: z.string(),
|
||||||
|
title: z.string(),
|
||||||
|
author_name: z.string(),
|
||||||
|
author_email: z.string(),
|
||||||
|
}).optional(),
|
||||||
|
pipeline: z.object({
|
||||||
|
id: z.number(),
|
||||||
|
project_id: z.number(),
|
||||||
|
status: z.string(),
|
||||||
|
ref: z.string(),
|
||||||
|
sha: z.string(),
|
||||||
|
}).optional(),
|
||||||
|
web_url: z.string().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for listing pipelines
|
||||||
|
export const ListPipelinesSchema = z.object({
|
||||||
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
|
scope: z.enum(['running', 'pending', 'finished', 'branches', 'tags']).optional().describe("The scope of pipelines"),
|
||||||
|
status: z.enum(['created', 'waiting_for_resource', 'preparing', 'pending', 'running', 'success', 'failed', 'canceled', 'skipped', 'manual', 'scheduled']).optional().describe("The status of pipelines"),
|
||||||
|
ref: z.string().optional().describe("The ref of pipelines"),
|
||||||
|
sha: z.string().optional().describe("The SHA of pipelines"),
|
||||||
|
yaml_errors: z.boolean().optional().describe("Returns pipelines with invalid configurations"),
|
||||||
|
username: z.string().optional().describe("The username of the user who triggered pipelines"),
|
||||||
|
updated_after: z.string().optional().describe("Return pipelines updated after the specified date"),
|
||||||
|
updated_before: z.string().optional().describe("Return pipelines updated before the specified date"),
|
||||||
|
order_by: z.enum(['id', 'status', 'ref', 'updated_at', 'user_id']).optional().describe("Order pipelines by"),
|
||||||
|
sort: z.enum(['asc', 'desc']).optional().describe("Sort pipelines"),
|
||||||
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
|
per_page: z.number().optional().describe("Number of items per page (max 100)"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for getting a specific pipeline
|
||||||
|
export const GetPipelineSchema = z.object({
|
||||||
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
|
pipeline_id: z.number().describe("The ID of the pipeline"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for listing jobs in a pipeline
|
||||||
|
export const ListPipelineJobsSchema = z.object({
|
||||||
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
|
pipeline_id: z.number().describe("The ID of the pipeline"),
|
||||||
|
scope: z.enum(['created', 'pending', 'running', 'failed', 'success', 'canceled', 'skipped', 'manual']).optional().describe("The scope of jobs to show"),
|
||||||
|
include_retried: z.boolean().optional().describe("Whether to include retried jobs"),
|
||||||
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
|
per_page: z.number().optional().describe("Number of items per page (max 100)"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for the input parameters for pipeline job operations
|
||||||
|
export const GetPipelineJobOutputSchema = z.object({
|
||||||
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
|
job_id: z.number().describe("The ID of the job"),
|
||||||
|
});
|
||||||
|
|
||||||
// Namespace related schemas
|
// Namespace related schemas
|
||||||
|
|
||||||
// Base schema for project-related operations
|
// Base schema for project-related operations
|
||||||
@ -107,7 +220,7 @@ export const GitLabRepositorySchema = z.object({
|
|||||||
jobs_enabled: z.boolean().optional(),
|
jobs_enabled: z.boolean().optional(),
|
||||||
snippets_enabled: z.boolean().optional(),
|
snippets_enabled: z.boolean().optional(),
|
||||||
can_create_merge_request_in: z.boolean().optional(),
|
can_create_merge_request_in: z.boolean().optional(),
|
||||||
resolve_outdated_diff_discussions: z.boolean().optional(),
|
resolve_outdated_diff_discussions: z.boolean().nullable().optional(),
|
||||||
shared_runners_enabled: z.boolean().optional(),
|
shared_runners_enabled: z.boolean().optional(),
|
||||||
shared_with_groups: z
|
shared_with_groups: z
|
||||||
.array(
|
.array(
|
||||||
@ -220,6 +333,22 @@ export const GitLabReferenceSchema = z.object({
|
|||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Milestones rest api output schemas
|
||||||
|
export const GitLabMilestonesSchema = z.object({
|
||||||
|
id: z.number(),
|
||||||
|
iid: z.number(),
|
||||||
|
project_id: z.number(),
|
||||||
|
title: z.string(),
|
||||||
|
description: z.string().nullable(),
|
||||||
|
due_date: z.string().nullable(),
|
||||||
|
start_date: z.string().nullable(),
|
||||||
|
state: z.string(),
|
||||||
|
updated_at: z.string(),
|
||||||
|
created_at: z.string(),
|
||||||
|
expired: z.boolean(),
|
||||||
|
web_url: z.string().optional()
|
||||||
|
});
|
||||||
|
|
||||||
// Input schemas for operations
|
// Input schemas for operations
|
||||||
export const CreateRepositoryOptionsSchema = z.object({
|
export const CreateRepositoryOptionsSchema = z.object({
|
||||||
name: z.string(),
|
name: z.string(),
|
||||||
@ -286,7 +415,7 @@ export const GitLabUserSchema = z.object({
|
|||||||
username: z.string(), // Changed from login to match GitLab API
|
username: z.string(), // Changed from login to match GitLab API
|
||||||
id: z.number(),
|
id: z.number(),
|
||||||
name: z.string(),
|
name: z.string(),
|
||||||
avatar_url: z.string(),
|
avatar_url: z.string().nullable(),
|
||||||
web_url: z.string(), // Changed from html_url to match GitLab API
|
web_url: z.string(), // Changed from html_url to match GitLab API
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -294,7 +423,7 @@ export const GitLabMilestoneSchema = z.object({
|
|||||||
id: z.number(),
|
id: z.number(),
|
||||||
iid: z.number(), // Added to match GitLab API
|
iid: z.number(), // Added to match GitLab API
|
||||||
title: z.string(),
|
title: z.string(),
|
||||||
description: z.string(),
|
description: z.string().nullable().default(""),
|
||||||
state: z.string(),
|
state: z.string(),
|
||||||
web_url: z.string(), // Changed from html_url to match GitLab API
|
web_url: z.string(), // Changed from html_url to match GitLab API
|
||||||
});
|
});
|
||||||
@ -304,7 +433,7 @@ export const GitLabIssueSchema = z.object({
|
|||||||
iid: z.number(), // Added to match GitLab API
|
iid: z.number(), // Added to match GitLab API
|
||||||
project_id: z.number(), // Added to match GitLab API
|
project_id: z.number(), // Added to match GitLab API
|
||||||
title: z.string(),
|
title: z.string(),
|
||||||
description: z.string(), // Changed from body to match GitLab API
|
description: z.string().nullable().default(""), // Changed from body to match GitLab API
|
||||||
state: z.string(),
|
state: z.string(),
|
||||||
author: GitLabUserSchema,
|
author: GitLabUserSchema,
|
||||||
assignees: z.array(GitLabUserSchema),
|
assignees: z.array(GitLabUserSchema),
|
||||||
@ -491,6 +620,22 @@ export const CreateMergeRequestNoteSchema = ProjectParamsSchema.extend({
|
|||||||
created_at: z.string().optional().describe("Date the note was created at (ISO 8601 format)"),
|
created_at: z.string().optional().describe("Date the note was created at (ISO 8601 format)"),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Input schema for updating an issue discussion note
|
||||||
|
export const UpdateIssueNoteSchema = ProjectParamsSchema.extend({
|
||||||
|
issue_iid: z.number().describe("The IID of an issue"),
|
||||||
|
discussion_id: z.string().describe("The ID of a thread"),
|
||||||
|
note_id: z.number().describe("The ID of a thread note"),
|
||||||
|
body: z.string().describe("The content of the note or reply"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Input schema for adding a note to an existing issue discussion
|
||||||
|
export const CreateIssueNoteSchema = ProjectParamsSchema.extend({
|
||||||
|
issue_iid: z.number().describe("The IID of an issue"),
|
||||||
|
discussion_id: z.string().describe("The ID of a thread"),
|
||||||
|
body: z.string().describe("The content of the note or reply"),
|
||||||
|
created_at: z.string().optional().describe("Date the note was created at (ISO 8601 format)"),
|
||||||
|
});
|
||||||
|
|
||||||
// API Operation Parameter Schemas
|
// API Operation Parameter Schemas
|
||||||
|
|
||||||
export const CreateOrUpdateFileSchema = ProjectParamsSchema.extend({
|
export const CreateOrUpdateFileSchema = ProjectParamsSchema.extend({
|
||||||
@ -705,6 +850,88 @@ export const ListIssuesSchema = z.object({
|
|||||||
per_page: z.number().optional().describe("Number of items per page"),
|
per_page: z.number().optional().describe("Number of items per page"),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Merge Requests API operation schemas
|
||||||
|
export const ListMergeRequestsSchema = z.object({
|
||||||
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
|
assignee_id: z
|
||||||
|
.number()
|
||||||
|
.optional()
|
||||||
|
.describe("Returns merge requests assigned to the given user ID"),
|
||||||
|
assignee_username: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Returns merge requests assigned to the given username"),
|
||||||
|
author_id: z
|
||||||
|
.number()
|
||||||
|
.optional()
|
||||||
|
.describe("Returns merge requests created by the given user ID"),
|
||||||
|
author_username: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Returns merge requests created by the given username"),
|
||||||
|
reviewer_id: z
|
||||||
|
.number()
|
||||||
|
.optional()
|
||||||
|
.describe("Returns merge requests which have the user as a reviewer"),
|
||||||
|
reviewer_username: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Returns merge requests which have the user as a reviewer"),
|
||||||
|
created_after: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests created after the given time"),
|
||||||
|
created_before: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests created before the given time"),
|
||||||
|
updated_after: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests updated after the given time"),
|
||||||
|
updated_before: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests updated before the given time"),
|
||||||
|
labels: z.array(z.string()).optional().describe("Array of label names"),
|
||||||
|
milestone: z.string().optional().describe("Milestone title"),
|
||||||
|
scope: z
|
||||||
|
.enum(["created_by_me", "assigned_to_me", "all"])
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests from a specific scope"),
|
||||||
|
search: z.string().optional().describe("Search for specific terms"),
|
||||||
|
state: z
|
||||||
|
.enum(["opened", "closed", "locked", "merged", "all"])
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests with a specific state"),
|
||||||
|
order_by: z
|
||||||
|
.enum(["created_at", "updated_at", "priority", "label_priority", "milestone_due", "popularity"])
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests ordered by the given field"),
|
||||||
|
sort: z
|
||||||
|
.enum(["asc", "desc"])
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests sorted in ascending or descending order"),
|
||||||
|
target_branch: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests targeting a specific branch"),
|
||||||
|
source_branch: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Return merge requests from a specific source branch"),
|
||||||
|
wip: z
|
||||||
|
.enum(["yes", "no"])
|
||||||
|
.optional()
|
||||||
|
.describe("Filter merge requests against their wip status"),
|
||||||
|
with_labels_details: z
|
||||||
|
.boolean()
|
||||||
|
.optional()
|
||||||
|
.describe("Return more details for each label"),
|
||||||
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
|
per_page: z.number().optional().describe("Number of items per page"),
|
||||||
|
});
|
||||||
|
|
||||||
export const GetIssueSchema = z.object({
|
export const GetIssueSchema = z.object({
|
||||||
project_id: z.string().describe("Project ID or URL-encoded path"),
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
issue_iid: z.number().describe("The internal ID of the project issue"),
|
issue_iid: z.number().describe("The internal ID of the project issue"),
|
||||||
@ -757,6 +984,15 @@ export const ListIssueLinksSchema = z.object({
|
|||||||
issue_iid: z.number().describe("The internal ID of a project's issue"),
|
issue_iid: z.number().describe("The internal ID of a project's issue"),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const ListIssueDiscussionsSchema = z.object({
|
||||||
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
|
issue_iid: z.number().describe("The internal ID of the project issue"),
|
||||||
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
|
per_page: z.number().optional().describe("Number of items per page"),
|
||||||
|
sort: z.enum(["asc", "desc"]).optional().describe("Return issue discussions sorted in ascending or descending order"),
|
||||||
|
order_by: z.enum(["created_at", "updated_at"]).optional().describe("Return issue discussions ordered by created_at or updated_at fields"),
|
||||||
|
});
|
||||||
|
|
||||||
export const GetIssueLinkSchema = z.object({
|
export const GetIssueLinkSchema = z.object({
|
||||||
project_id: z.string().describe("Project ID or URL-encoded path"),
|
project_id: z.string().describe("Project ID or URL-encoded path"),
|
||||||
issue_iid: z.number().describe("The internal ID of a project's issue"),
|
issue_iid: z.number().describe("The internal ID of a project's issue"),
|
||||||
@ -1040,6 +1276,63 @@ export const CreateMergeRequestThreadSchema = ProjectParamsSchema.extend({
|
|||||||
created_at: z.string().optional().describe("Date the thread was created at (ISO 8601 format)"),
|
created_at: z.string().optional().describe("Date the thread was created at (ISO 8601 format)"),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Milestone related schemas
|
||||||
|
// Schema for listing project milestones
|
||||||
|
export const ListProjectMilestonesSchema = ProjectParamsSchema.extend({
|
||||||
|
iids: z.array(z.number()).optional().describe("Return only the milestones having the given iid"),
|
||||||
|
state: z.enum(["active", "closed"]).optional().describe("Return only active or closed milestones"),
|
||||||
|
title: z.string().optional().describe("Return only milestones with a title matching the provided string"),
|
||||||
|
search: z.string().optional().describe("Return only milestones with a title or description matching the provided string"),
|
||||||
|
include_ancestors: z.boolean().optional().describe("Include ancestor groups"),
|
||||||
|
updated_before: z.string().optional().describe("Return milestones updated before the specified date (ISO 8601 format)"),
|
||||||
|
updated_after: z.string().optional().describe("Return milestones updated after the specified date (ISO 8601 format)"),
|
||||||
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
|
per_page: z.number().optional().describe("Number of items per page (max 100)"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for getting a single milestone
|
||||||
|
export const GetProjectMilestoneSchema = ProjectParamsSchema.extend({
|
||||||
|
milestone_id: z.number().describe("The ID of a project milestone"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for creating a new milestone
|
||||||
|
export const CreateProjectMilestoneSchema = ProjectParamsSchema.extend({
|
||||||
|
title: z.string().describe("The title of the milestone"),
|
||||||
|
description: z.string().optional().describe("The description of the milestone"),
|
||||||
|
due_date: z.string().optional().describe("The due date of the milestone (YYYY-MM-DD)"),
|
||||||
|
start_date: z.string().optional().describe("The start date of the milestone (YYYY-MM-DD)"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for editing a milestone
|
||||||
|
export const EditProjectMilestoneSchema = GetProjectMilestoneSchema.extend({
|
||||||
|
title: z.string().optional().describe("The title of the milestone"),
|
||||||
|
description: z.string().optional().describe("The description of the milestone"),
|
||||||
|
due_date: z.string().optional().describe("The due date of the milestone (YYYY-MM-DD)"),
|
||||||
|
start_date: z.string().optional().describe("The start date of the milestone (YYYY-MM-DD)"),
|
||||||
|
state_event: z.enum(["close", "activate"]).optional().describe("The state event of the milestone"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for deleting a milestone
|
||||||
|
export const DeleteProjectMilestoneSchema = GetProjectMilestoneSchema;
|
||||||
|
|
||||||
|
// Schema for getting issues assigned to a milestone
|
||||||
|
export const GetMilestoneIssuesSchema = GetProjectMilestoneSchema;
|
||||||
|
|
||||||
|
// Schema for getting merge requests assigned to a milestone
|
||||||
|
export const GetMilestoneMergeRequestsSchema = GetProjectMilestoneSchema.extend({
|
||||||
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
|
per_page: z.number().optional().describe("Number of items per page (max 100)"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Schema for promoting a project milestone to a group milestone
|
||||||
|
export const PromoteProjectMilestoneSchema = GetProjectMilestoneSchema;
|
||||||
|
|
||||||
|
// Schema for getting burndown chart events for a milestone
|
||||||
|
export const GetMilestoneBurndownEventsSchema = GetProjectMilestoneSchema.extend({
|
||||||
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
|
per_page: z.number().optional().describe("Number of items per page (max 100)"),
|
||||||
|
});
|
||||||
|
|
||||||
// Export types
|
// Export types
|
||||||
export type GitLabAuthor = z.infer<typeof GitLabAuthorSchema>;
|
export type GitLabAuthor = z.infer<typeof GitLabAuthorSchema>;
|
||||||
export type GitLabFork = z.infer<typeof GitLabForkSchema>;
|
export type GitLabFork = z.infer<typeof GitLabForkSchema>;
|
||||||
@ -1075,6 +1368,9 @@ export type GitLabMergeRequestDiff = z.infer<
|
|||||||
>;
|
>;
|
||||||
export type CreateNoteOptions = z.infer<typeof CreateNoteSchema>;
|
export type CreateNoteOptions = z.infer<typeof CreateNoteSchema>;
|
||||||
export type GitLabIssueLink = z.infer<typeof GitLabIssueLinkSchema>;
|
export type GitLabIssueLink = z.infer<typeof GitLabIssueLinkSchema>;
|
||||||
|
export type ListIssueDiscussionsOptions = z.infer<typeof ListIssueDiscussionsSchema>;
|
||||||
|
export type UpdateIssueNoteOptions = z.infer<typeof UpdateIssueNoteSchema>;
|
||||||
|
export type CreateIssueNoteOptions = z.infer<typeof CreateIssueNoteSchema>;
|
||||||
export type GitLabNamespace = z.infer<typeof GitLabNamespaceSchema>;
|
export type GitLabNamespace = z.infer<typeof GitLabNamespaceSchema>;
|
||||||
export type GitLabNamespaceExistsResponse = z.infer<
|
export type GitLabNamespaceExistsResponse = z.infer<
|
||||||
typeof GitLabNamespaceExistsResponseSchema
|
typeof GitLabNamespaceExistsResponseSchema
|
||||||
@ -1092,3 +1388,18 @@ export type GetRepositoryTreeOptions = z.infer<typeof GetRepositoryTreeSchema>;
|
|||||||
export type MergeRequestThreadPosition = z.infer<typeof MergeRequestThreadPositionSchema>;
|
export type MergeRequestThreadPosition = z.infer<typeof MergeRequestThreadPositionSchema>;
|
||||||
export type CreateMergeRequestThreadOptions = z.infer<typeof CreateMergeRequestThreadSchema>;
|
export type CreateMergeRequestThreadOptions = z.infer<typeof CreateMergeRequestThreadSchema>;
|
||||||
export type CreateMergeRequestNoteOptions = z.infer<typeof CreateMergeRequestNoteSchema>;
|
export type CreateMergeRequestNoteOptions = z.infer<typeof CreateMergeRequestNoteSchema>;
|
||||||
|
export type GitLabPipelineJob = z.infer<typeof GitLabPipelineJobSchema>;
|
||||||
|
export type GitLabPipeline = z.infer<typeof GitLabPipelineSchema>;
|
||||||
|
export type ListPipelinesOptions = z.infer<typeof ListPipelinesSchema>;
|
||||||
|
export type GetPipelineOptions = z.infer<typeof GetPipelineSchema>;
|
||||||
|
export type ListPipelineJobsOptions = z.infer<typeof ListPipelineJobsSchema>;
|
||||||
|
export type GitLabMilestones = z.infer<typeof GitLabMilestonesSchema>;
|
||||||
|
export type ListProjectMilestonesOptions = z.infer<typeof ListProjectMilestonesSchema>;
|
||||||
|
export type GetProjectMilestoneOptions = z.infer<typeof GetProjectMilestoneSchema>;
|
||||||
|
export type CreateProjectMilestoneOptions = z.infer<typeof CreateProjectMilestoneSchema>;
|
||||||
|
export type EditProjectMilestoneOptions = z.infer<typeof EditProjectMilestoneSchema>;
|
||||||
|
export type DeleteProjectMilestoneOptions = z.infer<typeof DeleteProjectMilestoneSchema>;
|
||||||
|
export type GetMilestoneIssuesOptions = z.infer<typeof GetMilestoneIssuesSchema>;
|
||||||
|
export type GetMilestoneMergeRequestsOptions = z.infer<typeof GetMilestoneMergeRequestsSchema>;
|
||||||
|
export type PromoteProjectMilestoneOptions = z.infer<typeof PromoteProjectMilestoneSchema>;
|
||||||
|
export type GetMilestoneBurndownEventsOptions = z.infer<typeof GetMilestoneBurndownEventsSchema>;
|
||||||
|
18
scripts/image_push.sh
Normal file
18
scripts/image_push.sh
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
if [ -z "$1" ]; then
|
||||||
|
echo "Error: docker user name required."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
DOCKER_USER=$1
|
||||||
|
IMAGE_NAME=gitlab-mcp
|
||||||
|
IMAGE_VERSION=$(jq -r '.version' package.json)
|
||||||
|
|
||||||
|
echo "${DOCKER_USER}/${IMAGE_NAME}:${IMAGE_VERSION}"
|
||||||
|
docker build --platform=linux/arm64 -t "${DOCKER_USER}/${IMAGE_NAME}:latest" .
|
||||||
|
|
||||||
|
docker tag "${DOCKER_USER}/${IMAGE_NAME}:latest" "${DOCKER_USER}/${IMAGE_NAME}:${IMAGE_VERSION}"
|
||||||
|
|
||||||
|
docker push "${DOCKER_USER}/${IMAGE_NAME}:latest"
|
||||||
|
docker push "${DOCKER_USER}/${IMAGE_NAME}:${IMAGE_VERSION}"
|
Reference in New Issue
Block a user