/ utils / preflightChecks.tsx
preflightChecks.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import axios from 'axios';
  3  import React, { useEffect, useState } from 'react';
  4  import { logEvent } from 'src/services/analytics/index.js';
  5  import { Spinner } from '../components/Spinner.js';
  6  import { getOauthConfig } from '../constants/oauth.js';
  7  import { useTimeout } from '../hooks/useTimeout.js';
  8  import { Box, Text } from '../ink.js';
  9  import { getSSLErrorHint } from '../services/api/errorUtils.js';
 10  import { getUserAgent } from './http.js';
 11  import { logError } from './log.js';
 12  export interface PreflightCheckResult {
 13    success: boolean;
 14    error?: string;
 15    sslHint?: string;
 16  }
 17  async function checkEndpoints(): Promise<PreflightCheckResult> {
 18    try {
 19      const oauthConfig = getOauthConfig();
 20      const tokenUrl = new URL(oauthConfig.TOKEN_URL);
 21      const endpoints = [`${oauthConfig.BASE_API_URL}/api/hello`, `${tokenUrl.origin}/v1/oauth/hello`];
 22      const checkEndpoint = async (url: string): Promise<PreflightCheckResult> => {
 23        try {
 24          const response = await axios.get(url, {
 25            headers: {
 26              'User-Agent': getUserAgent()
 27            }
 28          });
 29          if (response.status !== 200) {
 30            const hostname = new URL(url).hostname;
 31            return {
 32              success: false,
 33              error: `Failed to connect to ${hostname}: Status ${response.status}`
 34            };
 35          }
 36          return {
 37            success: true
 38          };
 39        } catch (error) {
 40          const hostname = new URL(url).hostname;
 41          const sslHint = getSSLErrorHint(error);
 42          return {
 43            success: false,
 44            error: `Failed to connect to ${hostname}: ${error instanceof Error ? (error as ErrnoException).code || error.message : String(error)}`,
 45            sslHint: sslHint ?? undefined
 46          };
 47        }
 48      };
 49      const results = await Promise.all(endpoints.map(checkEndpoint));
 50      const failedResult = results.find(result => !result.success);
 51      if (failedResult) {
 52        // Log failure to Statsig
 53        logEvent('tengu_preflight_check_failed', {
 54          isConnectivityError: false,
 55          hasErrorMessage: !!failedResult.error,
 56          isSSLError: !!failedResult.sslHint
 57        });
 58      }
 59      return failedResult || {
 60        success: true
 61      };
 62    } catch (error) {
 63      logError(error as Error);
 64  
 65      // Log to Statsig
 66      logEvent('tengu_preflight_check_failed', {
 67        isConnectivityError: true
 68      });
 69      return {
 70        success: false,
 71        error: `Connectivity check error: ${error instanceof Error ? (error as ErrnoException).code || error.message : String(error)}`
 72      };
 73    }
 74  }
 75  interface PreflightStepProps {
 76    onSuccess: () => void;
 77  }
 78  export function PreflightStep(t0) {
 79    const $ = _c(12);
 80    const {
 81      onSuccess
 82    } = t0;
 83    const [result, setResult] = useState(null);
 84    const [isChecking, setIsChecking] = useState(true);
 85    const showSpinner = useTimeout(1000) && isChecking;
 86    let t1;
 87    let t2;
 88    if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
 89      t1 = () => {
 90        const run = async function run() {
 91          const checkResult = await checkEndpoints();
 92          setResult(checkResult);
 93          setIsChecking(false);
 94        };
 95        run();
 96      };
 97      t2 = [];
 98      $[0] = t1;
 99      $[1] = t2;
100    } else {
101      t1 = $[0];
102      t2 = $[1];
103    }
104    useEffect(t1, t2);
105    let t3;
106    let t4;
107    if ($[2] !== onSuccess || $[3] !== result) {
108      t3 = () => {
109        if (result?.success) {
110          onSuccess();
111        } else {
112          if (result && !result.success) {
113            const timer = setTimeout(_temp, 100);
114            return () => clearTimeout(timer);
115          }
116        }
117      };
118      t4 = [result, onSuccess];
119      $[2] = onSuccess;
120      $[3] = result;
121      $[4] = t3;
122      $[5] = t4;
123    } else {
124      t3 = $[4];
125      t4 = $[5];
126    }
127    useEffect(t3, t4);
128    let t5;
129    if ($[6] !== isChecking || $[7] !== result || $[8] !== showSpinner) {
130      t5 = isChecking && showSpinner ? <Box paddingLeft={1}><Spinner /><Text>Checking connectivity...</Text></Box> : !result?.success && !isChecking && <Box flexDirection="column" gap={1}><Text color="error">Unable to connect to Anthropic services</Text><Text color="error">{result?.error}</Text>{result?.sslHint ? <Box flexDirection="column" gap={1}><Text>{result.sslHint}</Text><Text color="suggestion">See https://code.claude.com/docs/en/network-config</Text></Box> : <Box flexDirection="column" gap={1}><Text>Please check your internet connection and network settings.</Text><Text>Note: Claude Code might not be available in your country. Check supported countries at{" "}<Text color="suggestion">https://anthropic.com/supported-countries</Text></Text></Box>}</Box>;
131      $[6] = isChecking;
132      $[7] = result;
133      $[8] = showSpinner;
134      $[9] = t5;
135    } else {
136      t5 = $[9];
137    }
138    let t6;
139    if ($[10] !== t5) {
140      t6 = <Box flexDirection="column" gap={1} paddingLeft={1}>{t5}</Box>;
141      $[10] = t5;
142      $[11] = t6;
143    } else {
144      t6 = $[11];
145    }
146    return t6;
147  }
148  function _temp() {
149    return process.exit(1);
150  }
151  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["axios","React","useEffect","useState","logEvent","Spinner","getOauthConfig","useTimeout","Box","Text","getSSLErrorHint","getUserAgent","logError","PreflightCheckResult","success","error","sslHint","checkEndpoints","Promise","oauthConfig","tokenUrl","URL","TOKEN_URL","endpoints","BASE_API_URL","origin","checkEndpoint","url","response","get","headers","status","hostname","Error","ErrnoException","code","message","String","undefined","results","all","map","failedResult","find","result","isConnectivityError","hasErrorMessage","isSSLError","PreflightStepProps","onSuccess","PreflightStep","t0","$","_c","setResult","isChecking","setIsChecking","showSpinner","t1","t2","Symbol","for","run","checkResult","t3","t4","timer","setTimeout","_temp","clearTimeout","t5","t6","process","exit"],"sources":["preflightChecks.tsx"],"sourcesContent":["import axios from 'axios'\nimport React, { useEffect, useState } from 'react'\nimport { logEvent } from 'src/services/analytics/index.js'\nimport { Spinner } from '../components/Spinner.js'\nimport { getOauthConfig } from '../constants/oauth.js'\nimport { useTimeout } from '../hooks/useTimeout.js'\nimport { Box, Text } from '../ink.js'\nimport { getSSLErrorHint } from '../services/api/errorUtils.js'\nimport { getUserAgent } from './http.js'\nimport { logError } from './log.js'\n\nexport interface PreflightCheckResult {\n  success: boolean\n  error?: string\n  sslHint?: string\n}\n\nasync function checkEndpoints(): Promise<PreflightCheckResult> {\n  try {\n    const oauthConfig = getOauthConfig()\n    const tokenUrl = new URL(oauthConfig.TOKEN_URL)\n    const endpoints = [\n      `${oauthConfig.BASE_API_URL}/api/hello`,\n      `${tokenUrl.origin}/v1/oauth/hello`,\n    ]\n\n    const checkEndpoint = async (\n      url: string,\n    ): Promise<PreflightCheckResult> => {\n      try {\n        const response = await axios.get(url, {\n          headers: { 'User-Agent': getUserAgent() },\n        })\n        if (response.status !== 200) {\n          const hostname = new URL(url).hostname\n          return {\n            success: false,\n            error: `Failed to connect to ${hostname}: Status ${response.status}`,\n          }\n        }\n        return { success: true }\n      } catch (error) {\n        const hostname = new URL(url).hostname\n        const sslHint = getSSLErrorHint(error)\n        return {\n          success: false,\n          error: `Failed to connect to ${hostname}: ${error instanceof Error ? (error as ErrnoException).code || error.message : String(error)}`,\n          sslHint: sslHint ?? undefined,\n        }\n      }\n    }\n\n    const results = await Promise.all(endpoints.map(checkEndpoint))\n    const failedResult = results.find(result => !result.success)\n\n    if (failedResult) {\n      // Log failure to Statsig\n      logEvent('tengu_preflight_check_failed', {\n        isConnectivityError: false,\n        hasErrorMessage: !!failedResult.error,\n        isSSLError: !!failedResult.sslHint,\n      })\n    }\n\n    return failedResult || { success: true }\n  } catch (error) {\n    logError(error as Error)\n\n    // Log to Statsig\n    logEvent('tengu_preflight_check_failed', {\n      isConnectivityError: true,\n    })\n\n    return {\n      success: false,\n      error: `Connectivity check error: ${error instanceof Error ? (error as ErrnoException).code || error.message : String(error)}`,\n    }\n  }\n}\n\ninterface PreflightStepProps {\n  onSuccess: () => void\n}\n\nexport function PreflightStep({\n  onSuccess,\n}: PreflightStepProps): React.ReactNode {\n  const [result, setResult] = useState<PreflightCheckResult | null>(null)\n  const [isChecking, setIsChecking] = useState(true)\n\n  // delay showing the check since it's so fast that we normally\n  // want to just immediately show the next step without a flash\n  const showSpinner = useTimeout(1000) && isChecking\n\n  useEffect(() => {\n    async function run() {\n      const checkResult = await checkEndpoints()\n      setResult(checkResult)\n      setIsChecking(false)\n    }\n    void run()\n  }, [])\n\n  useEffect(() => {\n    if (result?.success) {\n      onSuccess()\n    } else if (result && !result.success) {\n      const timer = setTimeout(() => process.exit(1), 100)\n      return () => clearTimeout(timer)\n    }\n  }, [result, onSuccess])\n\n  return (\n    <Box flexDirection=\"column\" gap={1} paddingLeft={1}>\n      {isChecking && showSpinner ? (\n        <Box paddingLeft={1}>\n          <Spinner />\n          <Text>Checking connectivity...</Text>\n        </Box>\n      ) : (\n        !result?.success &&\n        !isChecking && (\n          <Box flexDirection=\"column\" gap={1}>\n            <Text color=\"error\">Unable to connect to Anthropic services</Text>\n            <Text color=\"error\">{result?.error}</Text>\n            {result?.sslHint ? (\n              <Box flexDirection=\"column\" gap={1}>\n                <Text>{result.sslHint}</Text>\n                <Text color=\"suggestion\">\n                  See https://code.claude.com/docs/en/network-config\n                </Text>\n              </Box>\n            ) : (\n              <Box flexDirection=\"column\" gap={1}>\n                <Text>\n                  Please check your internet connection and network settings.\n                </Text>\n                <Text>\n                  Note: Claude Code might not be available in your country.\n                  Check supported countries at{' '}\n                  <Text color=\"suggestion\">\n                    https://anthropic.com/supported-countries\n                  </Text>\n                </Text>\n              </Box>\n            )}\n          </Box>\n        )\n      )}\n    </Box>\n  )\n}\n"],"mappings":"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","ignoreList":[]}