2025-01-05 20:29:41 +00:00
name : AI Bot
on :
issues :
types : [ opened]
issue_comment :
types : [ created]
pull_request_review_comment :
types : [ created]
2025-01-05 20:35:55 +00:00
pull_request :
types : [ opened, edited, synchronize]
2025-01-05 20:29:41 +00:00
jobs :
respond-to-commands :
runs-on : ubuntu-latest
if : |
2025-01-05 20:35:55 +00:00
(github.actor == 'f') &&
((github.event_name == 'issues' && contains(github.event.issue.body, '/ai')) ||
2025-01-05 20:29:41 +00:00
(github.event_name == 'issue_comment' && contains(github.event.comment.body, '/ai')) ||
2025-01-05 20:35:55 +00:00
(github.event_name == 'pull_request_review_comment' && contains(github.event.comment.body, '/ai')) ||
(github.event_name == 'pull_request' && contains(github.event.pull_request.body, '/ai')))
2025-01-05 20:29:41 +00:00
permissions :
contents : write
pull-requests : write
issues : write
steps :
- uses : actions/checkout@v3
with :
fetch-depth : 0
2025-01-05 21:42:58 +00:00
token : ${{ secrets.PAT_TOKEN }}
2025-01-05 20:29:41 +00:00
- name : Setup Node.js
uses : actions/setup-node@v3
with :
node-version : '18'
- name : Install dependencies
run : npm install openai@^4.0.0 @octokit/rest@^19.0.0
- name : Process command
id : process
env :
2025-01-05 21:42:58 +00:00
GH_TOKEN : ${{ secrets.PAT_TOKEN }}
2025-01-05 20:29:41 +00:00
OPENAI_API_KEY : ${{ secrets.OPENAI_API_KEY }}
run : |
node << 'EOF'
const OpenAI = require('openai');
const { Octokit } = require('@octokit/rest');
async function main() {
const openai = new OpenAI({
apiKey : process.env.OPENAI_API_KEY
});
const octokit = new Octokit({
2025-01-05 21:42:58 +00:00
auth : process.env.GH_TOKEN
2025-01-05 20:29:41 +00:00
});
const eventName = process.env.GITHUB_EVENT_NAME;
const eventPath = process.env.GITHUB_EVENT_PATH;
const event = require(eventPath);
2025-01-05 20:35:55 +00:00
// Double check user authorization
const actor = event.sender?.login || event.pull_request?.user?.login || event.issue?.user?.login;
if (actor !== 'f') {
console.log('Unauthorized user attempted to use the bot:', actor);
return;
}
2025-01-05 20:29:41 +00:00
// Get command and context
let command = '';
let issueNumber = null;
2025-01-05 20:35:55 +00:00
let isPullRequest = false;
2025-01-05 20:29:41 +00:00
if (eventName === 'issues') {
command = event.issue.body;
issueNumber = event.issue.number;
} else if (eventName === 'issue_comment') {
command = event.comment.body;
issueNumber = event.issue.number;
2025-01-05 20:35:55 +00:00
isPullRequest = !!event.issue.pull_request;
2025-01-05 20:29:41 +00:00
} else if (eventName === 'pull_request_review_comment') {
command = event.comment.body;
issueNumber = event.pull_request.number;
2025-01-05 20:35:55 +00:00
isPullRequest = true;
} else if (eventName === 'pull_request') {
command = event.pull_request.body;
issueNumber = event.pull_request.number;
isPullRequest = true;
2025-01-05 20:29:41 +00:00
}
if (!command.startsWith('/ai')) {
return;
}
// Extract the actual command after /ai
const aiCommand = command.substring(3).trim();
2025-01-05 20:35:55 +00:00
2025-01-05 20:46:18 +00:00
// Handle resolve conflicts command
if (aiCommand === 'resolve' || aiCommand === 'fix conflicts') {
if (!isPullRequest) {
2025-01-05 21:24:07 +00:00
console.log('Command rejected : Not a pull request');
2025-01-05 20:46:18 +00:00
await octokit.issues.createComment({
owner : event.repository.owner.login,
repo : event.repository.name,
issue_number : issueNumber,
body : '❌ The resolve command can only be used on pull requests.'
});
return;
}
2025-01-05 21:02:16 +00:00
try {
2025-01-05 21:24:07 +00:00
console.log('Starting resolve command execution...');
2025-01-05 21:02:16 +00:00
// Get PR details
2025-01-05 21:24:07 +00:00
console.log('Fetching PR details...');
2025-01-05 21:02:16 +00:00
const { data: pr } = await octokit.pulls.get({
owner : event.repository.owner.login,
repo : event.repository.name,
pull_number : issueNumber
});
2025-01-05 21:54:02 +00:00
console.log(`Original PR found : #${issueNumber} from ${pr.user.login}`);
2025-01-05 21:02:16 +00:00
// Get the PR diff to extract the new prompt
2025-01-05 21:24:07 +00:00
console.log('Fetching PR file changes...');
2025-01-05 21:02:16 +00:00
const { data: files } = await octokit.pulls.listFiles({
owner : event.repository.owner.login,
repo : event.repository.name,
pull_number : issueNumber
});
2025-01-05 21:24:07 +00:00
console.log(`Found ${files.length} changed files`);
2025-01-05 21:02:16 +00:00
// Extract prompt from changes
2025-01-05 21:24:07 +00:00
console.log('Analyzing changes to extract prompt information...');
2025-01-05 23:03:41 +00:00
const prompts = new Map(); // Use Map to deduplicate prompts by title
2025-01-05 21:02:16 +00:00
2025-01-05 23:17:35 +00:00
// Helper function to normalize prompt titles
const normalizeTitle = (title) => {
title = title.trim();
// Remove "Act as" or "Act as a" or "Act as an" from start if present
title = title.replace(/^Act as (?:a |an )?/i, '');
// Add "Act as" prefix
return `Act as ${title}`;
};
2025-01-05 23:20:13 +00:00
// First try to find prompts in README
let foundInReadme = false;
2025-01-05 21:02:16 +00:00
for (const file of files) {
2025-01-05 21:24:07 +00:00
console.log(`Processing file : ${file.filename}`);
2025-01-05 21:02:16 +00:00
if (file.filename === 'README.md') {
const patch = file.patch || '';
const addedLines = patch.split('\n')
.filter(line => line.startsWith('+'))
.map(line => line.substring(1))
.join('\n');
2025-01-05 22:41:01 +00:00
console.log('Attempting to extract prompts from README changes...');
2025-01-05 23:17:35 +00:00
const promptMatches = [...addedLines.matchAll(/## (?:Act as (?:a |an )?)?([^\n]+)\n(?:Contributed by:[^\n]*\n)?(?:> )?([^#]+?)(?=\n##|\n\n##|$)/ig)];
2025-01-05 22:41:01 +00:00
for (const match of promptMatches) {
2025-01-05 23:17:35 +00:00
const actName = normalizeTitle(match[1]);
2025-01-05 22:41:01 +00:00
const promptText = match[2].trim();
2025-01-05 21:02:16 +00:00
const contributorLine = addedLines.match(/Contributed by : \[@([^\]]+)\]\(https:\/\/github\.com\/([^\)]+)\)/);
2025-01-05 22:41:01 +00:00
const contributorInfo = contributorLine
? `Contributed by : [ @${contributorLine[1]}](https://github.com/${contributorLine[2]})`
: `Contributed by : [ @${pr.user.login}](https://github.com/${pr.user.login})`;
2025-01-05 23:17:35 +00:00
prompts.set(actName.toLowerCase(), { actName, promptText, contributorInfo });
2025-01-05 23:03:41 +00:00
console.log(`Found prompt in README : "${actName}" `);
2025-01-05 23:20:13 +00:00
foundInReadme = true;
2025-01-05 23:03:41 +00:00
}
2025-01-05 23:20:13 +00:00
}
}
// Only look in CSV if we didn't find anything in README
if (!foundInReadme) {
console.log('No prompts found in README, checking CSV...');
for (const file of files) {
if (file.filename === 'prompts.csv') {
const patch = file.patch || '';
const addedLines = patch.split('\n')
.filter(line => line.startsWith('+'))
.map(line => line.substring(1))
.filter(line => line.trim()); // Remove empty lines
console.log('Attempting to extract prompts from CSV changes...');
for (const line of addedLines) {
// Parse CSV line considering escaped quotes
const matches = [...line.matchAll(/"([^"]*(?:""[^"]*)*)"/g)];
if (matches.length >= 2) {
const actName = normalizeTitle(matches[0][1].replace(/""/g, '"').trim());
const promptText = matches[1][1].replace(/""/g, '"').trim();
2025-01-05 23:03:41 +00:00
const contributorInfo = `Contributed by : [ @${pr.user.login}](https://github.com/${pr.user.login})`;
2025-01-05 23:17:35 +00:00
prompts.set(actName.toLowerCase(), { actName, promptText, contributorInfo });
2025-01-05 23:03:41 +00:00
console.log(`Found prompt in CSV : "${actName}" `);
}
}
2025-01-05 20:53:06 +00:00
}
2025-01-05 20:46:18 +00:00
}
}
2025-01-05 21:02:16 +00:00
2025-01-05 23:03:41 +00:00
if (prompts.size === 0) {
2025-01-05 21:24:07 +00:00
console.log('Failed to extract prompt information');
2025-01-05 21:02:16 +00:00
await octokit.issues.createComment({
owner : event.repository.owner.login,
repo : event.repository.name,
issue_number : issueNumber,
body : '❌ Could not extract prompt information from changes'
});
return;
2025-01-05 20:46:18 +00:00
}
2025-01-05 21:54:02 +00:00
// Get content from main branch
2025-01-05 21:24:07 +00:00
console.log('Fetching current content from main branch...');
2025-01-05 21:02:16 +00:00
const { data: readmeFile } = await octokit.repos.getContent({
2025-01-05 20:46:18 +00:00
owner : event.repository.owner.login,
repo : event.repository.name,
2025-01-05 21:02:16 +00:00
path : 'README.md' ,
2025-01-05 21:15:22 +00:00
ref : 'main'
2025-01-05 20:46:18 +00:00
});
2025-01-05 21:02:16 +00:00
const { data: csvFile } = await octokit.repos.getContent({
2025-01-05 20:46:18 +00:00
owner : event.repository.owner.login,
repo : event.repository.name,
2025-01-05 21:02:16 +00:00
path : 'prompts.csv' ,
2025-01-05 21:15:22 +00:00
ref : 'main'
2025-01-05 20:46:18 +00:00
});
2025-01-05 21:54:02 +00:00
// Prepare new content
2025-01-05 21:24:07 +00:00
console.log('Preparing content updates...');
2025-01-05 21:15:22 +00:00
let readmeContent = Buffer.from(readmeFile.content, 'base64').toString('utf-8');
2025-01-05 22:41:01 +00:00
let csvContent = Buffer.from(csvFile.content, 'base64').toString('utf-8');
if (!csvContent.endsWith('\n')) csvContent += '\n';
2025-01-05 23:03:41 +00:00
// Convert Map to array for processing
const promptsArray = Array.from(prompts.values());
2025-01-05 22:41:01 +00:00
// Process each prompt
2025-01-05 23:03:41 +00:00
for (const { actName, promptText, contributorInfo } of promptsArray) {
2025-01-05 22:41:01 +00:00
// Remove markdown quote character and trim whitespace
const cleanPrompt = promptText.replace(/^>\s*/gm, '').trim();
// For README : Add quote to each line
const readmePrompt = cleanPrompt.split('\n')
.map(line => `> ${line.trim()}`)
.join('\n');
const newSection = `## ${actName}\n${contributorInfo}\n\n${readmePrompt}\n\n`;
// For CSV : Convert to single paragraph
const csvPrompt = cleanPrompt.replace(/\n+/g, ' ').trim();
// Insert the new section before Contributors in README
const contributorsIndex = readmeContent.indexOf('## Contributors');
if (contributorsIndex === -1) {
2025-01-05 22:38:05 +00:00
console.log('Contributors section not found, appending to end');
readmeContent += newSection;
2025-01-05 22:41:01 +00:00
} else {
2025-01-05 22:38:05 +00:00
console.log('Inserting before Contributors section');
readmeContent = readmeContent.slice(0, contributorsIndex) + newSection + readmeContent.slice(contributorsIndex);
2025-01-05 22:41:01 +00:00
}
2025-01-05 21:15:22 +00:00
2025-01-05 22:41:01 +00:00
// Add to CSV content
csvContent += `"${actName.replace(/"/g, '""')}","${csvPrompt.replace(/"/g, '""')}"\n`;
}
2025-01-05 21:20:17 +00:00
2025-01-05 21:54:02 +00:00
// Create new branch
2025-01-05 23:03:41 +00:00
const branchName = `prompt/${promptsArray.map(p => p.actName.toLowerCase().replace(/[^a-z0-9]+/g, '-')).join('-')}`;
2025-01-05 21:54:02 +00:00
console.log(`Creating new branch : ${branchName}`);
2025-01-05 22:16:09 +00:00
// Check if branch exists and delete it
try {
console.log('Checking if branch already exists...');
const { data: existingRef } = await octokit.git.getRef({
owner : event.repository.owner.login,
repo : event.repository.name,
ref : `heads/${branchName}`
});
if (existingRef) {
2025-01-05 22:21:32 +00:00
// Check for existing PRs from this branch
console.log('Checking for existing PRs from this branch...');
const { data: existingPRs } = await octokit.pulls.list({
owner : event.repository.owner.login,
repo : event.repository.name,
head : `${event.repository.owner.login}:${branchName}`,
state : 'open'
});
// Close any existing PRs
for (const pr of existingPRs) {
console.log(`Closing existing PR #${pr.number}...`);
await octokit.pulls.update({
owner : event.repository.owner.login,
repo : event.repository.name,
pull_number : pr.number,
state : 'closed'
});
}
2025-01-05 22:16:09 +00:00
console.log('Branch exists, deleting it...');
await octokit.git.deleteRef({
owner : event.repository.owner.login,
repo : event.repository.name,
ref : `heads/${branchName}`
});
console.log('Existing branch deleted');
}
} catch (error) {
// 404 means branch doesn't exist, which is fine
if (error.status !== 404) {
throw error;
}
console.log('Branch does not exist, proceeding with creation');
}
2025-01-05 21:54:02 +00:00
// Get main branch ref
const { data: mainRef } = await octokit.git.getRef({
owner : event.repository.owner.login,
repo : event.repository.name,
ref : 'heads/main'
2025-01-05 21:27:40 +00:00
});
2025-01-05 21:54:02 +00:00
// Create new branch
await octokit.git.createRef({
owner : event.repository.owner.login,
repo : event.repository.name,
ref : `refs/heads/${branchName}`,
sha : mainRef.object.sha
2025-01-05 21:15:22 +00:00
});
2025-01-05 21:54:02 +00:00
2025-01-05 21:59:32 +00:00
// Get current files from the new branch
console.log('Getting current file SHAs...');
const { data: currentReadme } = await octokit.repos.getContent({
owner : event.repository.owner.login,
repo : event.repository.name,
path : 'README.md' ,
ref : branchName
});
const { data: currentCsv } = await octokit.repos.getContent({
owner : event.repository.owner.login,
repo : event.repository.name,
path : 'prompts.csv' ,
ref : branchName
});
2025-01-05 21:54:02 +00:00
// Update files with correct author
2025-01-05 21:59:32 +00:00
console.log('Updating README.md...');
2025-01-05 21:54:02 +00:00
await octokit.repos.createOrUpdateFileContents({
owner : event.repository.owner.login,
repo : event.repository.name,
path : 'README.md' ,
2025-01-05 23:03:41 +00:00
message : promptsArray.length === 1
? `feat : Add "${promptsArray[0].actName}" to README`
2025-01-05 22:54:52 +00:00
: `feat : Add multiple prompts to README`,
2025-01-05 21:54:02 +00:00
content : Buffer.from(readmeContent).toString('base64'),
branch : branchName,
2025-01-05 21:59:32 +00:00
sha : currentReadme.sha,
2025-01-05 21:54:02 +00:00
committer : {
name : pr.user.login,
email : `${pr.user.login}@users.noreply.github.com`
},
author : {
name : pr.user.login,
email : `${pr.user.login}@users.noreply.github.com`
}
2025-01-05 21:02:16 +00:00
});
2025-01-05 21:54:02 +00:00
2025-01-05 21:59:32 +00:00
console.log('Updating prompts.csv...');
2025-01-05 21:54:02 +00:00
await octokit.repos.createOrUpdateFileContents({
owner : event.repository.owner.login,
repo : event.repository.name,
path : 'prompts.csv' ,
2025-01-05 23:03:41 +00:00
message : promptsArray.length === 1
? `feat : Add "${promptsArray[0].actName}" to prompts.csv`
2025-01-05 22:54:52 +00:00
: `feat : Add multiple prompts to prompts.csv`,
2025-01-05 21:54:02 +00:00
content : Buffer.from(csvContent).toString('base64'),
branch : branchName,
2025-01-05 21:59:32 +00:00
sha : currentCsv.sha,
2025-01-05 21:54:02 +00:00
committer : {
name : pr.user.login,
email : `${pr.user.login}@users.noreply.github.com`
},
author : {
name : pr.user.login,
email : `${pr.user.login}@users.noreply.github.com`
}
2025-01-05 21:02:16 +00:00
});
2025-01-05 21:54:02 +00:00
// Create new PR
2025-01-05 23:03:41 +00:00
const prTitle = promptsArray.length === 1
? `feat : Add "${promptsArray[0].actName}"`
: `feat : Add multiple prompts (${promptsArray.map(p => `"${p.actName}"`).join(', ')})`;
2025-01-05 22:54:52 +00:00
2025-01-05 23:03:41 +00:00
const prBody = promptsArray.length === 1
? `This PR supersedes #${issueNumber} with proper formatting. Original PR by @${pr.user.login}. Added "${promptsArray[0].actName}" to README.md and prompts.csv, preserving original attribution.`
: `This PR supersedes #${issueNumber} with proper formatting. Original PR by @${pr.user.login}.\n\nAdded the following prompts:\n${promptsArray.map(p => `- "${p.actName}"`).join('\n')}\n\nAll prompts have been added to README.md and prompts.csv, preserving original attribution.`;
2025-01-05 22:54:52 +00:00
2025-01-05 21:54:02 +00:00
const { data: newPr } = await octokit.pulls.create({
owner : event.repository.owner.login,
repo : event.repository.name,
2025-01-05 22:54:52 +00:00
title : prTitle,
2025-01-05 21:54:02 +00:00
head : branchName,
base : 'main' ,
2025-01-05 22:54:52 +00:00
body : prBody
2025-01-05 21:57:48 +00:00
});
2025-01-05 21:54:02 +00:00
// Comment on original PR
2025-01-05 21:02:16 +00:00
await octokit.issues.createComment({
owner : event.repository.owner.login,
repo : event.repository.name,
issue_number : issueNumber,
2025-01-05 21:54:02 +00:00
body : `I've created a new PR #${newPr.number} with your contribution properly formatted. This PR will be closed in favor of the new one.`
});
// Close original PR
await octokit.pulls.update({
owner : event.repository.owner.login,
repo : event.repository.name,
pull_number : issueNumber,
state : 'closed'
2025-01-05 21:02:16 +00:00
});
2025-01-05 21:54:02 +00:00
console.log(`Created new PR #${newPr.number} and closed original PR #${issueNumber}`);
2025-01-05 20:46:18 +00:00
} catch (error) {
2025-01-05 21:24:07 +00:00
console.error('Error details:', error);
2025-01-05 20:46:18 +00:00
await octokit.issues.createComment({
owner : event.repository.owner.login,
repo : event.repository.name,
issue_number : issueNumber,
2025-01-05 21:54:02 +00:00
body : `❌ Error while trying to create new PR:\n\`\`\`\n${error.message}\n\`\`\``
2025-01-05 20:46:18 +00:00
});
}
return;
}
2025-01-05 20:35:55 +00:00
// Handle rename command specifically
if (aiCommand.startsWith('rename') || aiCommand === 'suggest title') {
if (!isPullRequest) {
await octokit.issues.createComment({
owner : event.repository.owner.login,
repo : event.repository.name,
issue_number : issueNumber,
body : '❌ The rename command can only be used on pull requests.'
});
return;
}
// Get PR details for context
const { data: pr } = await octokit.pulls.get({
owner : event.repository.owner.login,
repo : event.repository.name,
pull_number : issueNumber
});
2025-01-05 20:39:17 +00:00
// Get the list of files changed in the PR
const { data: files } = await octokit.pulls.listFiles({
2025-01-05 20:35:55 +00:00
owner : event.repository.owner.login,
repo : event.repository.name,
2025-01-05 20:39:17 +00:00
pull_number : issueNumber
2025-01-05 20:35:55 +00:00
});
2025-01-05 20:39:17 +00:00
// Process file changes
const fileChanges = await Promise.all(files.map(async file => {
if (file.status === 'removed') {
return `Deleted : ${file.filename}`;
}
// Get file content for added or modified files
if (file.status === 'added' || file.status === 'modified') {
const patch = file.patch || '';
return `${file.status === 'added' ? 'Added' : 'Modified'}: ${file.filename}\nChanges:\n${patch}`;
}
return `${file.status}: ${file.filename}`;
}));
2025-01-05 20:35:55 +00:00
const completion = await openai.chat.completions.create({
model : "gpt-3.5-turbo" ,
messages : [
{
role : "system" ,
2025-01-05 20:42:16 +00:00
content : "You are a helpful assistant that generates clear and concise pull request titles. Follow these rules:\n1. Use conventional commit style (feat:, fix:, docs:, etc.)\n2. Focus on WHAT changed, not HOW or WHERE\n3. Keep it short and meaningful\n4. Don't mention file names or technical implementation details\n5. Return ONLY the new title, nothing else\n\nGood examples:\n- feat: Add \"Act as a Career Coach\"\n- fix: Correct typo in Linux Terminal prompt\n- docs: Update installation instructions\n- refactor: Improve error handling"
2025-01-05 20:35:55 +00:00
},
{
role : "user" ,
2025-01-05 20:42:16 +00:00
content : `Based on these file changes, generate a concise PR title:\n\n${fileChanges.join('\n\n')}`
2025-01-05 20:35:55 +00:00
}
] ,
2025-01-05 20:42:16 +00:00
temperature : 0.5 ,
2025-01-05 20:35:55 +00:00
max_tokens : 60
});
const newTitle = completion.choices[0].message.content.trim();
// Update PR title
await octokit.pulls.update({
owner : event.repository.owner.login,
repo : event.repository.name,
pull_number : issueNumber,
title : newTitle
});
// Add comment about the rename
await octokit.issues.createComment({
owner : event.repository.owner.login,
repo : event.repository.name,
issue_number : issueNumber,
2025-01-05 20:39:17 +00:00
body: `✨ Updated PR title to : "${newTitle}" \n\nBased on the following changes:\n\`\`\`diff\n${fileChanges.join('\n')}\n\`\`\``
2025-01-05 20:35:55 +00:00
});
return;
}
2025-01-05 20:29:41 +00:00
2025-01-05 20:35:55 +00:00
// Handle other commands
2025-01-05 20:29:41 +00:00
const completion = await openai.chat.completions.create({
model : "gpt-3.5-turbo" ,
messages : [
{
role : "system" ,
content : "You are a helpful AI assistant that helps with GitHub repositories. You can suggest code changes, fix issues, and improve code quality."
},
{
role : "user" ,
content : aiCommand
}
] ,
temperature : 0.7 ,
max_tokens : 2000
});
const response = completion.choices[0].message.content;
// If response contains code changes, create a new branch and PR
if (response.includes('```')) {
const branchName = `ai-bot/fix-${issueNumber}`;
// Create new branch
const defaultBranch = event.repository.default_branch;
const ref = await octokit.git.getRef({
owner : event.repository.owner.login,
repo : event.repository.name,
ref : `heads/${defaultBranch}`
});
await octokit.git.createRef({
owner : event.repository.owner.login,
repo : event.repository.name,
ref : `refs/heads/${branchName}`,
sha : ref.data.object.sha
});
// Extract code changes and file paths from response
const codeBlocks = response.match(/```[\s\S]*?```/g);
for (const block of codeBlocks) {
const [_, filePath, ...codeLines] = block.split('\n');
const content = Buffer.from(codeLines.join('\n')).toString('base64');
await octokit.repos.createOrUpdateFileContents({
owner : event.repository.owner.login,
repo : event.repository.name,
path : filePath,
message: `AI Bot : Apply suggested changes for #${issueNumber}`,
content,
branch : branchName
});
}
// Create PR
await octokit.pulls.create({
owner : event.repository.owner.login,
repo : event.repository.name,
title: `AI Bot : Fix for #${issueNumber}`,
body : `This PR was automatically generated in response to #${issueNumber}\n\nChanges proposed:\n${response}`,
head : branchName,
base : defaultBranch
});
}
// Add comment with response
await octokit.issues.createComment({
owner : event.repository.owner.login,
repo : event.repository.name,
issue_number : issueNumber,
body : `AI Bot Response:\n\n${response}`
});
}
main().catch(error => {
console.error('Error:', error);
process.exit(1);
});
EOF
- name : Handle errors
if : failure()
uses : actions/github-script@v6
with :
script : |
const issueNumber = context.issue.number || context.payload.pull_request.number;
await github.rest.issues.createComment({
owner : context.repo.owner,
repo : context.repo.repo,
issue_number : issueNumber,
body : '❌ Sorry, there was an error processing your command. Please try again or contact the repository maintainers.'
});