π File detail
components/TeleportError.tsx
π― Use case
This file lives under βcomponents/β, which covers shared React UI pieces. On the API surface it exposes TeleportLocalErrorType, TeleportError, and getTeleportErrors β mainly types, interfaces, or factory objects. Dependencies touch React UI and src. It composes internal code from ink, ConsoleOAuthFlow, CustomSelect, design-system, and TeleportStash (relative imports).
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
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';
π€ Exports (heuristic)
TeleportLocalErrorTypeTeleportErrorgetTeleportErrors
π External import roots
Package roots from from "β¦" (relative paths omitted).
reactsrc
π₯οΈ Source preview
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":";AAAA,OAAOA,KAAK,IAAIC,WAAW,EAAEC,SAAS,EAAEC,QAAQ,QAAQ,OAAO;AAC/D,SACEC,eAAe,EACfC,uBAAuB,QAClB,8CAA8C;AACrD,SAASC,oBAAoB,QAAQ,+BAA+B;AACpE,SAASC,GAAG,EAAEC,IAAI,QAAQ,WAAW;AACrC,SAASC,gBAAgB,QAAQ,uBAAuB;AACxD,SAASC,MAAM,QAAQ,yBAAyB;AAChD,SAASC,MAAM,QAAQ,2BAA2B;AAClD,SAASC,aAAa,QAAQ,oBAAoB;AAElD,OAAO,KAAKC,sBAAsB,GAAG,YAAY,GAAG,eAAe;AAEnE,KAAKC,kBAAkB,GAAG;EACxBC,UAAU,EAAE,GAAG,GAAG,IAAI;EACtBC,cAAc,CAAC,EAAEC,WAAW,CAACJ,sBAAsB,CAAC;AACtD,CAAC;;AAED;AACA;AACA;AACA;AACA,MAAMK,sBAAsB,EAAED,WAAW,CAACJ,sBAAsB,CAAC,GAAG,IAAIM,GAAG,CAAC,CAAC;AAE7E,OAAO,SAAAC,cAAAC,EAAA;EAAA,MAAAC,CAAA,GAAAC,EAAA;EAAuB;IAAAR,UAAA;IAAAC,cAAA,EAAAQ;EAAA,IAAAH,EAGT;EADnB,MAAAL,cAAA,GAAAQ,EAAuC,KAAvCC,SAAuC,GAAvCP,sBAAuC,GAAvCM,EAAuC;EAEvC,OAAAE,YAAA,EAAAC,eAAA,IACExB,QAAQ,CAAgC,IAAI,CAAC;EAC/C,OAAAyB,WAAA,EAAAC,cAAA,IAAsC1B,QAAQ,CAAU,KAAK,CAAC;EAAA,IAAA2B,EAAA;EAAA,IAAAR,CAAA,QAAAN,cAAA,IAAAM,CAAA,QAAAP,UAAA;IAG9Be,EAAA,SAAAA,CAAA;MAC9B,MAAAC,aAAA,GAAsB,MAAMC,iBAAiB,CAAC,CAAC;MAC/C,MAAAC,cAAA,GAAuB,IAAId,GAAG,CAC5Be,KAAK,CAAAC,IAAK,CAACJ,aAAa,CAAC,CAAAK,MAAO,CAC9BC,KAAA,IAAmC,CAACrB,cAAc,CAAAsB,GAAI,CAACD,KAAK,CAC9D,CACF,CAAC;MAGD,IAAIJ,cAAc,CAAAM,IAAK,KAAK,CAAC;QAC3BxB,UAAU,CAAC,CAAC;QAAA;MAAA;MAKd,IAAIkB,cAAc,CAAAK,GAAI,CAAC,YAAY,CAAC;QAClCX,eAAe,CAAC,YAAY,CAAC;MAAA;QACxB,IAAIM,cAAc,CAAAK,GAAI,CAAC,eAAe,CAAC;UAC5CX,eAAe,CAAC,eAAe,CAAC;QAAA;MACjC;IAAA,CACF;IAAAL,CAAA,MAAAN,cAAA;IAAAM,CAAA,MAAAP,UAAA;IAAAO,CAAA,MAAAQ,EAAA;EAAA;IAAAA,EAAA,GAAAR,CAAA;EAAA;EApBD,MAAAkB,WAAA,GAAoBV,EAoBY;EAAA,IAAAW,EAAA;EAAA,IAAAC,EAAA;EAAA,IAAApB,CAAA,QAAAkB,WAAA;IAGtBC,EAAA,GAAAA,CAAA;MACHD,WAAW,CAAC,CAAC;IAAA,CACnB;IAAEE,EAAA,IAACF,WAAW,CAAC;IAAAlB,CAAA,MAAAkB,WAAA;IAAAlB,CAAA,MAAAmB,EAAA;IAAAnB,CAAA,MAAAoB,EAAA;EAAA;IAAAD,EAAA,GAAAnB,CAAA;IAAAoB,EAAA,GAAApB,CAAA;EAAA;EAFhBpB,SAAS,CAACuC,EAET,EAAEC,EAAa,CAAC;EAEjB,MAAAC,QAAA,GAAiBC,KAEX;EAAA,IAAAC,EAAA;EAAA,IAAAvB,CAAA,QAAAkB,WAAA;IAEkCK,EAAA,GAAAA,CAAA;MACtChB,cAAc,CAAC,KAAK,CAAC;MAChBW,WAAW,CAAC,CAAC;IAAA,CACnB;IAAAlB,CAAA,MAAAkB,WAAA;IAAAlB,CAAA,MAAAuB,EAAA;EAAA;IAAAA,EAAA,GAAAvB,CAAA;EAAA;EAHD,MAAAwB,mBAAA,GAA4BD,EAGX;EAAA,IAAAE,EAAA;EAAA,IAAAzB,CAAA,QAAA0B,MAAA,CAAAC,GAAA;IAE2BF,EAAA,GAAAA,CAAA;MAC1ClB,cAAc,CAAC,IAAI,CAAC;IAAA,CACrB;IAAAP,CAAA,MAAAyB,EAAA;EAAA;IAAAA,EAAA,GAAAzB,CAAA;EAAA;EAFD,MAAA4B,uBAAA,GAAgCH,EAEZ;EAAA,IAAAI,EAAA;EAAA,IAAA7B,CAAA,QAAA0B,MAAA,CAAAC,GAAA;IAGlBE,EAAA,GAAAC,KAAA;MACE,IAAIA,KAAK,KAAK,OAAO;QACnBF,uBAAuB,CAAC,CAAC;MAAA;QAGzBP,QAAQ,CAAC,CAAC;MAAA;IACX,CACF;IAAArB,CAAA,MAAA6B,EAAA;EAAA;IAAAA,EAAA,GAAA7B,CAAA;EAAA;EARH,MAAA+B,uBAAA,GAAgCF,EAU/B;EAAA,IAAAG,EAAA;EAAA,IAAAhC,CAAA,SAAAkB,WAAA;IAEuCc,EAAA,GAAAA,CAAA;MACjCd,WAAW,CAAC,CAAC;IAAA,CACnB;IAAAlB,CAAA,OAAAkB,WAAA;IAAAlB,CAAA,OAAAgC,EAAA;EAAA;IAAAA,EAAA,GAAAhC,CAAA;EAAA;EAFD,MAAAiC,mBAAA,GAA4BD,EAEX;EAGjB,IAAI,CAAC5B,YAAY;IAAA,OACR,IAAI;EAAA;EAGb,QAAQA,YAAY;IAAA,KACb,eAAe;MAAA;QAAA,IAAA8B,EAAA;QAAA,IAAAlC,CAAA,SAAAiC,mBAAA;UAEhBC,EAAA,IAAC,aAAa,CACQD,kBAAmB,CAAnBA,oBAAkB,CAAC,CAC7BZ,QAAQ,CAARA,SAAO,CAAC,GAClB;UAAArB,CAAA,OAAAiC,mBAAA;UAAAjC,CAAA,OAAAkC,EAAA;QAAA;UAAAA,EAAA,GAAAlC,CAAA;QAAA;QAAA,OAHFkC,EAGE;MAAA;IAAA,KAGD,YAAY;MAAA;QACf,IAAI5B,WAAW;UAAA,IAAA4B,EAAA;UAAA,IAAAlC,CAAA,SAAAwB,mBAAA;YAEXU,EAAA,IAAC,gBAAgB,CACPV,MAAmB,CAAnBA,oBAAkB,CAAC,CACtB,IAAO,CAAP,OAAO,CACK,gBAAU,CAAV,UAAU,GAC3B;YAAAxB,CAAA,OAAAwB,mBAAA;YAAAxB,CAAA,OAAAkC,EAAA;UAAA;YAAAA,EAAA,GAAAlC,CAAA;UAAA;UAAA,OAJFkC,EAIE;QAAA;QAEL,IAAAA,EAAA;QAAA,IAAAlC,CAAA,SAAA0B,MAAA,CAAAC,GAAA;UAIGO,EAAA,IAAC,GAAG,CAAe,aAAQ,CAAR,QAAQ,CACzB,CAAC,IAAI,CAAC,QAAQ,CAAR,KAAO,CAAC,CAAC,sCAAsC,EAApD,IAAI,CACL,CAAC,IAAI,CAAC,QAAQ,CAAR,KAAO,CAAC,CAAC,6DAEf,EAFC,IAAI,CAGP,EALC,GAAG,CAKE;UAAAlC,CAAA,OAAAkC,EAAA;QAAA;UAAAA,EAAA,GAAAlC,CAAA;QAAA;QAAA,IAAAmC,GAAA;QAAA,IAAAnC,CAAA,SAAA0B,MAAA,CAAAC,GAAA;UANRQ,GAAA,IAAC,MAAM,CAAO,KAAkB,CAAlB,kBAAkB,CAAWd,QAAQ,CAARA,SAAO,CAAC,CACjD,CAAAa,EAKK,CACL,CAAC,MAAM,CACI,OAGR,CAHQ,EACP;cAAAE,KAAA,EAAS,2BAA2B;cAAAN,KAAA,EAAS;YAAQ,CAAC,EACtD;cAAAM,KAAA,EAAS,MAAM;cAAAN,KAAA,EAAS;YAAO,CAAC,CAClC,CAAC,CACSC,QAAuB,CAAvBA,wBAAsB,CAAC,GAErC,EAdC,MAAM,CAcE;UAAA/B,CAAA,OAAAmC,GAAA;QAAA;UAAAA,GAAA,GAAAnC,CAAA;QAAA;QAAA,OAdTmC,GAcS;MAAA;EAGf;AAAC;;AAGH;AACA;AACA;AACA;AAlHO,SAAAb,MAAA;EAqCHtC,oBAAoB,CAAC,CAAC,CAAC;AAAA;AA8E3B,OAAO,eAAe0B,iBAAiBA,CAAA,CAAE,EAAE2B,OAAO,CAChDxC,GAAG,CAACN,sBAAsB,CAAC,CAC5B,CAAC;EACA,MAAM+C,MAAM,GAAG,IAAIzC,GAAG,CAACN,sBAAsB,CAAC,CAAC,CAAC;EAEhD,MAAM,CAACgD,UAAU,EAAEC,UAAU,CAAC,GAAG,MAAMH,OAAO,CAACI,GAAG,CAAC,CACjD1D,uBAAuB,CAAC,CAAC,EACzBD,eAAe,CAAC,CAAC,CAClB,CAAC;EAEF,IAAIyD,UAAU,EAAE;IACdD,MAAM,CAACI,GAAG,CAAC,YAAY,CAAC;EAC1B;EACA,IAAI,CAACF,UAAU,EAAE;IACfF,MAAM,CAACI,GAAG,CAAC,eAAe,CAAC;EAC7B;EAEA,OAAOJ,MAAM;AACf","ignoreList":[]}