@oflow-ai/oflow-cli 0.1.0 → 0.1.2
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/cli.js +217 -85
- package/dist/commands/index.js +7 -1
- package/dist/index.js +42 -11
- package/dist/updater.d.ts +17 -0
- package/dist/updater.js +167 -0
- package/package.json +1 -1
package/dist/updater.js
ADDED
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.checkForUpdates = checkForUpdates;
|
|
7
|
+
exports.performUpdate = performUpdate;
|
|
8
|
+
const chalk_1 = __importDefault(require("chalk"));
|
|
9
|
+
const child_process_1 = require("child_process");
|
|
10
|
+
const util_1 = require("util");
|
|
11
|
+
const execAsync = (0, util_1.promisify)(child_process_1.exec);
|
|
12
|
+
const PACKAGE_NAME = '@oflow-ai/oflow-cli';
|
|
13
|
+
const NPM_REGISTRY = 'https://registry.npmjs.org';
|
|
14
|
+
/**
|
|
15
|
+
* 获取当前安装的版本
|
|
16
|
+
*/
|
|
17
|
+
async function getCurrentVersion() {
|
|
18
|
+
try {
|
|
19
|
+
const packageJson = require('../../package.json');
|
|
20
|
+
return packageJson.version;
|
|
21
|
+
}
|
|
22
|
+
catch {
|
|
23
|
+
return '0.0.0';
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* 从npm获取最新版本
|
|
28
|
+
*/
|
|
29
|
+
async function getLatestVersion() {
|
|
30
|
+
try {
|
|
31
|
+
const { stdout } = await execAsync(`npm view ${PACKAGE_NAME} version`, {
|
|
32
|
+
timeout: 10000
|
|
33
|
+
});
|
|
34
|
+
return stdout.trim();
|
|
35
|
+
}
|
|
36
|
+
catch {
|
|
37
|
+
// 如果npm命令失败,尝试直接访问registry
|
|
38
|
+
try {
|
|
39
|
+
const https = require('https');
|
|
40
|
+
return new Promise((resolve, reject) => {
|
|
41
|
+
https.get(`${NPM_REGISTRY}/${PACKAGE_NAME.replace('/', '%2F')}/latest`, (res) => {
|
|
42
|
+
let data = '';
|
|
43
|
+
res.on('data', (chunk) => { data += chunk; });
|
|
44
|
+
res.on('end', () => {
|
|
45
|
+
try {
|
|
46
|
+
const json = JSON.parse(data);
|
|
47
|
+
resolve(json.version);
|
|
48
|
+
}
|
|
49
|
+
catch {
|
|
50
|
+
reject(new Error('Failed to parse response'));
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
}).on('error', reject);
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
return '0.0.0';
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* 比较版本号
|
|
63
|
+
*/
|
|
64
|
+
function compareVersions(v1, v2) {
|
|
65
|
+
const parts1 = v1.split('.').map(Number);
|
|
66
|
+
const parts2 = v2.split('.').map(Number);
|
|
67
|
+
for (let i = 0; i < Math.max(parts1.length, parts2.length); i++) {
|
|
68
|
+
const p1 = parts1[i] || 0;
|
|
69
|
+
const p2 = parts2[i] || 0;
|
|
70
|
+
if (p1 > p2)
|
|
71
|
+
return 1;
|
|
72
|
+
if (p1 < p2)
|
|
73
|
+
return -1;
|
|
74
|
+
}
|
|
75
|
+
return 0;
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* 检查是否有更新
|
|
79
|
+
*/
|
|
80
|
+
async function checkForUpdates() {
|
|
81
|
+
const currentVersion = await getCurrentVersion();
|
|
82
|
+
// 检查缓存(每天只检查一次)
|
|
83
|
+
const cacheKey = 'oflow_last_update_check';
|
|
84
|
+
const cachedData = getCachedUpdateInfo(cacheKey);
|
|
85
|
+
if (cachedData && Date.now() - cachedData.timestamp < 24 * 60 * 60 * 1000) {
|
|
86
|
+
return {
|
|
87
|
+
hasUpdate: compareVersions(currentVersion, cachedData.latestVersion) < 0,
|
|
88
|
+
currentVersion,
|
|
89
|
+
latestVersion: cachedData.latestVersion
|
|
90
|
+
};
|
|
91
|
+
}
|
|
92
|
+
const latestVersion = await getLatestVersion();
|
|
93
|
+
// 缓存结果
|
|
94
|
+
setCachedUpdateInfo(cacheKey, {
|
|
95
|
+
latestVersion,
|
|
96
|
+
timestamp: Date.now()
|
|
97
|
+
});
|
|
98
|
+
return {
|
|
99
|
+
hasUpdate: compareVersions(currentVersion, latestVersion) < 0,
|
|
100
|
+
currentVersion,
|
|
101
|
+
latestVersion
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* 执行更新
|
|
106
|
+
*/
|
|
107
|
+
async function performUpdate() {
|
|
108
|
+
console.log(chalk_1.default.cyan('\n🔄 Updating oflow CLI...\n'));
|
|
109
|
+
try {
|
|
110
|
+
const { stdout, stderr } = await execAsync(`npm install -g ${PACKAGE_NAME}@latest`, {
|
|
111
|
+
timeout: 60000
|
|
112
|
+
});
|
|
113
|
+
console.log(chalk_1.default.green('\n✓ oflow CLI updated successfully!'));
|
|
114
|
+
console.log(chalk_1.default.gray('Please restart oflow to use the new version.\n'));
|
|
115
|
+
return { success: true, message: 'Update completed' };
|
|
116
|
+
}
|
|
117
|
+
catch (error) {
|
|
118
|
+
const errorMsg = error instanceof Error ? error.message : String(error);
|
|
119
|
+
console.log(chalk_1.default.red('\n✗ Update failed'));
|
|
120
|
+
console.log(chalk_1.default.gray('You can manually update with: npm install -g @oflow-ai/oflow-cli@latest\n'));
|
|
121
|
+
return { success: false, message: errorMsg };
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* 获取缓存的更新信息
|
|
126
|
+
*/
|
|
127
|
+
function getCachedUpdateInfo(key) {
|
|
128
|
+
try {
|
|
129
|
+
const fs = require('fs');
|
|
130
|
+
const path = require('path');
|
|
131
|
+
const os = require('os');
|
|
132
|
+
const cacheFile = path.join(os.homedir(), '.oflow', 'update-cache.json');
|
|
133
|
+
if (fs.existsSync(cacheFile)) {
|
|
134
|
+
const data = JSON.parse(fs.readFileSync(cacheFile, 'utf-8'));
|
|
135
|
+
return data[key] || null;
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
catch {
|
|
139
|
+
// Ignore cache errors
|
|
140
|
+
}
|
|
141
|
+
return null;
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* 缓存更新信息
|
|
145
|
+
*/
|
|
146
|
+
function setCachedUpdateInfo(key, value) {
|
|
147
|
+
try {
|
|
148
|
+
const fs = require('fs');
|
|
149
|
+
const path = require('path');
|
|
150
|
+
const os = require('os');
|
|
151
|
+
const cacheDir = path.join(os.homedir(), '.oflow');
|
|
152
|
+
const cacheFile = path.join(cacheDir, 'update-cache.json');
|
|
153
|
+
if (!fs.existsSync(cacheDir)) {
|
|
154
|
+
fs.mkdirSync(cacheDir, { recursive: true });
|
|
155
|
+
}
|
|
156
|
+
let data = {};
|
|
157
|
+
if (fs.existsSync(cacheFile)) {
|
|
158
|
+
data = JSON.parse(fs.readFileSync(cacheFile, 'utf-8'));
|
|
159
|
+
}
|
|
160
|
+
data[key] = value;
|
|
161
|
+
fs.writeFileSync(cacheFile, JSON.stringify(data, null, 2));
|
|
162
|
+
}
|
|
163
|
+
catch {
|
|
164
|
+
// Ignore cache errors
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"updater.js","sourceRoot":"","sources":["../src/updater.ts"],"names":[],"mappings":";;;;;AA+EA,0CA4BC;AAKD,sCAmBC;AAnID,kDAA0B;AAC1B,iDAAqC;AACrC,+BAAiC;AAEjC,MAAM,SAAS,GAAG,IAAA,gBAAS,EAAC,oBAAI,CAAC,CAAC;AAElC,MAAM,YAAY,GAAG,qBAAqB,CAAC;AAC3C,MAAM,YAAY,GAAG,4BAA4B,CAAC;AAQlD;;GAEG;AACH,KAAK,UAAU,iBAAiB;IAC9B,IAAI,CAAC;QACH,MAAM,WAAW,GAAG,OAAO,CAAC,oBAAoB,CAAC,CAAC;QAClD,OAAO,WAAW,CAAC,OAAO,CAAC;IAC7B,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,OAAO,CAAC;IACjB,CAAC;AACH,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,gBAAgB;IAC7B,IAAI,CAAC;QACH,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,YAAY,YAAY,UAAU,EAAE;YACrE,OAAO,EAAE,KAAK;SACf,CAAC,CAAC;QACH,OAAO,MAAM,CAAC,IAAI,EAAE,CAAC;IACvB,CAAC;IAAC,MAAM,CAAC;QACP,2BAA2B;QAC3B,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;YAC/B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;gBACrC,KAAK,CAAC,GAAG,CAAC,GAAG,YAAY,IAAI,YAAY,CAAC,OAAO,CAAC,GAAG,EAAE,KAAK,CAAC,SAAS,EAAE,CAAC,GAAQ,EAAE,EAAE;oBACnF,IAAI,IAAI,GAAG,EAAE,CAAC;oBACd,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAa,EAAE,EAAE,GAAG,IAAI,IAAI,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;oBACtD,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;wBACjB,IAAI,CAAC;4BACH,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;4BAC9B,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;wBACxB,CAAC;wBAAC,MAAM,CAAC;4BACP,MAAM,CAAC,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC,CAAC;wBAChD,CAAC;oBACH,CAAC,CAAC,CAAC;gBACL,CAAC,CAAC,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;YACzB,CAAC,CAAC,CAAC;QACL,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,OAAO,CAAC;QACjB,CAAC;IACH,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,eAAe,CAAC,EAAU,EAAE,EAAU;IAC7C,MAAM,MAAM,GAAG,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IACzC,MAAM,MAAM,GAAG,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAEzC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QAChE,MAAM,EAAE,GAAG,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC1B,MAAM,EAAE,GAAG,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC1B,IAAI,EAAE,GAAG,EAAE;YAAE,OAAO,CAAC,CAAC;QACtB,IAAI,EAAE,GAAG,EAAE;YAAE,OAAO,CAAC,CAAC,CAAC;IACzB,CAAC;IACD,OAAO,CAAC,CAAC;AACX,CAAC;AAED;;GAEG;AACI,KAAK,UAAU,eAAe;IACnC,MAAM,cAAc,GAAG,MAAM,iBAAiB,EAAE,CAAC;IAEjD,gBAAgB;IAChB,MAAM,QAAQ,GAAG,yBAAyB,CAAC;IAC3C,MAAM,UAAU,GAAG,mBAAmB,CAAC,QAAQ,CAAC,CAAC;IAEjD,IAAI,UAAU,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,UAAU,CAAC,SAAS,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,CAAC;QAC1E,OAAO;YACL,SAAS,EAAE,eAAe,CAAC,cAAc,EAAE,UAAU,CAAC,aAAa,CAAC,GAAG,CAAC;YACxE,cAAc;YACd,aAAa,EAAE,UAAU,CAAC,aAAa;SACxC,CAAC;IACJ,CAAC;IAED,MAAM,aAAa,GAAG,MAAM,gBAAgB,EAAE,CAAC;IAE/C,OAAO;IACP,mBAAmB,CAAC,QAAQ,EAAE;QAC5B,aAAa;QACb,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;KACtB,CAAC,CAAC;IAEH,OAAO;QACL,SAAS,EAAE,eAAe,CAAC,cAAc,EAAE,aAAa,CAAC,GAAG,CAAC;QAC7D,cAAc;QACd,aAAa;KACd,CAAC;AACJ,CAAC;AAED;;GAEG;AACI,KAAK,UAAU,aAAa;IACjC,OAAO,CAAC,GAAG,CAAC,eAAK,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC,CAAC;IAExD,IAAI,CAAC;QACH,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,kBAAkB,YAAY,SAAS,EAAE;YAClF,OAAO,EAAE,KAAK;SACf,CAAC,CAAC;QAEH,OAAO,CAAC,GAAG,CAAC,eAAK,CAAC,KAAK,CAAC,qCAAqC,CAAC,CAAC,CAAC;QAChE,OAAO,CAAC,GAAG,CAAC,eAAK,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAC,CAAC;QAE1E,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,kBAAkB,EAAE,CAAC;IACxD,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,QAAQ,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACxE,OAAO,CAAC,GAAG,CAAC,eAAK,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC,CAAC;QAC5C,OAAO,CAAC,GAAG,CAAC,eAAK,CAAC,IAAI,CAAC,2EAA2E,CAAC,CAAC,CAAC;QAErG,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,CAAC;IAC/C,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,mBAAmB,CAAC,GAAW;IACtC,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;QACzB,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;QAC7B,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;QAEzB,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,QAAQ,EAAE,mBAAmB,CAAC,CAAC;QAEzE,IAAI,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;YAC7B,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC;YAC7D,OAAO,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC;QAC3B,CAAC;IACH,CAAC;IAAC,MAAM,CAAC;QACP,sBAAsB;IACxB,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;GAEG;AACH,SAAS,mBAAmB,CAAC,GAAW,EAAE,KAAmD;IAC3F,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;QACzB,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;QAC7B,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;QAEzB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,QAAQ,CAAC,CAAC;QACnD,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,mBAAmB,CAAC,CAAC;QAE3D,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7B,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC9C,CAAC;QAED,IAAI,IAAI,GAAwB,EAAE,CAAC;QACnC,IAAI,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;YAC7B,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC;QACzD,CAAC;QAED,IAAI,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;QAClB,EAAE,CAAC,aAAa,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IAC7D,CAAC;IAAC,MAAM,CAAC;QACP,sBAAsB;IACxB,CAAC;AACH,CAAC","sourcesContent":["import chalk from 'chalk';\nimport { exec } from 'child_process';\nimport { promisify } from 'util';\n\nconst execAsync = promisify(exec);\n\nconst PACKAGE_NAME = '@oflow-ai/oflow-cli';\nconst NPM_REGISTRY = 'https://registry.npmjs.org';\n\ninterface UpdateInfo {\n  hasUpdate: boolean;\n  currentVersion: string;\n  latestVersion: string;\n}\n\n/**\n * 获取当前安装的版本\n */\nasync function getCurrentVersion(): Promise<string> {\n  try {\n    const packageJson = require('../../package.json');\n    return packageJson.version;\n  } catch {\n    return '0.0.0';\n  }\n}\n\n/**\n * 从npm获取最新版本\n */\nasync function getLatestVersion(): Promise<string> {\n  try {\n    const { stdout } = await execAsync(`npm view ${PACKAGE_NAME} version`, {\n      timeout: 10000\n    });\n    return stdout.trim();\n  } catch {\n    // 如果npm命令失败，尝试直接访问registry\n    try {\n      const https = require('https');\n      return new Promise((resolve, reject) => {\n        https.get(`${NPM_REGISTRY}/${PACKAGE_NAME.replace('/', '%2F')}/latest`, (res: any) => {\n          let data = '';\n          res.on('data', (chunk: string) => { data += chunk; });\n          res.on('end', () => {\n            try {\n              const json = JSON.parse(data);\n              resolve(json.version);\n            } catch {\n              reject(new Error('Failed to parse response'));\n            }\n          });\n        }).on('error', reject);\n      });\n    } catch {\n      return '0.0.0';\n    }\n  }\n}\n\n/**\n * 比较版本号\n */\nfunction compareVersions(v1: string, v2: string): number {\n  const parts1 = v1.split('.').map(Number);\n  const parts2 = v2.split('.').map(Number);\n\n  for (let i = 0; i < Math.max(parts1.length, parts2.length); i++) {\n    const p1 = parts1[i] || 0;\n    const p2 = parts2[i] || 0;\n    if (p1 > p2) return 1;\n    if (p1 < p2) return -1;\n  }\n  return 0;\n}\n\n/**\n * 检查是否有更新\n */\nexport async function checkForUpdates(): Promise<UpdateInfo> {\n  const currentVersion = await getCurrentVersion();\n  \n  // 检查缓存（每天只检查一次）\n  const cacheKey = 'oflow_last_update_check';\n  const cachedData = getCachedUpdateInfo(cacheKey);\n  \n  if (cachedData && Date.now() - cachedData.timestamp < 24 * 60 * 60 * 1000) {\n    return {\n      hasUpdate: compareVersions(currentVersion, cachedData.latestVersion) < 0,\n      currentVersion,\n      latestVersion: cachedData.latestVersion\n    };\n  }\n\n  const latestVersion = await getLatestVersion();\n  \n  // 缓存结果\n  setCachedUpdateInfo(cacheKey, { \n    latestVersion, \n    timestamp: Date.now() \n  });\n\n  return {\n    hasUpdate: compareVersions(currentVersion, latestVersion) < 0,\n    currentVersion,\n    latestVersion\n  };\n}\n\n/**\n * 执行更新\n */\nexport async function performUpdate(): Promise<{ success: boolean; message: string }> {\n  console.log(chalk.cyan('\\n🔄 Updating oflow CLI...\\n'));\n  \n  try {\n    const { stdout, stderr } = await execAsync(`npm install -g ${PACKAGE_NAME}@latest`, {\n      timeout: 60000\n    });\n    \n    console.log(chalk.green('\\n✓ oflow CLI updated successfully!'));\n    console.log(chalk.gray('Please restart oflow to use the new version.\\n'));\n    \n    return { success: true, message: 'Update completed' };\n  } catch (error) {\n    const errorMsg = error instanceof Error ? error.message : String(error);\n    console.log(chalk.red('\\n✗ Update failed'));\n    console.log(chalk.gray('You can manually update with: npm install -g @oflow-ai/oflow-cli@latest\\n'));\n    \n    return { success: false, message: errorMsg };\n  }\n}\n\n/**\n * 获取缓存的更新信息\n */\nfunction getCachedUpdateInfo(key: string): { latestVersion: string; timestamp: number } | null {\n  try {\n    const fs = require('fs');\n    const path = require('path');\n    const os = require('os');\n    \n    const cacheFile = path.join(os.homedir(), '.oflow', 'update-cache.json');\n    \n    if (fs.existsSync(cacheFile)) {\n      const data = JSON.parse(fs.readFileSync(cacheFile, 'utf-8'));\n      return data[key] || null;\n    }\n  } catch {\n    // Ignore cache errors\n  }\n  return null;\n}\n\n/**\n * 缓存更新信息\n */\nfunction setCachedUpdateInfo(key: string, value: { latestVersion: string; timestamp: number }): void {\n  try {\n    const fs = require('fs');\n    const path = require('path');\n    const os = require('os');\n    \n    const cacheDir = path.join(os.homedir(), '.oflow');\n    const cacheFile = path.join(cacheDir, 'update-cache.json');\n    \n    if (!fs.existsSync(cacheDir)) {\n      fs.mkdirSync(cacheDir, { recursive: true });\n    }\n    \n    let data: Record<string, any> = {};\n    if (fs.existsSync(cacheFile)) {\n      data = JSON.parse(fs.readFileSync(cacheFile, 'utf-8'));\n    }\n    \n    data[key] = value;\n    fs.writeFileSync(cacheFile, JSON.stringify(data, null, 2));\n  } catch {\n    // Ignore cache errors\n  }\n}\n"]}
|