@shareai-lab/kode 1.1.21 → 1.1.23
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.
package/dist/components/Logo.js
CHANGED
|
@@ -5,6 +5,7 @@ import { PRODUCT_NAME } from "../constants/product.js";
|
|
|
5
5
|
import { getAnthropicApiKey, getGlobalConfig } from "../utils/config.js";
|
|
6
6
|
import { getCwd } from "../utils/state.js";
|
|
7
7
|
import { getModelManager } from "../utils/model.js";
|
|
8
|
+
import { MACRO } from "../constants/macros.js";
|
|
8
9
|
const MIN_LOGO_WIDTH = 50;
|
|
9
10
|
const DEFAULT_UPDATE_COMMANDS = [
|
|
10
11
|
"bun add -g @shareai-lab/kode@latest",
|
|
@@ -37,7 +38,7 @@ function Logo({
|
|
|
37
38
|
marginRight: 2,
|
|
38
39
|
width
|
|
39
40
|
},
|
|
40
|
-
updateBannerVersion ? /* @__PURE__ */ React.createElement(Box, { flexDirection: "column" }, /* @__PURE__ */ React.createElement(Text, { color: "yellow" }, "New version available: ", updateBannerVersion
|
|
41
|
+
updateBannerVersion ? /* @__PURE__ */ React.createElement(Box, { flexDirection: "column" }, /* @__PURE__ */ React.createElement(Text, { color: "yellow" }, "New version available: ", updateBannerVersion, " (current: ", MACRO.VERSION, ")"), /* @__PURE__ */ React.createElement(Text, null, "Run the following command to update:"), /* @__PURE__ */ React.createElement(Text, null, " ", updateBannerCommands?.[1] ?? DEFAULT_UPDATE_COMMANDS[1]), process.platform !== "win32" && /* @__PURE__ */ React.createElement(Text, { dimColor: true }, 'Note: you may need to prefix with "sudo" on macOS/Linux.')) : null,
|
|
41
42
|
/* @__PURE__ */ React.createElement(Text, null, /* @__PURE__ */ React.createElement(Text, { color: theme.kode }, "\u273B"), " Welcome to", " ", /* @__PURE__ */ React.createElement(Text, { bold: true }, PRODUCT_NAME), " ", /* @__PURE__ */ React.createElement(Text, null, "research preview!")),
|
|
42
43
|
/* @__PURE__ */ React.createElement(React.Fragment, null, /* @__PURE__ */ React.createElement(Box, { paddingLeft: 2, flexDirection: "column", gap: 1 }, /* @__PURE__ */ React.createElement(Text, { color: theme.secondaryText, italic: true }, "/help for help"), /* @__PURE__ */ React.createElement(Text, { color: theme.secondaryText }, "cwd: ", getCwd())), hasOverrides && /* @__PURE__ */ React.createElement(
|
|
43
44
|
Box,
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/components/Logo.tsx"],
|
|
4
|
-
"sourcesContent": ["import { Box, Text, Newline } from 'ink'\nimport * as React from 'react'\nimport { getTheme } from '../utils/theme'\nimport { PRODUCT_NAME } from '../constants/product'\nimport { getAnthropicApiKey, getGlobalConfig } from '../utils/config'\nimport { getCwd } from '../utils/state'\nimport { AsciiLogo } from './AsciiLogo'\nimport type { WrappedClient } from '../services/mcpClient'\nimport { getModelManager } from '../utils/model'\n\nexport const MIN_LOGO_WIDTH = 50\n\nconst DEFAULT_UPDATE_COMMANDS = [\n 'bun add -g @shareai-lab/kode@latest',\n 'npm install -g @shareai-lab/kode@latest',\n] as const\n\nexport function Logo({\n mcpClients,\n isDefaultModel = false,\n updateBannerVersion,\n updateBannerCommands,\n}: {\n mcpClients: WrappedClient[]\n isDefaultModel?: boolean\n updateBannerVersion?: string | null\n updateBannerCommands?: string[] | null\n}): React.ReactNode {\n const width = Math.max(MIN_LOGO_WIDTH, getCwd().length + 12)\n const theme = getTheme()\n const config = getGlobalConfig()\n\n const modelManager = getModelManager()\n const mainModelName = modelManager.getModelName('main')\n const currentModel = mainModelName || 'No model configured'\n const apiKey = getAnthropicApiKey()\n const hasOverrides = Boolean(\n process.env.ANTHROPIC_API_KEY ||\n process.env.DISABLE_PROMPT_CACHING ||\n process.env.API_TIMEOUT_MS ||\n process.env.MAX_THINKING_TOKENS,\n )\n\n return (\n <Box flexDirection=\"column\">\n <Box\n borderColor={theme.kode}\n borderStyle=\"round\"\n flexDirection=\"column\"\n gap={1}\n paddingLeft={1}\n marginRight={2}\n width={width}\n >\n {updateBannerVersion ? (\n <Box flexDirection=\"column\">\n <Text color=\"yellow\">New version available: {updateBannerVersion}</Text>\n <Text>Run the following command to update:</Text>\n <Text>\n {' '}\n {updateBannerCommands?.[
|
|
5
|
-
"mappings": "AAAA,SAAS,KAAK,YAAqB;AACnC,YAAY,WAAW;AACvB,SAAS,gBAAgB;AACzB,SAAS,oBAAoB;AAC7B,SAAS,oBAAoB,uBAAuB;AACpD,SAAS,cAAc;AAGvB,SAAS,uBAAuB;
|
|
4
|
+
"sourcesContent": ["import { Box, Text, Newline } from 'ink'\nimport * as React from 'react'\nimport { getTheme } from '../utils/theme'\nimport { PRODUCT_NAME } from '../constants/product'\nimport { getAnthropicApiKey, getGlobalConfig } from '../utils/config'\nimport { getCwd } from '../utils/state'\nimport { AsciiLogo } from './AsciiLogo'\nimport type { WrappedClient } from '../services/mcpClient'\nimport { getModelManager } from '../utils/model'\nimport { MACRO } from '../constants/macros'\n\nexport const MIN_LOGO_WIDTH = 50\n\nconst DEFAULT_UPDATE_COMMANDS = [\n 'bun add -g @shareai-lab/kode@latest',\n 'npm install -g @shareai-lab/kode@latest',\n] as const\n\nexport function Logo({\n mcpClients,\n isDefaultModel = false,\n updateBannerVersion,\n updateBannerCommands,\n}: {\n mcpClients: WrappedClient[]\n isDefaultModel?: boolean\n updateBannerVersion?: string | null\n updateBannerCommands?: string[] | null\n}): React.ReactNode {\n const width = Math.max(MIN_LOGO_WIDTH, getCwd().length + 12)\n const theme = getTheme()\n const config = getGlobalConfig()\n\n const modelManager = getModelManager()\n const mainModelName = modelManager.getModelName('main')\n const currentModel = mainModelName || 'No model configured'\n const apiKey = getAnthropicApiKey()\n const hasOverrides = Boolean(\n process.env.ANTHROPIC_API_KEY ||\n process.env.DISABLE_PROMPT_CACHING ||\n process.env.API_TIMEOUT_MS ||\n process.env.MAX_THINKING_TOKENS,\n )\n\n return (\n <Box flexDirection=\"column\">\n <Box\n borderColor={theme.kode}\n borderStyle=\"round\"\n flexDirection=\"column\"\n gap={1}\n paddingLeft={1}\n marginRight={2}\n width={width}\n >\n {updateBannerVersion ? (\n <Box flexDirection=\"column\">\n <Text color=\"yellow\">New version available: {updateBannerVersion} (current: {MACRO.VERSION})</Text>\n <Text>Run the following command to update:</Text>\n <Text>\n {' '}\n {updateBannerCommands?.[1] ?? DEFAULT_UPDATE_COMMANDS[1]}\n </Text>\n {process.platform !== 'win32' && (\n <Text dimColor>\n Note: you may need to prefix with \"sudo\" on macOS/Linux.\n </Text>\n )}\n </Box>\n ) : null}\n <Text>\n <Text color={theme.kode}>\u273B</Text> Welcome to{' '}\n <Text bold>{PRODUCT_NAME}</Text> <Text>research preview!</Text>\n </Text>\n {/* <AsciiLogo /> */}\n\n <>\n <Box paddingLeft={2} flexDirection=\"column\" gap={1}>\n <Text color={theme.secondaryText} italic>\n /help for help\n </Text>\n <Text color={theme.secondaryText}>cwd: {getCwd()}</Text>\n </Box>\n\n {hasOverrides && (\n <Box\n borderColor={theme.secondaryBorder}\n borderStyle=\"single\"\n borderBottom={false}\n borderLeft={false}\n borderRight={false}\n borderTop={true}\n flexDirection=\"column\"\n marginLeft={2}\n marginRight={1}\n paddingTop={1}\n >\n <Box marginBottom={1}>\n <Text color={theme.secondaryText}>Overrides (via env):</Text>\n </Box>\n {process.env.ANTHROPIC_API_KEY && apiKey ? (\n <Text color={theme.secondaryText}>\n \u2022 API Key:{' '}\n <Text bold>sk-ant-\u2026{apiKey!.slice(-width + 25)}</Text>\n </Text>\n ) : null}\n {process.env.DISABLE_PROMPT_CACHING ? (\n <Text color={theme.secondaryText}>\n \u2022 Prompt caching:{' '}\n <Text color={theme.error} bold>\n off\n </Text>\n </Text>\n ) : null}\n {process.env.API_TIMEOUT_MS ? (\n <Text color={theme.secondaryText}>\n \u2022 API timeout:{' '}\n <Text bold>{process.env.API_TIMEOUT_MS}ms</Text>\n </Text>\n ) : null}\n {process.env.MAX_THINKING_TOKENS ? (\n <Text color={theme.secondaryText}>\n \u2022 Max thinking tokens:{' '}\n <Text bold>{process.env.MAX_THINKING_TOKENS}</Text>\n </Text>\n ) : null}\n {process.env.ANTHROPIC_BASE_URL ? (\n <Text color={theme.secondaryText}>\n \u2022 API Base URL:{' '}\n <Text bold>{process.env.ANTHROPIC_BASE_URL}</Text>\n </Text>\n ) : null}\n </Box>\n )}\n </>\n {mcpClients.length ? (\n <Box\n borderColor={theme.secondaryBorder}\n borderStyle=\"single\"\n borderBottom={false}\n borderLeft={false}\n borderRight={false}\n borderTop={true}\n flexDirection=\"column\"\n marginLeft={2}\n marginRight={1}\n paddingTop={1}\n >\n <Box marginBottom={1}>\n <Text color={theme.secondaryText}>MCP Servers:</Text>\n </Box>\n {mcpClients.map((client, idx) => (\n <Box key={idx} width={width - 6}>\n <Text color={theme.secondaryText}>\u2022 {client.name}</Text>\n <Box flexGrow={1} />\n <Text\n bold\n color={\n client.type === 'connected' ? theme.success : theme.error\n }\n >\n {client.type === 'connected' ? 'connected' : 'failed'}\n </Text>\n </Box>\n ))}\n </Box>\n ) : null}\n </Box>\n </Box>\n )\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -31,11 +31,6 @@ async function assertMinVersion() {
|
|
|
31
31
|
if (versionConfig.minVersion && lt(MACRO.VERSION, versionConfig.minVersion)) {
|
|
32
32
|
const suggestions = await getUpdateCommandSuggestions();
|
|
33
33
|
const cmdLines = suggestions.map((c) => ` ${c}`).join("\n");
|
|
34
|
-
console.error(`
|
|
35
|
-
\u60A8\u7684 ${PRODUCT_NAME} \u7248\u672C (${MACRO.VERSION}) \u8FC7\u4F4E\uFF0C\u9700\u8981\u5347\u7EA7\u5230 ${versionConfig.minVersion} \u6216\u66F4\u9AD8\u7248\u672C\u3002
|
|
36
|
-
\u8BF7\u624B\u52A8\u6267\u884C\u4EE5\u4E0B\u4EFB\u4E00\u547D\u4EE4\u8FDB\u884C\u5347\u7EA7\uFF1A
|
|
37
|
-
${cmdLines}
|
|
38
|
-
`);
|
|
39
34
|
process.exit(1);
|
|
40
35
|
}
|
|
41
36
|
} catch (error) {
|
|
@@ -337,8 +332,11 @@ async function checkAndNotifyUpdate() {
|
|
|
337
332
|
lastSuggestedVersion: latest
|
|
338
333
|
});
|
|
339
334
|
const suggestions = await getUpdateCommandSuggestions();
|
|
340
|
-
|
|
341
|
-
console.log(
|
|
335
|
+
console.log(`New version available: ${latest} (current: ${MACRO.VERSION})`);
|
|
336
|
+
console.log("Run the following command to update:");
|
|
337
|
+
for (const command of suggestions) {
|
|
338
|
+
console.log(` ${command}`);
|
|
339
|
+
}
|
|
342
340
|
} else {
|
|
343
341
|
saveGlobalConfig({ ...config, lastUpdateCheckAt: now });
|
|
344
342
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/utils/autoUpdater.ts"],
|
|
4
|
-
"sourcesContent": ["import { homedir } from 'os'\nimport { join } from 'path'\nimport {\n existsSync,\n mkdirSync,\n appendFileSync,\n readFileSync,\n constants,\n writeFileSync,\n unlinkSync,\n statSync,\n} from 'fs'\nimport { platform } from 'process'\nimport { execFileNoThrow } from './execFileNoThrow'\nimport { spawn } from 'child_process'\nimport { logError } from './log'\nimport { accessSync } from 'fs'\nimport { CLAUDE_BASE_DIR } from './env'\nimport { logEvent, getDynamicConfig } from '../services/statsig'\nimport { lt, gt } from 'semver'\nimport { MACRO } from '../constants/macros'\nimport { PRODUCT_COMMAND, PRODUCT_NAME } from '../constants/product'\nimport { getGlobalConfig, saveGlobalConfig, isAutoUpdaterDisabled } from './config'\nimport { env } from './env'\nexport type InstallStatus =\n | 'success'\n | 'no_permissions'\n | 'install_failed'\n | 'in_progress'\n\nexport type AutoUpdaterResult = {\n version: string | null\n status: InstallStatus\n}\n\nexport type VersionConfig = {\n minVersion: string\n}\n\n/**\n * Checks if the current version meets the minimum required version from Statsig config\n * Terminates the process with an error message if the version is too old\n */\nexport async function assertMinVersion(): Promise<void> {\n try {\n const versionConfig = await getDynamicConfig<VersionConfig>(\n 'tengu_version_config',\n { minVersion: '0.0.0' },\n )\n\n if (\n versionConfig.minVersion &&\n lt(MACRO.VERSION, versionConfig.minVersion)\n ) {\n const suggestions = await getUpdateCommandSuggestions()\n const cmdLines = suggestions.map(c => ` ${c}`).join('\\n')\n console.error(`\n\u60A8\u7684 ${PRODUCT_NAME} \u7248\u672C (${MACRO.VERSION}) \u8FC7\u4F4E\uFF0C\u9700\u8981\u5347\u7EA7\u5230 ${versionConfig.minVersion} \u6216\u66F4\u9AD8\u7248\u672C\u3002\n\u8BF7\u624B\u52A8\u6267\u884C\u4EE5\u4E0B\u4EFB\u4E00\u547D\u4EE4\u8FDB\u884C\u5347\u7EA7\uFF1A\n${cmdLines}\n`)\n process.exit(1)\n }\n } catch (error) {\n logError(`Error checking minimum version: ${error}`)\n }\n}\n\n// Lock file for auto-updater to prevent concurrent updates\nexport const LOCK_FILE_PATH = join(CLAUDE_BASE_DIR, '.update.lock')\nconst LOCK_TIMEOUT_MS = 5 * 60 * 1000 // 5 minute timeout for locks\n\n/**\n * Attempts to acquire a lock for auto-updater\n * @returns {boolean} true if lock was acquired, false if another process holds the lock\n */\nfunction acquireLock(): boolean {\n try {\n // Ensure the base directory exists\n if (!existsSync(CLAUDE_BASE_DIR)) {\n mkdirSync(CLAUDE_BASE_DIR, { recursive: true })\n }\n\n // Check if lock file exists and is not stale\n if (existsSync(LOCK_FILE_PATH)) {\n const stats = statSync(LOCK_FILE_PATH)\n const age = Date.now() - stats.mtimeMs\n\n // If lock file is older than timeout, consider it stale\n if (age < LOCK_TIMEOUT_MS) {\n return false\n }\n\n // Lock is stale, we can take over\n try {\n unlinkSync(LOCK_FILE_PATH)\n } catch (err) {\n logError(`Failed to remove stale lock file: ${err}`)\n return false\n }\n }\n\n // Create lock file with current pid\n writeFileSync(LOCK_FILE_PATH, `${process.pid}`, 'utf8')\n return true\n } catch (err) {\n logError(`Failed to acquire lock: ${err}`)\n return false\n }\n}\n\n/**\n * Releases the update lock if it's held by this process\n */\nfunction releaseLock(): void {\n try {\n if (existsSync(LOCK_FILE_PATH)) {\n const lockData = readFileSync(LOCK_FILE_PATH, 'utf8')\n if (lockData === `${process.pid}`) {\n unlinkSync(LOCK_FILE_PATH)\n }\n }\n } catch (err) {\n logError(`Failed to release lock: ${err}`)\n }\n}\n\nexport async function checkNpmPermissions(): Promise<{\n hasPermissions: boolean\n npmPrefix: string | null\n}> {\n try {\n const prefixResult = await execFileNoThrow('npm', [\n '-g',\n 'config',\n 'get',\n 'prefix',\n ])\n if (prefixResult.code !== 0) {\n logError('Failed to check npm permissions')\n return { hasPermissions: false, npmPrefix: null }\n }\n\n const prefix = prefixResult.stdout.trim()\n\n let testWriteResult = false\n try {\n accessSync(prefix, constants.W_OK)\n testWriteResult = true\n } catch {\n testWriteResult = false\n }\n\n if (testWriteResult) {\n return { hasPermissions: true, npmPrefix: prefix }\n }\n\n logError('Insufficient permissions for global npm install.')\n return { hasPermissions: false, npmPrefix: prefix }\n } catch (error) {\n logError(`Failed to verify npm global install permissions: ${error}`)\n return { hasPermissions: false, npmPrefix: null }\n }\n}\n\nexport async function setupNewPrefix(prefix: string): Promise<void> {\n if (!acquireLock()) {\n // Log the lock contention to statsig\n logEvent('tengu_auto_updater_prefix_lock_contention', {\n pid: String(process.pid),\n currentVersion: MACRO.VERSION,\n prefix,\n })\n throw new Error('Another process is currently setting up npm prefix')\n }\n\n try {\n // Create directory if it doesn't exist\n if (!existsSync(prefix)) {\n mkdirSync(prefix, { recursive: true })\n }\n\n // Set npm prefix\n const setPrefix = await execFileNoThrow('npm', [\n '-g',\n 'config',\n 'set',\n 'prefix',\n prefix,\n ])\n\n if (setPrefix.code !== 0) {\n throw new Error(`Failed to set npm prefix: ${setPrefix.stderr}`)\n }\n\n // Update shell config files\n const pathUpdate = `\\n# npm global path\\nexport PATH=\"${prefix}/bin:$PATH\"\\n`\n\n if (platform === 'win32') {\n // On Windows, update user PATH environment variable\n const setxResult = await execFileNoThrow('setx', [\n 'PATH',\n `${process.env.PATH};${prefix}`,\n ])\n if (setxResult.code !== 0) {\n throw new Error(\n `Failed to update PATH on Windows: ${setxResult.stderr}`,\n )\n }\n } else {\n // Unix-like systems\n const shellConfigs = [\n // Bash\n join(homedir(), '.bashrc'),\n join(homedir(), '.bash_profile'),\n // Zsh\n join(homedir(), '.zshrc'),\n // Fish\n join(homedir(), '.config', 'fish', 'config.fish'),\n ]\n\n for (const config of shellConfigs) {\n if (existsSync(config)) {\n try {\n const content = readFileSync(config, 'utf8')\n if (!content.includes(prefix)) {\n if (config.includes('fish')) {\n // Fish shell has different syntax\n const fishPath = `\\n# npm global path\\nset -gx PATH ${prefix}/bin $PATH\\n`\n appendFileSync(config, fishPath)\n } else {\n appendFileSync(config, pathUpdate)\n }\n\n logEvent('npm_prefix_path_updated', {\n configPath: config,\n })\n }\n } catch (err) {\n // Log but don't throw - continue with other configs\n logEvent('npm_prefix_path_update_failed', {\n configPath: config,\n error:\n err instanceof Error\n ? err.message.slice(0, 200)\n : String(err).slice(0, 200),\n })\n logError(`Failed to update shell config ${config}: ${err}`)\n }\n }\n }\n }\n } finally {\n releaseLock()\n }\n}\n\nexport function getDefaultNpmPrefix(): string {\n return join(homedir(), '.npm-global')\n}\n\nexport function getPermissionsCommand(npmPrefix: string): string {\n const windowsCommand = `icacls \"${npmPrefix}\" /grant \"%USERNAME%:(OI)(CI)F\"`\n const prefixPath = npmPrefix || '$(npm -g config get prefix)'\n const unixCommand = `sudo chown -R $USER:$(id -gn) ${prefixPath} && sudo chmod -R u+w ${prefixPath}`\n\n return platform === 'win32' ? windowsCommand : unixCommand\n}\n\nexport async function getLatestVersion(): Promise<string | null> {\n // 1) Try npm CLI (fast when available)\n try {\n const abortController = new AbortController()\n setTimeout(() => abortController.abort(), 5000)\n const result = await execFileNoThrow(\n 'npm',\n ['view', MACRO.PACKAGE_URL, 'version'],\n abortController.signal,\n )\n if (result.code === 0) {\n const v = result.stdout.trim()\n if (v) return v\n }\n } catch {}\n\n // 2) Fallback: fetch npm registry (works in Bun/Node without npm)\n try {\n const controller = new AbortController()\n const timer = setTimeout(() => controller.abort(), 5000)\n const res = await fetch(\n `https://registry.npmjs.org/${encodeURIComponent(MACRO.PACKAGE_URL)}`,\n {\n method: 'GET',\n headers: {\n Accept: 'application/vnd.npm.install-v1+json',\n 'User-Agent': `${PRODUCT_NAME}/${MACRO.VERSION}`,\n },\n signal: controller.signal,\n },\n )\n clearTimeout(timer)\n if (!res.ok) return null\n const json: any = await res.json().catch(() => null)\n const latest = json && json['dist-tags'] && json['dist-tags'].latest\n return typeof latest === 'string' ? latest : null\n } catch {\n return null\n }\n}\n\nexport async function installGlobalPackage(): Promise<InstallStatus> {\n // Detect preferred package manager and install accordingly\n if (!acquireLock()) {\n logError('Another process is currently installing an update')\n // Log the lock contention to statsig\n logEvent('tengu_auto_updater_lock_contention', {\n pid: String(process.pid),\n currentVersion: MACRO.VERSION,\n })\n return 'in_progress'\n }\n\n try {\n const manager = await detectPackageManager()\n if (manager === 'npm') {\n const { hasPermissions } = await checkNpmPermissions()\n if (!hasPermissions) {\n return 'no_permissions'\n }\n // Stream\u5B9E\u65F6\u8F93\u51FA\uFF0C\u51CF\u5C11\u7528\u6237\u7B49\u5F85\u611F\n const code = await runStreaming('npm', ['install', '-g', MACRO.PACKAGE_URL])\n if (code !== 0) {\n logError(`Failed to install new version via npm (exit ${code})`)\n return 'install_failed'\n }\n return 'success'\n }\n\n if (manager === 'bun') {\n const code = await runStreaming('bun', ['add', '-g', `${MACRO.PACKAGE_URL}@latest`])\n if (code !== 0) {\n logError(`Failed to install new version via bun (exit ${code})`)\n return 'install_failed'\n }\n return 'success'\n }\n\n // Fallback to npm if unknown\n const { hasPermissions } = await checkNpmPermissions()\n if (!hasPermissions) return 'no_permissions'\n const code = await runStreaming('npm', ['install', '-g', MACRO.PACKAGE_URL])\n if (code !== 0) return 'install_failed'\n return 'success'\n } finally {\n // Ensure we always release the lock\n releaseLock()\n }\n}\n\nexport type PackageManager = 'npm' | 'bun'\n\nexport async function detectPackageManager(): Promise<PackageManager> {\n // Respect explicit override if provided later via config/env (future-proof)\n try {\n // Heuristic 1: npm available and global root resolvable\n const npmRoot = await execFileNoThrow('npm', ['-g', 'root'])\n if (npmRoot.code === 0 && npmRoot.stdout.trim()) {\n return 'npm'\n }\n } catch {}\n\n try {\n // Heuristic 2: running on a system with bun and installed path hints bun\n const bunVer = await execFileNoThrow('bun', ['--version'])\n if (bunVer.code === 0) {\n // BUN_INSTALL defaults to ~/.bun; if our package lives under that tree, prefer bun\n // If npm not detected but bun is available, choose bun\n return 'bun'\n }\n } catch {}\n\n // Default to npm when uncertain\n return 'npm'\n}\n\nfunction runStreaming(cmd: string, args: string[]): Promise<number> {\n return new Promise(resolve => {\n // \u6253\u5370\u6B63\u5728\u4F7F\u7528\u7684\u5305\u7BA1\u7406\u5668\u4E0E\u547D\u4EE4\uFF0C\u589E\u5F3A\u900F\u660E\u5EA6\n try {\n // eslint-disable-next-line no-console\n console.log(`> ${cmd} ${args.join(' ')}`)\n } catch {}\n\n const child = spawn(cmd, args, {\n stdio: 'inherit',\n env: process.env,\n })\n child.on('close', code => resolve(code ?? 0))\n child.on('error', () => resolve(1))\n })\n}\n\n/**\n * Generate human-friendly update commands for the detected package manager.\n * Also includes an alternative manager command as fallback for users.\n */\nexport async function getUpdateCommandSuggestions(): Promise<string[]> {\n // Prefer Bun first, then npm (consistent, simple UX). Include @latest.\n return [\n `bun add -g ${MACRO.PACKAGE_URL}@latest`,\n `npm install -g ${MACRO.PACKAGE_URL}@latest`,\n ]\n}\n\n/**\n * Non-blocking update notifier (daily)\n * - Respects CI and disabled auto-updater\n * - Uses env.hasInternetAccess() to quickly skip offline cases\n * - Stores last check timestamp + last suggested version in global config\n */\nexport async function checkAndNotifyUpdate(): Promise<void> {\n try {\n if (process.env.NODE_ENV === 'test') return\n if (await isAutoUpdaterDisabled()) return\n if (await env.getIsDocker()) return\n if (!(await env.hasInternetAccess())) return\n\n const config: any = getGlobalConfig()\n const now = Date.now()\n const DAY_MS = 24 * 60 * 60 * 1000\n const lastCheck = Number(config.lastUpdateCheckAt || 0)\n if (lastCheck && now - lastCheck < DAY_MS) return\n\n const latest = await getLatestVersion()\n if (!latest) {\n // Still record the check to avoid spamming\n saveGlobalConfig({ ...config, lastUpdateCheckAt: now })\n return\n }\n\n if (gt(latest, MACRO.VERSION)) {\n // Update stored state and print a low-noise hint\n saveGlobalConfig({\n ...config,\n lastUpdateCheckAt: now,\n lastSuggestedVersion: latest,\n })\n const suggestions = await getUpdateCommandSuggestions()\n const first = suggestions[0]\n console.log(`New version available: ${latest}. Recommended: ${first}`)\n } else {\n saveGlobalConfig({ ...config, lastUpdateCheckAt: now })\n }\n } catch (error) {\n // Never block or throw; just log and move on\n logError(`update-notify: ${error}`)\n }\n}\n"],
|
|
5
|
-
"mappings": "AAAA,SAAS,eAAe;AACxB,SAAS,YAAY;AACrB;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,gBAAgB;AACzB,SAAS,uBAAuB;AAChC,SAAS,aAAa;AACtB,SAAS,gBAAgB;AACzB,SAAS,kBAAkB;AAC3B,SAAS,uBAAuB;AAChC,SAAS,UAAU,wBAAwB;AAC3C,SAAS,IAAI,UAAU;AACvB,SAAS,aAAa;AACtB,SAA0B,oBAAoB;AAC9C,SAAS,iBAAiB,kBAAkB,6BAA6B;AACzE,SAAS,WAAW;AAoBpB,eAAsB,mBAAkC;AACtD,MAAI;AACF,UAAM,gBAAgB,MAAM;AAAA,MAC1B;AAAA,MACA,EAAE,YAAY,QAAQ;AAAA,IACxB;AAEA,QACE,cAAc,cACd,GAAG,MAAM,SAAS,cAAc,UAAU,GAC1C;AACA,YAAM,cAAc,MAAM,4BAA4B;AACtD,YAAM,WAAW,YAAY,IAAI,OAAK,OAAO,CAAC,EAAE,EAAE,KAAK,IAAI;AAC3D,cAAQ,
|
|
4
|
+
"sourcesContent": ["import { homedir } from 'os'\nimport { join } from 'path'\nimport {\n existsSync,\n mkdirSync,\n appendFileSync,\n readFileSync,\n constants,\n writeFileSync,\n unlinkSync,\n statSync,\n} from 'fs'\nimport { platform } from 'process'\nimport { execFileNoThrow } from './execFileNoThrow'\nimport { spawn } from 'child_process'\nimport { logError } from './log'\nimport { accessSync } from 'fs'\nimport { CLAUDE_BASE_DIR } from './env'\nimport { logEvent, getDynamicConfig } from '../services/statsig'\nimport { lt, gt } from 'semver'\nimport { MACRO } from '../constants/macros'\nimport { PRODUCT_COMMAND, PRODUCT_NAME } from '../constants/product'\nimport { getGlobalConfig, saveGlobalConfig, isAutoUpdaterDisabled } from './config'\nimport { env } from './env'\nexport type InstallStatus =\n | 'success'\n | 'no_permissions'\n | 'install_failed'\n | 'in_progress'\n\nexport type AutoUpdaterResult = {\n version: string | null\n status: InstallStatus\n}\n\nexport type VersionConfig = {\n minVersion: string\n}\n\n/**\n * Checks if the current version meets the minimum required version from Statsig config\n * Terminates the process with an error message if the version is too old\n */\nexport async function assertMinVersion(): Promise<void> {\n try {\n const versionConfig = await getDynamicConfig<VersionConfig>(\n 'tengu_version_config',\n { minVersion: '0.0.0' },\n )\n\n if (\n versionConfig.minVersion &&\n lt(MACRO.VERSION, versionConfig.minVersion)\n ) {\n const suggestions = await getUpdateCommandSuggestions()\n const cmdLines = suggestions.map(c => ` ${c}`).join('\\n')\n process.exit(1)\n }\n } catch (error) {\n logError(`Error checking minimum version: ${error}`)\n }\n}\n\n// Lock file for auto-updater to prevent concurrent updates\nexport const LOCK_FILE_PATH = join(CLAUDE_BASE_DIR, '.update.lock')\nconst LOCK_TIMEOUT_MS = 5 * 60 * 1000 // 5 minute timeout for locks\n\n/**\n * Attempts to acquire a lock for auto-updater\n * @returns {boolean} true if lock was acquired, false if another process holds the lock\n */\nfunction acquireLock(): boolean {\n try {\n // Ensure the base directory exists\n if (!existsSync(CLAUDE_BASE_DIR)) {\n mkdirSync(CLAUDE_BASE_DIR, { recursive: true })\n }\n\n // Check if lock file exists and is not stale\n if (existsSync(LOCK_FILE_PATH)) {\n const stats = statSync(LOCK_FILE_PATH)\n const age = Date.now() - stats.mtimeMs\n\n // If lock file is older than timeout, consider it stale\n if (age < LOCK_TIMEOUT_MS) {\n return false\n }\n\n // Lock is stale, we can take over\n try {\n unlinkSync(LOCK_FILE_PATH)\n } catch (err) {\n logError(`Failed to remove stale lock file: ${err}`)\n return false\n }\n }\n\n // Create lock file with current pid\n writeFileSync(LOCK_FILE_PATH, `${process.pid}`, 'utf8')\n return true\n } catch (err) {\n logError(`Failed to acquire lock: ${err}`)\n return false\n }\n}\n\n/**\n * Releases the update lock if it's held by this process\n */\nfunction releaseLock(): void {\n try {\n if (existsSync(LOCK_FILE_PATH)) {\n const lockData = readFileSync(LOCK_FILE_PATH, 'utf8')\n if (lockData === `${process.pid}`) {\n unlinkSync(LOCK_FILE_PATH)\n }\n }\n } catch (err) {\n logError(`Failed to release lock: ${err}`)\n }\n}\n\nexport async function checkNpmPermissions(): Promise<{\n hasPermissions: boolean\n npmPrefix: string | null\n}> {\n try {\n const prefixResult = await execFileNoThrow('npm', [\n '-g',\n 'config',\n 'get',\n 'prefix',\n ])\n if (prefixResult.code !== 0) {\n logError('Failed to check npm permissions')\n return { hasPermissions: false, npmPrefix: null }\n }\n\n const prefix = prefixResult.stdout.trim()\n\n let testWriteResult = false\n try {\n accessSync(prefix, constants.W_OK)\n testWriteResult = true\n } catch {\n testWriteResult = false\n }\n\n if (testWriteResult) {\n return { hasPermissions: true, npmPrefix: prefix }\n }\n\n logError('Insufficient permissions for global npm install.')\n return { hasPermissions: false, npmPrefix: prefix }\n } catch (error) {\n logError(`Failed to verify npm global install permissions: ${error}`)\n return { hasPermissions: false, npmPrefix: null }\n }\n}\n\nexport async function setupNewPrefix(prefix: string): Promise<void> {\n if (!acquireLock()) {\n // Log the lock contention to statsig\n logEvent('tengu_auto_updater_prefix_lock_contention', {\n pid: String(process.pid),\n currentVersion: MACRO.VERSION,\n prefix,\n })\n throw new Error('Another process is currently setting up npm prefix')\n }\n\n try {\n // Create directory if it doesn't exist\n if (!existsSync(prefix)) {\n mkdirSync(prefix, { recursive: true })\n }\n\n // Set npm prefix\n const setPrefix = await execFileNoThrow('npm', [\n '-g',\n 'config',\n 'set',\n 'prefix',\n prefix,\n ])\n\n if (setPrefix.code !== 0) {\n throw new Error(`Failed to set npm prefix: ${setPrefix.stderr}`)\n }\n\n // Update shell config files\n const pathUpdate = `\\n# npm global path\\nexport PATH=\"${prefix}/bin:$PATH\"\\n`\n\n if (platform === 'win32') {\n // On Windows, update user PATH environment variable\n const setxResult = await execFileNoThrow('setx', [\n 'PATH',\n `${process.env.PATH};${prefix}`,\n ])\n if (setxResult.code !== 0) {\n throw new Error(\n `Failed to update PATH on Windows: ${setxResult.stderr}`,\n )\n }\n } else {\n // Unix-like systems\n const shellConfigs = [\n // Bash\n join(homedir(), '.bashrc'),\n join(homedir(), '.bash_profile'),\n // Zsh\n join(homedir(), '.zshrc'),\n // Fish\n join(homedir(), '.config', 'fish', 'config.fish'),\n ]\n\n for (const config of shellConfigs) {\n if (existsSync(config)) {\n try {\n const content = readFileSync(config, 'utf8')\n if (!content.includes(prefix)) {\n if (config.includes('fish')) {\n // Fish shell has different syntax\n const fishPath = `\\n# npm global path\\nset -gx PATH ${prefix}/bin $PATH\\n`\n appendFileSync(config, fishPath)\n } else {\n appendFileSync(config, pathUpdate)\n }\n\n logEvent('npm_prefix_path_updated', {\n configPath: config,\n })\n }\n } catch (err) {\n // Log but don't throw - continue with other configs\n logEvent('npm_prefix_path_update_failed', {\n configPath: config,\n error:\n err instanceof Error\n ? err.message.slice(0, 200)\n : String(err).slice(0, 200),\n })\n logError(`Failed to update shell config ${config}: ${err}`)\n }\n }\n }\n }\n } finally {\n releaseLock()\n }\n}\n\nexport function getDefaultNpmPrefix(): string {\n return join(homedir(), '.npm-global')\n}\n\nexport function getPermissionsCommand(npmPrefix: string): string {\n const windowsCommand = `icacls \"${npmPrefix}\" /grant \"%USERNAME%:(OI)(CI)F\"`\n const prefixPath = npmPrefix || '$(npm -g config get prefix)'\n const unixCommand = `sudo chown -R $USER:$(id -gn) ${prefixPath} && sudo chmod -R u+w ${prefixPath}`\n\n return platform === 'win32' ? windowsCommand : unixCommand\n}\n\nexport async function getLatestVersion(): Promise<string | null> {\n // 1) Try npm CLI (fast when available)\n try {\n const abortController = new AbortController()\n setTimeout(() => abortController.abort(), 5000)\n const result = await execFileNoThrow(\n 'npm',\n ['view', MACRO.PACKAGE_URL, 'version'],\n abortController.signal,\n )\n if (result.code === 0) {\n const v = result.stdout.trim()\n if (v) return v\n }\n } catch {}\n\n // 2) Fallback: fetch npm registry (works in Bun/Node without npm)\n try {\n const controller = new AbortController()\n const timer = setTimeout(() => controller.abort(), 5000)\n const res = await fetch(\n `https://registry.npmjs.org/${encodeURIComponent(MACRO.PACKAGE_URL)}`,\n {\n method: 'GET',\n headers: {\n Accept: 'application/vnd.npm.install-v1+json',\n 'User-Agent': `${PRODUCT_NAME}/${MACRO.VERSION}`,\n },\n signal: controller.signal,\n },\n )\n clearTimeout(timer)\n if (!res.ok) return null\n const json: any = await res.json().catch(() => null)\n const latest = json && json['dist-tags'] && json['dist-tags'].latest\n return typeof latest === 'string' ? latest : null\n } catch {\n return null\n }\n}\n\nexport async function installGlobalPackage(): Promise<InstallStatus> {\n // Detect preferred package manager and install accordingly\n if (!acquireLock()) {\n logError('Another process is currently installing an update')\n // Log the lock contention to statsig\n logEvent('tengu_auto_updater_lock_contention', {\n pid: String(process.pid),\n currentVersion: MACRO.VERSION,\n })\n return 'in_progress'\n }\n\n try {\n const manager = await detectPackageManager()\n if (manager === 'npm') {\n const { hasPermissions } = await checkNpmPermissions()\n if (!hasPermissions) {\n return 'no_permissions'\n }\n // Stream\u5B9E\u65F6\u8F93\u51FA\uFF0C\u51CF\u5C11\u7528\u6237\u7B49\u5F85\u611F\n const code = await runStreaming('npm', ['install', '-g', MACRO.PACKAGE_URL])\n if (code !== 0) {\n logError(`Failed to install new version via npm (exit ${code})`)\n return 'install_failed'\n }\n return 'success'\n }\n\n if (manager === 'bun') {\n const code = await runStreaming('bun', ['add', '-g', `${MACRO.PACKAGE_URL}@latest`])\n if (code !== 0) {\n logError(`Failed to install new version via bun (exit ${code})`)\n return 'install_failed'\n }\n return 'success'\n }\n\n // Fallback to npm if unknown\n const { hasPermissions } = await checkNpmPermissions()\n if (!hasPermissions) return 'no_permissions'\n const code = await runStreaming('npm', ['install', '-g', MACRO.PACKAGE_URL])\n if (code !== 0) return 'install_failed'\n return 'success'\n } finally {\n // Ensure we always release the lock\n releaseLock()\n }\n}\n\nexport type PackageManager = 'npm' | 'bun'\n\nexport async function detectPackageManager(): Promise<PackageManager> {\n // Respect explicit override if provided later via config/env (future-proof)\n try {\n // Heuristic 1: npm available and global root resolvable\n const npmRoot = await execFileNoThrow('npm', ['-g', 'root'])\n if (npmRoot.code === 0 && npmRoot.stdout.trim()) {\n return 'npm'\n }\n } catch {}\n\n try {\n // Heuristic 2: running on a system with bun and installed path hints bun\n const bunVer = await execFileNoThrow('bun', ['--version'])\n if (bunVer.code === 0) {\n // BUN_INSTALL defaults to ~/.bun; if our package lives under that tree, prefer bun\n // If npm not detected but bun is available, choose bun\n return 'bun'\n }\n } catch {}\n\n // Default to npm when uncertain\n return 'npm'\n}\n\nfunction runStreaming(cmd: string, args: string[]): Promise<number> {\n return new Promise(resolve => {\n // \u6253\u5370\u6B63\u5728\u4F7F\u7528\u7684\u5305\u7BA1\u7406\u5668\u4E0E\u547D\u4EE4\uFF0C\u589E\u5F3A\u900F\u660E\u5EA6\n try {\n // eslint-disable-next-line no-console\n console.log(`> ${cmd} ${args.join(' ')}`)\n } catch {}\n\n const child = spawn(cmd, args, {\n stdio: 'inherit',\n env: process.env,\n })\n child.on('close', code => resolve(code ?? 0))\n child.on('error', () => resolve(1))\n })\n}\n\n/**\n * Generate human-friendly update commands for the detected package manager.\n * Also includes an alternative manager command as fallback for users.\n */\nexport async function getUpdateCommandSuggestions(): Promise<string[]> {\n // Prefer Bun first, then npm (consistent, simple UX). Include @latest.\n return [\n `bun add -g ${MACRO.PACKAGE_URL}@latest`,\n `npm install -g ${MACRO.PACKAGE_URL}@latest`,\n ]\n}\n\n/**\n * Non-blocking update notifier (daily)\n * - Respects CI and disabled auto-updater\n * - Uses env.hasInternetAccess() to quickly skip offline cases\n * - Stores last check timestamp + last suggested version in global config\n */\nexport async function checkAndNotifyUpdate(): Promise<void> {\n try {\n if (process.env.NODE_ENV === 'test') return\n if (await isAutoUpdaterDisabled()) return\n if (await env.getIsDocker()) return\n if (!(await env.hasInternetAccess())) return\n\n const config: any = getGlobalConfig()\n const now = Date.now()\n const DAY_MS = 24 * 60 * 60 * 1000\n const lastCheck = Number(config.lastUpdateCheckAt || 0)\n if (lastCheck && now - lastCheck < DAY_MS) return\n\n const latest = await getLatestVersion()\n if (!latest) {\n // Still record the check to avoid spamming\n saveGlobalConfig({ ...config, lastUpdateCheckAt: now })\n return\n }\n\n if (gt(latest, MACRO.VERSION)) {\n // Update stored state and print a low-noise hint\n saveGlobalConfig({\n ...config,\n lastUpdateCheckAt: now,\n lastSuggestedVersion: latest,\n })\n const suggestions = await getUpdateCommandSuggestions()\n console.log(`New version available: ${latest} (current: ${MACRO.VERSION})`)\n console.log('Run the following command to update:')\n for (const command of suggestions) {\n console.log(` ${command}`)\n }\n } else {\n saveGlobalConfig({ ...config, lastUpdateCheckAt: now })\n }\n } catch (error) {\n // Never block or throw; just log and move on\n logError(`update-notify: ${error}`)\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["hasPermissions", "code"]
|
|
7
7
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@shareai-lab/kode",
|
|
3
|
-
"version": "1.1.
|
|
3
|
+
"version": "1.1.23",
|
|
4
4
|
"bin": {
|
|
5
5
|
"kode": "cli.js",
|
|
6
6
|
"kwa": "cli.js",
|
|
@@ -51,6 +51,8 @@
|
|
|
51
51
|
"@commander-js/extra-typings": "^13.1.0",
|
|
52
52
|
"@inkjs/ui": "^2.0.0",
|
|
53
53
|
"@modelcontextprotocol/sdk": "^1.15.1",
|
|
54
|
+
"@shareai-lab/kode": "1.1.19-dev.1",
|
|
55
|
+
"@statsig/client-core": "^3.18.2",
|
|
54
56
|
"@statsig/js-client": "^3.18.2",
|
|
55
57
|
"@types/lodash-es": "^4.17.12",
|
|
56
58
|
"@types/react": "^19.1.8",
|
|
@@ -75,6 +77,8 @@
|
|
|
75
77
|
"lru-cache": "^11.1.0",
|
|
76
78
|
"marked": "^15.0.12",
|
|
77
79
|
"nanoid": "^5.1.5",
|
|
80
|
+
"node-fetch": "^3.3.2",
|
|
81
|
+
"node-html-parser": "^7.0.1",
|
|
78
82
|
"openai": "^4.104.0",
|
|
79
83
|
"react": "18.3.1",
|
|
80
84
|
"semver": "^7.7.2",
|
|
@@ -83,6 +87,7 @@
|
|
|
83
87
|
"string-width": "^7.2.0",
|
|
84
88
|
"strip-ansi": "^7.1.0",
|
|
85
89
|
"tsx": "^4.20.3",
|
|
90
|
+
"turndown": "^7.2.0",
|
|
86
91
|
"undici": "^7.11.0",
|
|
87
92
|
"wrap-ansi": "^9.0.0",
|
|
88
93
|
"zod": "^3.25.76",
|