@motivation-labs/crosscheck 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (56) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +92 -0
  3. package/crosscheck.config.example.yml +75 -0
  4. package/dist/cli.d.ts +3 -0
  5. package/dist/cli.d.ts.map +1 -0
  6. package/dist/cli.js +40 -0
  7. package/dist/cli.js.map +1 -0
  8. package/dist/commands/init.d.ts +2 -0
  9. package/dist/commands/init.d.ts.map +1 -0
  10. package/dist/commands/init.js +75 -0
  11. package/dist/commands/init.js.map +1 -0
  12. package/dist/commands/review.d.ts +2 -0
  13. package/dist/commands/review.d.ts.map +1 -0
  14. package/dist/commands/review.js +69 -0
  15. package/dist/commands/review.js.map +1 -0
  16. package/dist/commands/serve.d.ts +2 -0
  17. package/dist/commands/serve.d.ts.map +1 -0
  18. package/dist/commands/serve.js +92 -0
  19. package/dist/commands/serve.js.map +1 -0
  20. package/dist/commands/status.d.ts +2 -0
  21. package/dist/commands/status.d.ts.map +1 -0
  22. package/dist/commands/status.js +57 -0
  23. package/dist/commands/status.js.map +1 -0
  24. package/dist/commands/watch.d.ts +2 -0
  25. package/dist/commands/watch.d.ts.map +1 -0
  26. package/dist/commands/watch.js +177 -0
  27. package/dist/commands/watch.js.map +1 -0
  28. package/dist/config/loader.d.ts +5 -0
  29. package/dist/config/loader.d.ts.map +1 -0
  30. package/dist/config/loader.js +34 -0
  31. package/dist/config/loader.js.map +1 -0
  32. package/dist/config/schema.d.ts +267 -0
  33. package/dist/config/schema.d.ts.map +1 -0
  34. package/dist/config/schema.js +47 -0
  35. package/dist/config/schema.js.map +1 -0
  36. package/dist/github/client.d.ts +16 -0
  37. package/dist/github/client.d.ts.map +1 -0
  38. package/dist/github/client.js +89 -0
  39. package/dist/github/client.js.map +1 -0
  40. package/dist/github/detector.d.ts +6 -0
  41. package/dist/github/detector.d.ts.map +1 -0
  42. package/dist/github/detector.js +35 -0
  43. package/dist/github/detector.js.map +1 -0
  44. package/dist/github/webhook.d.ts +27 -0
  45. package/dist/github/webhook.d.ts.map +1 -0
  46. package/dist/github/webhook.js +48 -0
  47. package/dist/github/webhook.js.map +1 -0
  48. package/dist/reviewers/claude.d.ts +7 -0
  49. package/dist/reviewers/claude.d.ts.map +1 -0
  50. package/dist/reviewers/claude.js +66 -0
  51. package/dist/reviewers/claude.js.map +1 -0
  52. package/dist/reviewers/codex.d.ts +7 -0
  53. package/dist/reviewers/codex.d.ts.map +1 -0
  54. package/dist/reviewers/codex.js +59 -0
  55. package/dist/reviewers/codex.js.map +1 -0
  56. package/package.json +60 -0
