source dump of claude code
0
fork

Configure Feed

Select the types of activity you want to include in your feed.

at main 587 lines 87 kB view raw
1import { execa } from 'execa'; 2import React, { useCallback, useState } from 'react'; 3import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; 4import { WorkflowMultiselectDialog } from '../../components/WorkflowMultiselectDialog.js'; 5import { GITHUB_ACTION_SETUP_DOCS_URL } from '../../constants/github-app.js'; 6import { useExitOnCtrlCDWithKeybindings } from '../../hooks/useExitOnCtrlCDWithKeybindings.js'; 7import type { KeyboardEvent } from '../../ink/events/keyboard-event.js'; 8import { Box } from '../../ink.js'; 9import type { LocalJSXCommandOnDone } from '../../types/command.js'; 10import { getAnthropicApiKey, isAnthropicAuthEnabled } from '../../utils/auth.js'; 11import { openBrowser } from '../../utils/browser.js'; 12import { execFileNoThrow } from '../../utils/execFileNoThrow.js'; 13import { getGithubRepo } from '../../utils/git.js'; 14import { plural } from '../../utils/stringUtils.js'; 15import { ApiKeyStep } from './ApiKeyStep.js'; 16import { CheckExistingSecretStep } from './CheckExistingSecretStep.js'; 17import { CheckGitHubStep } from './CheckGitHubStep.js'; 18import { ChooseRepoStep } from './ChooseRepoStep.js'; 19import { CreatingStep } from './CreatingStep.js'; 20import { ErrorStep } from './ErrorStep.js'; 21import { ExistingWorkflowStep } from './ExistingWorkflowStep.js'; 22import { InstallAppStep } from './InstallAppStep.js'; 23import { OAuthFlowStep } from './OAuthFlowStep.js'; 24import { SuccessStep } from './SuccessStep.js'; 25import { setupGitHubActions } from './setupGitHubActions.js'; 26import type { State, Warning, Workflow } from './types.js'; 27import { WarningsStep } from './WarningsStep.js'; 28const INITIAL_STATE: State = { 29 step: 'check-gh', 30 selectedRepoName: '', 31 currentRepo: '', 32 useCurrentRepo: false, 33 // Default to false, will be set to true if repo detected 34 apiKeyOrOAuthToken: '', 35 useExistingKey: true, 36 currentWorkflowInstallStep: 0, 37 warnings: [], 38 secretExists: false, 39 secretName: 'ANTHROPIC_API_KEY', 40 useExistingSecret: true, 41 workflowExists: false, 42 selectedWorkflows: ['claude', 'claude-review'] as Workflow[], 43 selectedApiKeyOption: 'new' as 'existing' | 'new' | 'oauth', 44 authType: 'api_key' 45}; 46function InstallGitHubApp(props: { 47 onDone: (message: string) => void; 48}): React.ReactNode { 49 const [existingApiKey] = useState(() => getAnthropicApiKey()); 50 const [state, setState] = useState({ 51 ...INITIAL_STATE, 52 useExistingKey: !!existingApiKey, 53 selectedApiKeyOption: (existingApiKey ? 'existing' : isAnthropicAuthEnabled() ? 'oauth' : 'new') as 'existing' | 'new' | 'oauth' 54 }); 55 useExitOnCtrlCDWithKeybindings(); 56 React.useEffect(() => { 57 logEvent('tengu_install_github_app_started', {}); 58 }, []); 59 const checkGitHubCLI = useCallback(async () => { 60 const warnings: Warning[] = []; 61 62 // Check if gh is installed 63 const ghVersionResult = await execa('gh --version', { 64 shell: true, 65 reject: false 66 }); 67 if (ghVersionResult.exitCode !== 0) { 68 warnings.push({ 69 title: 'GitHub CLI not found', 70 message: 'GitHub CLI (gh) does not appear to be installed or accessible.', 71 instructions: ['Install GitHub CLI from https://cli.github.com/', 'macOS: brew install gh', 'Windows: winget install --id GitHub.cli', 'Linux: See installation instructions at https://github.com/cli/cli#installation'] 72 }); 73 } 74 75 // Check auth status 76 const authResult = await execa('gh auth status -a', { 77 shell: true, 78 reject: false 79 }); 80 if (authResult.exitCode !== 0) { 81 warnings.push({ 82 title: 'GitHub CLI not authenticated', 83 message: 'GitHub CLI does not appear to be authenticated.', 84 instructions: ['Run: gh auth login', 'Follow the prompts to authenticate with GitHub', 'Or set up authentication using environment variables or other methods'] 85 }); 86 } else { 87 // Check if required scopes are present in the Token scopes line 88 const tokenScopesMatch = authResult.stdout.match(/Token scopes:.*$/m); 89 if (tokenScopesMatch) { 90 const scopes = tokenScopesMatch[0]; 91 const missingScopes: string[] = []; 92 if (!scopes.includes('repo')) { 93 missingScopes.push('repo'); 94 } 95 if (!scopes.includes('workflow')) { 96 missingScopes.push('workflow'); 97 } 98 if (missingScopes.length > 0) { 99 // Missing required scopes - exit immediately 100 setState(prev => ({ 101 ...prev, 102 step: 'error', 103 error: `GitHub CLI is missing required permissions: ${missingScopes.join(', ')}.`, 104 errorReason: 'Missing required scopes', 105 errorInstructions: [`Your GitHub CLI authentication is missing the "${missingScopes.join('" and "')}" ${plural(missingScopes.length, 'scope')} needed to manage GitHub Actions and secrets.`, '', 'To fix this, run:', ' gh auth refresh -h github.com -s repo,workflow', '', 'This will add the necessary permissions to manage workflows and secrets.'] 106 })); 107 return; 108 } 109 } 110 } 111 112 // Check if in a git repo and get remote URL 113 const currentRepo = (await getGithubRepo()) ?? ''; 114 logEvent('tengu_install_github_app_step_completed', { 115 step: 'check-gh' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 116 }); 117 setState(prev_0 => ({ 118 ...prev_0, 119 warnings, 120 currentRepo, 121 selectedRepoName: currentRepo, 122 useCurrentRepo: !!currentRepo, 123 // Set to false if no repo detected 124 step: warnings.length > 0 ? 'warnings' : 'choose-repo' 125 })); 126 }, []); 127 React.useEffect(() => { 128 if (state.step === 'check-gh') { 129 void checkGitHubCLI(); 130 } 131 }, [state.step, checkGitHubCLI]); 132 const runSetupGitHubActions = useCallback(async (apiKeyOrOAuthToken: string | null, secretName: string) => { 133 setState(prev_1 => ({ 134 ...prev_1, 135 step: 'creating', 136 currentWorkflowInstallStep: 0 137 })); 138 try { 139 await setupGitHubActions(state.selectedRepoName, apiKeyOrOAuthToken, secretName, () => { 140 setState(prev_4 => ({ 141 ...prev_4, 142 currentWorkflowInstallStep: prev_4.currentWorkflowInstallStep + 1 143 })); 144 }, state.workflowAction === 'skip', state.selectedWorkflows, state.authType, { 145 useCurrentRepo: state.useCurrentRepo, 146 workflowExists: state.workflowExists, 147 secretExists: state.secretExists 148 }); 149 logEvent('tengu_install_github_app_step_completed', { 150 step: 'creating' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 151 }); 152 setState(prev_5 => ({ 153 ...prev_5, 154 step: 'success' 155 })); 156 } catch (error) { 157 const errorMessage = error instanceof Error ? error.message : 'Failed to set up GitHub Actions'; 158 if (errorMessage.includes('workflow file already exists')) { 159 logEvent('tengu_install_github_app_error', { 160 reason: 'workflow_file_exists' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 161 }); 162 setState(prev_2 => ({ 163 ...prev_2, 164 step: 'error', 165 error: 'A Claude workflow file already exists in this repository.', 166 errorReason: 'Workflow file conflict', 167 errorInstructions: ['The file .github/workflows/claude.yml already exists', 'You can either:', ' 1. Delete the existing file and run this command again', ' 2. Update the existing file manually using the template from:', ` ${GITHUB_ACTION_SETUP_DOCS_URL}`] 168 })); 169 } else { 170 logEvent('tengu_install_github_app_error', { 171 reason: 'setup_github_actions_failed' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 172 }); 173 setState(prev_3 => ({ 174 ...prev_3, 175 step: 'error', 176 error: errorMessage, 177 errorReason: 'GitHub Actions setup failed', 178 errorInstructions: [] 179 })); 180 } 181 } 182 }, [state.selectedRepoName, state.workflowAction, state.selectedWorkflows, state.useCurrentRepo, state.workflowExists, state.secretExists, state.authType]); 183 async function openGitHubAppInstallation() { 184 const installUrl = 'https://github.com/apps/claude'; 185 await openBrowser(installUrl); 186 } 187 async function checkRepositoryPermissions(repoName: string): Promise<{ 188 hasAccess: boolean; 189 error?: string; 190 }> { 191 try { 192 const result = await execFileNoThrow('gh', ['api', `repos/${repoName}`, '--jq', '.permissions.admin']); 193 if (result.code === 0) { 194 const hasAdmin = result.stdout.trim() === 'true'; 195 return { 196 hasAccess: hasAdmin 197 }; 198 } 199 if (result.stderr.includes('404') || result.stderr.includes('Not Found')) { 200 return { 201 hasAccess: false, 202 error: 'repository_not_found' 203 }; 204 } 205 return { 206 hasAccess: false 207 }; 208 } catch { 209 return { 210 hasAccess: false 211 }; 212 } 213 } 214 async function checkExistingWorkflowFile(repoName_0: string): Promise<boolean> { 215 const checkFileResult = await execFileNoThrow('gh', ['api', `repos/${repoName_0}/contents/.github/workflows/claude.yml`, '--jq', '.sha']); 216 return checkFileResult.code === 0; 217 } 218 async function checkExistingSecret() { 219 const checkSecretsResult = await execFileNoThrow('gh', ['secret', 'list', '--app', 'actions', '--repo', state.selectedRepoName]); 220 if (checkSecretsResult.code === 0) { 221 const lines = checkSecretsResult.stdout.split('\n'); 222 const hasAnthropicKey = lines.some((line: string) => { 223 return /^ANTHROPIC_API_KEY\s+/.test(line); 224 }); 225 if (hasAnthropicKey) { 226 setState(prev_6 => ({ 227 ...prev_6, 228 secretExists: true, 229 step: 'check-existing-secret' 230 })); 231 } else { 232 // No existing secret found 233 if (existingApiKey) { 234 // User has local key, skip to creating with it 235 setState(prev_7 => ({ 236 ...prev_7, 237 apiKeyOrOAuthToken: existingApiKey, 238 useExistingKey: true 239 })); 240 await runSetupGitHubActions(existingApiKey, state.secretName); 241 } else { 242 // No local key, go to API key step 243 setState(prev_8 => ({ 244 ...prev_8, 245 step: 'api-key' 246 })); 247 } 248 } 249 } else { 250 // Error checking secrets 251 if (existingApiKey) { 252 // User has local key, skip to creating with it 253 setState(prev_9 => ({ 254 ...prev_9, 255 apiKeyOrOAuthToken: existingApiKey, 256 useExistingKey: true 257 })); 258 await runSetupGitHubActions(existingApiKey, state.secretName); 259 } else { 260 // No local key, go to API key step 261 setState(prev_10 => ({ 262 ...prev_10, 263 step: 'api-key' 264 })); 265 } 266 } 267 } 268 const handleSubmit = async () => { 269 if (state.step === 'warnings') { 270 logEvent('tengu_install_github_app_step_completed', { 271 step: 'warnings' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 272 }); 273 setState(prev_11 => ({ 274 ...prev_11, 275 step: 'install-app' 276 })); 277 setTimeout(openGitHubAppInstallation, 0); 278 } else if (state.step === 'choose-repo') { 279 let repoName_1 = state.useCurrentRepo ? state.currentRepo : state.selectedRepoName; 280 if (!repoName_1.trim()) { 281 return; 282 } 283 const repoWarnings: Warning[] = []; 284 if (repoName_1.includes('github.com')) { 285 const match = repoName_1.match(/github\.com[:/]([^/]+\/[^/]+)(\.git)?$/); 286 if (!match) { 287 repoWarnings.push({ 288 title: 'Invalid GitHub URL format', 289 message: 'The repository URL format appears to be invalid.', 290 instructions: ['Use format: owner/repo or https://github.com/owner/repo', 'Example: anthropics/claude-cli'] 291 }); 292 } else { 293 repoName_1 = match[1]?.replace(/\.git$/, '') || ''; 294 } 295 } 296 if (!repoName_1.includes('/')) { 297 repoWarnings.push({ 298 title: 'Repository format warning', 299 message: 'Repository should be in format "owner/repo"', 300 instructions: ['Use format: owner/repo', 'Example: anthropics/claude-cli'] 301 }); 302 } 303 const permissionCheck = await checkRepositoryPermissions(repoName_1); 304 if (permissionCheck.error === 'repository_not_found') { 305 repoWarnings.push({ 306 title: 'Repository not found', 307 message: `Repository ${repoName_1} was not found or you don't have access.`, 308 instructions: [`Check that the repository name is correct: ${repoName_1}`, 'Ensure you have access to this repository', 'For private repositories, make sure your GitHub token has the "repo" scope', 'You can add the repo scope with: gh auth refresh -h github.com -s repo,workflow'] 309 }); 310 } else if (!permissionCheck.hasAccess) { 311 repoWarnings.push({ 312 title: 'Admin permissions required', 313 message: `You might need admin permissions on ${repoName_1} to set up GitHub Actions.`, 314 instructions: ['Repository admins can install GitHub Apps and set secrets', 'Ask a repository admin to run this command if setup fails', 'Alternatively, you can use the manual setup instructions'] 315 }); 316 } 317 const workflowExists = await checkExistingWorkflowFile(repoName_1); 318 if (repoWarnings.length > 0) { 319 const allWarnings = [...state.warnings, ...repoWarnings]; 320 setState(prev_12 => ({ 321 ...prev_12, 322 selectedRepoName: repoName_1, 323 workflowExists, 324 warnings: allWarnings, 325 step: 'warnings' 326 })); 327 } else { 328 logEvent('tengu_install_github_app_step_completed', { 329 step: 'choose-repo' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 330 }); 331 setState(prev_13 => ({ 332 ...prev_13, 333 selectedRepoName: repoName_1, 334 workflowExists, 335 step: 'install-app' 336 })); 337 setTimeout(openGitHubAppInstallation, 0); 338 } 339 } else if (state.step === 'install-app') { 340 logEvent('tengu_install_github_app_step_completed', { 341 step: 'install-app' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 342 }); 343 if (state.workflowExists) { 344 setState(prev_14 => ({ 345 ...prev_14, 346 step: 'check-existing-workflow' 347 })); 348 } else { 349 setState(prev_15 => ({ 350 ...prev_15, 351 step: 'select-workflows' 352 })); 353 } 354 } else if (state.step === 'check-existing-workflow') { 355 return; 356 } else if (state.step === 'select-workflows') { 357 // Handled by the WorkflowMultiselectDialog component 358 return; 359 } else if (state.step === 'check-existing-secret') { 360 logEvent('tengu_install_github_app_step_completed', { 361 step: 'check-existing-secret' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 362 }); 363 if (state.useExistingSecret) { 364 await runSetupGitHubActions(null, state.secretName); 365 } else { 366 // User wants to use a new secret name with their API key 367 await runSetupGitHubActions(state.apiKeyOrOAuthToken, state.secretName); 368 } 369 } else if (state.step === 'api-key') { 370 // In the new flow, api-key step only appears when user has no existing key 371 // They either entered a new key or will create OAuth token 372 if (state.selectedApiKeyOption === 'oauth') { 373 // OAuth flow already handled by handleCreateOAuthToken 374 return; 375 } 376 377 // If user selected 'existing' option, use the existing API key 378 const apiKeyToUse = state.selectedApiKeyOption === 'existing' ? existingApiKey : state.apiKeyOrOAuthToken; 379 if (!apiKeyToUse) { 380 logEvent('tengu_install_github_app_error', { 381 reason: 'api_key_missing' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 382 }); 383 setState(prev_16 => ({ 384 ...prev_16, 385 step: 'error', 386 error: 'API key is required' 387 })); 388 return; 389 } 390 391 // Store the API key being used (either existing or newly entered) 392 setState(prev_17 => ({ 393 ...prev_17, 394 apiKeyOrOAuthToken: apiKeyToUse, 395 useExistingKey: state.selectedApiKeyOption === 'existing' 396 })); 397 398 // Check if ANTHROPIC_API_KEY secret already exists 399 const checkSecretsResult_0 = await execFileNoThrow('gh', ['secret', 'list', '--app', 'actions', '--repo', state.selectedRepoName]); 400 if (checkSecretsResult_0.code === 0) { 401 const lines_0 = checkSecretsResult_0.stdout.split('\n'); 402 const hasAnthropicKey_0 = lines_0.some((line_0: string) => { 403 return /^ANTHROPIC_API_KEY\s+/.test(line_0); 404 }); 405 if (hasAnthropicKey_0) { 406 logEvent('tengu_install_github_app_step_completed', { 407 step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 408 }); 409 setState(prev_18 => ({ 410 ...prev_18, 411 secretExists: true, 412 step: 'check-existing-secret' 413 })); 414 } else { 415 logEvent('tengu_install_github_app_step_completed', { 416 step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 417 }); 418 // No existing secret, proceed to creating 419 await runSetupGitHubActions(apiKeyToUse, state.secretName); 420 } 421 } else { 422 logEvent('tengu_install_github_app_step_completed', { 423 step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 424 }); 425 // Error checking secrets, proceed anyway 426 await runSetupGitHubActions(apiKeyToUse, state.secretName); 427 } 428 } 429 }; 430 const handleRepoUrlChange = (value: string) => { 431 setState(prev_19 => ({ 432 ...prev_19, 433 selectedRepoName: value 434 })); 435 }; 436 const handleApiKeyChange = (value_0: string) => { 437 setState(prev_20 => ({ 438 ...prev_20, 439 apiKeyOrOAuthToken: value_0 440 })); 441 }; 442 const handleApiKeyOptionChange = (option: 'existing' | 'new' | 'oauth') => { 443 setState(prev_21 => ({ 444 ...prev_21, 445 selectedApiKeyOption: option 446 })); 447 }; 448 const handleCreateOAuthToken = useCallback(() => { 449 logEvent('tengu_install_github_app_step_completed', { 450 step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 451 }); 452 setState(prev_22 => ({ 453 ...prev_22, 454 step: 'oauth-flow' 455 })); 456 }, []); 457 const handleOAuthSuccess = useCallback((token: string) => { 458 logEvent('tengu_install_github_app_step_completed', { 459 step: 'oauth-flow' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 460 }); 461 setState(prev_23 => ({ 462 ...prev_23, 463 apiKeyOrOAuthToken: token, 464 useExistingKey: false, 465 secretName: 'CLAUDE_CODE_OAUTH_TOKEN', 466 authType: 'oauth_token' 467 })); 468 void runSetupGitHubActions(token, 'CLAUDE_CODE_OAUTH_TOKEN'); 469 }, [runSetupGitHubActions]); 470 const handleOAuthCancel = useCallback(() => { 471 setState(prev_24 => ({ 472 ...prev_24, 473 step: 'api-key' 474 })); 475 }, []); 476 const handleSecretNameChange = (value_1: string) => { 477 if (value_1 && !/^[a-zA-Z0-9_]+$/.test(value_1)) return; 478 setState(prev_25 => ({ 479 ...prev_25, 480 secretName: value_1 481 })); 482 }; 483 const handleToggleUseCurrentRepo = (useCurrentRepo: boolean) => { 484 setState(prev_26 => ({ 485 ...prev_26, 486 useCurrentRepo, 487 selectedRepoName: useCurrentRepo ? prev_26.currentRepo : '' 488 })); 489 }; 490 const handleToggleUseExistingKey = (useExistingKey: boolean) => { 491 setState(prev_27 => ({ 492 ...prev_27, 493 useExistingKey 494 })); 495 }; 496 const handleToggleUseExistingSecret = (useExistingSecret: boolean) => { 497 setState(prev_28 => ({ 498 ...prev_28, 499 useExistingSecret, 500 secretName: useExistingSecret ? 'ANTHROPIC_API_KEY' : '' 501 })); 502 }; 503 const handleWorkflowAction = async (action: 'update' | 'skip' | 'exit') => { 504 if (action === 'exit') { 505 props.onDone('Installation cancelled by user'); 506 return; 507 } 508 logEvent('tengu_install_github_app_step_completed', { 509 step: 'check-existing-workflow' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 510 }); 511 setState(prev_29 => ({ 512 ...prev_29, 513 workflowAction: action 514 })); 515 if (action === 'skip' || action === 'update') { 516 // Check if user has existing local API key 517 if (existingApiKey) { 518 await checkExistingSecret(); 519 } else { 520 // No local key, go straight to API key step 521 setState(prev_30 => ({ 522 ...prev_30, 523 step: 'api-key' 524 })); 525 } 526 } 527 }; 528 function handleDismissKeyDown(e: KeyboardEvent): void { 529 e.preventDefault(); 530 if (state.step === 'success') { 531 logEvent('tengu_install_github_app_completed', {}); 532 } 533 props.onDone(state.step === 'success' ? 'GitHub Actions setup complete!' : state.error ? `Couldn't install GitHub App: ${state.error}\nFor manual setup instructions, see: ${GITHUB_ACTION_SETUP_DOCS_URL}` : `GitHub App installation failed\nFor manual setup instructions, see: ${GITHUB_ACTION_SETUP_DOCS_URL}`); 534 } 535 switch (state.step) { 536 case 'check-gh': 537 return <CheckGitHubStep />; 538 case 'warnings': 539 return <WarningsStep warnings={state.warnings} onContinue={handleSubmit} />; 540 case 'choose-repo': 541 return <ChooseRepoStep currentRepo={state.currentRepo} useCurrentRepo={state.useCurrentRepo} repoUrl={state.selectedRepoName} onRepoUrlChange={handleRepoUrlChange} onToggleUseCurrentRepo={handleToggleUseCurrentRepo} onSubmit={handleSubmit} />; 542 case 'install-app': 543 return <InstallAppStep repoUrl={state.selectedRepoName} onSubmit={handleSubmit} />; 544 case 'check-existing-workflow': 545 return <ExistingWorkflowStep repoName={state.selectedRepoName} onSelectAction={handleWorkflowAction} />; 546 case 'check-existing-secret': 547 return <CheckExistingSecretStep useExistingSecret={state.useExistingSecret} secretName={state.secretName} onToggleUseExistingSecret={handleToggleUseExistingSecret} onSecretNameChange={handleSecretNameChange} onSubmit={handleSubmit} />; 548 case 'api-key': 549 return <ApiKeyStep existingApiKey={existingApiKey} useExistingKey={state.useExistingKey} apiKeyOrOAuthToken={state.apiKeyOrOAuthToken} onApiKeyChange={handleApiKeyChange} onToggleUseExistingKey={handleToggleUseExistingKey} onSubmit={handleSubmit} onCreateOAuthToken={isAnthropicAuthEnabled() ? handleCreateOAuthToken : undefined} selectedOption={state.selectedApiKeyOption} onSelectOption={handleApiKeyOptionChange} />; 550 case 'creating': 551 return <CreatingStep currentWorkflowInstallStep={state.currentWorkflowInstallStep} secretExists={state.secretExists} useExistingSecret={state.useExistingSecret} secretName={state.secretName} skipWorkflow={state.workflowAction === 'skip'} selectedWorkflows={state.selectedWorkflows} />; 552 case 'success': 553 return <Box tabIndex={0} autoFocus onKeyDown={handleDismissKeyDown}> 554 <SuccessStep secretExists={state.secretExists} useExistingSecret={state.useExistingSecret} secretName={state.secretName} skipWorkflow={state.workflowAction === 'skip'} /> 555 </Box>; 556 case 'error': 557 return <Box tabIndex={0} autoFocus onKeyDown={handleDismissKeyDown}> 558 <ErrorStep error={state.error} errorReason={state.errorReason} errorInstructions={state.errorInstructions} /> 559 </Box>; 560 case 'select-workflows': 561 return <WorkflowMultiselectDialog defaultSelections={state.selectedWorkflows} onSubmit={selectedWorkflows => { 562 logEvent('tengu_install_github_app_step_completed', { 563 step: 'select-workflows' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS 564 }); 565 setState(prev_31 => ({ 566 ...prev_31, 567 selectedWorkflows 568 })); 569 // Check if user has existing local API key 570 if (existingApiKey) { 571 void checkExistingSecret(); 572 } else { 573 // No local key, go straight to API key step 574 setState(prev_32 => ({ 575 ...prev_32, 576 step: 'api-key' 577 })); 578 } 579 }} />; 580 case 'oauth-flow': 581 return <OAuthFlowStep onSuccess={handleOAuthSuccess} onCancel={handleOAuthCancel} />; 582 } 583} 584export async function call(onDone: LocalJSXCommandOnDone): Promise<React.ReactNode> { 585 return <InstallGitHubApp onDone={onDone} />; 586} 587//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["execa","React","useCallback","useState","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","WorkflowMultiselectDialog","GITHUB_ACTION_SETUP_DOCS_URL","useExitOnCtrlCDWithKeybindings","KeyboardEvent","Box","LocalJSXCommandOnDone","getAnthropicApiKey","isAnthropicAuthEnabled","openBrowser","execFileNoThrow","getGithubRepo","plural","ApiKeyStep","CheckExistingSecretStep","CheckGitHubStep","ChooseRepoStep","CreatingStep","ErrorStep","ExistingWorkflowStep","InstallAppStep","OAuthFlowStep","SuccessStep","setupGitHubActions","State","Warning","Workflow","WarningsStep","INITIAL_STATE","step","selectedRepoName","currentRepo","useCurrentRepo","apiKeyOrOAuthToken","useExistingKey","currentWorkflowInstallStep","warnings","secretExists","secretName","useExistingSecret","workflowExists","selectedWorkflows","selectedApiKeyOption","authType","InstallGitHubApp","props","onDone","message","ReactNode","existingApiKey","state","setState","useEffect","checkGitHubCLI","ghVersionResult","shell","reject","exitCode","push","title","instructions","authResult","tokenScopesMatch","stdout","match","scopes","missingScopes","includes","length","prev","error","join","errorReason","errorInstructions","runSetupGitHubActions","workflowAction","errorMessage","Error","reason","openGitHubAppInstallation","installUrl","checkRepositoryPermissions","repoName","Promise","hasAccess","result","code","hasAdmin","trim","stderr","checkExistingWorkflowFile","checkFileResult","checkExistingSecret","checkSecretsResult","lines","split","hasAnthropicKey","some","line","test","handleSubmit","setTimeout","repoWarnings","replace","permissionCheck","allWarnings","apiKeyToUse","handleRepoUrlChange","value","handleApiKeyChange","handleApiKeyOptionChange","option","handleCreateOAuthToken","handleOAuthSuccess","token","handleOAuthCancel","handleSecretNameChange","handleToggleUseCurrentRepo","handleToggleUseExistingKey","handleToggleUseExistingSecret","handleWorkflowAction","action","handleDismissKeyDown","e","preventDefault","undefined","call"],"sources":["install-github-app.tsx"],"sourcesContent":["import { execa } from 'execa'\nimport React, { useCallback, useState } from 'react'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from 'src/services/analytics/index.js'\nimport { WorkflowMultiselectDialog } from '../../components/WorkflowMultiselectDialog.js'\nimport { GITHUB_ACTION_SETUP_DOCS_URL } from '../../constants/github-app.js'\nimport { useExitOnCtrlCDWithKeybindings } from '../../hooks/useExitOnCtrlCDWithKeybindings.js'\nimport type { KeyboardEvent } from '../../ink/events/keyboard-event.js'\nimport { Box } from '../../ink.js'\nimport type { LocalJSXCommandOnDone } from '../../types/command.js'\nimport { getAnthropicApiKey, isAnthropicAuthEnabled } from '../../utils/auth.js'\nimport { openBrowser } from '../../utils/browser.js'\nimport { execFileNoThrow } from '../../utils/execFileNoThrow.js'\nimport { getGithubRepo } from '../../utils/git.js'\nimport { plural } from '../../utils/stringUtils.js'\nimport { ApiKeyStep } from './ApiKeyStep.js'\nimport { CheckExistingSecretStep } from './CheckExistingSecretStep.js'\nimport { CheckGitHubStep } from './CheckGitHubStep.js'\nimport { ChooseRepoStep } from './ChooseRepoStep.js'\nimport { CreatingStep } from './CreatingStep.js'\nimport { ErrorStep } from './ErrorStep.js'\nimport { ExistingWorkflowStep } from './ExistingWorkflowStep.js'\nimport { InstallAppStep } from './InstallAppStep.js'\nimport { OAuthFlowStep } from './OAuthFlowStep.js'\nimport { SuccessStep } from './SuccessStep.js'\nimport { setupGitHubActions } from './setupGitHubActions.js'\nimport type { State, Warning, Workflow } from './types.js'\nimport { WarningsStep } from './WarningsStep.js'\n\nconst INITIAL_STATE: State = {\n  step: 'check-gh',\n  selectedRepoName: '',\n  currentRepo: '',\n  useCurrentRepo: false, // Default to false, will be set to true if repo detected\n  apiKeyOrOAuthToken: '',\n  useExistingKey: true,\n  currentWorkflowInstallStep: 0,\n  warnings: [],\n  secretExists: false,\n  secretName: 'ANTHROPIC_API_KEY',\n  useExistingSecret: true,\n  workflowExists: false,\n  selectedWorkflows: ['claude', 'claude-review'] as Workflow[],\n  selectedApiKeyOption: 'new' as 'existing' | 'new' | 'oauth',\n  authType: 'api_key',\n}\n\nfunction InstallGitHubApp(props: {\n  onDone: (message: string) => void\n}): React.ReactNode {\n  const [existingApiKey] = useState(() => getAnthropicApiKey())\n  const [state, setState] = useState({\n    ...INITIAL_STATE,\n    useExistingKey: !!existingApiKey,\n    selectedApiKeyOption: (existingApiKey\n      ? 'existing'\n      : isAnthropicAuthEnabled()\n        ? 'oauth'\n        : 'new') as 'existing' | 'new' | 'oauth',\n  })\n  useExitOnCtrlCDWithKeybindings()\n\n  React.useEffect(() => {\n    logEvent('tengu_install_github_app_started', {})\n  }, [])\n\n  const checkGitHubCLI = useCallback(async () => {\n    const warnings: Warning[] = []\n\n    // Check if gh is installed\n    const ghVersionResult = await execa('gh --version', {\n      shell: true,\n      reject: false,\n    })\n    if (ghVersionResult.exitCode !== 0) {\n      warnings.push({\n        title: 'GitHub CLI not found',\n        message:\n          'GitHub CLI (gh) does not appear to be installed or accessible.',\n        instructions: [\n          'Install GitHub CLI from https://cli.github.com/',\n          'macOS: brew install gh',\n          'Windows: winget install --id GitHub.cli',\n          'Linux: See installation instructions at https://github.com/cli/cli#installation',\n        ],\n      })\n    }\n\n    // Check auth status\n    const authResult = await execa('gh auth status -a', {\n      shell: true,\n      reject: false,\n    })\n    if (authResult.exitCode !== 0) {\n      warnings.push({\n        title: 'GitHub CLI not authenticated',\n        message: 'GitHub CLI does not appear to be authenticated.',\n        instructions: [\n          'Run: gh auth login',\n          'Follow the prompts to authenticate with GitHub',\n          'Or set up authentication using environment variables or other methods',\n        ],\n      })\n    } else {\n      // Check if required scopes are present in the Token scopes line\n      const tokenScopesMatch = authResult.stdout.match(/Token scopes:.*$/m)\n      if (tokenScopesMatch) {\n        const scopes = tokenScopesMatch[0]\n        const missingScopes: string[] = []\n\n        if (!scopes.includes('repo')) {\n          missingScopes.push('repo')\n        }\n        if (!scopes.includes('workflow')) {\n          missingScopes.push('workflow')\n        }\n\n        if (missingScopes.length > 0) {\n          // Missing required scopes - exit immediately\n          setState(prev => ({\n            ...prev,\n            step: 'error',\n            error: `GitHub CLI is missing required permissions: ${missingScopes.join(', ')}.`,\n            errorReason: 'Missing required scopes',\n            errorInstructions: [\n              `Your GitHub CLI authentication is missing the \"${missingScopes.join('\" and \"')}\" ${plural(missingScopes.length, 'scope')} needed to manage GitHub Actions and secrets.`,\n              '',\n              'To fix this, run:',\n              '  gh auth refresh -h github.com -s repo,workflow',\n              '',\n              'This will add the necessary permissions to manage workflows and secrets.',\n            ],\n          }))\n          return\n        }\n      }\n    }\n\n    // Check if in a git repo and get remote URL\n    const currentRepo = (await getGithubRepo()) ?? ''\n\n    logEvent('tengu_install_github_app_step_completed', {\n      step: 'check-gh' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n\n    setState(prev => ({\n      ...prev,\n      warnings,\n      currentRepo,\n      selectedRepoName: currentRepo,\n      useCurrentRepo: !!currentRepo, // Set to false if no repo detected\n      step: warnings.length > 0 ? 'warnings' : 'choose-repo',\n    }))\n  }, [])\n\n  React.useEffect(() => {\n    if (state.step === 'check-gh') {\n      void checkGitHubCLI()\n    }\n  }, [state.step, checkGitHubCLI])\n\n  const runSetupGitHubActions = useCallback(\n    async (apiKeyOrOAuthToken: string | null, secretName: string) => {\n      setState(prev => ({\n        ...prev,\n        step: 'creating',\n        currentWorkflowInstallStep: 0,\n      }))\n\n      try {\n        await setupGitHubActions(\n          state.selectedRepoName,\n          apiKeyOrOAuthToken,\n          secretName,\n          () => {\n            setState(prev => ({\n              ...prev,\n              currentWorkflowInstallStep: prev.currentWorkflowInstallStep + 1,\n            }))\n          },\n          state.workflowAction === 'skip',\n          state.selectedWorkflows,\n          state.authType,\n          {\n            useCurrentRepo: state.useCurrentRepo,\n            workflowExists: state.workflowExists,\n            secretExists: state.secretExists,\n          },\n        )\n        logEvent('tengu_install_github_app_step_completed', {\n          step: 'creating' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        setState(prev => ({ ...prev, step: 'success' }))\n      } catch (error) {\n        const errorMessage =\n          error instanceof Error\n            ? error.message\n            : 'Failed to set up GitHub Actions'\n\n        if (errorMessage.includes('workflow file already exists')) {\n          logEvent('tengu_install_github_app_error', {\n            reason:\n              'workflow_file_exists' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          })\n          setState(prev => ({\n            ...prev,\n            step: 'error',\n            error: 'A Claude workflow file already exists in this repository.',\n            errorReason: 'Workflow file conflict',\n            errorInstructions: [\n              'The file .github/workflows/claude.yml already exists',\n              'You can either:',\n              '  1. Delete the existing file and run this command again',\n              '  2. Update the existing file manually using the template from:',\n              `     ${GITHUB_ACTION_SETUP_DOCS_URL}`,\n            ],\n          }))\n        } else {\n          logEvent('tengu_install_github_app_error', {\n            reason:\n              'setup_github_actions_failed' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          })\n\n          setState(prev => ({\n            ...prev,\n            step: 'error',\n            error: errorMessage,\n            errorReason: 'GitHub Actions setup failed',\n            errorInstructions: [],\n          }))\n        }\n      }\n    },\n    [\n      state.selectedRepoName,\n      state.workflowAction,\n      state.selectedWorkflows,\n      state.useCurrentRepo,\n      state.workflowExists,\n      state.secretExists,\n      state.authType,\n    ],\n  )\n\n  async function openGitHubAppInstallation() {\n    const installUrl = 'https://github.com/apps/claude'\n    await openBrowser(installUrl)\n  }\n\n  async function checkRepositoryPermissions(\n    repoName: string,\n  ): Promise<{ hasAccess: boolean; error?: string }> {\n    try {\n      const result = await execFileNoThrow('gh', [\n        'api',\n        `repos/${repoName}`,\n        '--jq',\n        '.permissions.admin',\n      ])\n\n      if (result.code === 0) {\n        const hasAdmin = result.stdout.trim() === 'true'\n        return { hasAccess: hasAdmin }\n      }\n\n      if (\n        result.stderr.includes('404') ||\n        result.stderr.includes('Not Found')\n      ) {\n        return {\n          hasAccess: false,\n          error: 'repository_not_found',\n        }\n      }\n\n      return { hasAccess: false }\n    } catch {\n      return { hasAccess: false }\n    }\n  }\n\n  async function checkExistingWorkflowFile(repoName: string): Promise<boolean> {\n    const checkFileResult = await execFileNoThrow('gh', [\n      'api',\n      `repos/${repoName}/contents/.github/workflows/claude.yml`,\n      '--jq',\n      '.sha',\n    ])\n\n    return checkFileResult.code === 0\n  }\n\n  async function checkExistingSecret() {\n    const checkSecretsResult = await execFileNoThrow('gh', [\n      'secret',\n      'list',\n      '--app',\n      'actions',\n      '--repo',\n      state.selectedRepoName,\n    ])\n\n    if (checkSecretsResult.code === 0) {\n      const lines = checkSecretsResult.stdout.split('\\n')\n      const hasAnthropicKey = lines.some((line: string) => {\n        return /^ANTHROPIC_API_KEY\\s+/.test(line)\n      })\n\n      if (hasAnthropicKey) {\n        setState(prev => ({\n          ...prev,\n          secretExists: true,\n          step: 'check-existing-secret',\n        }))\n      } else {\n        // No existing secret found\n        if (existingApiKey) {\n          // User has local key, skip to creating with it\n          setState(prev => ({\n            ...prev,\n            apiKeyOrOAuthToken: existingApiKey,\n            useExistingKey: true,\n          }))\n          await runSetupGitHubActions(existingApiKey, state.secretName)\n        } else {\n          // No local key, go to API key step\n          setState(prev => ({ ...prev, step: 'api-key' }))\n        }\n      }\n    } else {\n      // Error checking secrets\n      if (existingApiKey) {\n        // User has local key, skip to creating with it\n        setState(prev => ({\n          ...prev,\n          apiKeyOrOAuthToken: existingApiKey,\n          useExistingKey: true,\n        }))\n        await runSetupGitHubActions(existingApiKey, state.secretName)\n      } else {\n        // No local key, go to API key step\n        setState(prev => ({ ...prev, step: 'api-key' }))\n      }\n    }\n  }\n\n  const handleSubmit = async () => {\n    if (state.step === 'warnings') {\n      logEvent('tengu_install_github_app_step_completed', {\n        step: 'warnings' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      setState(prev => ({ ...prev, step: 'install-app' }))\n      setTimeout(openGitHubAppInstallation, 0)\n    } else if (state.step === 'choose-repo') {\n      let repoName = state.useCurrentRepo\n        ? state.currentRepo\n        : state.selectedRepoName\n\n      if (!repoName.trim()) {\n        return\n      }\n\n      const repoWarnings: Warning[] = []\n\n      if (repoName.includes('github.com')) {\n        const match = repoName.match(/github\\.com[:/]([^/]+\\/[^/]+)(\\.git)?$/)\n        if (!match) {\n          repoWarnings.push({\n            title: 'Invalid GitHub URL format',\n            message: 'The repository URL format appears to be invalid.',\n            instructions: [\n              'Use format: owner/repo or https://github.com/owner/repo',\n              'Example: anthropics/claude-cli',\n            ],\n          })\n        } else {\n          repoName = match[1]?.replace(/\\.git$/, '') || ''\n        }\n      }\n\n      if (!repoName.includes('/')) {\n        repoWarnings.push({\n          title: 'Repository format warning',\n          message: 'Repository should be in format \"owner/repo\"',\n          instructions: [\n            'Use format: owner/repo',\n            'Example: anthropics/claude-cli',\n          ],\n        })\n      }\n\n      const permissionCheck = await checkRepositoryPermissions(repoName)\n\n      if (permissionCheck.error === 'repository_not_found') {\n        repoWarnings.push({\n          title: 'Repository not found',\n          message: `Repository ${repoName} was not found or you don't have access.`,\n          instructions: [\n            `Check that the repository name is correct: ${repoName}`,\n            'Ensure you have access to this repository',\n            'For private repositories, make sure your GitHub token has the \"repo\" scope',\n            'You can add the repo scope with: gh auth refresh -h github.com -s repo,workflow',\n          ],\n        })\n      } else if (!permissionCheck.hasAccess) {\n        repoWarnings.push({\n          title: 'Admin permissions required',\n          message: `You might need admin permissions on ${repoName} to set up GitHub Actions.`,\n          instructions: [\n            'Repository admins can install GitHub Apps and set secrets',\n            'Ask a repository admin to run this command if setup fails',\n            'Alternatively, you can use the manual setup instructions',\n          ],\n        })\n      }\n\n      const workflowExists = await checkExistingWorkflowFile(repoName)\n\n      if (repoWarnings.length > 0) {\n        const allWarnings = [...state.warnings, ...repoWarnings]\n        setState(prev => ({\n          ...prev,\n          selectedRepoName: repoName,\n          workflowExists,\n          warnings: allWarnings,\n          step: 'warnings',\n        }))\n      } else {\n        logEvent('tengu_install_github_app_step_completed', {\n          step: 'choose-repo' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        setState(prev => ({\n          ...prev,\n          selectedRepoName: repoName,\n          workflowExists,\n          step: 'install-app',\n        }))\n        setTimeout(openGitHubAppInstallation, 0)\n      }\n    } else if (state.step === 'install-app') {\n      logEvent('tengu_install_github_app_step_completed', {\n        step: 'install-app' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      if (state.workflowExists) {\n        setState(prev => ({ ...prev, step: 'check-existing-workflow' }))\n      } else {\n        setState(prev => ({ ...prev, step: 'select-workflows' }))\n      }\n    } else if (state.step === 'check-existing-workflow') {\n      return\n    } else if (state.step === 'select-workflows') {\n      // Handled by the WorkflowMultiselectDialog component\n      return\n    } else if (state.step === 'check-existing-secret') {\n      logEvent('tengu_install_github_app_step_completed', {\n        step: 'check-existing-secret' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      if (state.useExistingSecret) {\n        await runSetupGitHubActions(null, state.secretName)\n      } else {\n        // User wants to use a new secret name with their API key\n        await runSetupGitHubActions(state.apiKeyOrOAuthToken, state.secretName)\n      }\n    } else if (state.step === 'api-key') {\n      // In the new flow, api-key step only appears when user has no existing key\n      // They either entered a new key or will create OAuth token\n      if (state.selectedApiKeyOption === 'oauth') {\n        // OAuth flow already handled by handleCreateOAuthToken\n        return\n      }\n\n      // If user selected 'existing' option, use the existing API key\n      const apiKeyToUse =\n        state.selectedApiKeyOption === 'existing'\n          ? existingApiKey\n          : state.apiKeyOrOAuthToken\n\n      if (!apiKeyToUse) {\n        logEvent('tengu_install_github_app_error', {\n          reason:\n            'api_key_missing' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        setState(prev => ({\n          ...prev,\n          step: 'error',\n          error: 'API key is required',\n        }))\n        return\n      }\n\n      // Store the API key being used (either existing or newly entered)\n      setState(prev => ({\n        ...prev,\n        apiKeyOrOAuthToken: apiKeyToUse,\n        useExistingKey: state.selectedApiKeyOption === 'existing',\n      }))\n\n      // Check if ANTHROPIC_API_KEY secret already exists\n      const checkSecretsResult = await execFileNoThrow('gh', [\n        'secret',\n        'list',\n        '--app',\n        'actions',\n        '--repo',\n        state.selectedRepoName,\n      ])\n\n      if (checkSecretsResult.code === 0) {\n        const lines = checkSecretsResult.stdout.split('\\n')\n        const hasAnthropicKey = lines.some((line: string) => {\n          return /^ANTHROPIC_API_KEY\\s+/.test(line)\n        })\n\n        if (hasAnthropicKey) {\n          logEvent('tengu_install_github_app_step_completed', {\n            step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          })\n          setState(prev => ({\n            ...prev,\n            secretExists: true,\n            step: 'check-existing-secret',\n          }))\n        } else {\n          logEvent('tengu_install_github_app_step_completed', {\n            step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          })\n          // No existing secret, proceed to creating\n          await runSetupGitHubActions(apiKeyToUse, state.secretName)\n        }\n      } else {\n        logEvent('tengu_install_github_app_step_completed', {\n          step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        // Error checking secrets, proceed anyway\n        await runSetupGitHubActions(apiKeyToUse, state.secretName)\n      }\n    }\n  }\n\n  const handleRepoUrlChange = (value: string) => {\n    setState(prev => ({ ...prev, selectedRepoName: value }))\n  }\n\n  const handleApiKeyChange = (value: string) => {\n    setState(prev => ({ ...prev, apiKeyOrOAuthToken: value }))\n  }\n\n  const handleApiKeyOptionChange = (option: 'existing' | 'new' | 'oauth') => {\n    setState(prev => ({ ...prev, selectedApiKeyOption: option }))\n  }\n\n  const handleCreateOAuthToken = useCallback(() => {\n    logEvent('tengu_install_github_app_step_completed', {\n      step: 'api-key' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n    setState(prev => ({ ...prev, step: 'oauth-flow' }))\n  }, [])\n\n  const handleOAuthSuccess = useCallback(\n    (token: string) => {\n      logEvent('tengu_install_github_app_step_completed', {\n        step: 'oauth-flow' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      setState(prev => ({\n        ...prev,\n        apiKeyOrOAuthToken: token,\n        useExistingKey: false,\n        secretName: 'CLAUDE_CODE_OAUTH_TOKEN',\n        authType: 'oauth_token',\n      }))\n      void runSetupGitHubActions(token, 'CLAUDE_CODE_OAUTH_TOKEN')\n    },\n    [runSetupGitHubActions],\n  )\n\n  const handleOAuthCancel = useCallback(() => {\n    setState(prev => ({ ...prev, step: 'api-key' }))\n  }, [])\n\n  const handleSecretNameChange = (value: string) => {\n    if (value && !/^[a-zA-Z0-9_]+$/.test(value)) return\n    setState(prev => ({ ...prev, secretName: value }))\n  }\n\n  const handleToggleUseCurrentRepo = (useCurrentRepo: boolean) => {\n    setState(prev => ({\n      ...prev,\n      useCurrentRepo,\n      selectedRepoName: useCurrentRepo ? prev.currentRepo : '',\n    }))\n  }\n\n  const handleToggleUseExistingKey = (useExistingKey: boolean) => {\n    setState(prev => ({ ...prev, useExistingKey }))\n  }\n\n  const handleToggleUseExistingSecret = (useExistingSecret: boolean) => {\n    setState(prev => ({\n      ...prev,\n      useExistingSecret,\n      secretName: useExistingSecret ? 'ANTHROPIC_API_KEY' : '',\n    }))\n  }\n\n  const handleWorkflowAction = async (action: 'update' | 'skip' | 'exit') => {\n    if (action === 'exit') {\n      props.onDone('Installation cancelled by user')\n      return\n    }\n\n    logEvent('tengu_install_github_app_step_completed', {\n      step: 'check-existing-workflow' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n\n    setState(prev => ({ ...prev, workflowAction: action }))\n\n    if (action === 'skip' || action === 'update') {\n      // Check if user has existing local API key\n      if (existingApiKey) {\n        await checkExistingSecret()\n      } else {\n        // No local key, go straight to API key step\n        setState(prev => ({ ...prev, step: 'api-key' }))\n      }\n    }\n  }\n\n  function handleDismissKeyDown(e: KeyboardEvent): void {\n    e.preventDefault()\n    if (state.step === 'success') {\n      logEvent('tengu_install_github_app_completed', {})\n    }\n    props.onDone(\n      state.step === 'success'\n        ? 'GitHub Actions setup complete!'\n        : state.error\n          ? `Couldn't install GitHub App: ${state.error}\\nFor manual setup instructions, see: ${GITHUB_ACTION_SETUP_DOCS_URL}`\n          : `GitHub App installation failed\\nFor manual setup instructions, see: ${GITHUB_ACTION_SETUP_DOCS_URL}`,\n    )\n  }\n\n  switch (state.step) {\n    case 'check-gh':\n      return <CheckGitHubStep />\n    case 'warnings':\n      return (\n        <WarningsStep warnings={state.warnings} onContinue={handleSubmit} />\n      )\n    case 'choose-repo':\n      return (\n        <ChooseRepoStep\n          currentRepo={state.currentRepo}\n          useCurrentRepo={state.useCurrentRepo}\n          repoUrl={state.selectedRepoName}\n          onRepoUrlChange={handleRepoUrlChange}\n          onToggleUseCurrentRepo={handleToggleUseCurrentRepo}\n          onSubmit={handleSubmit}\n        />\n      )\n    case 'install-app':\n      return (\n        <InstallAppStep\n          repoUrl={state.selectedRepoName}\n          onSubmit={handleSubmit}\n        />\n      )\n    case 'check-existing-workflow':\n      return (\n        <ExistingWorkflowStep\n          repoName={state.selectedRepoName}\n          onSelectAction={handleWorkflowAction}\n        />\n      )\n    case 'check-existing-secret':\n      return (\n        <CheckExistingSecretStep\n          useExistingSecret={state.useExistingSecret}\n          secretName={state.secretName}\n          onToggleUseExistingSecret={handleToggleUseExistingSecret}\n          onSecretNameChange={handleSecretNameChange}\n          onSubmit={handleSubmit}\n        />\n      )\n    case 'api-key':\n      return (\n        <ApiKeyStep\n          existingApiKey={existingApiKey}\n          useExistingKey={state.useExistingKey}\n          apiKeyOrOAuthToken={state.apiKeyOrOAuthToken}\n          onApiKeyChange={handleApiKeyChange}\n          onToggleUseExistingKey={handleToggleUseExistingKey}\n          onSubmit={handleSubmit}\n          onCreateOAuthToken={\n            isAnthropicAuthEnabled() ? handleCreateOAuthToken : undefined\n          }\n          selectedOption={state.selectedApiKeyOption}\n          onSelectOption={handleApiKeyOptionChange}\n        />\n      )\n    case 'creating':\n      return (\n        <CreatingStep\n          currentWorkflowInstallStep={state.currentWorkflowInstallStep}\n          secretExists={state.secretExists}\n          useExistingSecret={state.useExistingSecret}\n          secretName={state.secretName}\n          skipWorkflow={state.workflowAction === 'skip'}\n          selectedWorkflows={state.selectedWorkflows}\n        />\n      )\n    case 'success':\n      return (\n        <Box tabIndex={0} autoFocus onKeyDown={handleDismissKeyDown}>\n          <SuccessStep\n            secretExists={state.secretExists}\n            useExistingSecret={state.useExistingSecret}\n            secretName={state.secretName}\n            skipWorkflow={state.workflowAction === 'skip'}\n          />\n        </Box>\n      )\n    case 'error':\n      return (\n        <Box tabIndex={0} autoFocus onKeyDown={handleDismissKeyDown}>\n          <ErrorStep\n            error={state.error}\n            errorReason={state.errorReason}\n            errorInstructions={state.errorInstructions}\n          />\n        </Box>\n      )\n    case 'select-workflows':\n      return (\n        <WorkflowMultiselectDialog\n          defaultSelections={state.selectedWorkflows}\n          onSubmit={selectedWorkflows => {\n            logEvent('tengu_install_github_app_step_completed', {\n              step: 'select-workflows' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n            })\n            setState(prev => ({\n              ...prev,\n              selectedWorkflows,\n            }))\n            // Check if user has existing local API key\n            if (existingApiKey) {\n              void checkExistingSecret()\n            } else {\n              // No local key, go straight to API key step\n              setState(prev => ({ ...prev, step: 'api-key' }))\n            }\n          }}\n        />\n      )\n    case 'oauth-flow':\n      return (\n        <OAuthFlowStep\n          onSuccess={handleOAuthSuccess}\n          onCancel={handleOAuthCancel}\n        />\n      )\n  }\n}\n\nexport async function call(\n  onDone: LocalJSXCommandOnDone,\n): Promise<React.ReactNode> {\n  return <InstallGitHubApp onDone={onDone} />\n}\n"],"mappings":"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","ignoreList":[]}