generated from ztimson/template
Compare commits
2 Commits
45d43c0977
...
c27802ca44
| Author | SHA1 | Date | |
|---|---|---|---|
| c27802ca44 | |||
| 2b11979b66 |
11
.env
Normal file
11
.env
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
; DO NOT MODIFY! this is an example environment file
|
||||||
|
; Create a copy called .env.local with the needed settings
|
||||||
|
|
||||||
|
GIT_HOST=
|
||||||
|
GIT_OWNER=
|
||||||
|
GIT_REPO=
|
||||||
|
GIT_TOKEN=
|
||||||
|
PULL_REQUEST=
|
||||||
|
AI_HOST=
|
||||||
|
AI_MODEL=
|
||||||
|
AI_TOKEN=
|
||||||
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
.env.local
|
||||||
30
package-lock.json
generated
30
package-lock.json
generated
@@ -9,8 +9,10 @@
|
|||||||
"version": "0.0.0",
|
"version": "0.0.0",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ztimson/ai-utils": "0.2.3",
|
"@ztimson/ai-utils": "^0.2.4",
|
||||||
"@ztimson/utils": "^0.28.3"
|
"@ztimson/node-utils": "^1.0.7",
|
||||||
|
"@ztimson/utils": "^0.28.3",
|
||||||
|
"dotenv": "^17.2.3"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@anthropic-ai/sdk": {
|
"node_modules/@anthropic-ai/sdk": {
|
||||||
@@ -286,9 +288,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@ztimson/ai-utils": {
|
"node_modules/@ztimson/ai-utils": {
|
||||||
"version": "0.2.3",
|
"version": "0.2.4",
|
||||||
"resolved": "https://registry.npmjs.org/@ztimson/ai-utils/-/ai-utils-0.2.3.tgz",
|
"resolved": "https://registry.npmjs.org/@ztimson/ai-utils/-/ai-utils-0.2.4.tgz",
|
||||||
"integrity": "sha512-3BnB7Xx4WaQbcYucaUNN9wCqZlhWZKyp/emEdHa4SBNUcyQAMl3VuVo6Ulle0vScsyOKXhPXN+5ij3sv1dk02w==",
|
"integrity": "sha512-dCkUCTXlKDztXRta4LrfOmgmhxYwegG/0dEr9jNXgbFb8j0M6bS5y9ZOn3m/tK6NDNdUzHKDBA/QDl17Vu7oyA==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@anthropic-ai/sdk": "^0.67.0",
|
"@anthropic-ai/sdk": "^0.67.0",
|
||||||
@@ -311,9 +313,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@ztimson/node-utils": {
|
"node_modules/@ztimson/node-utils": {
|
||||||
"version": "1.0.4",
|
"version": "1.0.7",
|
||||||
"resolved": "https://registry.npmjs.org/@ztimson/node-utils/-/node-utils-1.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/@ztimson/node-utils/-/node-utils-1.0.7.tgz",
|
||||||
"integrity": "sha512-EVI0CQYvrzRV7f9pmy13piyFfF9htYS4Kf2W/dW4HzSBEPTO//mGPZthWyQXnxvMeG6p3e5hAPlQ4hAmrRDv6A==",
|
"integrity": "sha512-BqrEMfW3/FKyYnTnL1uzxHUZFUeg2FaRGjL9tWAFkw8M99HcN9HoStdrPL6JfLSudvCwDbGzUJSEP6GL0E8uoQ==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/@ztimson/utils": {
|
"node_modules/@ztimson/utils": {
|
||||||
@@ -682,6 +684,18 @@
|
|||||||
"node": ">=8"
|
"node": ">=8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/dotenv": {
|
||||||
|
"version": "17.2.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.2.3.tgz",
|
||||||
|
"integrity": "sha512-JVUnt+DUIzu87TABbhPmNfVdBDt18BLOWjMUFJMSi/Qqg7NTYtabbvSNJGOJ7afbRuv9D/lngizHtP7QyLQ+9w==",
|
||||||
|
"license": "BSD-2-Clause",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=12"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://dotenvx.com"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/dunder-proto": {
|
"node_modules/dunder-proto": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
|
||||||
|
|||||||
@@ -9,8 +9,9 @@
|
|||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"description": "",
|
"description": "",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ztimson/ai-utils": "0.2.3",
|
"@ztimson/ai-utils": "^0.2.4",
|
||||||
|
"@ztimson/node-utils": "^1.0.7",
|
||||||
"@ztimson/utils": "^0.28.3",
|
"@ztimson/utils": "^0.28.3",
|
||||||
"@ztimson/node-utils": "^1.0.4"
|
"dotenv": "^17.2.3"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,10 @@ import {$} from '@ztimson/node-utils';
|
|||||||
import * as os from 'node:os';
|
import * as os from 'node:os';
|
||||||
import * as path from 'node:path';
|
import * as path from 'node:path';
|
||||||
import * as fs from 'node:fs';
|
import * as fs from 'node:fs';
|
||||||
|
import * as dotenv from 'dotenv';
|
||||||
|
|
||||||
|
dotenv.config();
|
||||||
|
dotenv.config({path: '.env.local', override: true});
|
||||||
|
|
||||||
(async () => {
|
(async () => {
|
||||||
const
|
const
|
||||||
@@ -16,9 +20,15 @@ import * as fs from 'node:fs';
|
|||||||
model = process.env['AI_MODEL'],
|
model = process.env['AI_MODEL'],
|
||||||
token = process.env['AI_TOKEN'];
|
token = process.env['AI_TOKEN'];
|
||||||
|
|
||||||
|
|
||||||
const comments = [];
|
const comments = [];
|
||||||
const commit = (await $`git log -1 --pretty=format:%H`).trim();
|
const commit = await $`cd ${root} && git log -1 --pretty=format:%H`;
|
||||||
const gitDiff = await $`git diff HEAD^..HEAD`;
|
const target = await $`cd ${root} && git rev-parse HEAD`;
|
||||||
|
const dest = await $`cd ${root} && git symbolic-ref refs/remotes/origin/HEAD`;
|
||||||
|
const gitDiff = await $`git diff ${dest} ${target}`;
|
||||||
|
const markdown = !!git;
|
||||||
|
|
||||||
|
console.log(`Inspecting: ${root} (${commit})\n`);
|
||||||
|
|
||||||
let options = {ollama: {model, host}};
|
let options = {ollama: {model, host}};
|
||||||
if(host === 'anthropic') options = {anthropic: {model, token}};
|
if(host === 'anthropic') options = {anthropic: {model, token}};
|
||||||
@@ -27,7 +37,7 @@ import * as fs from 'node:fs';
|
|||||||
...options,
|
...options,
|
||||||
model: [host, model],
|
model: [host, model],
|
||||||
path: process.env['path'] || os.tmpdir(),
|
path: process.env['path'] || os.tmpdir(),
|
||||||
system: 'You are a code reviewer. Analyze the git diff and use the `recommend` tool for EACH issue you find. You must call `recommend` exactly once for every bug or improvement opportunity. After making all recommendations, provide a brief summary.',
|
system: `You are a code reviewer. Analyze the git diff and use the \`recommend\` tool for EACH issue you find. You must call \`recommend\` exactly once for every bug or improvement opportunity. After making all recommendations, provide a summary using 100 words or less in ${markdown ? 'markdown' : 'unstyled text'}.`,
|
||||||
tools: [{
|
tools: [{
|
||||||
name: 'read_file',
|
name: 'read_file',
|
||||||
description: 'Read contents of a file',
|
description: 'Read contents of a file',
|
||||||
@@ -47,7 +57,7 @@ import * as fs from 'node:fs';
|
|||||||
description: 'REQUIRED: Call this once for every bug, improvement, or concern identified in the review.',
|
description: 'REQUIRED: Call this once for every bug, improvement, or concern identified in the review.',
|
||||||
args: {
|
args: {
|
||||||
file: {type: 'string', description: 'File path'},
|
file: {type: 'string', description: 'File path'},
|
||||||
line: {type: 'number', description: 'Line number in new file', optional: true},
|
line: {type: 'number', description: 'Line number in new file'},
|
||||||
comment: {type: 'string', description: 'Review comment explaining the issue'}
|
comment: {type: 'string', description: 'Review comment explaining the issue'}
|
||||||
},
|
},
|
||||||
fn: (args) => {
|
fn: (args) => {
|
||||||
@@ -61,19 +71,28 @@ import * as fs from 'node:fs';
|
|||||||
}]
|
}]
|
||||||
});
|
});
|
||||||
|
|
||||||
const summary = await ai.language.ask(gitDiff);
|
if(!gitDiff) {
|
||||||
const res = await fetch(`${git}/api/v1/repos/${owner}/${repo}/pulls/${pr}/reviews`, {
|
console.warn('No diff found');
|
||||||
method: 'POST',
|
return process.exit();
|
||||||
headers: {
|
}
|
||||||
'Authorization': `token ${auth}`,
|
|
||||||
'Content-Type': 'application/json'
|
const messages = await ai.language.ask(gitDiff);
|
||||||
},
|
const summary = messages.pop().content;
|
||||||
body: JSON.stringify({
|
if(git) {
|
||||||
body: summary,
|
const res = await fetch(`${git}/api/v1/repos/${owner}/${repo}/pulls/${pr}/reviews`, {
|
||||||
commit_id: commit,
|
method: 'POST',
|
||||||
event: 'COMMENT',
|
headers: {
|
||||||
comments,
|
'Authorization': `token ${auth}`,
|
||||||
})
|
'Content-Type': 'application/json'
|
||||||
});
|
},
|
||||||
if(!res.ok) throw new Error(`${res.status} ${await res.text()}`);
|
body: JSON.stringify({
|
||||||
|
body: summary,
|
||||||
|
commit_id: commit,
|
||||||
|
event: 'COMMENT',
|
||||||
|
comments,
|
||||||
|
})
|
||||||
|
});
|
||||||
|
if(!res.ok) throw new Error(`${res.status} ${await res.text()}`);
|
||||||
|
}
|
||||||
|
console.log(comments.map(c => `${c.path}${c.new_position ? `:${c.new_position}` : ''}\n${c.body}`).join('\n\n') + '\n\n' + summary);
|
||||||
})();
|
})();
|
||||||
|
|||||||
Reference in New Issue
Block a user