@sentry/warden 0.0.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.
- package/.agents/skills/find-bugs/SKILL.md +75 -0
- package/.agents/skills/vercel-react-best-practices/AGENTS.md +2934 -0
- package/.agents/skills/vercel-react-best-practices/SKILL.md +136 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-api-routes.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-dependencies.md +51 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-parallel.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-preload.md +50 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
- package/.claude/settings.json +57 -0
- package/.claude/settings.local.json +88 -0
- package/.claude/skills/agent-prompt/SKILL.md +54 -0
- package/.claude/skills/agent-prompt/references/agentic-patterns.md +94 -0
- package/.claude/skills/agent-prompt/references/anti-patterns.md +140 -0
- package/.claude/skills/agent-prompt/references/context-design.md +124 -0
- package/.claude/skills/agent-prompt/references/core-principles.md +75 -0
- package/.claude/skills/agent-prompt/references/model-guidance.md +118 -0
- package/.claude/skills/agent-prompt/references/output-formats.md +98 -0
- package/.claude/skills/agent-prompt/references/skill-structure.md +115 -0
- package/.claude/skills/agent-prompt/references/system-prompts.md +115 -0
- package/.claude/skills/notseer/SKILL.md +131 -0
- package/.claude/skills/skill-writer/SKILL.md +140 -0
- package/.claude/skills/testing-guidelines/SKILL.md +132 -0
- package/.claude/skills/warden-skill/SKILL.md +250 -0
- package/.claude/skills/warden-skill/references/config-schema.md +133 -0
- package/.dex/config.toml +2 -0
- package/.github/workflows/ci.yml +33 -0
- package/.github/workflows/release.yml +54 -0
- package/.github/workflows/warden.yml +40 -0
- package/AGENTS.md +89 -0
- package/CONTRIBUTING.md +60 -0
- package/LICENSE +105 -0
- package/README.md +43 -0
- package/SPEC.md +263 -0
- package/action.yml +87 -0
- package/assets/favicon.png +0 -0
- package/assets/warden-icon-bw.svg +5 -0
- package/assets/warden-icon-purple.png +0 -0
- package/assets/warden-icon-purple.svg +5 -0
- package/docs/.claude/settings.local.json +11 -0
- package/docs/astro.config.mjs +43 -0
- package/docs/package.json +19 -0
- package/docs/pnpm-lock.yaml +4000 -0
- package/docs/public/favicon.svg +5 -0
- package/docs/src/components/Code.astro +141 -0
- package/docs/src/components/PackageManagerTabs.astro +183 -0
- package/docs/src/components/Terminal.astro +212 -0
- package/docs/src/layouts/Base.astro +380 -0
- package/docs/src/pages/cli.astro +167 -0
- package/docs/src/pages/config.astro +394 -0
- package/docs/src/pages/guide.astro +449 -0
- package/docs/src/pages/index.astro +490 -0
- package/docs/src/styles/global.css +551 -0
- package/docs/tsconfig.json +3 -0
- package/docs/vercel.json +5 -0
- package/eslint.config.js +33 -0
- package/package.json +73 -0
- package/src/action/index.ts +1 -0
- package/src/action/main.ts +868 -0
- package/src/cli/args.test.ts +477 -0
- package/src/cli/args.ts +415 -0
- package/src/cli/commands/add.ts +447 -0
- package/src/cli/commands/init.test.ts +136 -0
- package/src/cli/commands/init.ts +132 -0
- package/src/cli/commands/setup-app/browser.ts +38 -0
- package/src/cli/commands/setup-app/credentials.ts +45 -0
- package/src/cli/commands/setup-app/manifest.ts +48 -0
- package/src/cli/commands/setup-app/server.ts +172 -0
- package/src/cli/commands/setup-app.ts +156 -0
- package/src/cli/commands/sync.ts +114 -0
- package/src/cli/context.ts +131 -0
- package/src/cli/files.test.ts +155 -0
- package/src/cli/files.ts +89 -0
- package/src/cli/fix.test.ts +310 -0
- package/src/cli/fix.ts +387 -0
- package/src/cli/git.test.ts +119 -0
- package/src/cli/git.ts +318 -0
- package/src/cli/index.ts +14 -0
- package/src/cli/main.ts +672 -0
- package/src/cli/output/box.ts +235 -0
- package/src/cli/output/formatters.test.ts +187 -0
- package/src/cli/output/formatters.ts +269 -0
- package/src/cli/output/icons.ts +13 -0
- package/src/cli/output/index.ts +44 -0
- package/src/cli/output/ink-runner.tsx +337 -0
- package/src/cli/output/jsonl.test.ts +347 -0
- package/src/cli/output/jsonl.ts +126 -0
- package/src/cli/output/reporter.ts +435 -0
- package/src/cli/output/tasks.ts +374 -0
- package/src/cli/output/tty.test.ts +117 -0
- package/src/cli/output/tty.ts +60 -0
- package/src/cli/output/verbosity.test.ts +40 -0
- package/src/cli/output/verbosity.ts +31 -0
- package/src/cli/terminal.test.ts +148 -0
- package/src/cli/terminal.ts +301 -0
- package/src/config/index.ts +3 -0
- package/src/config/loader.test.ts +313 -0
- package/src/config/loader.ts +103 -0
- package/src/config/schema.ts +168 -0
- package/src/config/writer.test.ts +119 -0
- package/src/config/writer.ts +84 -0
- package/src/diff/classify.test.ts +162 -0
- package/src/diff/classify.ts +92 -0
- package/src/diff/coalesce.test.ts +208 -0
- package/src/diff/coalesce.ts +133 -0
- package/src/diff/context.test.ts +226 -0
- package/src/diff/context.ts +201 -0
- package/src/diff/index.ts +4 -0
- package/src/diff/parser.test.ts +212 -0
- package/src/diff/parser.ts +149 -0
- package/src/event/context.ts +132 -0
- package/src/event/index.ts +2 -0
- package/src/event/schedule-context.ts +101 -0
- package/src/examples/examples.integration.test.ts +66 -0
- package/src/examples/index.test.ts +101 -0
- package/src/examples/index.ts +122 -0
- package/src/examples/setup.ts +25 -0
- package/src/index.ts +115 -0
- package/src/output/dedup.test.ts +419 -0
- package/src/output/dedup.ts +607 -0
- package/src/output/github-checks.test.ts +300 -0
- package/src/output/github-checks.ts +476 -0
- package/src/output/github-issues.ts +329 -0
- package/src/output/index.ts +5 -0
- package/src/output/issue-renderer.ts +197 -0
- package/src/output/renderer.test.ts +727 -0
- package/src/output/renderer.ts +217 -0
- package/src/output/stale.test.ts +375 -0
- package/src/output/stale.ts +155 -0
- package/src/output/types.ts +34 -0
- package/src/sdk/index.ts +1 -0
- package/src/sdk/runner.test.ts +806 -0
- package/src/sdk/runner.ts +1232 -0
- package/src/skills/index.ts +36 -0
- package/src/skills/loader.test.ts +300 -0
- package/src/skills/loader.ts +423 -0
- package/src/skills/remote.test.ts +704 -0
- package/src/skills/remote.ts +604 -0
- package/src/triggers/matcher.test.ts +277 -0
- package/src/triggers/matcher.ts +152 -0
- package/src/types/index.ts +194 -0
- package/src/utils/async.ts +18 -0
- package/src/utils/index.test.ts +84 -0
- package/src/utils/index.ts +50 -0
- package/tsconfig.json +25 -0
- package/vitest.config.ts +8 -0
- package/vitest.integration.config.ts +11 -0
- package/warden.toml +19 -0
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cross-platform browser opener.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { exec } from 'node:child_process';
|
|
6
|
+
import { platform } from 'node:os';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Open a URL in the default browser.
|
|
10
|
+
* Returns a promise that resolves when the browser open command has been executed.
|
|
11
|
+
*/
|
|
12
|
+
export function openBrowser(url: string): Promise<void> {
|
|
13
|
+
return new Promise((resolve, reject) => {
|
|
14
|
+
const os = platform();
|
|
15
|
+
let command: string;
|
|
16
|
+
|
|
17
|
+
switch (os) {
|
|
18
|
+
case 'darwin':
|
|
19
|
+
command = `open "${url}"`;
|
|
20
|
+
break;
|
|
21
|
+
case 'win32':
|
|
22
|
+
command = `start "" "${url}"`;
|
|
23
|
+
break;
|
|
24
|
+
default:
|
|
25
|
+
// Linux and others
|
|
26
|
+
command = `xdg-open "${url}"`;
|
|
27
|
+
break;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
exec(command, (error) => {
|
|
31
|
+
if (error) {
|
|
32
|
+
reject(error);
|
|
33
|
+
} else {
|
|
34
|
+
resolve();
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Exchange temporary code for GitHub App credentials via GitHub API.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
export interface AppCredentials {
|
|
6
|
+
id: number;
|
|
7
|
+
name: string;
|
|
8
|
+
pem: string;
|
|
9
|
+
htmlUrl: string;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Exchange a temporary code for GitHub App credentials.
|
|
14
|
+
* This uses the GitHub API to convert the code into full app credentials.
|
|
15
|
+
*/
|
|
16
|
+
export async function exchangeCodeForCredentials(code: string): Promise<AppCredentials> {
|
|
17
|
+
const url = `https://api.github.com/app-manifests/${code}/conversions`;
|
|
18
|
+
|
|
19
|
+
const response = await fetch(url, {
|
|
20
|
+
method: 'POST',
|
|
21
|
+
headers: {
|
|
22
|
+
Accept: 'application/vnd.github+json',
|
|
23
|
+
'X-GitHub-Api-Version': '2022-11-28',
|
|
24
|
+
},
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
if (!response.ok) {
|
|
28
|
+
const errorText = await response.text();
|
|
29
|
+
throw new Error(`Failed to exchange code for credentials: ${response.status} ${response.statusText}\n${errorText}`);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const data = (await response.json()) as {
|
|
33
|
+
id: number;
|
|
34
|
+
name: string;
|
|
35
|
+
pem: string;
|
|
36
|
+
html_url: string;
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
return {
|
|
40
|
+
id: data.id,
|
|
41
|
+
name: data.name,
|
|
42
|
+
pem: data.pem,
|
|
43
|
+
htmlUrl: data.html_url,
|
|
44
|
+
};
|
|
45
|
+
}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GitHub App manifest builder for the setup-app flow.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
export interface ManifestOptions {
|
|
6
|
+
name?: string;
|
|
7
|
+
port: number;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export interface GitHubAppManifest {
|
|
11
|
+
name: string;
|
|
12
|
+
url: string;
|
|
13
|
+
hook_attributes: {
|
|
14
|
+
url: string;
|
|
15
|
+
active: boolean;
|
|
16
|
+
};
|
|
17
|
+
redirect_url: string;
|
|
18
|
+
public: boolean;
|
|
19
|
+
default_permissions: Record<string, string>;
|
|
20
|
+
default_events: string[];
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Build a GitHub App manifest for Warden.
|
|
25
|
+
*/
|
|
26
|
+
export function buildManifest(options: ManifestOptions): GitHubAppManifest {
|
|
27
|
+
const name = options.name ?? 'Warden';
|
|
28
|
+
|
|
29
|
+
return {
|
|
30
|
+
name,
|
|
31
|
+
url: 'https://github.com/getsentry/warden',
|
|
32
|
+
hook_attributes: {
|
|
33
|
+
// URL is required by GitHub even when webhooks are disabled
|
|
34
|
+
url: 'https://example.com/webhook',
|
|
35
|
+
active: false,
|
|
36
|
+
},
|
|
37
|
+
redirect_url: `http://localhost:${options.port}/callback`,
|
|
38
|
+
public: false,
|
|
39
|
+
default_permissions: {
|
|
40
|
+
contents: 'read',
|
|
41
|
+
pull_requests: 'write',
|
|
42
|
+
issues: 'write',
|
|
43
|
+
checks: 'write',
|
|
44
|
+
metadata: 'read',
|
|
45
|
+
},
|
|
46
|
+
default_events: ['pull_request'],
|
|
47
|
+
};
|
|
48
|
+
}
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Local HTTP server for GitHub App manifest flow.
|
|
3
|
+
* Serves a form that POSTs the manifest to GitHub, then receives the callback.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { createServer, type Server, type IncomingMessage, type ServerResponse } from 'node:http';
|
|
7
|
+
import { URL } from 'node:url';
|
|
8
|
+
import type { GitHubAppManifest } from './manifest.js';
|
|
9
|
+
|
|
10
|
+
export interface CallbackResult {
|
|
11
|
+
code: string;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export interface ServerOptions {
|
|
15
|
+
port: number;
|
|
16
|
+
expectedState: string;
|
|
17
|
+
timeoutMs: number;
|
|
18
|
+
manifest: GitHubAppManifest;
|
|
19
|
+
org?: string;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Build the HTML page that auto-submits the manifest form to GitHub.
|
|
24
|
+
*/
|
|
25
|
+
function buildStartPage(manifest: GitHubAppManifest, state: string, org?: string): string {
|
|
26
|
+
const githubUrl = org
|
|
27
|
+
? `https://github.com/organizations/${org}/settings/apps/new?state=${state}`
|
|
28
|
+
: `https://github.com/settings/apps/new?state=${state}`;
|
|
29
|
+
|
|
30
|
+
const manifestJson = JSON.stringify(manifest);
|
|
31
|
+
|
|
32
|
+
return `<!DOCTYPE html>
|
|
33
|
+
<html>
|
|
34
|
+
<head>
|
|
35
|
+
<title>Creating GitHub App...</title>
|
|
36
|
+
<style>
|
|
37
|
+
body { font-family: -apple-system, BlinkMacSystemFont, sans-serif; text-align: center; padding: 50px; }
|
|
38
|
+
.spinner { border: 4px solid #f3f3f3; border-top: 4px solid #3498db; border-radius: 50%; width: 40px; height: 40px; animation: spin 1s linear infinite; margin: 20px auto; }
|
|
39
|
+
@keyframes spin { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } }
|
|
40
|
+
</style>
|
|
41
|
+
</head>
|
|
42
|
+
<body>
|
|
43
|
+
<h1>Redirecting to GitHub...</h1>
|
|
44
|
+
<div class="spinner"></div>
|
|
45
|
+
<p>If you are not redirected automatically, click the button below.</p>
|
|
46
|
+
<form id="manifest-form" action="${githubUrl}" method="post">
|
|
47
|
+
<input type="hidden" name="manifest" value='${manifestJson.replace(/'/g, ''')}'>
|
|
48
|
+
<button type="submit" style="padding: 10px 20px; font-size: 16px; cursor: pointer;">Continue to GitHub</button>
|
|
49
|
+
</form>
|
|
50
|
+
<script>
|
|
51
|
+
// Auto-submit the form after a brief delay
|
|
52
|
+
setTimeout(function() {
|
|
53
|
+
document.getElementById('manifest-form').submit();
|
|
54
|
+
}, 500);
|
|
55
|
+
</script>
|
|
56
|
+
</body>
|
|
57
|
+
</html>`;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Create and start a local HTTP server for the manifest flow.
|
|
62
|
+
* - GET / or /start: Serves the form that POSTs to GitHub
|
|
63
|
+
* - GET /callback: Receives the callback from GitHub with the code
|
|
64
|
+
*/
|
|
65
|
+
export function startCallbackServer(options: ServerOptions): {
|
|
66
|
+
server: Server;
|
|
67
|
+
waitForCallback: Promise<CallbackResult>;
|
|
68
|
+
close: () => void;
|
|
69
|
+
startUrl: string;
|
|
70
|
+
} {
|
|
71
|
+
let resolveCallback: (result: CallbackResult) => void;
|
|
72
|
+
let rejectCallback: (error: Error) => void;
|
|
73
|
+
|
|
74
|
+
const waitForCallback = new Promise<CallbackResult>((resolve, reject) => {
|
|
75
|
+
resolveCallback = resolve;
|
|
76
|
+
rejectCallback = reject;
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
const server = createServer((req: IncomingMessage, res: ServerResponse) => {
|
|
80
|
+
const url = new URL(req.url || '/', `http://localhost:${options.port}`);
|
|
81
|
+
|
|
82
|
+
// Serve the start page that auto-submits to GitHub
|
|
83
|
+
if (req.method === 'GET' && (url.pathname === '/' || url.pathname === '/start')) {
|
|
84
|
+
res.writeHead(200, { 'Content-Type': 'text/html' });
|
|
85
|
+
res.end(buildStartPage(options.manifest, options.expectedState, options.org));
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Handle callback from GitHub
|
|
90
|
+
if (req.method === 'GET' && url.pathname === '/callback') {
|
|
91
|
+
const code = url.searchParams.get('code');
|
|
92
|
+
const state = url.searchParams.get('state');
|
|
93
|
+
|
|
94
|
+
// Validate state parameter (CSRF protection)
|
|
95
|
+
if (state !== options.expectedState) {
|
|
96
|
+
res.writeHead(400, { 'Content-Type': 'text/html' });
|
|
97
|
+
res.end(`
|
|
98
|
+
<!DOCTYPE html>
|
|
99
|
+
<html>
|
|
100
|
+
<head><title>Error</title></head>
|
|
101
|
+
<body>
|
|
102
|
+
<h1>Error: Invalid state parameter</h1>
|
|
103
|
+
<p>This may be a CSRF attack. Please try again.</p>
|
|
104
|
+
</body>
|
|
105
|
+
</html>
|
|
106
|
+
`);
|
|
107
|
+
rejectCallback(new Error('Invalid state parameter - possible CSRF attack'));
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
if (!code) {
|
|
112
|
+
res.writeHead(400, { 'Content-Type': 'text/html' });
|
|
113
|
+
res.end(`
|
|
114
|
+
<!DOCTYPE html>
|
|
115
|
+
<html>
|
|
116
|
+
<head><title>Error</title></head>
|
|
117
|
+
<body>
|
|
118
|
+
<h1>Error: Missing code parameter</h1>
|
|
119
|
+
<p>GitHub did not provide the expected authorization code.</p>
|
|
120
|
+
</body>
|
|
121
|
+
</html>
|
|
122
|
+
`);
|
|
123
|
+
rejectCallback(new Error('Missing code parameter in callback'));
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// Success - send response and resolve promise
|
|
128
|
+
res.writeHead(200, { 'Content-Type': 'text/html' });
|
|
129
|
+
res.end(`
|
|
130
|
+
<!DOCTYPE html>
|
|
131
|
+
<html>
|
|
132
|
+
<head>
|
|
133
|
+
<title>Success</title>
|
|
134
|
+
<style>
|
|
135
|
+
body { font-family: -apple-system, BlinkMacSystemFont, sans-serif; text-align: center; padding: 50px; }
|
|
136
|
+
h1 { color: #28a745; }
|
|
137
|
+
</style>
|
|
138
|
+
</head>
|
|
139
|
+
<body>
|
|
140
|
+
<h1>GitHub App Created!</h1>
|
|
141
|
+
<p>You can close this window and return to the terminal.</p>
|
|
142
|
+
</body>
|
|
143
|
+
</html>
|
|
144
|
+
`);
|
|
145
|
+
|
|
146
|
+
resolveCallback({ code });
|
|
147
|
+
return;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// 404 for anything else
|
|
151
|
+
res.writeHead(404);
|
|
152
|
+
res.end('Not found');
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
// Bind only to localhost for security
|
|
156
|
+
server.listen(options.port, '127.0.0.1');
|
|
157
|
+
|
|
158
|
+
// Set up timeout
|
|
159
|
+
const timeoutId = setTimeout(() => {
|
|
160
|
+
rejectCallback(new Error(`Timeout: No callback received within ${options.timeoutMs / 1000} seconds`));
|
|
161
|
+
server.close();
|
|
162
|
+
}, options.timeoutMs);
|
|
163
|
+
|
|
164
|
+
const close = () => {
|
|
165
|
+
clearTimeout(timeoutId);
|
|
166
|
+
server.close();
|
|
167
|
+
};
|
|
168
|
+
|
|
169
|
+
const startUrl = `http://localhost:${options.port}/start`;
|
|
170
|
+
|
|
171
|
+
return { server, waitForCallback, close, startUrl };
|
|
172
|
+
}
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Setup GitHub App command.
|
|
3
|
+
* Creates a GitHub App via the manifest flow for Warden to post as a custom bot.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { randomBytes } from 'node:crypto';
|
|
7
|
+
import chalk from 'chalk';
|
|
8
|
+
import type { SetupAppOptions } from '../args.js';
|
|
9
|
+
import type { Reporter } from '../output/reporter.js';
|
|
10
|
+
import { getGitHubRepoUrl } from '../git.js';
|
|
11
|
+
import { buildManifest } from './setup-app/manifest.js';
|
|
12
|
+
import { startCallbackServer } from './setup-app/server.js';
|
|
13
|
+
import { openBrowser } from './setup-app/browser.js';
|
|
14
|
+
import { exchangeCodeForCredentials } from './setup-app/credentials.js';
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Run the setup-app command.
|
|
18
|
+
*/
|
|
19
|
+
export async function runSetupApp(options: SetupAppOptions, reporter: Reporter): Promise<number> {
|
|
20
|
+
const { port, timeout, org, name, open } = options;
|
|
21
|
+
|
|
22
|
+
reporter.bold('SETUP GITHUB APP');
|
|
23
|
+
reporter.blank();
|
|
24
|
+
|
|
25
|
+
// Generate state token for CSRF protection
|
|
26
|
+
const state = randomBytes(16).toString('hex');
|
|
27
|
+
|
|
28
|
+
// Build manifest
|
|
29
|
+
const manifest = buildManifest({ name, port });
|
|
30
|
+
|
|
31
|
+
// Show what permissions will be requested
|
|
32
|
+
reporter.text('This will create a GitHub App with the following permissions:');
|
|
33
|
+
reporter.text(` ${chalk.dim('•')} contents: read ${chalk.dim('- Read repository files')}`);
|
|
34
|
+
reporter.text(` ${chalk.dim('•')} pull_requests: write ${chalk.dim('- Post review comments')}`);
|
|
35
|
+
reporter.text(` ${chalk.dim('•')} issues: write ${chalk.dim('- Create/update issues')}`);
|
|
36
|
+
reporter.text(` ${chalk.dim('•')} checks: write ${chalk.dim('- Create check runs')}`);
|
|
37
|
+
reporter.text(` ${chalk.dim('•')} metadata: read ${chalk.dim('- Read repository metadata')}`);
|
|
38
|
+
reporter.blank();
|
|
39
|
+
|
|
40
|
+
// Start local server (serves the form and handles callback)
|
|
41
|
+
reporter.step(`Starting local server on http://localhost:${port}...`);
|
|
42
|
+
|
|
43
|
+
const serverHandle = startCallbackServer({
|
|
44
|
+
port,
|
|
45
|
+
expectedState: state,
|
|
46
|
+
timeoutMs: timeout * 1000,
|
|
47
|
+
manifest,
|
|
48
|
+
org,
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
// Handle server errors (e.g., port already in use)
|
|
52
|
+
serverHandle.server.on('error', (error: NodeJS.ErrnoException) => {
|
|
53
|
+
if (error.code === 'EADDRINUSE') {
|
|
54
|
+
reporter.error(`Port ${port} is already in use. Try a different port with --port <number>`);
|
|
55
|
+
} else {
|
|
56
|
+
reporter.error(`Server error: ${error.message}`);
|
|
57
|
+
}
|
|
58
|
+
process.exit(1);
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
try {
|
|
62
|
+
// Open browser to our local server (which will POST to GitHub)
|
|
63
|
+
if (open) {
|
|
64
|
+
reporter.step('Opening browser...');
|
|
65
|
+
try {
|
|
66
|
+
await openBrowser(serverHandle.startUrl);
|
|
67
|
+
} catch {
|
|
68
|
+
reporter.warning('Could not open browser automatically.');
|
|
69
|
+
reporter.blank();
|
|
70
|
+
reporter.text('Open this URL in your browser:');
|
|
71
|
+
reporter.text(chalk.cyan(serverHandle.startUrl));
|
|
72
|
+
}
|
|
73
|
+
} else {
|
|
74
|
+
reporter.blank();
|
|
75
|
+
reporter.text('Open this URL in your browser:');
|
|
76
|
+
reporter.text(chalk.cyan(serverHandle.startUrl));
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
reporter.blank();
|
|
80
|
+
reporter.text(`On the GitHub page, click ${chalk.cyan('"Create GitHub App"')} to continue.`);
|
|
81
|
+
reporter.blank();
|
|
82
|
+
reporter.text(chalk.dim('Waiting for GitHub callback... (Ctrl+C to cancel)'));
|
|
83
|
+
|
|
84
|
+
// Wait for callback
|
|
85
|
+
const { code } = await serverHandle.waitForCallback;
|
|
86
|
+
|
|
87
|
+
// Exchange code for credentials
|
|
88
|
+
reporter.blank();
|
|
89
|
+
reporter.step('Exchanging code for credentials...');
|
|
90
|
+
const credentials = await exchangeCodeForCredentials(code);
|
|
91
|
+
|
|
92
|
+
// Success!
|
|
93
|
+
reporter.blank();
|
|
94
|
+
reporter.success('GitHub App created!');
|
|
95
|
+
reporter.blank();
|
|
96
|
+
reporter.text(` App ID: ${chalk.cyan(credentials.id)}`);
|
|
97
|
+
reporter.text(` App Name: ${chalk.cyan(credentials.name)}`);
|
|
98
|
+
reporter.text(` App URL: ${chalk.cyan(credentials.htmlUrl)}`);
|
|
99
|
+
reporter.blank();
|
|
100
|
+
|
|
101
|
+
// Next steps - in correct order!
|
|
102
|
+
const githubRepoUrl = getGitHubRepoUrl(process.cwd());
|
|
103
|
+
reporter.bold('Next steps:');
|
|
104
|
+
reporter.blank();
|
|
105
|
+
|
|
106
|
+
// Step 1: Install the app (must happen first!)
|
|
107
|
+
reporter.text(` ${chalk.cyan('1.')} Install the app on your repository:`);
|
|
108
|
+
reporter.text(` ${chalk.cyan(credentials.htmlUrl + '/installations/new')}`);
|
|
109
|
+
reporter.blank();
|
|
110
|
+
|
|
111
|
+
// Step 2: Add secrets
|
|
112
|
+
reporter.text(` ${chalk.cyan('2.')} Add these secrets to your repository:`);
|
|
113
|
+
if (githubRepoUrl) {
|
|
114
|
+
reporter.text(` ${chalk.cyan(githubRepoUrl + '/settings/secrets/actions')}`);
|
|
115
|
+
}
|
|
116
|
+
reporter.blank();
|
|
117
|
+
reporter.text(` ${chalk.white('WARDEN_APP_ID')} ${credentials.id}`);
|
|
118
|
+
reporter.text(` ${chalk.white('WARDEN_PRIVATE_KEY')} ${chalk.dim('(copy the key below)')}`);
|
|
119
|
+
reporter.blank();
|
|
120
|
+
|
|
121
|
+
// Private key with clear instructions
|
|
122
|
+
reporter.text(` ${chalk.cyan('Private Key')} ${chalk.dim('(copy entire block including BEGIN/END lines):')}`);
|
|
123
|
+
reporter.blank();
|
|
124
|
+
// Indent the private key for readability
|
|
125
|
+
const pemLines = credentials.pem.trim().split('\n');
|
|
126
|
+
for (const line of pemLines) {
|
|
127
|
+
reporter.text(` ${chalk.dim(line)}`);
|
|
128
|
+
}
|
|
129
|
+
reporter.blank();
|
|
130
|
+
|
|
131
|
+
return 0;
|
|
132
|
+
} catch (error) {
|
|
133
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
134
|
+
reporter.error(message);
|
|
135
|
+
reporter.blank();
|
|
136
|
+
|
|
137
|
+
// Provide recovery guidance if the app might have been created
|
|
138
|
+
const githubRepoUrl = getGitHubRepoUrl(process.cwd());
|
|
139
|
+
reporter.text(chalk.dim('If the GitHub App was created before this error:'));
|
|
140
|
+
reporter.text(chalk.dim(' 1. Go to https://github.com/settings/apps' + (org ? ` (or your org's settings)` : '')));
|
|
141
|
+
reporter.text(chalk.dim(' 2. Find your app and click "Edit"'));
|
|
142
|
+
reporter.text(chalk.dim(' 3. Note the App ID from the "About" section'));
|
|
143
|
+
reporter.text(chalk.dim(' 4. Scroll to "Private keys" and click "Generate a private key"'));
|
|
144
|
+
reporter.text(chalk.dim(' 5. Install the app: click "Install App" in the sidebar'));
|
|
145
|
+
reporter.text(chalk.dim(' 6. Add secrets to your repository:'));
|
|
146
|
+
if (githubRepoUrl) {
|
|
147
|
+
reporter.text(chalk.dim(` ${githubRepoUrl}/settings/secrets/actions`));
|
|
148
|
+
}
|
|
149
|
+
reporter.text(chalk.dim(' - WARDEN_APP_ID: your App ID'));
|
|
150
|
+
reporter.text(chalk.dim(' - WARDEN_PRIVATE_KEY: contents of the downloaded .pem file'));
|
|
151
|
+
|
|
152
|
+
return 1;
|
|
153
|
+
} finally {
|
|
154
|
+
serverHandle.close();
|
|
155
|
+
}
|
|
156
|
+
}
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
import {
|
|
3
|
+
fetchRemote,
|
|
4
|
+
listCachedRemotes,
|
|
5
|
+
parseRemoteRef,
|
|
6
|
+
} from '../../skills/remote.js';
|
|
7
|
+
import type { Reporter } from '../output/reporter.js';
|
|
8
|
+
import type { CLIOptions } from '../args.js';
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Run the sync command.
|
|
12
|
+
* Updates cached remote skills to their latest versions.
|
|
13
|
+
* Pinned skills (with @sha) are skipped as they're immutable.
|
|
14
|
+
*/
|
|
15
|
+
export async function runSync(options: CLIOptions, reporter: Reporter): Promise<number> {
|
|
16
|
+
const cachedRemotes = listCachedRemotes();
|
|
17
|
+
|
|
18
|
+
if (cachedRemotes.length === 0) {
|
|
19
|
+
reporter.warning('No remote skills cached.');
|
|
20
|
+
reporter.tip('Add remote skills with: warden add --remote owner/repo --skill name');
|
|
21
|
+
return 0;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// If a specific remote is provided, only sync that one
|
|
25
|
+
const targetRepo = options.remote;
|
|
26
|
+
const remotesToSync = targetRepo
|
|
27
|
+
? cachedRemotes.filter(({ ref }) => ref === targetRepo || ref.startsWith(`${targetRepo}@`))
|
|
28
|
+
: cachedRemotes;
|
|
29
|
+
|
|
30
|
+
if (targetRepo && remotesToSync.length === 0) {
|
|
31
|
+
reporter.error(`Remote not found in cache: ${targetRepo}`);
|
|
32
|
+
reporter.blank();
|
|
33
|
+
reporter.text('Cached remotes:');
|
|
34
|
+
for (const { ref } of cachedRemotes) {
|
|
35
|
+
reporter.text(` - ${ref}`);
|
|
36
|
+
}
|
|
37
|
+
return 1;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// Separate pinned and unpinned remotes
|
|
41
|
+
const unpinnedRemotes = remotesToSync.filter(({ ref }) => {
|
|
42
|
+
const parsed = parseRemoteRef(ref);
|
|
43
|
+
return !parsed.sha;
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
const pinnedRemotes = remotesToSync.filter(({ ref }) => {
|
|
47
|
+
const parsed = parseRemoteRef(ref);
|
|
48
|
+
return !!parsed.sha;
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
if (unpinnedRemotes.length === 0) {
|
|
52
|
+
if (pinnedRemotes.length > 0) {
|
|
53
|
+
reporter.warning('All cached remotes are pinned to specific versions.');
|
|
54
|
+
reporter.text('Pinned remotes do not need syncing as they are immutable.');
|
|
55
|
+
for (const { ref } of pinnedRemotes) {
|
|
56
|
+
reporter.text(` - ${ref} ${chalk.dim('(pinned)')}`);
|
|
57
|
+
}
|
|
58
|
+
} else {
|
|
59
|
+
reporter.warning('No remotes to sync.');
|
|
60
|
+
}
|
|
61
|
+
return 0;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
reporter.bold('SYNC REMOTE SKILLS');
|
|
65
|
+
reporter.blank();
|
|
66
|
+
|
|
67
|
+
let updated = 0;
|
|
68
|
+
let errors = 0;
|
|
69
|
+
|
|
70
|
+
for (const { ref, entry } of unpinnedRemotes) {
|
|
71
|
+
reporter.step(`Syncing ${ref}...`);
|
|
72
|
+
|
|
73
|
+
const previousSha = entry.sha;
|
|
74
|
+
|
|
75
|
+
try {
|
|
76
|
+
const newSha = await fetchRemote(ref, { force: true });
|
|
77
|
+
|
|
78
|
+
if (newSha !== previousSha) {
|
|
79
|
+
reporter.success(`${ref}: updated ${chalk.dim(previousSha.slice(0, 7))} -> ${chalk.dim(newSha.slice(0, 7))}`);
|
|
80
|
+
updated++;
|
|
81
|
+
} else {
|
|
82
|
+
reporter.text(`${ref}: ${chalk.dim('already up to date')}`);
|
|
83
|
+
}
|
|
84
|
+
} catch (err) {
|
|
85
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
86
|
+
reporter.error(`${ref}: ${message}`);
|
|
87
|
+
errors++;
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
// Show pinned remotes that were skipped
|
|
92
|
+
if (pinnedRemotes.length > 0 && !targetRepo) {
|
|
93
|
+
reporter.blank();
|
|
94
|
+
reporter.text(chalk.dim('Skipped pinned remotes:'));
|
|
95
|
+
for (const { ref } of pinnedRemotes) {
|
|
96
|
+
reporter.text(chalk.dim(` - ${ref}`));
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// Summary
|
|
101
|
+
reporter.blank();
|
|
102
|
+
if (errors > 0) {
|
|
103
|
+
reporter.warning(`Synced with ${errors} error${errors === 1 ? '' : 's'}.`);
|
|
104
|
+
return 1;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
if (updated > 0) {
|
|
108
|
+
reporter.success(`Updated ${updated} remote${updated === 1 ? '' : 's'}.`);
|
|
109
|
+
} else {
|
|
110
|
+
reporter.success('All remotes up to date.');
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
return 0;
|
|
114
|
+
}
|