TeleportError.tsx
components/TeleportError.tsx
189
Lines
18824
Bytes
3
Exports
9
Imports
10
Keywords
What this is
This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.
Beginner explanation
This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.
How it is used
Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.
Expert explanation
Architecturally, this file intersects with ui-flow. It contains 189 lines, 9 detected imports, and 3 detected exports.
Important relationships
Detected exports
TeleportLocalErrorTypeTeleportErrorgetTeleportErrors
Keywords
checkerrorselsereactneedsloginerrorstoignoreoncanceltextteleportlocalerrortypeneedsgitstashoncomplete
Detected imports
react/compiler-runtimereactsrc/utils/background/remote/preconditions.jssrc/utils/gracefulShutdown.js../ink.js./ConsoleOAuthFlow.js./CustomSelect/index.js./design-system/Dialog.js./TeleportStash.js
Source notes
This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.
Full source
import { c as _c } from "react/compiler-runtime";
import React, { useCallback, useEffect, useState } from 'react';
import { checkIsGitClean, checkNeedsClaudeAiLogin } from 'src/utils/background/remote/preconditions.js';
import { gracefulShutdownSync } from 'src/utils/gracefulShutdown.js';
import { Box, Text } from '../ink.js';
import { ConsoleOAuthFlow } from './ConsoleOAuthFlow.js';
import { Select } from './CustomSelect/index.js';
import { Dialog } from './design-system/Dialog.js';
import { TeleportStash } from './TeleportStash.js';
export type TeleportLocalErrorType = 'needsLogin' | 'needsGitStash';
type TeleportErrorProps = {
onComplete: () => void;
errorsToIgnore?: ReadonlySet<TeleportLocalErrorType>;
};
// Module-level sentinel so the default parameter has stable identity.
// Previously `= new Set()` created a fresh Set every render, which put
// a new object in checkErrors' deps and caused the mount effect to
// re-fire on every render.
const EMPTY_ERRORS_TO_IGNORE: ReadonlySet<TeleportLocalErrorType> = new Set();
export function TeleportError(t0) {
const $ = _c(18);
const {
onComplete,
errorsToIgnore: t1
} = t0;
const errorsToIgnore = t1 === undefined ? EMPTY_ERRORS_TO_IGNORE : t1;
const [currentError, setCurrentError] = useState(null);
const [isLoggingIn, setIsLoggingIn] = useState(false);
let t2;
if ($[0] !== errorsToIgnore || $[1] !== onComplete) {
t2 = async () => {
const currentErrors = await getTeleportErrors();
const filteredErrors = new Set(Array.from(currentErrors).filter(error => !errorsToIgnore.has(error)));
if (filteredErrors.size === 0) {
onComplete();
return;
}
if (filteredErrors.has("needsLogin")) {
setCurrentError("needsLogin");
} else {
if (filteredErrors.has("needsGitStash")) {
setCurrentError("needsGitStash");
}
}
};
$[0] = errorsToIgnore;
$[1] = onComplete;
$[2] = t2;
} else {
t2 = $[2];
}
const checkErrors = t2;
let t3;
let t4;
if ($[3] !== checkErrors) {
t3 = () => {
checkErrors();
};
t4 = [checkErrors];
$[3] = checkErrors;
$[4] = t3;
$[5] = t4;
} else {
t3 = $[4];
t4 = $[5];
}
useEffect(t3, t4);
const onCancel = _temp;
let t5;
if ($[6] !== checkErrors) {
t5 = () => {
setIsLoggingIn(false);
checkErrors();
};
$[6] = checkErrors;
$[7] = t5;
} else {
t5 = $[7];
}
const handleLoginComplete = t5;
let t6;
if ($[8] === Symbol.for("react.memo_cache_sentinel")) {
t6 = () => {
setIsLoggingIn(true);
};
$[8] = t6;
} else {
t6 = $[8];
}
const handleLoginWithClaudeAI = t6;
let t7;
if ($[9] === Symbol.for("react.memo_cache_sentinel")) {
t7 = value => {
if (value === "login") {
handleLoginWithClaudeAI();
} else {
onCancel();
}
};
$[9] = t7;
} else {
t7 = $[9];
}
const handleLoginDialogSelect = t7;
let t8;
if ($[10] !== checkErrors) {
t8 = () => {
checkErrors();
};
$[10] = checkErrors;
$[11] = t8;
} else {
t8 = $[11];
}
const handleStashComplete = t8;
if (!currentError) {
return null;
}
switch (currentError) {
case "needsGitStash":
{
let t9;
if ($[12] !== handleStashComplete) {
t9 = <TeleportStash onStashAndContinue={handleStashComplete} onCancel={onCancel} />;
$[12] = handleStashComplete;
$[13] = t9;
} else {
t9 = $[13];
}
return t9;
}
case "needsLogin":
{
if (isLoggingIn) {
let t9;
if ($[14] !== handleLoginComplete) {
t9 = <ConsoleOAuthFlow onDone={handleLoginComplete} mode="login" forceLoginMethod="claudeai" />;
$[14] = handleLoginComplete;
$[15] = t9;
} else {
t9 = $[15];
}
return t9;
}
let t9;
if ($[16] === Symbol.for("react.memo_cache_sentinel")) {
t9 = <Box flexDirection="column"><Text dimColor={true}>Teleport requires a Claude.ai account.</Text><Text dimColor={true}>Your Claude Pro/Max subscription will be used by Claude Code.</Text></Box>;
$[16] = t9;
} else {
t9 = $[16];
}
let t10;
if ($[17] === Symbol.for("react.memo_cache_sentinel")) {
t10 = <Dialog title="Log in to Claude" onCancel={onCancel}>{t9}<Select options={[{
label: "Login with Claude account",
value: "login"
}, {
label: "Exit",
value: "exit"
}]} onChange={handleLoginDialogSelect} /></Dialog>;
$[17] = t10;
} else {
t10 = $[17];
}
return t10;
}
}
}
/**
* Gets current teleport errors that need to be resolved
* @returns Set of teleport error types that need to be handled
*/
function _temp() {
gracefulShutdownSync(0);
}
export async function getTeleportErrors(): Promise<Set<TeleportLocalErrorType>> {
const errors = new Set<TeleportLocalErrorType>();
const [needsLogin, isGitClean] = await Promise.all([checkNeedsClaudeAiLogin(), checkIsGitClean()]);
if (needsLogin) {
errors.add('needsLogin');
}
if (!isGitClean) {
errors.add('needsGitStash');
}
return errors;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useCallback","useEffect","useState","checkIsGitClean","checkNeedsClaudeAiLogin","gracefulShutdownSync","Box","Text","ConsoleOAuthFlow","Select","Dialog","TeleportStash","TeleportLocalErrorType","TeleportErrorProps","onComplete","errorsToIgnore","ReadonlySet","EMPTY_ERRORS_TO_IGNORE","Set","TeleportError","t0","$","_c","t1","undefined","currentError","setCurrentError","isLoggingIn","setIsLoggingIn","t2","currentErrors","getTeleportErrors","filteredErrors","Array","from","filter","error","has","size","checkErrors","t3","t4","onCancel","_temp","t5","handleLoginComplete","t6","Symbol","for","handleLoginWithClaudeAI","t7","value","handleLoginDialogSelect","t8","handleStashComplete","t9","t10","label","Promise","errors","needsLogin","isGitClean","all","add"],"sources":["TeleportError.tsx"],"sourcesContent":["import React, { useCallback, useEffect, useState } from 'react'\nimport {\n  checkIsGitClean,\n  checkNeedsClaudeAiLogin,\n} from 'src/utils/background/remote/preconditions.js'\nimport { gracefulShutdownSync } from 'src/utils/gracefulShutdown.js'\nimport { Box, Text } from '../ink.js'\nimport { ConsoleOAuthFlow } from './ConsoleOAuthFlow.js'\nimport { Select } from './CustomSelect/index.js'\nimport { Dialog } from './design-system/Dialog.js'\nimport { TeleportStash } from './TeleportStash.js'\n\nexport type TeleportLocalErrorType = 'needsLogin' | 'needsGitStash'\n\ntype TeleportErrorProps = {\n  onComplete: () => void\n  errorsToIgnore?: ReadonlySet<TeleportLocalErrorType>\n}\n\n// Module-level sentinel so the default parameter has stable identity.\n// Previously `= new Set()` created a fresh Set every render, which put\n// a new object in checkErrors' deps and caused the mount effect to\n// re-fire on every render.\nconst EMPTY_ERRORS_TO_IGNORE: ReadonlySet<TeleportLocalErrorType> = new Set()\n\nexport function TeleportError({\n  onComplete,\n  errorsToIgnore = EMPTY_ERRORS_TO_IGNORE,\n}: TeleportErrorProps): React.ReactNode {\n  const [currentError, setCurrentError] =\n    useState<TeleportLocalErrorType | null>(null)\n  const [isLoggingIn, setIsLoggingIn] = useState<boolean>(false)\n\n  // Check for errors on mount and when error resolution occurs\n  const checkErrors = useCallback(async () => {\n    const currentErrors = await getTeleportErrors()\n    const filteredErrors = new Set(\n      Array.from(currentErrors).filter(\n        (error: TeleportLocalErrorType) => !errorsToIgnore.has(error),\n      ),\n    )\n\n    // If no errors remain, call onComplete\n    if (filteredErrors.size === 0) {\n      onComplete()\n      return\n    }\n\n    // Set current error to handle (prioritize login over git)\n    if (filteredErrors.has('needsLogin')) {\n      setCurrentError('needsLogin')\n    } else if (filteredErrors.has('needsGitStash')) {\n      setCurrentError('needsGitStash')\n    }\n  }, [onComplete, errorsToIgnore])\n\n  // Check errors on mount\n  useEffect(() => {\n    void checkErrors()\n  }, [checkErrors])\n\n  const onCancel = useCallback(() => {\n    gracefulShutdownSync(0)\n  }, [])\n\n  const handleLoginComplete = useCallback(() => {\n    setIsLoggingIn(false)\n    void checkErrors()\n  }, [checkErrors])\n\n  const handleLoginWithClaudeAI = useCallback(() => {\n    setIsLoggingIn(true)\n  }, [setIsLoggingIn])\n\n  const handleLoginDialogSelect = useCallback(\n    (value: string) => {\n      if (value === 'login') {\n        handleLoginWithClaudeAI()\n      } else {\n        // User selected exit\n        onCancel()\n      }\n    },\n    [handleLoginWithClaudeAI, onCancel],\n  )\n\n  const handleStashComplete = useCallback(() => {\n    void checkErrors()\n  }, [checkErrors])\n\n  // Don't render anything if no current error (onComplete will be called)\n  if (!currentError) {\n    return null\n  }\n\n  switch (currentError) {\n    case 'needsGitStash':\n      return (\n        <TeleportStash\n          onStashAndContinue={handleStashComplete}\n          onCancel={onCancel}\n        />\n      )\n\n    case 'needsLogin': {\n      if (isLoggingIn) {\n        return (\n          <ConsoleOAuthFlow\n            onDone={handleLoginComplete}\n            mode=\"login\"\n            forceLoginMethod=\"claudeai\"\n          />\n        )\n      }\n\n      return (\n        <Dialog title=\"Log in to Claude\" onCancel={onCancel}>\n          <Box flexDirection=\"column\">\n            <Text dimColor>Teleport requires a Claude.ai account.</Text>\n            <Text dimColor>\n              Your Claude Pro/Max subscription will be used by Claude Code.\n            </Text>\n          </Box>\n          <Select\n            options={[\n              { label: 'Login with Claude account', value: 'login' },\n              { label: 'Exit', value: 'exit' },\n            ]}\n            onChange={handleLoginDialogSelect}\n          />\n        </Dialog>\n      )\n    }\n  }\n}\n\n/**\n * Gets current teleport errors that need to be resolved\n * @returns Set of teleport error types that need to be handled\n */\nexport async function getTeleportErrors(): Promise<\n  Set<TeleportLocalErrorType>\n> {\n  const errors = new Set<TeleportLocalErrorType>()\n\n  const [needsLogin, isGitClean] = await Promise.all([\n    checkNeedsClaudeAiLogin(),\n    checkIsGitClean(),\n  ])\n\n  if (needsLogin) {\n    errors.add('needsLogin')\n  }\n  if (!isGitClean) {\n    errors.add('needsGitStash')\n  }\n\n  return errors\n}\n"],"mappings":"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","ignoreList":[]}