@@ -0,0 +1,177 @@
1
+ import { execSync } from 'child_process';
2
+ import chalk from 'chalk';
3
+ import SmeeClient from 'smee-client';
4
+ import { createWebhookServer } from '../github/webhook.js';
5
+ import { createGithubClient, postReviewComment, registerRepoWebhook, deleteRepoWebhook, registerOrgWebhook, deleteOrgWebhook, } from '../github/client.js';
6
+ import { detectPROrigin, assignReviewer } from '../github/detector.js';
7
+ import { runCodexReview } from '../reviewers/codex.js';
8
+ import { runClaudeReview } from '../reviewers/claude.js';
9
+ import { loadConfig, getGithubToken, getWebhookSecret } from '../config/loader.js';
10
+ import { mkdtempSync, rmSync } from 'fs';
11
+ import { tmpdir } from 'os';
12
+ import { join } from 'path';
13
+ function detectCurrentRepo() {
14
+ try {
15
+ const remote = execSync('git remote get-url origin 2>/dev/null', { encoding: 'utf8' }).trim();
16
+ const m = remote.match(/github\.com[:/]([^/]+)\/([^/.]+)/);
17
+ if (m)
18
+ return { owner: m[1], repo: m[2] };
19
+ }
20
+ catch { /* ignore */ }
21
+ return null;
22
+ }
23
+ async function createSmeeChannel() {
24
+ const res = await fetch('https://smee.io/new', { method: 'HEAD', redirect: 'manual' });
25
+ const location = res.headers.get('location');
26
+ if (!location)
27
+ throw new Error('Could not create smee.io channel');
28
+ return location;
29
+ }
30
+ export async function runWatch(configPath) {
31
+ const config = loadConfig(configPath);
32
+ const token = getGithubToken();
33
+ const webhookSecret = getWebhookSecret();
34
+ const log = (msg) => console.log(`${chalk.dim(new Date().toLocaleTimeString())} ${msg}`);
35
+ // PR deduplication — skip if already reviewing this PR+SHA
36
+ const inFlight = new Set();
37
+ // Start local webhook server
38
+ const server = createWebhookServer(config, webhookSecret, async (event) => {
39
+ const { pull_request: pr, repository: repo } = event;
40
+ const owner = repo.owner.login;
41
+ const repoName = repo.name;
42
+ const prNumber = event.number;
43
+ const key = `${owner}/${repoName}#${prNumber}@${pr.head.sha}`;
44
+ if (inFlight.has(key)) {
45
+ log(chalk.dim(`PR #${prNumber} already in review — skipping duplicate event`));
46
+ return;
47
+ }
48
+ inFlight.add(key);
49
+ log(`${chalk.bold(`PR #${prNumber}`)} ${event.action}: ${pr.title}`);
50
+ const origin = detectPROrigin(pr.body ?? '', config);
51
+ const reviewer = assignReviewer(origin, config);
52
+ if (!reviewer) {
53
+ log(chalk.dim(` origin=${origin}, no reviewer — skipping`));
54
+ inFlight.delete(key);
55
+ return;
56
+ }
57
+ log(` ${chalk.dim('→')} origin=${chalk.yellow(origin)}, reviewer=${chalk.cyan(reviewer)}`);
58
+ const tmpDir = mkdtempSync(join(tmpdir(), 'crosscheck-repo-'));
59
+ try {
60
+ log(' cloning PR...');
61
+ execSync(`gh repo clone ${owner}/${repoName} ${tmpDir} -- --depth=50 --quiet`, { stdio: 'pipe' });
62
+ execSync(`git fetch origin pull/${prNumber}/head:pr-${prNumber}`, { cwd: tmpDir, stdio: 'pipe' });
63
+ execSync(`git checkout pr-${prNumber}`, { cwd: tmpDir, stdio: 'pipe' });
64
+ let reviewText;
65
+ if (reviewer === 'codex') {
66
+ reviewText = await runCodexReview(tmpDir, pr.base.ref, pr.title, config.quality, config.vendors.codex.model, config.vendors.codex.auth, log);
67
+ }
68
+ else {
69
+ reviewText = await runClaudeReview(tmpDir, pr.base.ref, pr.title, config.quality, config.vendors.claude, config.budget.per_review_usd, log);
70
+ }
71
+ const octokit = createGithubClient(token);
72
+ await postReviewComment(octokit, owner, repoName, prNumber, reviewText, reviewer);
73
+ log(chalk.green(` ✓ review posted to PR #${prNumber}`));
74
+ }
75
+ catch (err) {
76
+ const error = err;
77
+ log(chalk.red(` ✗ ${error.message ?? 'unknown error'}`));
78
+ }
79
+ finally {
80
+ rmSync(tmpDir, { force: true, recursive: true });
81
+ inFlight.delete(key);
82
+ }
83
+ }, log);
84
+ await new Promise(resolve => server.listen(config.server.port, resolve));
85
+ // Create smee.io channel and start proxy
86
+ log('Creating smee.io tunnel...');
87
+ const smeeUrl = await createSmeeChannel();
88
+ const target = `http://localhost:${config.server.port}${config.server.webhook_path}`;
89
+ const smee = new SmeeClient({ source: smeeUrl, target, logger: { info: () => { }, error: console.error } });
90
+ const smeeEvents = smee.start();
91
+ const registeredHooks = [];
92
+ if (config.orgs.length > 0) {
93
+ // Org-level webhooks take priority
94
+ for (const org of config.orgs) {
95
+ log(`Registering org webhook for ${org}...`);
96
+ try {
97
+ const hookId = await registerOrgWebhook(org, smeeUrl, webhookSecret, token);
98
+ registeredHooks.push({ type: 'org', org, hookId });
99
+ log(chalk.green(` ✓ org webhook registered for ${org} (id ${hookId})`));
100
+ }
101
+ catch (err) {
102
+ const error = err;
103
+ log(chalk.yellow(` Could not register org webhook for ${org}: ${error.message ?? 'unknown'}`));
104
+ log(chalk.dim(` Register manually at: https://github.com/organizations/${org}/settings/hooks`));
105
+ }
106
+ }
107
+ }
108
+ else if (config.repos.length > 0) {
109
+ // Explicit repo list
110
+ for (const { owner, name } of config.repos) {
111
+ log(`Registering repo webhook for ${owner}/${name}...`);
112
+ try {
113
+ const hookId = await registerRepoWebhook(owner, name, smeeUrl, webhookSecret, token);
114
+ registeredHooks.push({ type: 'repo', owner, repo: name, hookId });
115
+ log(chalk.green(` ✓ repo webhook registered for ${owner}/${name} (id ${hookId})`));
116
+ }
117
+ catch (err) {
118
+ const error = err;
119
+ log(chalk.yellow(` Could not register webhook for ${owner}/${name}: ${error.message ?? 'unknown'}`));
120
+ }
121
+ }
122
+ }
123
+ else {
124
+ // Auto-detect from git remote
125
+ const currentRepo = detectCurrentRepo();
126
+ if (!currentRepo) {
127
+ console.error(chalk.red('Could not detect a GitHub repo from git remote. Run inside a git repo or set repos/orgs in config.'));
128
+ smeeEvents.close();
129
+ server.close(() => process.exit(1));
130
+ return;
131
+ }
132
+ log(`Registering webhook on ${currentRepo.owner}/${currentRepo.repo}...`);
133
+ try {
134
+ const hookId = await registerRepoWebhook(currentRepo.owner, currentRepo.repo, smeeUrl, webhookSecret, token);
135
+ registeredHooks.push({ type: 'repo', owner: currentRepo.owner, repo: currentRepo.repo, hookId });
136
+ log(chalk.green(`Webhook registered (id ${hookId})`));
137
+ }
138
+ catch (err) {
139
+ const error = err;
140
+ log(chalk.yellow(`Could not auto-register webhook: ${error.message ?? 'unknown'}`));
141
+ log(chalk.dim(`Register manually: ${smeeUrl} → ${target}`));
142
+ }
143
+ }
144
+ console.log(chalk.bold('\ncrosscheck watch\n'));
145
+ if (config.orgs.length > 0) {
146
+ console.log(` orgs ${chalk.cyan(config.orgs.join(', '))}`);
147
+ }
148
+ else if (config.repos.length > 0) {
149
+ console.log(` repos ${chalk.cyan(config.repos.map(r => `${r.owner}/${r.name}`).join(', '))}`);
150
+ }
151
+ console.log(` mode ${chalk.cyan(config.mode)}`);
152
+ console.log(` quality ${chalk.cyan(config.quality.tier)}`);
153
+ console.log(` tunnel ${chalk.cyan(smeeUrl)}`);
154
+ console.log();
155
+ console.log(chalk.dim('Waiting for PR events — Ctrl+C to stop and clean up.\n'));
156
+ const cleanup = async () => {
157
+ console.log('\nCleaning up...');
158
+ smeeEvents.close();
159
+ for (const hook of registeredHooks) {
160
+ try {
161
+ if (hook.type === 'org') {
162
+ await deleteOrgWebhook(hook.org, hook.hookId, token);
163
+ console.log(` org webhook deregistered for ${hook.org}`);
164
+ }
165
+ else {
166
+ await deleteRepoWebhook(hook.owner, hook.repo, hook.hookId, token);
167
+ console.log(` repo webhook deregistered for ${hook.owner}/${hook.repo}`);
168
+ }
169
+ }
170
+ catch { /* best-effort cleanup */ }
171
+ }
172
+ server.close(() => process.exit(0));
173
+ };
174
+ process.on('SIGINT', () => { void cleanup(); });
175
+ process.on('SIGTERM', () => { void cleanup(); });
176
+ }
177
+ //# sourceMappingURL=watch.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"watch.js","sourceRoot":"","sources":["../../src/commands/watch.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { type Config } from './schema.js';
2
+ export declare function loadConfig(explicitPath?: string): Config;
3
+ export declare function getGithubToken(): string;
4
+ export declare function getWebhookSecret(): string;
5
+ //# sourceMappingURL=loader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"loader.d.ts","sourceRoot":"","sources":["../../src/config/loader.ts"],"names":[],"mappings":"AAIA,OAAO,EAAgB,KAAK,MAAM,EAAE,MAAM,aAAa,CAAA;AAavD,wBAAgB,UAAU,CAAC,YAAY,CAAC,EAAE,MAAM,GAAG,MAAM,CAMxD;AAED,wBAAgB,cAAc,IAAI,MAAM,CAIvC;AAED,wBAAgB,gBAAgB,IAAI,MAAM,CAIzC"}
@@ -0,0 +1,34 @@
1
+ import { readFileSync, existsSync } from 'fs';
2
+ import { resolve, join } from 'path';
3
+ import { homedir } from 'os';
4
+ import yaml from 'js-yaml';
5
+ import { ConfigSchema } from './schema.js';
6
+ const CONFIG_FILENAME = 'crosscheck.config.yml';
7
+ function findConfigFile() {
8
+ const candidates = [
9
+ resolve(process.cwd(), CONFIG_FILENAME),
10
+ resolve(process.cwd(), '.crosscheck.yml'),
11
+ join(homedir(), '.crosscheck', 'config.yml'),
12
+ ];
13
+ return candidates.find(existsSync) ?? null;
14
+ }
15
+ export function loadConfig(explicitPath) {
16
+ const configPath = explicitPath ?? findConfigFile();
17
+ if (!configPath)
18
+ return ConfigSchema.parse({});
19
+ const raw = yaml.load(readFileSync(configPath, 'utf8'));
20
+ return ConfigSchema.parse(raw);
21
+ }
22
+ export function getGithubToken() {
23
+ const token = process.env.GITHUB_TOKEN ?? process.env.GH_TOKEN;
24
+ if (!token)
25
+ throw new Error('GITHUB_TOKEN or GH_TOKEN environment variable is required');
26
+ return token;
27
+ }
28
+ export function getWebhookSecret() {
29
+ const secret = process.env.CROSSCHECK_WEBHOOK_SECRET ?? process.env.GITHUB_WEBHOOK_SECRET;
30
+ if (!secret)
31
+ throw new Error('CROSSCHECK_WEBHOOK_SECRET environment variable is required');
32
+ return secret;
33
+ }
34
+ //# sourceMappingURL=loader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"loader.js","sourceRoot":"","sources":["../../src/config/loader.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,IAAI,CAAA;AAC7C,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAA;AACpC,OAAO,EAAE,OAAO,EAAE,MAAM,IAAI,CAAA;AAC5B,OAAO,IAAI,MAAM,SAAS,CAAA;AAC1B,OAAO,EAAE,YAAY,EAAe,MAAM,aAAa,CAAA;AAEvD,MAAM,eAAe,GAAG,uBAAuB,CAAA;AAE/C,SAAS,cAAc;IACrB,MAAM,UAAU,GAAG;QACjB,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,eAAe,CAAC;QACvC,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,iBAAiB,CAAC;QACzC,IAAI,CAAC,OAAO,EAAE,EAAE,aAAa,EAAE,YAAY,CAAC;KAC7C,CAAA;IACD,OAAO,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,IAAI,CAAA;AAC5C,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,YAAqB;IAC9C,MAAM,UAAU,GAAG,YAAY,IAAI,cAAc,EAAE,CAAA;IACnD,IAAI,CAAC,UAAU;QAAE,OAAO,YAAY,CAAC,KAAK,CAAC,EAAE,CAAC,CAAA;IAE9C,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC,CAAA;IACvD,OAAO,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA;AAChC,CAAC;AAED,MAAM,UAAU,cAAc;IAC5B,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAA;IAC9D,IAAI,CAAC,KAAK;QAAE,MAAM,IAAI,KAAK,CAAC,2DAA2D,CAAC,CAAA;IACxF,OAAO,KAAK,CAAA;AACd,CAAC;AAED,MAAM,UAAU,gBAAgB;IAC9B,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,yBAAyB,IAAI,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAA;IACzF,IAAI,CAAC,MAAM;QAAE,MAAM,IAAI,KAAK,CAAC,4DAA4D,CAAC,CAAA;IAC1F,OAAO,MAAM,CAAA;AACf,CAAC"}
@@ -0,0 +1,267 @@
1
+ import { z } from 'zod';
2
+ export declare const VendorConfigSchema: z.ZodObject<{
3
+ enabled: z.ZodDefault<z.ZodBoolean>;
4
+ model: z.ZodOptional<z.ZodString>;
5
+ auth: z.ZodDefault<z.ZodEnum<["subscription", "api-key"]>>;
6
+ effort: z.ZodDefault<z.ZodEnum<["low", "medium", "high", "max"]>>;
7
+ }, "strip", z.ZodTypeAny, {
8
+ enabled: boolean;
9
+ auth: "subscription" | "api-key";
10
+ effort: "low" | "medium" | "high" | "max";
11
+ model?: string | undefined;
12
+ }, {
13
+ enabled?: boolean | undefined;
14
+ model?: string | undefined;
15
+ auth?: "subscription" | "api-key" | undefined;
16
+ effort?: "low" | "medium" | "high" | "max" | undefined;
17
+ }>;
18
+ export declare const QualityConfigSchema: z.ZodObject<{
19
+ tier: z.ZodDefault<z.ZodEnum<["fast", "balanced", "thorough"]>>;
20
+ focus: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
21
+ custom_prompt: z.ZodOptional<z.ZodString>;
22
+ }, "strip", z.ZodTypeAny, {
23
+ tier: "fast" | "balanced" | "thorough";
24
+ focus: string[];
25
+ custom_prompt?: string | undefined;
26
+ }, {
27
+ tier?: "fast" | "balanced" | "thorough" | undefined;
28
+ focus?: string[] | undefined;
29
+ custom_prompt?: string | undefined;
30
+ }>;
31
+ export declare const BudgetConfigSchema: z.ZodObject<{
32
+ codex_monthly_usd: z.ZodDefault<z.ZodNullable<z.ZodNumber>>;
33
+ per_review_usd: z.ZodDefault<z.ZodNumber>;
34
+ }, "strip", z.ZodTypeAny, {
35
+ codex_monthly_usd: number | null;
36
+ per_review_usd: number;
37
+ }, {
38
+ codex_monthly_usd?: number | null | undefined;
39
+ per_review_usd?: number | undefined;
40
+ }>;
41
+ export declare const RepoConfigSchema: z.ZodObject<{
42
+ owner: z.ZodString;
43
+ name: z.ZodString;
44
+ }, "strip", z.ZodTypeAny, {
45
+ owner: string;
46
+ name: string;
47
+ }, {
48
+ owner: string;
49
+ name: string;
50
+ }>;
51
+ export declare const RoutingConfigSchema: z.ZodObject<{
52
+ codex_reviews_patterns: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
53
+ claude_reviews_patterns: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
54
+ }, "strip", z.ZodTypeAny, {
55
+ codex_reviews_patterns: string[];
56
+ claude_reviews_patterns: string[];
57
+ }, {
58
+ codex_reviews_patterns?: string[] | undefined;
59
+ claude_reviews_patterns?: string[] | undefined;
60
+ }>;
61
+ export declare const ServerConfigSchema: z.ZodObject<{
62
+ port: z.ZodDefault<z.ZodNumber>;
63
+ webhook_path: z.ZodDefault<z.ZodString>;
64
+ }, "strip", z.ZodTypeAny, {
65
+ port: number;
66
+ webhook_path: string;
67
+ }, {
68
+ port?: number | undefined;
69
+ webhook_path?: string | undefined;
70
+ }>;
71
+ export declare const ConfigSchema: z.ZodObject<{
72
+ mode: z.ZodDefault<z.ZodEnum<["single-vendor", "cross-vendor"]>>;
73
+ vendors: z.ZodDefault<z.ZodObject<{
74
+ codex: z.ZodDefault<z.ZodObject<{
75
+ enabled: z.ZodDefault<z.ZodBoolean>;
76
+ model: z.ZodOptional<z.ZodString>;
77
+ auth: z.ZodDefault<z.ZodEnum<["subscription", "api-key"]>>;
78
+ effort: z.ZodDefault<z.ZodEnum<["low", "medium", "high", "max"]>>;
79
+ }, "strip", z.ZodTypeAny, {
80
+ enabled: boolean;
81
+ auth: "subscription" | "api-key";
82
+ effort: "low" | "medium" | "high" | "max";
83
+ model?: string | undefined;
84
+ }, {
85
+ enabled?: boolean | undefined;
86
+ model?: string | undefined;
87
+ auth?: "subscription" | "api-key" | undefined;
88
+ effort?: "low" | "medium" | "high" | "max" | undefined;
89
+ }>>;
90
+ claude: z.ZodDefault<z.ZodObject<{
91
+ enabled: z.ZodDefault<z.ZodBoolean>;
92
+ model: z.ZodOptional<z.ZodString>;
93
+ auth: z.ZodDefault<z.ZodEnum<["subscription", "api-key"]>>;
94
+ effort: z.ZodDefault<z.ZodEnum<["low", "medium", "high", "max"]>>;
95
+ }, "strip", z.ZodTypeAny, {
96
+ enabled: boolean;
97
+ auth: "subscription" | "api-key";
98
+ effort: "low" | "medium" | "high" | "max";
99
+ model?: string | undefined;
100
+ }, {
101
+ enabled?: boolean | undefined;
102
+ model?: string | undefined;
103
+ auth?: "subscription" | "api-key" | undefined;
104
+ effort?: "low" | "medium" | "high" | "max" | undefined;
105
+ }>>;
106
+ }, "strip", z.ZodTypeAny, {
107
+ codex: {
108
+ enabled: boolean;
109
+ auth: "subscription" | "api-key";
110
+ effort: "low" | "medium" | "high" | "max";
111
+ model?: string | undefined;
112
+ };
113
+ claude: {
114
+ enabled: boolean;
115
+ auth: "subscription" | "api-key";
116
+ effort: "low" | "medium" | "high" | "max";
117
+ model?: string | undefined;
118
+ };
119
+ }, {
120
+ codex?: {
121
+ enabled?: boolean | undefined;
122
+ model?: string | undefined;
123
+ auth?: "subscription" | "api-key" | undefined;
124
+ effort?: "low" | "medium" | "high" | "max" | undefined;
125
+ } | undefined;
126
+ claude?: {
127
+ enabled?: boolean | undefined;
128
+ model?: string | undefined;
129
+ auth?: "subscription" | "api-key" | undefined;
130
+ effort?: "low" | "medium" | "high" | "max" | undefined;
131
+ } | undefined;
132
+ }>>;
133
+ quality: z.ZodDefault<z.ZodObject<{
134
+ tier: z.ZodDefault<z.ZodEnum<["fast", "balanced", "thorough"]>>;
135
+ focus: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
136
+ custom_prompt: z.ZodOptional<z.ZodString>;
137
+ }, "strip", z.ZodTypeAny, {
138
+ tier: "fast" | "balanced" | "thorough";
139
+ focus: string[];
140
+ custom_prompt?: string | undefined;
141
+ }, {
142
+ tier?: "fast" | "balanced" | "thorough" | undefined;
143
+ focus?: string[] | undefined;
144
+ custom_prompt?: string | undefined;
145
+ }>>;
146
+ budget: z.ZodDefault<z.ZodObject<{
147
+ codex_monthly_usd: z.ZodDefault<z.ZodNullable<z.ZodNumber>>;
148
+ per_review_usd: z.ZodDefault<z.ZodNumber>;
149
+ }, "strip", z.ZodTypeAny, {
150
+ codex_monthly_usd: number | null;
151
+ per_review_usd: number;
152
+ }, {
153
+ codex_monthly_usd?: number | null | undefined;
154
+ per_review_usd?: number | undefined;
155
+ }>>;
156
+ orgs: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
157
+ repos: z.ZodDefault<z.ZodArray<z.ZodObject<{
158
+ owner: z.ZodString;
159
+ name: z.ZodString;
160
+ }, "strip", z.ZodTypeAny, {
161
+ owner: string;
162
+ name: string;
163
+ }, {
164
+ owner: string;
165
+ name: string;
166
+ }>, "many">>;
167
+ routing: z.ZodDefault<z.ZodObject<{
168
+ codex_reviews_patterns: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
169
+ claude_reviews_patterns: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
170
+ }, "strip", z.ZodTypeAny, {
171
+ codex_reviews_patterns: string[];
172
+ claude_reviews_patterns: string[];
173
+ }, {
174
+ codex_reviews_patterns?: string[] | undefined;
175
+ claude_reviews_patterns?: string[] | undefined;
176
+ }>>;
177
+ server: z.ZodDefault<z.ZodObject<{
178
+ port: z.ZodDefault<z.ZodNumber>;
179
+ webhook_path: z.ZodDefault<z.ZodString>;
180
+ }, "strip", z.ZodTypeAny, {
181
+ port: number;
182
+ webhook_path: string;
183
+ }, {
184
+ port?: number | undefined;
185
+ webhook_path?: string | undefined;
186
+ }>>;
187
+ }, "strip", z.ZodTypeAny, {
188
+ mode: "single-vendor" | "cross-vendor";
189
+ vendors: {
190
+ codex: {
191
+ enabled: boolean;
192
+ auth: "subscription" | "api-key";
193
+ effort: "low" | "medium" | "high" | "max";
194
+ model?: string | undefined;
195
+ };
196
+ claude: {
197
+ enabled: boolean;
198
+ auth: "subscription" | "api-key";
199
+ effort: "low" | "medium" | "high" | "max";
200
+ model?: string | undefined;
201
+ };
202
+ };
203
+ quality: {
204
+ tier: "fast" | "balanced" | "thorough";
205
+ focus: string[];
206
+ custom_prompt?: string | undefined;
207
+ };
208
+ budget: {
209
+ codex_monthly_usd: number | null;
210
+ per_review_usd: number;
211
+ };
212
+ orgs: string[];
213
+ repos: {
214
+ owner: string;
215
+ name: string;
216
+ }[];
217
+ routing: {
218
+ codex_reviews_patterns: string[];
219
+ claude_reviews_patterns: string[];
220
+ };
221
+ server: {
222
+ port: number;
223
+ webhook_path: string;
224
+ };
225
+ }, {
226
+ mode?: "single-vendor" | "cross-vendor" | undefined;
227
+ vendors?: {
228
+ codex?: {
229
+ enabled?: boolean | undefined;
230
+ model?: string | undefined;
231
+ auth?: "subscription" | "api-key" | undefined;
232
+ effort?: "low" | "medium" | "high" | "max" | undefined;
233
+ } | undefined;
234
+ claude?: {
235
+ enabled?: boolean | undefined;
236
+ model?: string | undefined;
237
+ auth?: "subscription" | "api-key" | undefined;
238
+ effort?: "low" | "medium" | "high" | "max" | undefined;
239
+ } | undefined;
240
+ } | undefined;
241
+ quality?: {
242
+ tier?: "fast" | "balanced" | "thorough" | undefined;
243
+ focus?: string[] | undefined;
244
+ custom_prompt?: string | undefined;
245
+ } | undefined;
246
+ budget?: {
247
+ codex_monthly_usd?: number | null | undefined;
248
+ per_review_usd?: number | undefined;
249
+ } | undefined;
250
+ orgs?: string[] | undefined;
251
+ repos?: {
252
+ owner: string;
253
+ name: string;
254
+ }[] | undefined;
255
+ routing?: {
256
+ codex_reviews_patterns?: string[] | undefined;
257
+ claude_reviews_patterns?: string[] | undefined;
258
+ } | undefined;
259
+ server?: {
260
+ port?: number | undefined;
261
+ webhook_path?: string | undefined;
262
+ } | undefined;
263
+ }>;
264
+ export type Config = z.infer<typeof ConfigSchema>;
265
+ export type VendorConfig = z.infer<typeof VendorConfigSchema>;
266
+ export type QualityConfig = z.infer<typeof QualityConfigSchema>;
267
+ //# sourceMappingURL=schema.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../src/config/schema.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAA;AAEvB,eAAO,MAAM,kBAAkB;;;;;;;;;;;;;;;EAK7B,CAAA;AAEF,eAAO,MAAM,mBAAmB;;;;;;;;;;;;EAI9B,CAAA;AAEF,eAAO,MAAM,kBAAkB;;;;;;;;;EAG7B,CAAA;AAEF,eAAO,MAAM,gBAAgB;;;;;;;;;EAG3B,CAAA;AAEF,eAAO,MAAM,mBAAmB;;;;;;;;;EAQ9B,CAAA;AAEF,eAAO,MAAM,kBAAkB;;;;;;;;;EAG7B,CAAA;AAEF,eAAO,MAAM,YAAY;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAYvB,CAAA;AAEF,MAAM,MAAM,MAAM,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,YAAY,CAAC,CAAA;AACjD,MAAM,MAAM,YAAY,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,kBAAkB,CAAC,CAAA;AAC7D,MAAM,MAAM,aAAa,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,mBAAmB,CAAC,CAAA"}
@@ -0,0 +1,47 @@
1
+ import { z } from 'zod';
2
+ export const VendorConfigSchema = z.object({
3
+ enabled: z.boolean().default(true),
4
+ model: z.string().optional(),
5
+ auth: z.enum(['subscription', 'api-key']).default('subscription'),
6
+ effort: z.enum(['low', 'medium', 'high', 'max']).default('medium'),
7
+ });
8
+ export const QualityConfigSchema = z.object({
9
+ tier: z.enum(['fast', 'balanced', 'thorough']).default('balanced'),
10
+ focus: z.array(z.string()).default([]),
11
+ custom_prompt: z.string().optional(),
12
+ });
13
+ export const BudgetConfigSchema = z.object({
14
+ codex_monthly_usd: z.number().nullable().default(null),
15
+ per_review_usd: z.number().default(2.0),
16
+ });
17
+ export const RepoConfigSchema = z.object({
18
+ owner: z.string(),
19
+ name: z.string(),
20
+ });
21
+ export const RoutingConfigSchema = z.object({
22
+ codex_reviews_patterns: z.array(z.string()).default([
23
+ 'Generated with \\[Claude Code\\]',
24
+ ]),
25
+ claude_reviews_patterns: z.array(z.string()).default([
26
+ 'Generated with \\[OpenAI Codex\\]',
27
+ 'Co-Authored-By: codex',
28
+ ]),
29
+ });
30
+ export const ServerConfigSchema = z.object({
31
+ port: z.number().default(7891),
32
+ webhook_path: z.string().default('/webhook'),
33
+ });
34
+ export const ConfigSchema = z.object({
35
+ mode: z.enum(['single-vendor', 'cross-vendor']).default('cross-vendor'),
36
+ vendors: z.object({
37
+ codex: VendorConfigSchema.default({}),
38
+ claude: VendorConfigSchema.default({}),
39
+ }).default({}),
40
+ quality: QualityConfigSchema.default({}),
41
+ budget: BudgetConfigSchema.default({}),
42
+ orgs: z.array(z.string()).default([]),
43
+ repos: z.array(RepoConfigSchema).default([]),
44
+ routing: RoutingConfigSchema.default({}),
45
+ server: ServerConfigSchema.default({}),
46
+ });
47
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.js","sourceRoot":"","sources":["../../src/config/schema.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,16 @@
1
+ import { Octokit } from 'octokit';
2
+ export declare function createGithubClient(token: string): import("@octokit/core").Octokit & {
3
+ paginate: import("@octokit/plugin-paginate-rest").PaginateInterface;
4
+ } & import("@octokit/plugin-paginate-graphql").paginateGraphQLInterface & import("@octokit/plugin-rest-endpoint-methods").Api & {
5
+ retry: {
6
+ retryRequest: (error: import("octokit").RequestError, retries: number, retryAfter: number) => import("octokit").RequestError;
7
+ };
8
+ };
9
+ export declare function verifyWebhookSignature(payload: string, signature: string, secret: string): boolean;
10
+ export declare function getPRDiff(octokit: Octokit, owner: string, repo: string, pullNumber: number): Promise<string>;
11
+ export declare function registerRepoWebhook(owner: string, repo: string, webhookUrl: string, secret: string, token: string): Promise<number>;
12
+ export declare function deleteRepoWebhook(owner: string, repo: string, hookId: number, token: string): Promise<void>;
13
+ export declare function registerOrgWebhook(org: string, webhookUrl: string, secret: string, token: string): Promise<number>;
14
+ export declare function deleteOrgWebhook(org: string, hookId: number, token: string): Promise<void>;
15
+ export declare function postReviewComment(octokit: Octokit, owner: string, repo: string, pullNumber: number, body: string, reviewer: string): Promise<void>;
16
+ //# sourceMappingURL=client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.d.ts","sourceRoot":"","sources":["../../src/github/client.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAA;AAGjC,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,MAAM;;;;;;EAE/C;AAED,wBAAgB,sBAAsB,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAOlG;AAED,wBAAsB,SAAS,CAC7B,OAAO,EAAE,OAAO,EAChB,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,UAAU,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,CAAC,CAQjB;AAED,wBAAsB,mBAAmB,CACvC,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,UAAU,EAAE,MAAM,EAClB,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,MAAM,GACZ,OAAO,CAAC,MAAM,CAAC,CAqBjB;AAED,wBAAsB,iBAAiB,CACrC,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,MAAM,GACZ,OAAO,CAAC,IAAI,CAAC,CAKf;AAED,wBAAsB,kBAAkB,CACtC,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,MAAM,GACZ,OAAO,CAAC,MAAM,CAAC,CAqBjB;AAED,wBAAsB,gBAAgB,CACpC,GAAG,EAAE,MAAM,EACX,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,MAAM,GACZ,OAAO,CAAC,IAAI,CAAC,CAKf;AAED,wBAAsB,iBAAiB,CACrC,OAAO,EAAE,OAAO,EAChB,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,UAAU,EAAE,MAAM,EAClB,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,MAAM,GACf,OAAO,CAAC,IAAI,CAAC,CAQf"}