mirror of
https://github.com/markwylde/claude-code-gitea-action.git
synced 2026-02-20 02:22:49 +08:00
Attempt to make this work
This commit is contained in:
@@ -536,13 +536,13 @@ ${context.directPrompt ? ` - DIRECT INSTRUCTION: A direct instruction was prov
|
|||||||
- Commit changes using mcp__local_git_ops__commit_files to the existing branch (works for both new and existing files).
|
- Commit changes using mcp__local_git_ops__commit_files to the existing branch (works for both new and existing files).
|
||||||
- Use mcp__local_git_ops__commit_files to commit files atomically in a single commit (supports single or multiple files).
|
- Use mcp__local_git_ops__commit_files to commit files atomically in a single commit (supports single or multiple files).
|
||||||
- CRITICAL: After committing, you MUST push the branch to the remote repository using mcp__local_git_ops__push_branch
|
- CRITICAL: After committing, you MUST push the branch to the remote repository using mcp__local_git_ops__push_branch
|
||||||
- When pushing changes with this tool and TRIGGER_USERNAME is not "Unknown", include a "Co-authored-by: ${context.triggerUsername} <${context.triggerUsername}@users.noreply.github.com>" line in the commit message.`
|
- When pushing changes with this tool and TRIGGER_USERNAME is not "Unknown", include a "Co-authored-by: ${context.triggerUsername} <${context.triggerUsername}@users.noreply.local>" line in the commit message.`
|
||||||
: `
|
: `
|
||||||
- You are already on the correct branch (${eventData.claudeBranch || "the PR branch"}). Do not create a new branch.
|
- You are already on the correct branch (${eventData.claudeBranch || "the PR branch"}). Do not create a new branch.
|
||||||
- Commit changes using mcp__local_git_ops__commit_files (works for both new and existing files)
|
- Commit changes using mcp__local_git_ops__commit_files (works for both new and existing files)
|
||||||
- Use mcp__local_git_ops__commit_files to commit files atomically in a single commit (supports single or multiple files).
|
- Use mcp__local_git_ops__commit_files to commit files atomically in a single commit (supports single or multiple files).
|
||||||
- CRITICAL: After committing, you MUST push the branch to the remote repository using mcp__local_git_ops__push_branch
|
- CRITICAL: After committing, you MUST push the branch to the remote repository using mcp__local_git_ops__push_branch
|
||||||
- When pushing changes and TRIGGER_USERNAME is not "Unknown", include a "Co-authored-by: ${context.triggerUsername} <${context.triggerUsername}@users.noreply.github.com>" line in the commit message.
|
- When pushing changes and TRIGGER_USERNAME is not "Unknown", include a "Co-authored-by: ${context.triggerUsername} <${context.triggerUsername}@users.noreply.local>" line in the commit message.
|
||||||
${
|
${
|
||||||
eventData.claudeBranch
|
eventData.claudeBranch
|
||||||
? `- Provide a URL to create a PR manually in this format:
|
? `- Provide a URL to create a PR manually in this format:
|
||||||
@@ -584,7 +584,7 @@ Important Notes:
|
|||||||
- Never create new comments. Only update the existing comment using ${eventData.eventName === "pull_request_review_comment" ? "mcp__github__update_pull_request_comment" : "mcp__github__update_issue_comment"} with comment_id: ${context.claudeCommentId}.
|
- Never create new comments. Only update the existing comment using ${eventData.eventName === "pull_request_review_comment" ? "mcp__github__update_pull_request_comment" : "mcp__github__update_issue_comment"} with comment_id: ${context.claudeCommentId}.
|
||||||
- This includes ALL responses: code reviews, answers to questions, progress updates, and final results.${eventData.isPR ? "\n- PR CRITICAL: After reading files and forming your response, you MUST post it by calling mcp__github__update_issue_comment. Do NOT just respond with a normal response, the user will not see it." : ""}
|
- This includes ALL responses: code reviews, answers to questions, progress updates, and final results.${eventData.isPR ? "\n- PR CRITICAL: After reading files and forming your response, you MUST post it by calling mcp__github__update_issue_comment. Do NOT just respond with a normal response, the user will not see it." : ""}
|
||||||
- You communicate exclusively by editing your single comment - not through any other means.
|
- You communicate exclusively by editing your single comment - not through any other means.
|
||||||
- Use this spinner HTML when work is in progress: <img src="https://github.com/user-attachments/assets/5ac382c7-e004-429b-8e35-7feb3e8f9c6f" width="14px" height="14px" style="vertical-align: middle; margin-left: 4px;" />
|
- Use this spinner HTML when work is in progress: <img src="" width="14px" height="14px" style="vertical-align: middle; margin-left: 4px;" />
|
||||||
${eventData.isPR && !eventData.claudeBranch ? `- Always push to the existing branch when triggered on a PR.` : `- IMPORTANT: You are already on the correct branch (${eventData.claudeBranch || "the created branch"}). Never create new branches when triggered on issues or closed/merged PRs.`}
|
${eventData.isPR && !eventData.claudeBranch ? `- Always push to the existing branch when triggered on a PR.` : `- IMPORTANT: You are already on the correct branch (${eventData.claudeBranch || "the created branch"}). Never create new branches when triggered on issues or closed/merged PRs.`}
|
||||||
- Use mcp__local_git_ops__commit_files for making commits (works for both new and existing files, single or multiple). Use mcp__local_git_ops__delete_files for deleting files (supports deleting single or multiple files atomically), or mcp__github__delete_file for deleting a single file. Edit files locally, and the tool will read the content from the same path on disk.
|
- Use mcp__local_git_ops__commit_files for making commits (works for both new and existing files, single or multiple). Use mcp__local_git_ops__delete_files for deleting files (supports deleting single or multiple files atomically), or mcp__github__delete_file for deleting a single file. Edit files locally, and the tool will read the content from the same path on disk.
|
||||||
Tool usage examples:
|
Tool usage examples:
|
||||||
@@ -621,7 +621,7 @@ What You CANNOT Do:
|
|||||||
- View CI/CD results or workflow run outputs (cannot access GitHub Actions logs or test results)
|
- View CI/CD results or workflow run outputs (cannot access GitHub Actions logs or test results)
|
||||||
|
|
||||||
When users ask you to perform actions you cannot do, politely explain the limitation and, when applicable, direct them to the FAQ for more information and workarounds:
|
When users ask you to perform actions you cannot do, politely explain the limitation and, when applicable, direct them to the FAQ for more information and workarounds:
|
||||||
"I'm unable to [specific action] due to [reason]. You can find more information and potential workarounds in the [FAQ](https://github.com/anthropics/claude-code-action/blob/main/FAQ.md)."
|
"I'm unable to [specific action] due to [reason]. Please check the documentation for more information and potential workarounds."
|
||||||
|
|
||||||
If a user asks for something outside these capabilities (and you have no other tools provided), politely explain that you cannot perform that action and suggest an alternative approach if possible.
|
If a user asks for something outside these capabilities (and you have no other tools provided), politely explain that you cannot perform that action and suggest an alternative approach if possible.
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,12 @@ import {
|
|||||||
} from "../github/context";
|
} from "../github/context";
|
||||||
import { GITEA_SERVER_URL } from "../github/api/config";
|
import { GITEA_SERVER_URL } from "../github/api/config";
|
||||||
import { checkAndDeleteEmptyBranch } from "../github/operations/branch-cleanup";
|
import { checkAndDeleteEmptyBranch } from "../github/operations/branch-cleanup";
|
||||||
import { branchHasChanges, fetchBranch, branchExists, remoteBranchExists } from "../github/utils/local-git";
|
import {
|
||||||
|
branchHasChanges,
|
||||||
|
fetchBranch,
|
||||||
|
branchExists,
|
||||||
|
remoteBranchExists,
|
||||||
|
} from "../github/utils/local-git";
|
||||||
|
|
||||||
async function run() {
|
async function run() {
|
||||||
try {
|
try {
|
||||||
@@ -108,14 +113,17 @@ async function run() {
|
|||||||
if (!containsPRUrl) {
|
if (!containsPRUrl) {
|
||||||
// Check if we're using Gitea or GitHub
|
// Check if we're using Gitea or GitHub
|
||||||
const giteaApiUrl = process.env.GITEA_API_URL?.trim();
|
const giteaApiUrl = process.env.GITEA_API_URL?.trim();
|
||||||
const isGitea = giteaApiUrl &&
|
const isGitea =
|
||||||
|
giteaApiUrl &&
|
||||||
giteaApiUrl !== "" &&
|
giteaApiUrl !== "" &&
|
||||||
!giteaApiUrl.includes("api.github.com") &&
|
!giteaApiUrl.includes("api.github.com") &&
|
||||||
!giteaApiUrl.includes("github.com");
|
!giteaApiUrl.includes("github.com");
|
||||||
|
|
||||||
if (isGitea) {
|
if (isGitea) {
|
||||||
// Use local git commands for Gitea
|
// Use local git commands for Gitea
|
||||||
console.log("Using local git commands for PR link check (Gitea mode)");
|
console.log(
|
||||||
|
"Using local git commands for PR link check (Gitea mode)",
|
||||||
|
);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Fetch latest changes from remote
|
// Fetch latest changes from remote
|
||||||
@@ -123,7 +131,10 @@ async function run() {
|
|||||||
await fetchBranch(baseBranch);
|
await fetchBranch(baseBranch);
|
||||||
|
|
||||||
// Check if branch exists and has changes
|
// Check if branch exists and has changes
|
||||||
const { hasChanges, branchSha, baseSha } = await branchHasChanges(claudeBranch, baseBranch);
|
const { hasChanges, branchSha, baseSha } = await branchHasChanges(
|
||||||
|
claudeBranch,
|
||||||
|
baseBranch,
|
||||||
|
);
|
||||||
|
|
||||||
if (branchSha && baseSha) {
|
if (branchSha && baseSha) {
|
||||||
if (hasChanges) {
|
if (hasChanges) {
|
||||||
@@ -150,7 +161,9 @@ async function run() {
|
|||||||
const remoteExists = await remoteBranchExists(claudeBranch);
|
const remoteExists = await remoteBranchExists(claudeBranch);
|
||||||
|
|
||||||
if (localExists || remoteExists) {
|
if (localExists || remoteExists) {
|
||||||
console.log(`Branch ${claudeBranch} exists but SHA comparison failed, adding PR link to be safe`);
|
console.log(
|
||||||
|
`Branch ${claudeBranch} exists but SHA comparison failed, adding PR link to be safe`,
|
||||||
|
);
|
||||||
const entityType = context.isPR ? "PR" : "Issue";
|
const entityType = context.isPR ? "PR" : "Issue";
|
||||||
const prTitle = encodeURIComponent(
|
const prTitle = encodeURIComponent(
|
||||||
`${entityType} #${context.entityNumber}: Changes from Claude`,
|
`${entityType} #${context.entityNumber}: Changes from Claude`,
|
||||||
|
|||||||
@@ -1,6 +1,11 @@
|
|||||||
import type { GitHubClient } from "../api/client";
|
import type { GitHubClient } from "../api/client";
|
||||||
import { GITEA_SERVER_URL } from "../api/config";
|
import { GITEA_SERVER_URL } from "../api/config";
|
||||||
import { branchHasChanges, fetchBranch, branchExists, remoteBranchExists } from "../utils/local-git";
|
import {
|
||||||
|
branchHasChanges,
|
||||||
|
fetchBranch,
|
||||||
|
branchExists,
|
||||||
|
remoteBranchExists,
|
||||||
|
} from "../utils/local-git";
|
||||||
|
|
||||||
export async function checkAndDeleteEmptyBranch(
|
export async function checkAndDeleteEmptyBranch(
|
||||||
client: GitHubClient,
|
client: GitHubClient,
|
||||||
@@ -15,7 +20,8 @@ export async function checkAndDeleteEmptyBranch(
|
|||||||
if (claudeBranch) {
|
if (claudeBranch) {
|
||||||
// Check if we're using Gitea or GitHub
|
// Check if we're using Gitea or GitHub
|
||||||
const giteaApiUrl = process.env.GITEA_API_URL?.trim();
|
const giteaApiUrl = process.env.GITEA_API_URL?.trim();
|
||||||
const isGitea = giteaApiUrl &&
|
const isGitea =
|
||||||
|
giteaApiUrl &&
|
||||||
giteaApiUrl !== "" &&
|
giteaApiUrl !== "" &&
|
||||||
!giteaApiUrl.includes("api.github.com") &&
|
!giteaApiUrl.includes("api.github.com") &&
|
||||||
!giteaApiUrl.includes("github.com");
|
!giteaApiUrl.includes("github.com");
|
||||||
@@ -30,7 +36,10 @@ export async function checkAndDeleteEmptyBranch(
|
|||||||
await fetchBranch(baseBranch);
|
await fetchBranch(baseBranch);
|
||||||
|
|
||||||
// Check if branch exists and has changes
|
// Check if branch exists and has changes
|
||||||
const { hasChanges, branchSha, baseSha } = await branchHasChanges(claudeBranch, baseBranch);
|
const { hasChanges, branchSha, baseSha } = await branchHasChanges(
|
||||||
|
claudeBranch,
|
||||||
|
baseBranch,
|
||||||
|
);
|
||||||
|
|
||||||
if (branchSha && baseSha) {
|
if (branchSha && baseSha) {
|
||||||
if (hasChanges) {
|
if (hasChanges) {
|
||||||
@@ -51,7 +60,9 @@ export async function checkAndDeleteEmptyBranch(
|
|||||||
const remoteExists = await remoteBranchExists(claudeBranch);
|
const remoteExists = await remoteBranchExists(claudeBranch);
|
||||||
|
|
||||||
if (localExists || remoteExists) {
|
if (localExists || remoteExists) {
|
||||||
console.log(`Branch ${claudeBranch} exists but SHA comparison failed, assuming it has commits`);
|
console.log(
|
||||||
|
`Branch ${claudeBranch} exists but SHA comparison failed, assuming it has commits`,
|
||||||
|
);
|
||||||
const branchUrl = `${GITEA_SERVER_URL}/${owner}/${repo}/tree/${claudeBranch}`;
|
const branchUrl = `${GITEA_SERVER_URL}/${owner}/${repo}/tree/${claudeBranch}`;
|
||||||
branchLink = `\n[View branch](${branchUrl})`;
|
branchLink = `\n[View branch](${branchUrl})`;
|
||||||
} else {
|
} else {
|
||||||
@@ -81,7 +92,11 @@ export async function checkAndDeleteEmptyBranch(
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Get base branch info for comparison
|
// Get base branch info for comparison
|
||||||
const baseResponse = await client.api.getBranch(owner, repo, baseBranch);
|
const baseResponse = await client.api.getBranch(
|
||||||
|
owner,
|
||||||
|
repo,
|
||||||
|
baseBranch,
|
||||||
|
);
|
||||||
|
|
||||||
const branchSha = branchResponse.data.commit.sha;
|
const branchSha = branchResponse.data.commit.sha;
|
||||||
const baseSha = baseResponse.data.commit.sha;
|
const baseSha = baseResponse.data.commit.sha;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { GITEA_SERVER_URL } from "../../api/config";
|
import { GITEA_SERVER_URL } from "../../api/config";
|
||||||
|
|
||||||
export const SPINNER_HTML =
|
export const SPINNER_HTML =
|
||||||
'<img src="https://github.com/user-attachments/assets/5ac382c7-e004-429b-8e35-7feb3e8f9c6f" width="14px" height="14px" style="vertical-align: middle; margin-left: 4px;" />';
|
'<img src="" width="14px" height="14px" style="vertical-align: middle; margin-left: 4px;" />';
|
||||||
|
|
||||||
export function createJobRunLink(
|
export function createJobRunLink(
|
||||||
owner: string,
|
owner: string,
|
||||||
|
|||||||
@@ -53,7 +53,14 @@ export async function getBranchSha(branchName: string): Promise<string | null> {
|
|||||||
/**
|
/**
|
||||||
* Check if a branch has commits different from base branch
|
* Check if a branch has commits different from base branch
|
||||||
*/
|
*/
|
||||||
export async function branchHasChanges(branchName: string, baseBranch: string): Promise<{ hasChanges: boolean; branchSha: string | null; baseSha: string | null }> {
|
export async function branchHasChanges(
|
||||||
|
branchName: string,
|
||||||
|
baseBranch: string,
|
||||||
|
): Promise<{
|
||||||
|
hasChanges: boolean;
|
||||||
|
branchSha: string | null;
|
||||||
|
baseSha: string | null;
|
||||||
|
}> {
|
||||||
try {
|
try {
|
||||||
const branchSha = await getBranchSha(branchName);
|
const branchSha = await getBranchSha(branchName);
|
||||||
const baseSha = await getBranchSha(baseBranch);
|
const baseSha = await getBranchSha(baseBranch);
|
||||||
@@ -65,7 +72,10 @@ export async function branchHasChanges(branchName: string, baseBranch: string):
|
|||||||
const hasChanges = branchSha !== baseSha;
|
const hasChanges = branchSha !== baseSha;
|
||||||
return { hasChanges, branchSha, baseSha };
|
return { hasChanges, branchSha, baseSha };
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(`Error comparing branches ${branchName} and ${baseBranch}:`, error);
|
console.error(
|
||||||
|
`Error comparing branches ${branchName} and ${baseBranch}:`,
|
||||||
|
error,
|
||||||
|
);
|
||||||
return { hasChanges: false, branchSha: null, baseSha: null };
|
return { hasChanges: false, branchSha: null, baseSha: null };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -78,7 +88,9 @@ export async function fetchBranch(branchName: string): Promise<boolean> {
|
|||||||
await $`git fetch origin ${branchName}`;
|
await $`git fetch origin ${branchName}`;
|
||||||
return true;
|
return true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.log(`Could not fetch branch ${branchName} from remote (may not exist yet)`);
|
console.log(
|
||||||
|
`Could not fetch branch ${branchName} from remote (may not exist yet)`,
|
||||||
|
);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -17,7 +17,7 @@ console.log(`[GITEA-MCP] REPO_OWNER: ${REPO_OWNER}`);
|
|||||||
console.log(`[GITEA-MCP] REPO_NAME: ${REPO_NAME}`);
|
console.log(`[GITEA-MCP] REPO_NAME: ${REPO_NAME}`);
|
||||||
console.log(`[GITEA-MCP] BRANCH_NAME: ${BRANCH_NAME}`);
|
console.log(`[GITEA-MCP] BRANCH_NAME: ${BRANCH_NAME}`);
|
||||||
console.log(`[GITEA-MCP] GITEA_API_URL: ${GITEA_API_URL}`);
|
console.log(`[GITEA-MCP] GITEA_API_URL: ${GITEA_API_URL}`);
|
||||||
console.log(`[GITEA-MCP] GITHUB_TOKEN: ${GITHUB_TOKEN ? '***' : 'undefined'}`);
|
console.log(`[GITEA-MCP] GITHUB_TOKEN: ${GITHUB_TOKEN ? "***" : "undefined"}`);
|
||||||
|
|
||||||
if (!REPO_OWNER || !REPO_NAME || !GITHUB_TOKEN) {
|
if (!REPO_OWNER || !REPO_NAME || !GITHUB_TOKEN) {
|
||||||
console.error(
|
console.error(
|
||||||
@@ -41,8 +41,8 @@ async function giteaRequest(
|
|||||||
console.log(`[GITEA-MCP] Making ${method} request to: ${url}`);
|
console.log(`[GITEA-MCP] Making ${method} request to: ${url}`);
|
||||||
|
|
||||||
const headers: Record<string, string> = {
|
const headers: Record<string, string> = {
|
||||||
"Authorization": `token ${GITHUB_TOKEN}`,
|
Authorization: `token ${GITHUB_TOKEN}`,
|
||||||
"Accept": "application/json",
|
Accept: "application/json",
|
||||||
};
|
};
|
||||||
|
|
||||||
if (body) {
|
if (body) {
|
||||||
@@ -60,7 +60,9 @@ async function giteaRequest(
|
|||||||
console.log(`[GITEA-MCP] Response: ${responseText.substring(0, 500)}...`);
|
console.log(`[GITEA-MCP] Response: ${responseText.substring(0, 500)}...`);
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error(`Gitea API request failed: ${response.status} ${responseText}`);
|
throw new Error(
|
||||||
|
`Gitea API request failed: ${response.status} ${responseText}`,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return responseText ? JSON.parse(responseText) : null;
|
return responseText ? JSON.parse(responseText) : null;
|
||||||
@@ -75,7 +77,9 @@ server.tool(
|
|||||||
},
|
},
|
||||||
async ({ issue_number }) => {
|
async ({ issue_number }) => {
|
||||||
try {
|
try {
|
||||||
const issue = await giteaRequest(`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/issues/${issue_number}`);
|
const issue = await giteaRequest(
|
||||||
|
`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/issues/${issue_number}`,
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -86,7 +90,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error getting issue: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error getting issue: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -108,8 +113,16 @@ server.tool(
|
|||||||
"Get all comments for a specific issue",
|
"Get all comments for a specific issue",
|
||||||
{
|
{
|
||||||
issue_number: z.number().describe("The issue number to fetch comments for"),
|
issue_number: z.number().describe("The issue number to fetch comments for"),
|
||||||
since: z.string().optional().describe("Only show comments updated after this time (ISO 8601 format)"),
|
since: z
|
||||||
before: z.string().optional().describe("Only show comments updated before this time (ISO 8601 format)"),
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Only show comments updated after this time (ISO 8601 format)"),
|
||||||
|
before: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe(
|
||||||
|
"Only show comments updated before this time (ISO 8601 format)",
|
||||||
|
),
|
||||||
},
|
},
|
||||||
async ({ issue_number, since, before }) => {
|
async ({ issue_number, since, before }) => {
|
||||||
try {
|
try {
|
||||||
@@ -134,8 +147,11 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
console.error(`[GITEA-MCP] Error getting issue comments: ${errorMessage}`);
|
error instanceof Error ? error.message : String(error);
|
||||||
|
console.error(
|
||||||
|
`[GITEA-MCP] Error getting issue comments: ${errorMessage}`,
|
||||||
|
);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
{
|
{
|
||||||
@@ -175,7 +191,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error adding issue comment: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error adding issue comment: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -216,8 +233,11 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
console.error(`[GITEA-MCP] Error updating issue comment: ${errorMessage}`);
|
error instanceof Error ? error.message : String(error);
|
||||||
|
console.error(
|
||||||
|
`[GITEA-MCP] Error updating issue comment: ${errorMessage}`,
|
||||||
|
);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
{
|
{
|
||||||
@@ -255,8 +275,11 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
console.error(`[GITEA-MCP] Error deleting issue comment: ${errorMessage}`);
|
error instanceof Error ? error.message : String(error);
|
||||||
|
console.error(
|
||||||
|
`[GITEA-MCP] Error deleting issue comment: ${errorMessage}`,
|
||||||
|
);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
{
|
{
|
||||||
@@ -280,7 +303,9 @@ server.tool(
|
|||||||
},
|
},
|
||||||
async ({ comment_id }) => {
|
async ({ comment_id }) => {
|
||||||
try {
|
try {
|
||||||
const comment = await giteaRequest(`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/issues/comments/${comment_id}`);
|
const comment = await giteaRequest(
|
||||||
|
`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/issues/comments/${comment_id}`,
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -291,7 +316,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error getting comment: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error getting comment: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -312,16 +338,40 @@ server.tool(
|
|||||||
"list_issues",
|
"list_issues",
|
||||||
"List issues in the repository",
|
"List issues in the repository",
|
||||||
{
|
{
|
||||||
state: z.enum(["open", "closed", "all"]).optional().describe("Issue state filter"),
|
state: z
|
||||||
labels: z.string().optional().describe("Comma-separated list of label names"),
|
.enum(["open", "closed", "all"])
|
||||||
|
.optional()
|
||||||
|
.describe("Issue state filter"),
|
||||||
|
labels: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Comma-separated list of label names"),
|
||||||
milestone: z.string().optional().describe("Milestone title to filter by"),
|
milestone: z.string().optional().describe("Milestone title to filter by"),
|
||||||
assignee: z.string().optional().describe("Username to filter issues assigned to"),
|
assignee: z
|
||||||
creator: z.string().optional().describe("Username to filter issues created by"),
|
.string()
|
||||||
mentioned: z.string().optional().describe("Username to filter issues that mention"),
|
.optional()
|
||||||
|
.describe("Username to filter issues assigned to"),
|
||||||
|
creator: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Username to filter issues created by"),
|
||||||
|
mentioned: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe("Username to filter issues that mention"),
|
||||||
page: z.number().optional().describe("Page number for pagination"),
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
limit: z.number().optional().describe("Number of items per page"),
|
limit: z.number().optional().describe("Number of items per page"),
|
||||||
},
|
},
|
||||||
async ({ state, labels, milestone, assignee, creator, mentioned, page, limit }) => {
|
async ({
|
||||||
|
state,
|
||||||
|
labels,
|
||||||
|
milestone,
|
||||||
|
assignee,
|
||||||
|
creator,
|
||||||
|
mentioned,
|
||||||
|
page,
|
||||||
|
limit,
|
||||||
|
}) => {
|
||||||
try {
|
try {
|
||||||
let endpoint = `/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/issues`;
|
let endpoint = `/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/issues`;
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
@@ -350,7 +400,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error listing issues: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error listing issues: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -374,9 +425,18 @@ server.tool(
|
|||||||
title: z.string().describe("Issue title"),
|
title: z.string().describe("Issue title"),
|
||||||
body: z.string().optional().describe("Issue body content"),
|
body: z.string().optional().describe("Issue body content"),
|
||||||
assignee: z.string().optional().describe("Username to assign the issue to"),
|
assignee: z.string().optional().describe("Username to assign the issue to"),
|
||||||
assignees: z.array(z.string()).optional().describe("Array of usernames to assign the issue to"),
|
assignees: z
|
||||||
milestone: z.number().optional().describe("Milestone ID to associate with the issue"),
|
.array(z.string())
|
||||||
labels: z.array(z.string()).optional().describe("Array of label names to apply to the issue"),
|
.optional()
|
||||||
|
.describe("Array of usernames to assign the issue to"),
|
||||||
|
milestone: z
|
||||||
|
.number()
|
||||||
|
.optional()
|
||||||
|
.describe("Milestone ID to associate with the issue"),
|
||||||
|
labels: z
|
||||||
|
.array(z.string())
|
||||||
|
.optional()
|
||||||
|
.describe("Array of label names to apply to the issue"),
|
||||||
},
|
},
|
||||||
async ({ title, body, assignee, assignees, milestone, labels }) => {
|
async ({ title, body, assignee, assignees, milestone, labels }) => {
|
||||||
try {
|
try {
|
||||||
@@ -403,7 +463,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error creating issue: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error creating issue: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -428,12 +489,30 @@ server.tool(
|
|||||||
title: z.string().optional().describe("New issue title"),
|
title: z.string().optional().describe("New issue title"),
|
||||||
body: z.string().optional().describe("New issue body content"),
|
body: z.string().optional().describe("New issue body content"),
|
||||||
assignee: z.string().optional().describe("Username to assign the issue to"),
|
assignee: z.string().optional().describe("Username to assign the issue to"),
|
||||||
assignees: z.array(z.string()).optional().describe("Array of usernames to assign the issue to"),
|
assignees: z
|
||||||
milestone: z.number().optional().describe("Milestone ID to associate with the issue"),
|
.array(z.string())
|
||||||
labels: z.array(z.string()).optional().describe("Array of label names to apply to the issue"),
|
.optional()
|
||||||
|
.describe("Array of usernames to assign the issue to"),
|
||||||
|
milestone: z
|
||||||
|
.number()
|
||||||
|
.optional()
|
||||||
|
.describe("Milestone ID to associate with the issue"),
|
||||||
|
labels: z
|
||||||
|
.array(z.string())
|
||||||
|
.optional()
|
||||||
|
.describe("Array of label names to apply to the issue"),
|
||||||
state: z.enum(["open", "closed"]).optional().describe("Issue state"),
|
state: z.enum(["open", "closed"]).optional().describe("Issue state"),
|
||||||
},
|
},
|
||||||
async ({ issue_number, title, body, assignee, assignees, milestone, labels, state }) => {
|
async ({
|
||||||
|
issue_number,
|
||||||
|
title,
|
||||||
|
body,
|
||||||
|
assignee,
|
||||||
|
assignees,
|
||||||
|
milestone,
|
||||||
|
labels,
|
||||||
|
state,
|
||||||
|
}) => {
|
||||||
try {
|
try {
|
||||||
const updateData: any = {};
|
const updateData: any = {};
|
||||||
|
|
||||||
@@ -460,7 +539,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error updating issue: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error updating issue: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -477,11 +557,7 @@ server.tool(
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Get repository information
|
// Get repository information
|
||||||
server.tool(
|
server.tool("get_repository", "Get repository information", {}, async () => {
|
||||||
"get_repository",
|
|
||||||
"Get repository information",
|
|
||||||
{},
|
|
||||||
async () => {
|
|
||||||
try {
|
try {
|
||||||
const repo = await giteaRequest(`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}`);
|
const repo = await giteaRequest(`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}`);
|
||||||
|
|
||||||
@@ -507,15 +583,17 @@ server.tool(
|
|||||||
isError: true,
|
isError: true,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
},
|
});
|
||||||
);
|
|
||||||
|
|
||||||
// Get pull requests
|
// Get pull requests
|
||||||
server.tool(
|
server.tool(
|
||||||
"list_pull_requests",
|
"list_pull_requests",
|
||||||
"List pull requests in the repository",
|
"List pull requests in the repository",
|
||||||
{
|
{
|
||||||
state: z.enum(["open", "closed", "all"]).optional().describe("Pull request state filter"),
|
state: z
|
||||||
|
.enum(["open", "closed", "all"])
|
||||||
|
.optional()
|
||||||
|
.describe("Pull request state filter"),
|
||||||
head: z.string().optional().describe("Head branch name"),
|
head: z.string().optional().describe("Head branch name"),
|
||||||
base: z.string().optional().describe("Base branch name"),
|
base: z.string().optional().describe("Base branch name"),
|
||||||
page: z.number().optional().describe("Page number for pagination"),
|
page: z.number().optional().describe("Page number for pagination"),
|
||||||
@@ -547,7 +625,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error listing pull requests: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error listing pull requests: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -572,7 +651,9 @@ server.tool(
|
|||||||
},
|
},
|
||||||
async ({ pull_number }) => {
|
async ({ pull_number }) => {
|
||||||
try {
|
try {
|
||||||
const pull = await giteaRequest(`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/pulls/${pull_number}`);
|
const pull = await giteaRequest(
|
||||||
|
`/api/v1/repos/${REPO_OWNER}/${REPO_NAME}/pulls/${pull_number}`,
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -583,7 +664,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error getting pull request: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error getting pull request: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
@@ -608,12 +690,33 @@ server.tool(
|
|||||||
body: z.string().optional().describe("Pull request body/description"),
|
body: z.string().optional().describe("Pull request body/description"),
|
||||||
head: z.string().describe("Head branch name"),
|
head: z.string().describe("Head branch name"),
|
||||||
base: z.string().describe("Base branch name"),
|
base: z.string().describe("Base branch name"),
|
||||||
assignee: z.string().optional().describe("Username to assign the pull request to"),
|
assignee: z
|
||||||
assignees: z.array(z.string()).optional().describe("Array of usernames to assign the pull request to"),
|
.string()
|
||||||
milestone: z.number().optional().describe("Milestone ID to associate with the pull request"),
|
.optional()
|
||||||
labels: z.array(z.string()).optional().describe("Array of label names to apply to the pull request"),
|
.describe("Username to assign the pull request to"),
|
||||||
|
assignees: z
|
||||||
|
.array(z.string())
|
||||||
|
.optional()
|
||||||
|
.describe("Array of usernames to assign the pull request to"),
|
||||||
|
milestone: z
|
||||||
|
.number()
|
||||||
|
.optional()
|
||||||
|
.describe("Milestone ID to associate with the pull request"),
|
||||||
|
labels: z
|
||||||
|
.array(z.string())
|
||||||
|
.optional()
|
||||||
|
.describe("Array of label names to apply to the pull request"),
|
||||||
},
|
},
|
||||||
async ({ title, body, head, base, assignee, assignees, milestone, labels }) => {
|
async ({
|
||||||
|
title,
|
||||||
|
body,
|
||||||
|
head,
|
||||||
|
base,
|
||||||
|
assignee,
|
||||||
|
assignees,
|
||||||
|
milestone,
|
||||||
|
labels,
|
||||||
|
}) => {
|
||||||
try {
|
try {
|
||||||
const pullData: any = { title, head, base };
|
const pullData: any = { title, head, base };
|
||||||
|
|
||||||
@@ -638,7 +741,8 @@ server.tool(
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
console.error(`[GITEA-MCP] Error creating pull request: ${errorMessage}`);
|
console.error(`[GITEA-MCP] Error creating pull request: ${errorMessage}`);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
|
|||||||
@@ -10,9 +10,15 @@ export async function prepareMcpConfig(
|
|||||||
console.log(`[MCP-INSTALL] Owner: ${owner}`);
|
console.log(`[MCP-INSTALL] Owner: ${owner}`);
|
||||||
console.log(`[MCP-INSTALL] Repo: ${repo}`);
|
console.log(`[MCP-INSTALL] Repo: ${repo}`);
|
||||||
console.log(`[MCP-INSTALL] Branch: ${branch}`);
|
console.log(`[MCP-INSTALL] Branch: ${branch}`);
|
||||||
console.log(`[MCP-INSTALL] GitHub token: ${githubToken ? '***' : 'undefined'}`);
|
console.log(
|
||||||
console.log(`[MCP-INSTALL] GITHUB_ACTION_PATH: ${process.env.GITHUB_ACTION_PATH}`);
|
`[MCP-INSTALL] GitHub token: ${githubToken ? "***" : "undefined"}`,
|
||||||
console.log(`[MCP-INSTALL] GITHUB_WORKSPACE: ${process.env.GITHUB_WORKSPACE}`);
|
);
|
||||||
|
console.log(
|
||||||
|
`[MCP-INSTALL] GITHUB_ACTION_PATH: ${process.env.GITHUB_ACTION_PATH}`,
|
||||||
|
);
|
||||||
|
console.log(
|
||||||
|
`[MCP-INSTALL] GITHUB_WORKSPACE: ${process.env.GITHUB_WORKSPACE}`,
|
||||||
|
);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const mcpConfig = {
|
const mcpConfig = {
|
||||||
@@ -29,7 +35,8 @@ export async function prepareMcpConfig(
|
|||||||
REPO_NAME: repo,
|
REPO_NAME: repo,
|
||||||
BRANCH_NAME: branch,
|
BRANCH_NAME: branch,
|
||||||
REPO_DIR: process.env.GITHUB_WORKSPACE || process.cwd(),
|
REPO_DIR: process.env.GITHUB_WORKSPACE || process.cwd(),
|
||||||
GITEA_API_URL: process.env.GITEA_API_URL || "https://api.github.com",
|
GITEA_API_URL:
|
||||||
|
process.env.GITEA_API_URL || "https://api.github.com",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
local_git_ops: {
|
local_git_ops: {
|
||||||
|
|||||||
@@ -21,7 +21,9 @@ console.log(`[LOCAL-GIT-MCP] REPO_NAME: ${REPO_NAME}`);
|
|||||||
console.log(`[LOCAL-GIT-MCP] BRANCH_NAME: ${BRANCH_NAME}`);
|
console.log(`[LOCAL-GIT-MCP] BRANCH_NAME: ${BRANCH_NAME}`);
|
||||||
console.log(`[LOCAL-GIT-MCP] REPO_DIR: ${REPO_DIR}`);
|
console.log(`[LOCAL-GIT-MCP] REPO_DIR: ${REPO_DIR}`);
|
||||||
console.log(`[LOCAL-GIT-MCP] GITEA_API_URL: ${GITEA_API_URL}`);
|
console.log(`[LOCAL-GIT-MCP] GITEA_API_URL: ${GITEA_API_URL}`);
|
||||||
console.log(`[LOCAL-GIT-MCP] GITHUB_TOKEN: ${GITHUB_TOKEN ? '***' : 'undefined'}`);
|
console.log(
|
||||||
|
`[LOCAL-GIT-MCP] GITHUB_TOKEN: ${GITHUB_TOKEN ? "***" : "undefined"}`,
|
||||||
|
);
|
||||||
|
|
||||||
if (!REPO_OWNER || !REPO_NAME || !BRANCH_NAME) {
|
if (!REPO_OWNER || !REPO_NAME || !BRANCH_NAME) {
|
||||||
console.error(
|
console.error(
|
||||||
@@ -63,7 +65,9 @@ function ensureGitUserConfigured(): void {
|
|||||||
runGitCommand("git config user.email");
|
runGitCommand("git config user.email");
|
||||||
console.log(`[LOCAL-GIT-MCP] Git user.email already configured`);
|
console.log(`[LOCAL-GIT-MCP] Git user.email already configured`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.log(`[LOCAL-GIT-MCP] Git user.email not configured, setting default`);
|
console.log(
|
||||||
|
`[LOCAL-GIT-MCP] Git user.email not configured, setting default`,
|
||||||
|
);
|
||||||
runGitCommand('git config user.email "claude@anthropic.com"');
|
runGitCommand('git config user.email "claude@anthropic.com"');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -72,7 +76,9 @@ function ensureGitUserConfigured(): void {
|
|||||||
runGitCommand("git config user.name");
|
runGitCommand("git config user.name");
|
||||||
console.log(`[LOCAL-GIT-MCP] Git user.name already configured`);
|
console.log(`[LOCAL-GIT-MCP] Git user.name already configured`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.log(`[LOCAL-GIT-MCP] Git user.name not configured, setting default`);
|
console.log(
|
||||||
|
`[LOCAL-GIT-MCP] Git user.name not configured, setting default`,
|
||||||
|
);
|
||||||
runGitCommand('git config user.name "Claude"');
|
runGitCommand('git config user.name "Claude"');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -134,7 +140,9 @@ server.tool(
|
|||||||
message: z.string().describe("Commit message"),
|
message: z.string().describe("Commit message"),
|
||||||
},
|
},
|
||||||
async ({ files, message }) => {
|
async ({ files, message }) => {
|
||||||
console.log(`[LOCAL-GIT-MCP] commit_files called with files: ${JSON.stringify(files)}, message: ${message}`);
|
console.log(
|
||||||
|
`[LOCAL-GIT-MCP] commit_files called with files: ${JSON.stringify(files)}, message: ${message}`,
|
||||||
|
);
|
||||||
try {
|
try {
|
||||||
// Ensure git user is configured before committing
|
// Ensure git user is configured before committing
|
||||||
ensureGitUserConfigured();
|
ensureGitUserConfigured();
|
||||||
@@ -151,7 +159,9 @@ server.tool(
|
|||||||
console.log(`[LOCAL-GIT-MCP] Committing with message: ${message}`);
|
console.log(`[LOCAL-GIT-MCP] Committing with message: ${message}`);
|
||||||
runGitCommand(`git commit -m "${message}"`);
|
runGitCommand(`git commit -m "${message}"`);
|
||||||
|
|
||||||
console.log(`[LOCAL-GIT-MCP] Successfully committed ${files.length} files`);
|
console.log(
|
||||||
|
`[LOCAL-GIT-MCP] Successfully committed ${files.length} files`,
|
||||||
|
);
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
{
|
{
|
||||||
@@ -353,7 +363,9 @@ server.tool("git_status", "Get the current git status", {}, async () => {
|
|||||||
const currentBranch = runGitCommand("git rev-parse --abbrev-ref HEAD");
|
const currentBranch = runGitCommand("git rev-parse --abbrev-ref HEAD");
|
||||||
|
|
||||||
console.log(`[LOCAL-GIT-MCP] Current branch: ${currentBranch}`);
|
console.log(`[LOCAL-GIT-MCP] Current branch: ${currentBranch}`);
|
||||||
console.log(`[LOCAL-GIT-MCP] Git status: ${status || "Working tree clean"}`);
|
console.log(
|
||||||
|
`[LOCAL-GIT-MCP] Git status: ${status || "Working tree clean"}`,
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
content: [
|
content: [
|
||||||
|
|||||||
Reference in New Issue
Block a user