@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.
Files changed (199) hide show
  1. package/.agents/skills/find-bugs/SKILL.md +75 -0
  2. package/.agents/skills/vercel-react-best-practices/AGENTS.md +2934 -0
  3. package/.agents/skills/vercel-react-best-practices/SKILL.md +136 -0
  4. package/.agents/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
  5. package/.agents/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
  6. package/.agents/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
  7. package/.agents/skills/vercel-react-best-practices/rules/async-api-routes.md +38 -0
  8. package/.agents/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
  9. package/.agents/skills/vercel-react-best-practices/rules/async-dependencies.md +51 -0
  10. package/.agents/skills/vercel-react-best-practices/rules/async-parallel.md +28 -0
  11. package/.agents/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
  12. package/.agents/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
  13. package/.agents/skills/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
  14. package/.agents/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
  15. package/.agents/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
  16. package/.agents/skills/vercel-react-best-practices/rules/bundle-preload.md +50 -0
  17. package/.agents/skills/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
  18. package/.agents/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
  19. package/.agents/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
  20. package/.agents/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
  21. package/.agents/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
  22. package/.agents/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
  23. package/.agents/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
  24. package/.agents/skills/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
  25. package/.agents/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
  26. package/.agents/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
  27. package/.agents/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
  28. package/.agents/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
  29. package/.agents/skills/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
  30. package/.agents/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
  31. package/.agents/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
  32. package/.agents/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
  33. package/.agents/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
  34. package/.agents/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
  35. package/.agents/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
  36. package/.agents/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
  37. package/.agents/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
  38. package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
  39. package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
  40. package/.agents/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
  41. package/.agents/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
  42. package/.agents/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
  43. package/.agents/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
  44. package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
  45. package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
  46. package/.agents/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
  47. package/.agents/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
  48. package/.agents/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
  49. package/.agents/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
  50. package/.agents/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
  51. package/.agents/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
  52. package/.agents/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
  53. package/.agents/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
  54. package/.agents/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
  55. package/.agents/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
  56. package/.agents/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
  57. package/.agents/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
  58. package/.agents/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
  59. package/.agents/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
  60. package/.agents/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
  61. package/.claude/settings.json +57 -0
  62. package/.claude/settings.local.json +88 -0
  63. package/.claude/skills/agent-prompt/SKILL.md +54 -0
  64. package/.claude/skills/agent-prompt/references/agentic-patterns.md +94 -0
  65. package/.claude/skills/agent-prompt/references/anti-patterns.md +140 -0
  66. package/.claude/skills/agent-prompt/references/context-design.md +124 -0
  67. package/.claude/skills/agent-prompt/references/core-principles.md +75 -0
  68. package/.claude/skills/agent-prompt/references/model-guidance.md +118 -0
  69. package/.claude/skills/agent-prompt/references/output-formats.md +98 -0
  70. package/.claude/skills/agent-prompt/references/skill-structure.md +115 -0
  71. package/.claude/skills/agent-prompt/references/system-prompts.md +115 -0
  72. package/.claude/skills/notseer/SKILL.md +131 -0
  73. package/.claude/skills/skill-writer/SKILL.md +140 -0
  74. package/.claude/skills/testing-guidelines/SKILL.md +132 -0
  75. package/.claude/skills/warden-skill/SKILL.md +250 -0
  76. package/.claude/skills/warden-skill/references/config-schema.md +133 -0
  77. package/.dex/config.toml +2 -0
  78. package/.github/workflows/ci.yml +33 -0
  79. package/.github/workflows/release.yml +54 -0
  80. package/.github/workflows/warden.yml +40 -0
  81. package/AGENTS.md +89 -0
  82. package/CONTRIBUTING.md +60 -0
  83. package/LICENSE +105 -0
  84. package/README.md +43 -0
  85. package/SPEC.md +263 -0
  86. package/action.yml +87 -0
  87. package/assets/favicon.png +0 -0
  88. package/assets/warden-icon-bw.svg +5 -0
  89. package/assets/warden-icon-purple.png +0 -0
  90. package/assets/warden-icon-purple.svg +5 -0
  91. package/docs/.claude/settings.local.json +11 -0
  92. package/docs/astro.config.mjs +43 -0
  93. package/docs/package.json +19 -0
  94. package/docs/pnpm-lock.yaml +4000 -0
  95. package/docs/public/favicon.svg +5 -0
  96. package/docs/src/components/Code.astro +141 -0
  97. package/docs/src/components/PackageManagerTabs.astro +183 -0
  98. package/docs/src/components/Terminal.astro +212 -0
  99. package/docs/src/layouts/Base.astro +380 -0
  100. package/docs/src/pages/cli.astro +167 -0
  101. package/docs/src/pages/config.astro +394 -0
  102. package/docs/src/pages/guide.astro +449 -0
  103. package/docs/src/pages/index.astro +490 -0
  104. package/docs/src/styles/global.css +551 -0
  105. package/docs/tsconfig.json +3 -0
  106. package/docs/vercel.json +5 -0
  107. package/eslint.config.js +33 -0
  108. package/package.json +73 -0
  109. package/src/action/index.ts +1 -0
  110. package/src/action/main.ts +868 -0
  111. package/src/cli/args.test.ts +477 -0
  112. package/src/cli/args.ts +415 -0
  113. package/src/cli/commands/add.ts +447 -0
  114. package/src/cli/commands/init.test.ts +136 -0
  115. package/src/cli/commands/init.ts +132 -0
  116. package/src/cli/commands/setup-app/browser.ts +38 -0
  117. package/src/cli/commands/setup-app/credentials.ts +45 -0
  118. package/src/cli/commands/setup-app/manifest.ts +48 -0
  119. package/src/cli/commands/setup-app/server.ts +172 -0
  120. package/src/cli/commands/setup-app.ts +156 -0
  121. package/src/cli/commands/sync.ts +114 -0
  122. package/src/cli/context.ts +131 -0
  123. package/src/cli/files.test.ts +155 -0
  124. package/src/cli/files.ts +89 -0
  125. package/src/cli/fix.test.ts +310 -0
  126. package/src/cli/fix.ts +387 -0
  127. package/src/cli/git.test.ts +119 -0
  128. package/src/cli/git.ts +318 -0
  129. package/src/cli/index.ts +14 -0
  130. package/src/cli/main.ts +672 -0
  131. package/src/cli/output/box.ts +235 -0
  132. package/src/cli/output/formatters.test.ts +187 -0
  133. package/src/cli/output/formatters.ts +269 -0
  134. package/src/cli/output/icons.ts +13 -0
  135. package/src/cli/output/index.ts +44 -0
  136. package/src/cli/output/ink-runner.tsx +337 -0
  137. package/src/cli/output/jsonl.test.ts +347 -0
  138. package/src/cli/output/jsonl.ts +126 -0
  139. package/src/cli/output/reporter.ts +435 -0
  140. package/src/cli/output/tasks.ts +374 -0
  141. package/src/cli/output/tty.test.ts +117 -0
  142. package/src/cli/output/tty.ts +60 -0
  143. package/src/cli/output/verbosity.test.ts +40 -0
  144. package/src/cli/output/verbosity.ts +31 -0
  145. package/src/cli/terminal.test.ts +148 -0
  146. package/src/cli/terminal.ts +301 -0
  147. package/src/config/index.ts +3 -0
  148. package/src/config/loader.test.ts +313 -0
  149. package/src/config/loader.ts +103 -0
  150. package/src/config/schema.ts +168 -0
  151. package/src/config/writer.test.ts +119 -0
  152. package/src/config/writer.ts +84 -0
  153. package/src/diff/classify.test.ts +162 -0
  154. package/src/diff/classify.ts +92 -0
  155. package/src/diff/coalesce.test.ts +208 -0
  156. package/src/diff/coalesce.ts +133 -0
  157. package/src/diff/context.test.ts +226 -0
  158. package/src/diff/context.ts +201 -0
  159. package/src/diff/index.ts +4 -0
  160. package/src/diff/parser.test.ts +212 -0
  161. package/src/diff/parser.ts +149 -0
  162. package/src/event/context.ts +132 -0
  163. package/src/event/index.ts +2 -0
  164. package/src/event/schedule-context.ts +101 -0
  165. package/src/examples/examples.integration.test.ts +66 -0
  166. package/src/examples/index.test.ts +101 -0
  167. package/src/examples/index.ts +122 -0
  168. package/src/examples/setup.ts +25 -0
  169. package/src/index.ts +115 -0
  170. package/src/output/dedup.test.ts +419 -0
  171. package/src/output/dedup.ts +607 -0
  172. package/src/output/github-checks.test.ts +300 -0
  173. package/src/output/github-checks.ts +476 -0
  174. package/src/output/github-issues.ts +329 -0
  175. package/src/output/index.ts +5 -0
  176. package/src/output/issue-renderer.ts +197 -0
  177. package/src/output/renderer.test.ts +727 -0
  178. package/src/output/renderer.ts +217 -0
  179. package/src/output/stale.test.ts +375 -0
  180. package/src/output/stale.ts +155 -0
  181. package/src/output/types.ts +34 -0
  182. package/src/sdk/index.ts +1 -0
  183. package/src/sdk/runner.test.ts +806 -0
  184. package/src/sdk/runner.ts +1232 -0
  185. package/src/skills/index.ts +36 -0
  186. package/src/skills/loader.test.ts +300 -0
  187. package/src/skills/loader.ts +423 -0
  188. package/src/skills/remote.test.ts +704 -0
  189. package/src/skills/remote.ts +604 -0
  190. package/src/triggers/matcher.test.ts +277 -0
  191. package/src/triggers/matcher.ts +152 -0
  192. package/src/types/index.ts +194 -0
  193. package/src/utils/async.ts +18 -0
  194. package/src/utils/index.test.ts +84 -0
  195. package/src/utils/index.ts +50 -0
  196. package/tsconfig.json +25 -0
  197. package/vitest.config.ts +8 -0
  198. package/vitest.integration.config.ts +11 -0
  199. 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, '&#39;')}'>
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
+ }