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.
package/dist/build-info.json
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
|
|
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
|
|
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 =
|
|
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 =
|
|
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,9 +1,8 @@
|
|
|
1
1
|
#!/usr/bin/env bash
|
|
2
2
|
# =================================================================
|
|
3
3
|
# OpenClaw 深度集成初始化工具 (create-instance.sh)
|
|
4
|
-
#
|
|
5
|
-
#
|
|
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
|
-
|
|
28
|
-
|
|
29
|
-
|
|
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
|
-
|
|
32
|
-
if
|
|
33
|
-
|
|
34
|
-
echo -e "${
|
|
35
|
-
|
|
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
|
-
|
|
43
|
-
|
|
44
|
-
echo -e "${
|
|
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
|
-
|
|
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
|
-
#
|
|
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
|
-
# ---
|
|
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
|
-
# ---
|
|
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
|
-
# ---
|
|
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
|
-
# ---
|
|
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
|
-
# ---
|
|
98
|
-
echo -e "${BLUE}🔄
|
|
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
|
-
# ---
|
|
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
|
-
# ---
|
|
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
|
-
|
|
254
|
-
|
|
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
|
-
"
|
|
262
|
-
"
|
|
263
|
-
"
|
|
242
|
+
"gateway": {
|
|
243
|
+
"mode": "local",
|
|
244
|
+
"port": ${PORT},
|
|
245
|
+
"bind": "127.0.0.1",
|
|
246
|
+
"autoStart": true
|
|
264
247
|
},
|
|
265
|
-
"
|
|
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
|
-
"
|
|
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
|
-
"
|
|
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
|
-
"
|
|
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
|
-
"
|
|
335
|
-
"
|
|
291
|
+
"kimi": {
|
|
292
|
+
"apiKey": "${KIMI_KEY}",
|
|
293
|
+
"baseURL": "https://api.moonshot.cn/v1",
|
|
294
|
+
"models": ["kimi/moonshot-v1-8k"]
|
|
336
295
|
},
|
|
337
|
-
"
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
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
|
-
|
|
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
|
-
# ---
|
|
522
|
-
echo -e "${
|
|
523
|
-
|
|
524
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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}"
|