openclaw-multi-auto 1.5.7 → 1.5.8

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.
@@ -1,5 +1,5 @@
1
1
  {
2
- "version": "1.5.7",
3
- "commit": "2b03a8323394f32899c77e83d9e49d24bdacdcb2",
4
- "builtAt": "2026-03-15T02:39:23.584Z"
2
+ "version": "1.5.8",
3
+ "commit": "d59a4444ec442856e44d6d919acf9d12d5bc0acd",
4
+ "builtAt": "2026-03-15T02:48:49.454Z"
5
5
  }
@@ -1 +1 @@
1
- 68cf9c5cbc9928498c8a9feff7725c6aa76f195d5c300f32ec49facb8437b9a3
1
+ 2d05c326c20e3a4286d096f6adfb7cdec3c32aff6cde0c9f0a59fb5ee06d681c
@@ -5,7 +5,7 @@ import fs, { constants } from "node:fs";
5
5
  import os from "node:os";
6
6
  import chalk, { Chalk } from "chalk";
7
7
  import { Logger } from "tslog";
8
- import json5 from "json5";
8
+ import JSON5 from "json5";
9
9
  import util, { promisify } from "node:util";
10
10
  import fs$1 from "node:fs/promises";
11
11
  import process$1 from "node:process";
@@ -400,7 +400,7 @@ function readLoggingConfig() {
400
400
  try {
401
401
  if (!fs.existsSync(configPath)) return;
402
402
  const raw = fs.readFileSync(configPath, "utf-8");
403
- const logging = json5.parse(raw)?.logging;
403
+ const logging = JSON5.parse(raw)?.logging;
404
404
  if (!logging || typeof logging !== "object" || Array.isArray(logging)) return;
405
405
  return logging;
406
406
  } catch {
@@ -4481,7 +4481,7 @@ function resolveOpenClawManifestBlock(params) {
4481
4481
  const raw = getFrontmatterString(params.frontmatter, params.key ?? "metadata");
4482
4482
  if (!raw) return;
4483
4483
  try {
4484
- const parsed = json5.parse(raw);
4484
+ const parsed = JSON5.parse(raw);
4485
4485
  if (!parsed || typeof parsed !== "object") return;
4486
4486
  const manifestKeys = [MANIFEST_KEY, ...LEGACY_MANIFEST_KEYS];
4487
4487
  for (const key of manifestKeys) {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "openclaw-multi-auto",
3
- "version": "1.5.7",
3
+ "version": "1.5.8",
4
4
  "description": "Multi-channel AI gateway with extensible messaging integrations",
5
5
  "keywords": [],
6
6
  "homepage": "https://github.com/openclaw/openclaw#readme",
@@ -1,9 +1,8 @@
1
1
  #!/usr/bin/env bash
2
2
  # =================================================================
3
3
  # OpenClaw 深度集成初始化工具 (create-instance.sh)
4
- # 功能:创建实例目录、复制文件、安装依赖、配置模型、启动服务
5
- # 增强:支持环境变量 OPENCLAW_SOURCE_DIR 定位源码,自动复制 docs 到根目录供 TUI 使用
6
- # 增强:支持 --profile 命令行习惯,自动配置 profile 软链接
4
+ # 功能:安装全局运行时、安装扩展、创建实例目录、复制文件、配置模型、启动服务
5
+ # 正确顺序:安装全局 openclaw 安装 page-action-cache 创建实例
7
6
  # =================================================================
8
7
 
9
8
  set -e
@@ -15,75 +14,130 @@ YELLOW='\033[1;33m'
15
14
  BLUE='\033[0;34m'
16
15
  NC='\033[0m'
17
16
 
18
- # --- 检查必要命令 ---
17
+ # --- 0. 检查必要命令 ---
18
+ echo -e "${BLUE}🔍 检查必要命令...${NC}"
19
19
  for cmd in node pnpm; do
20
20
  if ! command -v "$cmd" &> /dev/null; then
21
21
  echo -e "${RED}❌ 错误: 未找到命令 '$cmd',请先安装${NC}"
22
22
  exit 1
23
23
  fi
24
24
  done
25
+ echo -e "${GREEN}✅ 必要命令检查完成${NC}"
25
26
 
26
- # --- 1. 路径溯源与环境检查 ---
27
- if [ -n "$OPENCLAW_SOURCE_DIR" ] && [ -d "$OPENCLAW_SOURCE_DIR" ]; then
28
- SOURCE_CODE="$OPENCLAW_SOURCE_DIR"
29
- echo -e "${BLUE}📂 使用环境变量指定的源码路径: $SOURCE_CODE${NC}"
27
+ # --- 1. 设置全局目录 ---
28
+ GLOBAL_DIR="$HOME/openclaw-runtime"
29
+ echo -e "${BLUE}📂 全局运行时目录: $GLOBAL_DIR${NC}"
30
+
31
+ # 创建全局目录
32
+ if [ ! -d "$GLOBAL_DIR" ]; then
33
+ echo -e "${YELLOW}📁 创建全局目录...${NC}"
34
+ mkdir -p "$GLOBAL_DIR"
35
+ fi
36
+
37
+ cd "$GLOBAL_DIR"
38
+ echo -e "${GREEN}✅ 切换到全局目录: $(pwd)${NC}"
39
+
40
+ # --- 2. 安装全局 openclaw-multi-auto ---
41
+ echo -e "${BLUE}🤖 正在检查/安装全局 openclaw-multi-auto...${NC}"
42
+
43
+ # 检查是否已安装 openclaw-multi-auto
44
+ if pnpm list openclaw-multi-auto 2>&1 | grep -q "openclaw-multi-auto"; then
45
+ echo -e "${YELLOW}⚠️ openclaw-multi-auto 已在全局安装${NC}"
46
+ echo -e "${YELLOW}💡 如需更新,请使用: pnpm install openclaw-multi-auto@latest${NC}"
30
47
  else
31
- LOCAL_SOURCE="$HOME/openclaw-runtime/node_modules/openclaw-multi-auto"
32
- if [ -d "$LOCAL_SOURCE" ]; then
33
- SOURCE_CODE="$LOCAL_SOURCE"
34
- echo -e "${BLUE}📂 使用本地 pnpm 安装的源码路径: $SOURCE_CODE${NC}"
35
- else
36
- GLOBAL_PREFIX=$(npm config get prefix)
37
- SOURCE_CODE="${GLOBAL_PREFIX}/lib/node_modules/openclaw-multi-auto"
38
- echo -e "${YELLOW}⚠️ 未找到环境变量和本地路径,尝试使用全局安装路径: $SOURCE_CODE${NC}"
48
+ echo -e "${YELLOW}📦 正在安装 openclaw-multi-auto 到全局目录...${NC}"
49
+ if ! pnpm install openclaw-multi-auto --registry=https://registry.npmmirror.com 2>&1; then
50
+ echo -e "${RED}❌ openclaw-multi-auto 安装失败${NC}"
51
+ echo -e "${YELLOW}💡 提示: 请检查网络连接或包名是否正确${NC}"
52
+ exit 1
39
53
  fi
54
+ echo -e "${GREEN}✅ openclaw-multi-auto 安装成功${NC}"
40
55
  fi
41
56
 
42
- if [ ! -d "$SOURCE_CODE" ]; then
43
- echo -e "${RED}❌ 错误: 未找到 OpenClaw 源码目录: $SOURCE_CODE${NC}"
44
- echo -e "${YELLOW}请确认已正确安装 openclaw-multi-auto,或设置 OPENCLAW_SOURCE_DIR 环境变量指向正确路径。${NC}"
57
+ # 检查全局安装结果
58
+ if [ ! -d "node_modules/openclaw-multi-auto" ]; then
59
+ echo -e "${RED} 严重错误: openclaw-multi-auto 未正确安装到 node_modules${NC}"
45
60
  exit 1
46
61
  fi
47
62
 
48
- INSTANCES_BASE=$(eval echo "$HOME/openclaws")
63
+ SOURCE_CODE="$GLOBAL_DIR/node_modules/openclaw-multi-auto"
64
+ echo -e "${GREEN}✅ 全局 openclaw-multi-auto 路径: $SOURCE_CODE${NC}"
65
+
66
+ # --- 3. 安装 page-action-cache 扩展到全局目录 ---
67
+ echo -e "${BLUE}🤖 正在安装 page-action-cache 扩展到全局目录...${NC}"
68
+
69
+ # 创建 extensions 目录
70
+ EXTENSIONS_DIR="$GLOBAL_DIR/extensions"
71
+ if [ ! -d "$EXTENSIONS_DIR" ]; then
72
+ echo -e "${YELLOW}📁 创建 extensions 目录...${NC}"
73
+ mkdir -p "$EXTENSIONS_DIR"
74
+ fi
49
75
 
50
- # --- 2. 获取实例名称 ---
76
+ # 检查 page-action-cache 是否已安装
77
+ if [ -d "$EXTENSIONS_DIR/page-action-cache" ]; then
78
+ echo -e "${YELLOW}⚠️ page-action-cache 已在全局安装${NC}"
79
+ echo -e "${YELLOW}💡 如需更新,请手动更新${NC}"
80
+ else
81
+ echo -e "${YELLOW}📦 正在安装 page-action-cache 到全局 extensions 目录...${NC}"
82
+ if ! pnpm add page-action-cache --registry=https://registry.npmmirror.com 2>&1; then
83
+ echo -e "${RED}❌ page-action-cache 安装失败${NC}"
84
+ echo -e "${YELLOW}💡 提示: 请检查网络连接或包名是否正确${NC}"
85
+ exit 1
86
+ fi
87
+
88
+ # 复制 page-action-cache 到 extensions 目录
89
+ if [ -d "node_modules/page-action-cache" ]; then
90
+ echo -e "${YELLOW}📁 复制 page-action-cache 到 extensions 目录...${NC}"
91
+ cp -r "node_modules/page-action-cache" "$EXTENSIONS_DIR/"
92
+ echo -e "${GREEN}✅ page-action-cache 安装并复制成功${NC}"
93
+ else
94
+ echo -e "${YELLOW}⚠️ page-action-cache 已安装但未在 node_modules 中找到${NC}"
95
+ fi
96
+ fi
97
+
98
+ echo -e "${GREEN}✅ 全局扩展安装完成${NC}"
99
+ echo -e "${BLUE}📝 全局运行时已就绪${NC}"
100
+
101
+ # --- 4. 获取实例名称 ---
51
102
  INSTANCE_NAME="${1:-}"
52
103
  if [ -z "$INSTANCE_NAME" ]; then
53
104
  read -p "请输入实例名称 (默认: yunwei): " INPUT_NAME
54
105
  INSTANCE_NAME="${INPUT_NAME:-yunwei}"
55
106
  fi
56
107
 
57
- # --- 3. 获取端口 ---
108
+ # --- 5. 获取端口 ---
58
109
  if [ -n "$2" ]; then
59
110
  PORT="$2"
60
111
  else
112
+ INSTANCES_BASE=$(eval echo "$HOME/openclaws")
61
113
  EXISTING_COUNT=$(find "$INSTANCES_BASE" -maxdepth 1 -type d -name "*" ! -name ".*" 2>/dev/null | wc -l | tr -d ' ')
62
114
  PORT=$((18789 + EXISTING_COUNT * 100))
63
115
  fi
64
116
 
65
- # --- 4. 定义实例目录并清理旧残留 ---
117
+ # --- 6. 定义实例目录并清理旧残留 ---
118
+ INSTANCES_BASE=$(eval echo "$HOME/openclaws")
66
119
  INSTANCE_DIR="$INSTANCES_BASE/${INSTANCE_NAME}"
67
120
  DIST_DIR="${INSTANCE_DIR}/dist"
68
121
 
69
122
  echo -e "${BLUE}🏗️ 正在深度初始化实例: ${INSTANCE_NAME} (端口 ${PORT})${NC}"
70
123
  echo "SOURCE_CODE = $SOURCE_CODE"
124
+ echo "GLOBAL_DIR = $GLOBAL_DIR"
71
125
 
72
126
  if [ -d "$INSTANCE_DIR" ]; then
73
127
  echo -e "${YELLOW}⚠️ 清理旧实例目录...${NC}"
74
128
  rm -rf "$INSTANCE_DIR"
75
129
  fi
76
- [ -d "${INSTANCES_BASE}/node_modules" ] && rm -rf "${INSTANCES_BASE}/node_modules"
77
- [ -f "${INSTANCES_BASE}/pnpm-workspace.yaml" ] && rm -f "${INSTANCES_BASE}/pnpm-workspace.yaml"
78
130
 
79
- # --- 5. 完整目录结构创建 ---
131
+ # --- 7. 完整目录结构创建 ---
132
+ echo -e "${YELLOW}📁 创建实例目录结构...${NC}"
80
133
  mkdir -p "${INSTANCE_DIR}/config" "${INSTANCE_DIR}/tmp" "${INSTANCE_DIR}/workspace"
81
134
  mkdir -p "${INSTANCE_DIR}/state"
82
135
  mkdir -p "${INSTANCE_DIR}/workspace/memory"
83
136
  mkdir -p "${INSTANCE_DIR}/agents/main/agent/sessions"
84
137
  mkdir -p "${INSTANCE_DIR}/credentials" "${INSTANCE_DIR}/media"
138
+ echo -e "${GREEN}✅ 目录结构创建完成${NC}"
85
139
 
86
- # --- 6. 动态提取母本版本号 ---
140
+ # --- 8. 动态提取母本版本号 ---
87
141
  echo -e "${BLUE}🔍 正在提取母本版本信息...${NC}"
88
142
  PKG_PATH="${SOURCE_CODE}/package.json"
89
143
  if [ -f "$PKG_PATH" ]; then
@@ -94,13 +148,15 @@ else
94
148
  echo -e "${YELLOW}⚠️ 未找到 package.json,使用兜底版本: ${CURRENT_VERSION}${NC}"
95
149
  fi
96
150
 
97
- # --- 7. 核心代码同步 (完整复制) ---
98
- echo -e "${BLUE}🔄 完整复制 OpenClaw 源码目录...${NC}"
151
+ # --- 9. 从全局目录复制到实例目录 ---
152
+ echo -e "${BLUE}🔄 从全局运行时复制到实例目录...${NC}"
99
153
 
100
154
  # 定义需要复制的目录和文件
101
155
  DIRS_TO_COPY="dist extensions scripts skills docs ui assets"
102
156
  FILES_TO_COPY="package.json openclaw.mjs"
103
157
 
158
+ cd "$SOURCE_CODE"
159
+
104
160
  # 复制所有必要目录
105
161
  for dir in $DIRS_TO_COPY; do
106
162
  if [ -d "${SOURCE_CODE}/${dir}" ]; then
@@ -130,17 +186,13 @@ fi
130
186
 
131
187
  echo -e "${GREEN}✅ 核心代码复制完成${NC}"
132
188
 
189
+ # --- 10. 安装实例依赖 ---
133
190
  export SHARP_BINARY_HOST="https://npmmirror.com/mirrors/sharp"
134
191
  export SHARP_LIBVIPS_BINARY_HOST="https://npmmirror.com/mirrors/sharp-libvips"
135
192
 
136
193
  cd "${INSTANCE_DIR}"
137
194
  echo -e "${YELLOW}📍 安装路径确认: $(pwd)${NC}"
138
195
 
139
- if [ -f "${INSTANCES_BASE}/pnpm-workspace.yaml" ]; then
140
- echo -e "${YELLOW}⚠️ 检测到 ${INSTANCES_BASE}/pnpm-workspace.yaml,临时重命名以避免干扰${NC}"
141
- mv "${INSTANCES_BASE}/pnpm-workspace.yaml" "${INSTANCES_BASE}/pnpm-workspace.yaml.bak.$(date +%s)"
142
- fi
143
-
144
196
  echo -e "${BLUE}📦 安装依赖包...${NC}"
145
197
  pnpm install --shamefully-hoist --ignore-workspace --registry=https://registry.npmmirror.com
146
198
 
@@ -150,74 +202,7 @@ if [ ! -d "node_modules" ]; then
150
202
  fi
151
203
  echo -e "${GREEN}✅ 依赖安装成功,node_modules 已生成在实例目录${NC}"
152
204
 
153
- # --- 8.5. 安装 page-action-cache 扩展到全局目录 ---
154
- echo -e "${BLUE}🤖 正在安装 page-action-cache 扩展到全局目录...${NC}"
155
-
156
- # 全局目录路径
157
- GLOBAL_DIR="$HOME/openclaw-runtime"
158
- echo -e "${BLUE}📂 全局目录: $GLOBAL_DIR${NC}"
159
-
160
- # 检测可用的包管理器
161
- if command -v pnpm &> /dev/null; then
162
- PACKAGE_MANAGER="pnpm"
163
- echo -e "${BLUE}📦 检测到包管理器: pnpm${NC}"
164
- elif command -v npm &> /dev/null; then
165
- PACKAGE_MANAGER="npm"
166
- echo -e "${BLUE}📦 检测到包管理器: npm${NC}"
167
- else
168
- echo -e "${RED}❌ 未找到 npm 或 pnpm 命令${NC}"
169
- echo -e "${YELLOW}💡 请先安装 Node.js 和包管理器${NC}"
170
- exit 1
171
- fi
172
-
173
- # 确定包名
174
- PACKAGE_NAME="page-action-cache"
175
-
176
- # 安装 page-action-cache 到全局目录
177
- echo -e "${YELLOW}📦 正在安装 $PACKAGE_NAME 到全局目录...${NC}"
178
- cd "$GLOBAL_DIR"
179
-
180
- # 检查全局安装状态
181
- if $PACKAGE_MANAGER list $PACKAGE_NAME 2>&1 | grep -q "$PACKAGE_NAME"; then
182
- echo -e "${YELLOW}⚠️ $PACKAGE_NAME 已在全局安装${NC}"
183
- echo -e "${YELLOW}💡 如需更新,请使用: $PACKAGE_MANAGER install $PACKAGE_NAME@latest${NC}"
184
- else
185
- # 尝试从 npm 安装
186
- echo -ne "${YELLOW}📦 尝试从 npm registry 安装 $PACKAGE_NAME...${NC}"
187
- if [ "$PACKAGE_MANAGER" = "pnpm" ]; then
188
- pnpm add $PACKAGE_NAME --registry=https://registry.npmmirror.com 2>&1
189
- else
190
- npm install $PACKAGE_NAME --registry=https://registry.npmmirror.com --no-save 2>&1
191
- fi
192
-
193
- if [ $? -eq 0 ]; then
194
- echo -e "${GREEN}✅ $PACKAGE_NAME 安装成功到全局目录${NC}"
195
- else
196
- echo -e "${RED}❌ $PACKAGE_NAME 安装失败${NC}"
197
- echo -e "${YELLOW}💡 提示: 请检查网络连接或包名是否正确${NC}"
198
- fi
199
- fi
200
-
201
- # 检查全局安装结果
202
- INSTALLED=$($PACKAGE_MANAGER list $PACKAGE_NAME 2>&1 | grep -q "$PACKAGE_NAME")
203
- if [ $? -eq 0 ]; then
204
- echo -e "${GREEN}✅ $PACKAGE_NAME 已安装到全局目录${NC}"
205
-
206
- # 获取安装信息
207
- PACKAGE_INFO=$($PACKAGE_MANAGER list $PACKAGE_NAME 2>&1)
208
- echo -e "${BLUE}📦 $PACKAGE_NAME 全局安装信息:${NC}"
209
- echo "$PACKAGE_INFO"
210
- else
211
- echo -e "${YELLOW}⚠️ $PACKAGE_NAME 安装状态不确定${NC}"
212
- echo -e "${YELLOW}💡 提示: 请手动检查安装状态: $PACKAGE_MANAGER list $PACKAGE_NAME --global${NC}"
213
- fi
214
-
215
- echo -e "${GREEN}✅ page-action-cache 全局扩展安装完成${NC}"
216
- echo -e "${BLUE}📝 yunwei 实例将从全局目录复制代码和依赖${NC}"
217
- echo -e "${YELLOW}🔄 多实例架构:全局运行时 + 实例从运行时复制${NC}"
218
-
219
- # 返回到实例目录
220
- cd "${INSTANCE_DIR}"
205
+ # --- 11. 模型配置 ---
221
206
  MODEL_PROVIDER="${3:-}"
222
207
  if [ -z "$MODEL_PROVIDER" ]; then
223
208
  echo -e "${BLUE}--- 请选择模型供应商 ---${NC}"
@@ -238,7 +223,7 @@ if [ -z "$API_KEY" ]; then
238
223
  read -p "请输入您的 ${MODEL_PROVIDER} API Key: " API_KEY
239
224
  fi
240
225
 
241
- # --- 9. 生成深度配置文件 ---
226
+ # --- 12. 生成深度配置文件 ---
242
227
  DEFAULT_MODEL="zai/glm-5"
243
228
  ZAI_KEY=""; DS_KEY=""; MX_KEY=""; KIMI_KEY=""; QWEN_KEY=""
244
229
 
@@ -248,344 +233,88 @@ case "$MODEL_PROVIDER" in
248
233
  minimax) PRIMARY_MODEL="minimax-cn/MiniMax-M2.5-highspeed"; MX_KEY="$API_KEY" ;;
249
234
  kimi) PRIMARY_MODEL="kimi/moonshot-v1-8k"; KIMI_KEY="$API_KEY" ;;
250
235
  qwen) PRIMARY_MODEL="qwen/qwen-plus"; QWEN_KEY="$API_KEY" ;;
236
+ *) PRIMARY_MODEL="zai/glm-5"; ZAI_KEY="$API_KEY" ;;
251
237
  esac
252
238
 
253
- if date -u +"%Y-%m-%dT%H:%M:%S.%3NZ" 2>/dev/null; then
254
- TIMESTAMP=$(date -u +"%Y-%m-%dT%H:%M:%S.%3NZ")
255
- else
256
- TIMESTAMP=$(date -u +"%Y-%m-%dT%H:%M:%S.000Z")
257
- fi
258
-
259
- cat > "${INSTANCE_DIR}/config/openclaw.json" << EOF
239
+ echo -e "${BLUE}⚙️ 生成配置文件...${NC}"
240
+ cat > "${INSTANCE_DIR}/config/deep.json" << EOF
260
241
  {
261
- "meta": {
262
- "lastTouchedVersion": "${CURRENT_VERSION}",
263
- "lastTouchedAt": "${TIMESTAMP}"
242
+ "gateway": {
243
+ "mode": "local",
244
+ "port": ${PORT},
245
+ "bind": "127.0.0.1",
246
+ "autoStart": true
264
247
  },
265
- "models": {
248
+ "plugins": {
249
+ "entries": {
250
+ "page-action-cache": {
251
+ "path": "${INSTANCE_DIR}/extensions/page-action-cache",
252
+ "enabled": true,
253
+ "config": {
254
+ "enabled": true,
255
+ "autoUseCache": true,
256
+ "scenarioRecognitionEnabled": true,
257
+ "llmClassificationThreshold": 0.8,
258
+ "cacheLevelStrategy": "auto",
259
+ "defaultCacheLevel": "L1",
260
+ "pageChangeDetectionEnabled": true,
261
+ "changeInvalidationThreshold": 0.5,
262
+ "invalidationStrategy": "soft",
263
+ "variableExtractionEnabled": true,
264
+ "allowUserConfirmVariables": true,
265
+ "allowUserForcedRefresh": true,
266
+ "enableUserCacheErrorReport": true,
267
+ "trackExecutionStats": true,
268
+ "statsUpdateInterval": 60000
269
+ }
270
+ }
271
+ }
272
+ },
273
+ "model": {
274
+ "defaultProvider": "${MODEL_PROVIDER}",
266
275
  "providers": {
267
276
  "zai": {
268
- "baseUrl": "https://open.bigmodel.cn/api/paas/v4",
269
277
  "apiKey": "${ZAI_KEY}",
270
- "api": "openai-completions",
271
- "models": [
272
- {
273
- "id": "glm-5",
274
- "name": "GLM-5",
275
- "input": ["text"],
276
- "contextWindow": 32768,
277
- "maxTokens": 4096
278
- }
279
- ]
278
+ "baseURL": "https://open.bigmodel.cn/api/paas/v4",
279
+ "models": ["zai/glm-5"]
280
280
  },
281
281
  "deepseek": {
282
- "baseUrl": "https://api.deepseek.com",
283
282
  "apiKey": "${DS_KEY}",
284
- "api": "openai-completions",
285
- "models": [
286
- {
287
- "id": "deepseek-chat",
288
- "name": "DeepSeek-Chat",
289
- "input": ["text"],
290
- "contextWindow": 32768,
291
- "maxTokens": 4096
292
- }
293
- ]
283
+ "baseURL": "https://api.deepseek.com/v1",
284
+ "models": ["deepseek/deepseek-chat"]
294
285
  },
295
286
  "minimax-cn": {
296
- "baseUrl": "https://api.minimax.chat/v1",
297
287
  "apiKey": "${MX_KEY}",
298
- "api": "openai-completions",
299
- "models": [
300
- {
301
- "id": "MiniMax-M2.5-highspeed",
302
- "name": "MiniMax-M2.5-highspeed",
303
- "input": ["text"],
304
- "cost": { "input": 0.000002, "output": 0.000006 },
305
- "contextWindow": 65536,
306
- "maxTokens": 8192
307
- }
308
- ]
309
- },
310
- "ollama": {
311
- "baseUrl": "http://127.0.0.1:11434",
312
- "api": "ollama",
313
- "models": [
314
- {
315
- "id": "qwen3-vl:8b",
316
- "name": "Qwen3-VL-8B",
317
- "input": ["text", "image"],
318
- "contextWindow": 32768,
319
- "maxTokens": 4096
320
- }
321
- ]
322
- }
323
- }
324
- },
325
- "agents": {
326
- "defaults": {
327
- "model": { "primary": "${PRIMARY_MODEL}" },
328
- "imageModel": {
329
- "primary": "ollama/qwen3-vl:8b",
330
- "fallbacks": [
331
- "ollama/qwen3-vl:8b"
332
- ]
288
+ "baseURL": "https://api.minimax.chat/v1",
289
+ "models": ["minimax-cn/MiniMax-M2.5-highspeed"]
333
290
  },
334
- "models": {
335
- "ollama/qwen3-vl:8b": {}
291
+ "kimi": {
292
+ "apiKey": "${KIMI_KEY}",
293
+ "baseURL": "https://api.moonshot.cn/v1",
294
+ "models": ["kimi/moonshot-v1-8k"]
336
295
  },
337
- "workspace": "${INSTANCE_DIR}/workspace",
338
- "compaction": { "mode": "safeguard" }
339
- }
340
- },
341
- "commands": { "native": "auto", "restart": false },
342
- "gateway": { "port": ${PORT}, "mode": "local" },
343
- "page-action-cache": {
344
- "enabled": true,
345
- "config": {
346
- "cacheLevelStrategy": "l3-only",
347
- "pageChangeDetectionEnabled": true,
348
- "invalidationStrategy": "hard"
349
- }
296
+ "qwen": {
297
+ "apiKey": "${QWEN_KEY}",
298
+ "baseURL": "https://dashscope.aliyuncs.com/compatible-mode/v1",
299
+ "models": ["qwen/qwen-plus"]
300
+ }
350
301
  }
302
+ }
351
303
  }
352
304
  EOF
353
305
 
354
- # --- 10. 预装可选技能 (失败不影响主流程) ---
355
- echo -e "${BLUE}🤖 正在注入飞书运维机器人引导...${NC}"
356
- SKILLS_DIR="${INSTANCE_DIR}/workspace/skills"
357
- mkdir -p "$SKILLS_DIR"
358
-
359
- SKILL_PKG="jinyu-skill-feishu-config"
360
- REGISTRY="https://registry.npmmirror.com"
361
-
362
- TEMP_DIR=$(mktemp -d)
363
- cd "$TEMP_DIR"
364
-
365
- if npm pack "$SKILL_PKG@latest" --registry="$REGISTRY" --quiet > /dev/null; then
366
- TARBALL=$(ls *.tgz)
367
- mkdir -p "$SKILLS_DIR/feishu-config"
368
- tar -xzf "$TARBALL" -C "$SKILLS_DIR/feishu-config" --strip-components=1
369
- echo -e "${GREEN}✅ 技能 feishu-config 安装成功${NC}"
370
- else
371
- echo -e "${YELLOW}⚠️ 技能 feishu-config 下载失败,跳过(可手动安装)${NC}"
372
- fi
373
-
374
- cd - > /dev/null
375
- rm -rf "$TEMP_DIR"
376
-
377
- # --- 如果实例是 yunwei,安装角色包并初始化 workspace ---
378
- if [ "$INSTANCE_NAME" = "yunwei" ]; then
379
- echo -e "${BLUE}🛠️ 检测到运维实例,正在安装角色包 role-openclaw-yunwei...${NC}"
380
- TEMP_ROLE_DIR=$(mktemp -d)
381
- cd "$TEMP_ROLE_DIR"
382
- if npm pack role-openclaw-yunwei@latest --registry=https://registry.npmmirror.com --quiet > /dev/null; then
383
- TARBALL=$(ls *.tgz)
384
- tar -xzf "$TARBALL" --strip-components=1
385
- rm "$TARBALL"
386
-
387
- # 1. 将 memory、skills 等目录复制到 workspace 根目录(排除 .md 文件)
388
- for item in *; do
389
- if [ -d "$item" ]; then
390
- # 目录(如 memory、skills)复制到 workspace
391
- cp -r "$item" "$INSTANCE_DIR/workspace/"
392
- elif [ -f "$item" ]; then
393
- # 非 .md 文件(如 package.json)也复制到 workspace
394
- cp "$item" "$INSTANCE_DIR/workspace/" 2>/dev/null || true
395
- fi
396
- done
397
-
398
- # 2. 创建实例根目录下的 docs 目录,并将所有 .md 文件复制过去
399
- mkdir -p "$INSTANCE_DIR/docs/reference/templates"
400
- cp *.md "$INSTANCE_DIR/docs/reference/templates/" 2>/dev/null || true
401
-
402
- echo -e "${GREEN}✅ 角色包安装完成,workspace 已初始化${NC}"
403
- else
404
- echo -e "${YELLOW}⚠️ 角色包下载失败,请检查网络或手动安装${NC}"
405
- fi
406
- cd - > /dev/null
407
- rm -rf "$TEMP_ROLE_DIR"
408
- fi
409
-
410
- # --- 设置 profile 链接以便 openclaw --profile 使用 ---
411
- echo -e "${BLUE}🔗 配置 profile 链接以支持 --profile 命令行...${NC}"
412
- PROFILES_DIR="$HOME/.openclaw/profiles"
413
- mkdir -p "$PROFILES_DIR"
414
-
415
- # 配置文件软链接
416
- ln -sf "$INSTANCE_DIR/config/openclaw.json" "$PROFILES_DIR/$INSTANCE_NAME.json"
417
-
418
- # 状态目录软链接(profile 默认状态目录为 ~/.openclaw/profiles/<profile>/)
419
- # 使用 -sfn 以确保目录链接正确
420
- ln -sfn "$INSTANCE_DIR/state" "$PROFILES_DIR/$INSTANCE_NAME"
421
-
422
- echo -e "${GREEN}✅ 现在可以使用 'openclaw --profile $INSTANCE_NAME ...' 管理此实例${NC}"
423
-
424
- # --- 11. 生成增强型启动脚本 ---
425
- cat > "${INSTANCE_DIR}/start.sh" << 'EOF'
426
- #!/usr/bin/env bash
427
- # =================================================================
428
- # OpenClaw 实例启动脚本 (yunwei 优化版)
429
- # 功能:启动网关、运行 TUI 或其他命令
430
- # =================================================================
431
-
432
- # 确保全局安装的 openclaw 命令可用
433
- export PATH="$HOME/.npm-global/bin:$PATH"
434
-
435
- SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
436
- cd "$SCRIPT_DIR"
437
- INSTANCE_NAME=$(basename "$SCRIPT_DIR")
438
-
439
- # 导出核心配置路径
440
- export OPENCLAW_CONFIG_PATH="$SCRIPT_DIR/config/openclaw.json"
441
- export OPENCLAW_TMP_DIR="$SCRIPT_DIR/tmp"
442
- export OPENCLAW_STATE_DIR="$SCRIPT_DIR/state"
443
-
444
- # 从配置中提取 workspace 路径并导出
445
- if [ -f "$OPENCLAW_CONFIG_PATH" ]; then
446
- export OPENCLAW_WORKSPACE=$(node -e "
447
- const cfg = require('$OPENCLAW_CONFIG_PATH');
448
- console.log(cfg.agents?.defaults?.workspace || '');
449
- ")
450
- fi
451
-
452
- # 从配置中导出 API Key(如果存在 env 字段)
453
- if [ -f "$OPENCLAW_CONFIG_PATH" ]; then
454
- eval $(node -e "
455
- const env = require('$OPENCLAW_CONFIG_PATH').env || {};
456
- Object.keys(env).forEach(k => {
457
- if (env[k]) console.log('export ' + k + '=\"' + env[k] + '\"');
458
- });
459
- ")
460
- fi
461
-
462
- # macOS launchd 守护配置(可选)
463
- if [[ "$OSTYPE" == "darwin"* ]]; then
464
- PLIST_LABEL="ai.openclaw.$INSTANCE_NAME"
465
- PLIST_PATH="$HOME/Library/LaunchAgents/$PLIST_LABEL.plist"
466
- if [ ! -f "$PLIST_PATH" ]; then
467
- cat > "$PLIST_PATH" <<EOP
468
- <?xml version="1.0" encoding="UTF-8"?>
469
- <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
470
- <plist version="1.0">
471
- <dict>
472
- <key>Label</key><string>$PLIST_LABEL</string>
473
- <key>ProgramArguments</key><array><string>/bin/bash</string><string>$SCRIPT_DIR/start.sh</string></array>
474
- <key>RunAtLoad</key><true/><key>KeepAlive</key><true/>
475
- <key>WorkingDirectory</key><string>$SCRIPT_DIR</string>
476
- </dict>
477
- </plist>
478
- EOP
479
- launchctl load "$PLIST_PATH" 2>/dev/null || true
480
- fi
481
- fi
482
-
483
- # 定义可执行文件路径:优先使用全局 openclaw,否则用绝对路径 node
484
- if command -v openclaw &> /dev/null; then
485
- OPENCLAW_EXEC="openclaw"
486
- else
487
- # 使用完整复制的目录中的 openclaw.mjs 或 dist/index.js
488
- if [ -f "$SCRIPT_DIR/openclaw.mjs" ]; then
489
- OPENCLAW_EXEC="node $SCRIPT_DIR/openclaw.mjs"
490
- elif [ -f "$SCRIPT_DIR/dist/index.js" ]; then
491
- OPENCLAW_EXEC="node $SCRIPT_DIR/dist/index.js"
492
- else
493
- echo -e "${RED}❌ 错误: 未找到可执行文件${NC}"
494
- exit 1
495
- fi
496
- fi
497
-
498
- # 根据命令行参数执行对应操作
499
- if [ $# -eq 0 ]; then
500
- # 无参数:启动网关
501
- GATEWAY_PORT=$(node -e "console.log(require('$OPENCLAW_CONFIG_PATH').gateway.port)")
502
- echo "🚀 使用 $OPENCLAW_EXEC 启动 OpenClaw-$INSTANCE_NAME 网关..."
503
- exec $OPENCLAW_EXEC gateway run --port "$GATEWAY_PORT"
504
- else
505
- if [ "$1" = "tui" ]; then
506
- # TUI 必须在 workspace 目录下运行,以正确加载模板文件
507
- echo "🚀 启动 OpenClaw-$INSTANCE_NAME TUI (workspace mode)..."
508
- cd "$OPENCLAW_WORKSPACE"
509
- # 使用绝对路径调用 node,避免相对路径问题
510
- exec node "$SCRIPT_DIR/dist/index.js" tui
511
- else
512
- # 其他命令直接在实例目录执行
513
- echo "🚀 使用 $OPENCLAW_EXEC 执行 OpenClaw-$INSTANCE_NAME 命令: $@"
514
- exec $OPENCLAW_EXEC "$@"
515
- fi
516
- fi
517
- EOF
518
-
519
- chmod +x "${INSTANCE_DIR}/start.sh"
306
+ echo -e "${GREEN}✅ 配置文件生成完成${NC}"
520
307
 
521
- # --- 12. 执行拉起 ---
522
- echo -e "${BLUE} 正在后台启动服务...${NC}"
523
- nohup "${INSTANCE_DIR}/start.sh" > "${INSTANCE_DIR}/workspace/server.log" 2>&1 &
524
- PID=$!
308
+ # --- 13. 完成 ---
309
+ echo -e "${GREEN}🎉 ${INSTANCE_NAME} 实例创建完成!${NC}"
310
+ echo -e "${BLUE}📂 实例路径: ${INSTANCE_DIR}${NC}"
311
+ echo -e "${BLUE}🔌 端口: ${PORT}${NC}"
312
+ echo -e "${BLUE}🤖 模型: ${PRIMARY_MODEL}${NC}"
313
+ echo -e "${BLUE}📦 全局运行时: ${GLOBAL_DIR}${NC}"
314
+ echo -e "${BLUE}📝 配置文件: ${INSTANCE_DIR}/config/deep.json${NC}"
525
315
 
526
- sleep 5
316
+ echo -e "${YELLOW}--- 启动命令 ---${NC}"
317
+ echo -e "${BLUE}cd ${INSTANCE_DIR}${NC}"
318
+ echo -e "${BLUE}node openclaw.mjs gateway run --config config/deep.json${NC}"
527
319
 
528
- check_port() {
529
- if command -v lsof &> /dev/null; then
530
- lsof -i :$PORT -sTCP:LISTEN -t &> /dev/null
531
- elif command -v nc &> /dev/null; then
532
- nc -z localhost $PORT 2>/dev/null
533
- elif command -v curl &> /dev/null; then
534
- curl --head --connect-timeout 1 http://localhost:$PORT &> /dev/null
535
- else
536
- return 0
537
- fi
538
- }
539
-
540
- if kill -0 $PID 2>/dev/null; then
541
- if check_port; then
542
- echo -e "${GREEN}================================================${NC}"
543
- echo -e "${GREEN}🎉 实例 [${INSTANCE_NAME}] 初始化成功!${NC}"
544
- echo -e "🌐 本地网关: http://localhost:${PORT}"
545
- echo -e "📄 实时日志: tail -f ${INSTANCE_DIR}/workspace/server.log"
546
- echo -e "${GREEN}================================================${NC}"
547
-
548
- # --- 交互式菜单 ---
549
- if [ -t 0 ]; then
550
- echo
551
- echo -e "${BLUE}请选择下一步操作:${NC}"
552
- echo "1) 启动 TUI (终端交互界面)"
553
- echo "2) 打开 Web UI (浏览器)"
554
- echo "3) 退出"
555
- read -p "请输入数字 [1-3]: " choice
556
- case $choice in
557
- 1)
558
- echo -e "${YELLOW}启动 TUI...${NC}"
559
- cd "$INSTANCE_DIR"
560
- ./start.sh tui
561
- ;;
562
- 2)
563
- TOKEN=$(node -e "console.log(require('$INSTANCE_DIR/config/openclaw.json').gateway?.auth?.token || '')")
564
- if [ -n "$TOKEN" ]; then
565
- URL="http://localhost:$PORT/?token=$TOKEN"
566
- echo -e "${GREEN}请在浏览器中打开: ${BLUE}$URL${NC}"
567
- if [[ "$OSTYPE" == "darwin"* ]]; then
568
- open "$URL"
569
- elif command -v xdg-open &> /dev/null; then
570
- xdg-open "$URL"
571
- fi
572
- else
573
- echo -e "${RED}未找到 token,请手动查看配置文件。${NC}"
574
- fi
575
- ;;
576
- 3)
577
- echo "退出"
578
- exit 0
579
- ;;
580
- *)
581
- echo "无效选择"
582
- ;;
583
- esac
584
- fi
585
- else
586
- echo -e "${YELLOW}⚠️ 进程存在但端口 ${PORT} 未监听(可能启动较慢或检查工具问题),请查看日志: ${INSTANCE_DIR}/workspace/server.log${NC}"
587
- fi
588
- else
589
- echo -e "${RED}❌ 启动失败,请查看日志: tail -n 20 ${INSTANCE_DIR}/workspace/server.log${NC}"
590
- exit 1
591
- fi
320
+ echo -e "${GREEN}🚀 已就绪,可以启动实例!${NC}"