提交 7693a276 编写于 作者: D DCloud_LXH

Merge branch 'alpha' into prod_alpha

...@@ -5,3 +5,4 @@ unpackage/ ...@@ -5,3 +5,4 @@ unpackage/
.hbuilderx/ .hbuilderx/
__image_snapshots__/ __image_snapshots__/
__snapshot__ __snapshot__
/code-review/node_modules
\ No newline at end of file
image: node:18
stages:
- review
variables:
GITLAB_TOKEN: $GITLAB_ACCESS_TOKEN
DEEPSEEK_API_KEY: $DEEPSEEK_API_KEY
NPM_CONFIG_CACHE: "$CI_PROJECT_DIR/.npm"
NPM_CONFIG_REGISTRY: https://registry.npmmirror.com
GIT_STRATEGY: fetch
cache:
key: ${CI_COMMIT_REF_SLUG}
paths:
- .npm/
- code-review/node_modules/
policy: pull-push
before_script:
- cd code-review
- npm ci --cache .npm --prefer-offline
code_review:
stage: review
script:
- node index.js
rules:
- if: $CI_PIPELINE_SOURCE == "merge_request_event"
- if: $CI_PIPELINE_SOURCE == "push" && $CI_COMMIT_BRANCH != "main" && $CI_COMMIT_BRANCH != "master"
\ No newline at end of file
<script lang="uts"> <script lang="uts">
import { state, setLifeCycleNum, checkSystemTheme, setNetless } from '@/store/index.uts' import { state, setLifeCycleNum, checkSystemTheme, setNetless } from '@/store/index.uts'
// #ifdef APP-ANDROID // #ifdef APP-ANDROID || APP-HARMONY
let firstBackTime = 0 let firstBackTime = 0
// #endif // #endif
export default { export default {
...@@ -13,7 +13,7 @@ ...@@ -13,7 +13,7 @@
str: 'default globalData obj str', str: 'default globalData obj str',
num: 0, num: 0,
bool: false, bool: false,
}, } as UTSJSONObject,
null: null as string | null, null: null as string | null,
arr: [] as number[], arr: [] as number[],
mySet: new Set<string>(), mySet: new Set<string>(),
...@@ -131,7 +131,7 @@ ...@@ -131,7 +131,7 @@
}) })
// #endif // #endif
}, },
// #ifdef APP-ANDROID // #ifdef APP-ANDROID || APP-HARMONY
onLastPageBackPress: function () { onLastPageBackPress: function () {
// 自动化测试 // 自动化测试
setLifeCycleNum(state.lifeCycleNum - 1000) setLifeCycleNum(state.lifeCycleNum - 1000)
......
const { loadProjectConfig, generateReviewPrompt, getAIReview, getChanges, addReviewComment, processReview } = require('../index');
const fs = require('fs');
const path = require('path');
// 输出环境变量
console.log('测试环境变量:');
console.log('GITLAB_TOKEN:', process.env.GITLAB_TOKEN ? '已设置' : '未设置');
console.log('GITLAB_URL:', process.env.GITLAB_URL);
console.log('DEEPSEEK_API_KEY:', process.env.DEEPSEEK_API_KEY ? '已设置' : '未设置');
console.log('CI_PROJECT_ID:', process.env.CI_PROJECT_ID);
console.log('CI_MERGE_REQUEST_IID:', process.env.CI_MERGE_REQUEST_IID);
console.log('CI_COMMIT_SHA:', process.env.CI_COMMIT_SHA);
console.log('CI_COMMIT_BRANCH:', process.env.CI_COMMIT_BRANCH);
console.log('CI_PIPELINE_SOURCE:', process.env.CI_PIPELINE_SOURCE);
console.log('CI_PROJECT_DIR:', process.env.CI_PROJECT_DIR);
console.log('----------------------------------------');
describe('Code Review System', () => {
// 测试配置文件加载
describe('loadProjectConfig', () => {
it('should load project configuration correctly', async () => {
const config = loadProjectConfig();
console.log('项目配置:', JSON.stringify(config, null, 2));
expect(config).toBeDefined();
expect(config.language).toBeDefined();
expect(config.reviewGuidelines).toBeDefined();
expect(config.reviewRules).toBeDefined();
expect(config.ignoreFiles).toBeDefined();
});
});
// 测试提示词生成
describe('generateReviewPrompt', () => {
it('should generate review prompt correctly', () => {
const config = loadProjectConfig();
const changes = [
{
file: 'test.js',
diff: 'console.log("test");'
}
];
const prompt = generateReviewPrompt(config, JSON.stringify(changes));
console.log('评审提示词:', prompt);
expect(prompt).toContain(config.language);
expect(prompt).toContain(config.reviewGuidelines);
expect(prompt).toContain('test.js');
});
});
// 测试获取代码变更
describe('getChanges', () => {
it('should get changes from merge request', async () => {
const changes = await getChanges(
process.env.CI_PROJECT_ID,
'merge_request',
process.env.CI_MERGE_REQUEST_IID
);
console.log('合并请求变更:', JSON.stringify(changes, null, 2));
expect(Array.isArray(changes)).toBe(true);
if (changes.length > 0) {
expect(changes[0]).toHaveProperty('file');
expect(changes[0]).toHaveProperty('diff');
}
}, 30000); // 设置超时时间为 30 秒
it('should get changes from push', async () => {
const changes = await getChanges(
process.env.CI_PROJECT_ID,
'push',
process.env.CI_COMMIT_SHA
);
console.log('推送变更:', JSON.stringify(changes, null, 2));
expect(Array.isArray(changes)).toBe(true);
if (changes.length > 0) {
expect(changes[0]).toHaveProperty('file');
expect(changes[0]).toHaveProperty('diff');
}
}, 30000);
});
// 测试 AI 评审
describe('getAIReview', () => {
it('should get AI review for changes', async () => {
console.log('开始 AI 评审测试...');
console.log('加载项目配置...');
const config = loadProjectConfig();
console.log('项目配置加载完成');
console.log('获取代码变更...');
const sourceType = process.env.CI_PIPELINE_SOURCE === 'merge_request_event' ? 'merge_request' : 'push';
const sourceId = sourceType === 'merge_request' ? process.env.CI_MERGE_REQUEST_IID : process.env.CI_COMMIT_SHA;
console.log(`变更来源: ${sourceType}, ID: ${sourceId}`);
const changes = await getChanges(
process.env.CI_PROJECT_ID,
sourceType,
sourceId
);
console.log(`获取到 ${changes.length} 个文件变更`);
changes.forEach(change => {
console.log(`文件: ${change.file}`);
console.log(`变更内容:\n${change.diff}`);
});
console.log('生成评审提示词...');
const prompt = generateReviewPrompt(config, JSON.stringify(changes));
console.log('评审提示词生成完成');
console.log('调用 AI 进行评审...');
const review = await getAIReview(prompt);
console.log('AI 评审完成');
console.log('评审结果:', review);
expect(typeof review).toBe('string');
expect(review.length).toBeGreaterThan(0);
console.log('AI 评审测试完成');
}, 120000); // 设置测试超时时间为 120 秒
});
// 测试添加评审评论
describe('addReviewComment', () => {
// it('should add review comment to merge request', async () => {
// const review = 'Test review comment';
// console.log('添加合并请求评论:', review);
// await expect(addReviewComment(
// process.env.CI_PROJECT_ID,
// 'merge_request',
// process.env.CI_MERGE_REQUEST_IID,
// review
// )).resolves.not.toThrow();
// }, 30000);
it('should add review comment to commit', async () => {
const review = 'Test review comment';
console.log('添加提交评论:', review);
await expect(addReviewComment(
process.env.CI_PROJECT_ID,
'push',
process.env.CI_COMMIT_SHA,
review
)).resolves.not.toThrow();
}, 30000);
});
// 测试完整评审流程
describe('processReview', () => {
// it('should process review for merge request', async () => {
// console.log('开始处理合并请求评审...');
// await expect(processReview(
// process.env.CI_PROJECT_ID,
// 'merge_request',
// process.env.CI_MERGE_REQUEST_IID
// )).resolves.not.toThrow();
// console.log('合并请求评审完成');
// }, 120000); // 设置超时时间为 120 秒
it('should process review for push', async () => {
console.log('开始处理推送评审...');
await expect(processReview(
process.env.CI_PROJECT_ID,
'push',
process.env.CI_COMMIT_SHA
)).resolves.not.toThrow();
console.log('推送评审完成');
}, 120000);
});
});
\ No newline at end of file
// 设置测试环境变量
process.env.NODE_ENV = 'test';
// GitLab 相关配置
process.env.GITLAB_TOKEN = process.env.TEST_GITLAB_TOKEN;
process.env.GITLAB_URL = process.env.TEST_GITLAB_URL || 'http://git.dcloud.io';
// DeepSeek API 配置
process.env.DEEPSEEK_API_KEY = process.env.TEST_DEEPSEEK_API_KEY;
// 测试项目配置
process.env.TEST_PROJECT_ID = '602';
process.env.TEST_MERGE_REQUEST_IID = '1';
process.env.TEST_COMMIT_SHA = 'f07f0dfe33e4099256bb23412d004502973c55c8';
process.env.TEST_BRANCH = 'dev';
process.env.TEST_PIPELINE_SOURCE = 'merge_request_event';
process.env.CI_PROJECT_DIR = process.cwd();
// 添加新的配置
process.env.CI_PROJECT_ID = process.env.TEST_PROJECT_ID;
process.env.CI_MERGE_REQUEST_IID = process.env.TEST_MERGE_REQUEST_IID;
process.env.CI_COMMIT_SHA = process.env.TEST_COMMIT_SHA;
process.env.CI_COMMIT_BRANCH = process.env.TEST_BRANCH || 'dev';
process.env.CI_PIPELINE_SOURCE = process.env.TEST_PIPELINE_SOURCE || 'push';
process.env.CI_PROJECT_DIR = process.cwd() + '/../';
\ No newline at end of file
# uni-app x 示例项目代码评审配置文件
# 项目基本信息
project:
provider: "ablai" # AI 服务商,可选值:ablai(阿波罗), bailian(阿里百炼)。如果要使用国外的大模型,请选择 ablai;如果使用国内的大模型,请选择 bailian。
# 阿波罗支持的模型列表:https://api.ablai.top/models
# 阿里百炼支持的模型列表:https://bailian.console.aliyun.com/?tab=model#/model-market
aiModel: "gemini-2.5-pro-preview-05-06" # AI 模型,可选值:qwen-turbo-2025-04-28, deepseek-v3, gemini-2.5-pro-preview-05-06
maxTokens: 5000 # AI 模型最大输出 token 数。不同的模型支持的 token 数不同,请根据实际情况调整。输出 token 数越多,评审结果越准确,但成本也越高。
reviewGuidelines: |
**请扮演一位经验丰富的 uni-app x 高级开发者,专注于对 Git Commit 进行代码评审。**
我将提供本次提交的以下信息:
1. 提交日志(Commit Message)
2. 代码变更内容(Diff)
3. 变更涉及到的代码上下文(包含注释)
我的项目使用 **uni-app x 框架** 实现示例功能,使用 **jest 框架** 进行断言和截图测试。
---
**请基于这些信息,针对 本次提交引入的变更 进行代码评审。你的评审应聚焦于:**
### 1. 变更目的与实际代码一致性
* 代码变更是否准确、完整地实现了提交日志中描述的目的或修复的问题?
* 提交日志是否清晰地解释了本次变更的意图?
### 2. 代码质量 (针对变更部分)
* **可读性与清晰度:**
* 新添加或修改的代码是否容易理解?
* 变量、函数、类、方法的命名是否符合规范且表达清晰?
* 代码结构是否合理?
* **潜在问题:**
* 变更是否可能引入新的 bug 或意外行为?
* 是否可能引入性能瓶颈(例如,在循环中执行昂贵的数据库操作,或者未优化 Eloquent 查询)?
* 是否存在潜在的资源泄漏或错误处理不当?
* **代码规范与最佳实践:**
* 新代码是否遵循 UTS 编码标准和项目内部的编码规范?
* 是否考虑到多端适配性(如 iOS、Android、H5、小程序等)?
* 是否遵循了 uni-app x 的最佳实践(如组件化、状态管理等)?
* 是否遵循了 jest 的最佳实践?
* **安全性:**
* 变更是否引入了新的安全风险?(特别关注用户输入处理、数据库交互、跨站脚本 XSS、跨站请求伪造 CSRF 等)
* 对于敏感操作,是否确保了适当的授权和认证?
* 涉及到手机号验证、邮箱验证、实人认证、三要素认证等付费验证时,需要增加合理的防刷机制。
### 3. 代码注释
* 新添加的注释是否清晰、准确、简洁,能够有效解释代码意图或复杂部分?
* 变更是否导致现有注释过时、不准确或产生误导?
* 是否有需要添加但缺失的关键注释(例如,对公共方法或复杂逻辑的解释)?
### 4. 整体变更结构
* 这次提交的变更范围是否合理(即,只包含了与提交目的相关的修改,没有混入其他不相关的改动)?
* 如果变更跨越多个文件,它们之间的逻辑联系是否清晰?
---
**请以清晰的格式(如列表或分节)输出评审结果,并:**
* **引用** 提交日志来作为评审的起点。
* **具体指出** 发现问题的代码位置(尽可能引用文件路径和相关的 diff 行号)。
* **详细描述** 发现的问题是什么。
* **提供具体、可操作的改进建议**。
* **标注问题的建议严重程度**(例如:`严重`, `主要`, `次要`, `建议`)。
* 如果基于本次变更看,代码质量良好且没有发现明显问题,也请简要说明。
* 将总结内容放在最前面输出,总结内容需要包含"总体评分(优、良、中、差)"、"推荐级别(可上线、建议修改、必须修改)"、"总体评价",方便评审人员快速评估。
* 回复时使用中文。
---
**以下是本次 Git Commit 的相关信息:**
# 忽略文件
ignore:
- "public/build/**"
- "node_modules/**"
- "*.lock"
- "code-review/**"
- ".env"
- ".env.*"
- "package-lock.json"
- ".gitlab-ci.yml"
\ No newline at end of file
const { Gitlab } = require('@gitbeaker/rest');
const axios = require('axios');
const yaml = require('yaml');
const fs = require('fs');
const minimatch = require('minimatch');
require('dotenv').config();
// 阿里百炼 https://bailian.console.aliyun.com/
const BAILIAN_API_KEY = process.env.BAILIAN_API_KEY;
const BAILIAN_API_URL = 'https://dashscope.aliyuncs.com/compatible-mode/v1/chat/completions';
// 阿波罗AI https://api.ablai.top/personal
const ABLAI_API_KEY = process.env.ABLAI_API_KEY;
const ABLAI_API_URL = 'https://api.ablai.top/v1/chat/completions';
const GITLAB_TOKEN = process.env.GITLAB_TOKEN;
const GITLAB_URL = process.env.CI_SERVER_URL || 'http://git.dcloud.io';
const api = new Gitlab({
token: GITLAB_TOKEN,
host: GITLAB_URL
});
// AI 服务商配置
const AI_PROVIDERS = {
bailian: {
name: '阿里百炼',
apiKey: BAILIAN_API_KEY,
apiUrl: BAILIAN_API_URL,
envKey: 'BAILIAN_API_KEY'
},
ablai: {
name: '阿波罗',
apiKey: ABLAI_API_KEY,
apiUrl: ABLAI_API_URL,
envKey: 'ABLAI_API_KEY'
}
};
// 检查提交是否已经被评审过
async function isCommitReviewed(projectId, commitId) {
try {
const discussions = await api.CommitDiscussions.all(projectId, commitId);
return discussions.some(discussion =>
discussion.notes.some(note =>
note.body.includes('🤖 AI 代码评审结果')
)
);
} catch (error) {
console.error(`检查提交 ${commitId} 评审状态时出错:`, error);
return false;
}
}
// 加载项目配置
function loadProjectConfig() {
try {
// 在 GitLab CI 环境中,工作目录是 /builds/username/project-name/
const configPath = `${process.env.CI_PROJECT_DIR}/code-review/configs/code-review.yaml`;
const configContent = fs.readFileSync(configPath, 'utf8');
const config = yaml.parse(configContent);
if (!config || !config.project) {
throw new Error('配置文件格式错误');
}
return {
reviewGuidelines: config.project.reviewGuidelines || '',
ignoreFiles: config.ignore || [],
aiModel: config.project.aiModel || "qwen-turbo-2025-04-28",
provider: config.project.provider || 'ablai',
maxTokens: config.project.maxTokens || 5000
};
} catch (error) {
console.error('Error loading config:', error);
return null;
}
}
// 生成 AI 评审提示词
function generateReviewPrompt(projectConfig, changes, commitInfo = null) {
const { reviewGuidelines } = projectConfig;
// 格式化变更信息
const formattedChanges = changes.map(change => {
return `
#### 文件路径:${change.file}
##### 变更内容:
${change.diff}
${change.content ? `##### 文件完整内容:
${change.content}` : ''}
`;
}).join('\n');
// 添加 commit 信息
const commitInfoText = commitInfo ? `${commitInfo.message}` : '';
return `
${reviewGuidelines}
### 提交日志 (Commit Message):
${commitInfoText}
### 代码变更及上下文:
${formattedChanges}
`;
}
// 添加重试函数
async function retryWithDelay(fn, maxRetries = 5, delay = 3000) {
let lastError;
for (let i = 0; i < maxRetries; i++) {
try {
return await fn();
} catch (error) {
lastError = error;
if (error.response && error.response.status >= 500) {
console.log(`API 请求失败 (状态码: ${error.response.status}),${i + 1}/${maxRetries} 次重试...`);
if (i < maxRetries - 1) {
await new Promise(resolve => setTimeout(resolve, delay));
continue;
}
}
throw error;
}
}
throw lastError;
}
// 调用 AI API 进行评审
async function getAIReview(prompt, projectConfig) {
try {
console.log('调用 AI API...');
console.log(prompt);
const model = projectConfig.aiModel || "qwen-turbo-2025-04-28";
const provider = projectConfig.provider || 'ablai';
console.log('provider', provider);
// 获取服务商配置
const providerConfig = AI_PROVIDERS[provider];
if (!providerConfig) {
throw new Error(`不支持的服务商: ${provider}`);
}
if (!providerConfig.apiKey) {
throw new Error(`${providerConfig.name} API Key (${providerConfig.envKey}) 未设置`);
}
// 创建 axios 实例
const axiosInstance = axios.create({
proxy: false,
timeout: 600000 // 设置超时时间为 10 分钟
});
// 使用重试机制发送请求
const response = await retryWithDelay(async () => {
return await axiosInstance.post(providerConfig.apiUrl, {
model: model,
messages: [{ role: "user", content: prompt }],
temperature: 0.7,
max_tokens: projectConfig.maxTokens || 5000
}, {
headers: {
'Content-Type': 'application/json',
'Authorization': `Bearer ${providerConfig.apiKey}`
}
});
});
return response.data.choices[0].message.content;
} catch (error) {
console.error('Error calling AI API:', error);
if (error.code === 'ECONNABORTED') {
console.error('API 请求超时,请检查网络连接或增加超时时间');
}
throw error;
}
}
// 获取代码变更内容
async function getChanges(projectId, sourceType, sourceId) {
try {
let changes;
if (sourceType === 'merge_request') {
console.log(`获取合并请求 ${sourceId} 的代码变更...`);
changes = await api.MergeRequests.allDiffs(projectId, sourceId, {
accessRawDiffs: true
});
console.log(`成功获取合并请求 ${sourceId} 的代码变更,共 ${changes.length} 个文件`);
} else if (sourceType === 'push') {
console.log(`获取提交 ${sourceId} 的代码变更...`);
// 获取单个 commit 的变更
const diff = await api.Commits.showDiff(projectId, sourceId);
changes = diff.map(change => ({
new_path: change.new_path,
old_path: change.old_path,
diff: change.diff
}));
console.log(`成功获取提交 ${sourceId} 的代码变更,共 ${changes.length} 个文件`);
} else {
console.error(`不支持的类型: ${sourceType}`);
throw new Error(`不支持的类型: ${sourceType}`);
}
const projectConfig = loadProjectConfig();
const ignorePatterns = projectConfig.ignoreFiles || [];
// 获取变更文件的完整内容
const changesWithContent = await Promise.all(changes
.filter(change => {
// 检查文件是否在忽略列表中
return !ignorePatterns.some(pattern => {
// 使用 minimatch 进行 glob 模式匹配
const shouldIgnore =
(change.new_path && minimatch(change.new_path, pattern)) ||
(change.old_path && minimatch(change.old_path, pattern));
if (shouldIgnore) {
console.log(`忽略文件: ${change.new_path || change.old_path} (匹配模式: ${pattern})`);
}
return shouldIgnore;
});
})
.map(async change => {
const filePath = change.new_path || change.old_path;
try {
console.log(`正在获取文件 ${filePath} 的完整内容...`);
// 获取文件的完整内容
const fileContent = await api.RepositoryFiles.show(projectId, filePath, sourceId);
// 对 base64 编码的内容进行解码
const decodedContent = Buffer.from(fileContent.content, 'base64').toString('utf-8');
console.log(`成功获取文件 ${filePath} 的完整内容`);
return {
file: filePath,
diff: change.diff,
content: decodedContent
};
} catch (error) {
console.error(`无法获取文件 ${filePath} 的完整内容:`, error);
return {
file: filePath,
diff: change.diff
};
}
}));
console.log(`成功处理所有文件变更,共 ${changesWithContent.length} 个文件`);
return changesWithContent;
} catch (error) {
console.error('获取代码变更失败:', error);
throw error;
}
}
// 添加评审评论
async function addReviewComment(projectId, sourceType, sourceId, review) {
try {
console.log(`添加评审评论 - 项目ID: ${projectId}, 来源类型: ${sourceType}, 来源ID: ${sourceId}`);
if (!projectId) {
throw new Error('项目ID不能为空');
}
if (!sourceId) {
throw new Error('来源ID不能为空');
}
if (!review) {
throw new Error('评审内容不能为空');
}
const note = `🤖 AI 代码评审结果:\n\n${review}`;
if (sourceType === 'merge_request') {
console.log('正在为合并请求添加评论...');
await api.MergeRequestNotes.create(projectId, sourceId, note);
console.log('合并请求评论添加成功');
} else if (sourceType === 'push') {
console.log('正在为提交添加评论...');
await api.CommitDiscussions.create(projectId, sourceId, note);
console.log('提交评论添加成功');
} else {
throw new Error(`不支持的来源类型: ${sourceType}`);
}
} catch (error) {
console.error('添加评审评论失败:', {
error: error.message,
projectId,
sourceType,
sourceId,
reviewLength: review?.length
});
if (error.cause?.description) {
console.error('错误详情:', error.cause.description);
}
throw error;
}
}
// 主处理函数
async function processReview(projectId, sourceType, sourceId) {
try {
const projectConfig = loadProjectConfig();
if (!projectConfig) {
console.error('Project configuration not found');
process.exit(1);
}
if (sourceType === 'push') {
console.log(process.env.CI_COMMIT_BEFORE_SHA);
console.log(process.env.CI_COMMIT_SHA);
console.log(process.env.CI_COMMIT_BRANCH);
// 获取本次 push 的所有 commit
let commits;
if (process.env.CI_COMMIT_BEFORE_SHA && process.env.CI_COMMIT_SHA) {
commits = await api.Repositories.compare(projectId, process.env.CI_COMMIT_BEFORE_SHA, process.env.CI_COMMIT_SHA);
commits = commits.commits || [];
console.log('获取本次提交的信息:', commits);
} else {
commits = await api.Commits.all(projectId, {
ref_name: process.env.CI_COMMIT_BRANCH,
per_page: 1
});
console.log('获取首次提交的信息:', commits);
}
// 过滤掉合并分支的提交
commits = commits.filter(commit => !commit.message.startsWith('Merge branch'));
console.log(`获取到 ${commits.length} 个提交需要评审(已过滤合并分支的提交)`);
// 对每个 commit 进行评审
for (const commit of commits) {
console.log(`开始评审提交: ${commit.id}`);
console.log(`提交信息: ${commit.message}`);
// 检查提交是否已经被评审过
const isReviewed = await isCommitReviewed(projectId, commit.id);
if (isReviewed) {
console.log(`提交 ${commit.id} 已经评审过,跳过评审`);
continue;
}
// 获取该 commit 的变更
const changes = await getChanges(projectId, sourceType, commit.id);
if (changes.length === 0) {
console.log(`提交 ${commit.id} 没有代码变更,跳过评审`);
continue;
}
console.log(`提交 ${commit.id} 包含 ${changes.length} 个文件变更`);
// 生成评审提示词
const prompt = generateReviewPrompt(projectConfig, changes, {
author_name: commit.author_name,
created_at: commit.created_at,
message: commit.message,
ref_name: process.env.CI_COMMIT_BRANCH
});
// 获取 AI 评审结果
const review = await getAIReview(prompt, projectConfig);
// 添加评审评论到 commit
await addReviewComment(projectId, sourceType, commit.id, review);
console.log(`提交 ${commit.id} 评审完成`);
}
} else if (sourceType === 'merge_request') {
const changes = await getChanges(projectId, sourceType, sourceId);
if (changes.length === 0) {
console.log('No changes to review');
return;
}
// 获取合并请求信息
const mrInfo = await api.MergeRequests.show(projectId, sourceId);
const prompt = generateReviewPrompt(projectConfig, changes, {
author_name: mrInfo.author.name,
created_at: mrInfo.created_at,
message: mrInfo.description,
ref_name: mrInfo.source_branch
});
const review = await getAIReview(prompt, projectConfig);
await addReviewComment(projectId, sourceType, sourceId, review);
}
console.log('Review completed successfully');
} catch (error) {
console.error('Error processing review:', error);
if (error.cause?.description?.includes('401 Unauthorized')) {
console.error('GitLab API authentication failed. Please check your GITLAB_TOKEN.');
}
process.exit(1);
}
}
// 导出需要测试的函数
module.exports = {
loadProjectConfig,
generateReviewPrompt,
getAIReview,
getChanges,
addReviewComment,
processReview
};
// 只在直接运行 index.js 时执行
if (require.main === module) {
const projectId = process.env.CI_PROJECT_ID;
const sourceType = process.env.CI_PIPELINE_SOURCE === 'merge_request_event' ? 'merge_request' : 'push';
const sourceId = sourceType === 'merge_request' ? process.env.CI_MERGE_REQUEST_IID : process.env.CI_COMMIT_SHA;
if (!GITLAB_TOKEN) {
console.error('GITLAB_TOKEN is not set');
process.exit(1);
}
if (!projectId) {
console.error('CI_PROJECT_ID is not set');
process.exit(1);
}
if (!sourceId) {
console.error('Source ID is not set');
process.exit(1);
}
processReview(projectId, sourceType, sourceId);
}
\ No newline at end of file
此差异已折叠。
{
"name": "uni-ai-code-review",
"version": "1.0.0",
"description": "AI 代码评审系统",
"main": "index.js",
"scripts": {
"test": "jest",
"start": "node index.js"
},
"dependencies": {
"@gitbeaker/rest": "^42.5.0",
"axios": "^1.6.0",
"dotenv": "^16.3.1",
"minimatch": "^5.1.6",
"yaml": "^2.3.4"
},
"devDependencies": {
"jest": "^29.7.0"
},
"jest": {
"testEnvironment": "node",
"coverageDirectory": "./coverage",
"collectCoverageFrom": [
"**/*.js",
"!**/*.test.js",
"!__tests__/setup.js"
],
"setupFiles": [
"./__tests__/setup.js"
],
"testMatch": [
"**/__tests__/**/*.test.js"
]
}
}
此差异已折叠。
...@@ -99,7 +99,7 @@ ...@@ -99,7 +99,7 @@
flex-direction: row; flex-direction: row;
align-items: center; align-items: center;
padding: 12px 18px; padding: 12px 18px;
background-color: #fff; background-color: #ffffff;
} }
.uni-collapse-item .down_arrow { .uni-collapse-item .down_arrow {
......
...@@ -32,6 +32,10 @@ module.exports = { ...@@ -32,6 +32,10 @@ module.exports = {
"android": { "android": {
"id": "emulator-5554", "id": "emulator-5554",
"executablePath": "/Applications/HBuilderX-Dev.app/Contents/HBuilderX/plugins/uniappx-launcher/base/android_base.apk" "executablePath": "/Applications/HBuilderX-Dev.app/Contents/HBuilderX/plugins/uniappx-launcher/base/android_base.apk"
},
"ios": {
"id": "CA80343E-7D4C-401D-81BA-B5B7D446C11D",
"executablePath": "/Applications/HBuilderX-Dev.app/Contents/HBuilderX/plugins/uniappx-launcher/base/Pandora_simulator.app"
} }
} }
} }
......
此差异已折叠。
...@@ -19,3 +19,4 @@ ...@@ -19,3 +19,4 @@
<script type="module" src="/main"></script> <script type="module" src="/main"></script>
</body> </body>
</html> </html>
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册