remote-claude 1.0.2 → 1.0.4

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/.env.example CHANGED
@@ -18,6 +18,10 @@ ALLOWED_USERS=ou_xxxxx,ou_yyyyy
18
18
  # 支持多词命令,如:ccr code、/usr/local/bin/claude
19
19
  # CLAUDE_COMMAND=claude
20
20
 
21
+ # Codex CLI 命令(可选,默认 codex)
22
+ # 支持多词命令,如:/usr/local/bin/codex
23
+ # CODEX_COMMAND=codex
24
+
21
25
  # 流式卡片配置(可选)
22
26
  # 单张卡片最多显示的 block 数量,超限自动冻结并创建新卡片(默认 50)
23
27
  # MAX_CARD_BLOCKS=50
@@ -30,3 +34,7 @@ ALLOWED_USERS=ou_xxxxx,ou_yyyyy
30
34
  # 支持: DEBUG / INFO / WARNING / ERROR
31
35
  # SERVER_LOG_LEVEL=INFO
32
36
 
37
+ # SOCKS 代理兼容(可选)
38
+ # 系统有 SOCKS 代理但飞书可直连时,设为 1 绕过代理
39
+ # LARK_NO_PROXY=1
40
+
package/README.md CHANGED
@@ -179,7 +179,6 @@ remote-claude attach <会话名>
179
179
  "im:message:recall",
180
180
  "im:message:update",
181
181
  "search:docs:read",
182
- "search:suite_dataset:readonly",
183
182
  "sheets:spreadsheet.meta:read",
184
183
  "sheets:spreadsheet.meta:write_only",
185
184
  "sheets:spreadsheet:create",
@@ -191,7 +190,6 @@ remote-claude attach <会话名>
191
190
  "task:task:write",
192
191
  "task:task:writeonly",
193
192
  "task:tasklist:read",
194
- "task:tasklist:writeonly",
195
193
  "wiki:wiki:readonly"
196
194
  ]
197
195
  }
package/bin/cdx CHANGED
@@ -16,5 +16,13 @@ fi
16
16
  # 检查飞书配置
17
17
  source "$SCRIPT_DIR/scripts/check-env.sh" "$SCRIPT_DIR"
18
18
 
19
+ # 会话名:第一个参数非 - 开头时作为自定义名,否则用 PWD_时间戳
20
+ if [ -n "$1" ] && [[ "$1" != -* ]]; then
21
+ SESSION_NAME="$1"
22
+ shift
23
+ else
24
+ SESSION_NAME="${PWD}_$(date +%m%d_%H%M%S)"
25
+ fi
26
+
19
27
  uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" lark start
20
- uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "${PWD}_$(date +%m%d_%H%M%S)" --cli codex -- "$@"
28
+ uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "$SESSION_NAME" --cli codex -- "$@"
package/bin/cl CHANGED
@@ -16,5 +16,13 @@ fi
16
16
  # 检查飞书配置
17
17
  source "$SCRIPT_DIR/scripts/check-env.sh" "$SCRIPT_DIR"
18
18
 
19
+ # 会话名:第一个参数非 - 开头时作为自定义名,否则用 PWD_时间戳
20
+ if [ -n "$1" ] && [[ "$1" != -* ]]; then
21
+ SESSION_NAME="$1"
22
+ shift
23
+ else
24
+ SESSION_NAME="${PWD}_$(date +%m%d_%H%M%S)"
25
+ fi
26
+
19
27
  uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" lark start
20
- uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "${PWD}_$(date +%m%d_%H%M%S)" -- --dangerously-skip-permissions --permission-mode=dontAsk "$@"
28
+ uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "$SESSION_NAME" -- --dangerously-skip-permissions --permission-mode=dontAsk "$@"
package/bin/cla CHANGED
@@ -16,5 +16,13 @@ fi
16
16
  # 检查飞书配置
17
17
  source "$SCRIPT_DIR/scripts/check-env.sh" "$SCRIPT_DIR"
18
18
 
19
+ # 会话名:第一个参数非 - 开头时作为自定义名,否则用 PWD_时间戳
20
+ if [ -n "$1" ] && [[ "$1" != -* ]]; then
21
+ SESSION_NAME="$1"
22
+ shift
23
+ else
24
+ SESSION_NAME="${PWD}_$(date +%m%d_%H%M%S)"
25
+ fi
26
+
19
27
  uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" lark start
20
- uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "${PWD}_$(date +%m%d_%H%M%S)" -- "$@"
28
+ uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "$SESSION_NAME" -- "$@"
package/bin/cx CHANGED
@@ -16,5 +16,13 @@ fi
16
16
  # 检查飞书配置
17
17
  source "$SCRIPT_DIR/scripts/check-env.sh" "$SCRIPT_DIR"
18
18
 
19
+ # 会话名:第一个参数非 - 开头时作为自定义名,否则用 PWD_时间戳
20
+ if [ -n "$1" ] && [[ "$1" != -* ]]; then
21
+ SESSION_NAME="$1"
22
+ shift
23
+ else
24
+ SESSION_NAME="${PWD}_$(date +%m%d_%H%M%S)"
25
+ fi
26
+
19
27
  uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" lark start
20
- uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "${PWD}_$(date +%m%d_%H%M%S)" --cli codex -- --dangerously-bypass-approvals-and-sandbox "$@"
28
+ uv run --project "$SCRIPT_DIR" python3 "$SCRIPT_DIR/remote_claude.py" start "$SESSION_NAME" --cli codex -- --dangerously-bypass-approvals-and-sandbox "$@"
package/bin/remote-claude CHANGED
@@ -17,25 +17,16 @@ fi
17
17
  if [ "$1" = "log" ]; then
18
18
  LOG_DIR="/tmp/remote-claude"
19
19
  if [ -n "$2" ]; then
20
- # 指定 session 名,/ . 替换为 _(与 _safe_filename 一致)
21
- SESSION_SAFE=$(echo "$2" | tr '/.' '__')
22
- LOG_FILE="$LOG_DIR/${SESSION_SAFE}_messages.log"
20
+ # 指定 session 名,/ . 空格替换为 _(与 _log_filename 一致)
21
+ SESSION_LOG=$(echo "$2" | tr '/. ' '___')
22
+ LOG_FILE="$LOG_DIR/${SESSION_LOG}_messages.log"
23
23
  else
24
- # 找最后启动的 session(按创建时间排序,排除 lark.pid)
25
- if [[ "$OSTYPE" == "darwin"* ]]; then
26
- # macOS: stat -f "%B" 返回创建时间(秒)
27
- LATEST_PID=$(ls "$LOG_DIR"/_*.pid 2>/dev/null | while read f; do
28
- stat -f "%B:%N" "$f" 2>/dev/null
29
- done | sort -rn | head -1 | cut -d: -f2)
30
- else
31
- # Linux: stat -c "%W" 返回创建时间(秒)
32
- LATEST_PID=$(ls "$LOG_DIR"/_*.pid 2>/dev/null | while read f; do
33
- stat -c "%W:%N" "$f" 2>/dev/null
34
- done | sort -rn | head -1 | cut -d: -f2)
35
- fi
36
- if [ -n "$LATEST_PID" ]; then
37
- SESSION_SAFE=$(basename "$LATEST_PID" .pid)
38
- LOG_FILE="$LOG_DIR/${SESSION_SAFE}_messages.log"
24
+ # 找最后启动的 session(按 .name 文件修改时间排序)
25
+ LATEST_NAME=$(ls -t "$LOG_DIR"/*.name 2>/dev/null | head -1)
26
+ if [ -n "$LATEST_NAME" ]; then
27
+ SESSION_NAME=$(cat "$LATEST_NAME" 2>/dev/null)
28
+ SESSION_LOG=$(echo "$SESSION_NAME" | tr '/. ' '___')
29
+ LOG_FILE="$LOG_DIR/${SESSION_LOG}_messages.log"
39
30
  fi
40
31
  fi
41
32
 
package/client/client.py CHANGED
@@ -111,7 +111,7 @@ class RemoteClient:
111
111
  async def run(self):
112
112
  """运行客户端"""
113
113
  if not await self.connect():
114
- return
114
+ raise SystemExit(1)
115
115
 
116
116
  self.running = True
117
117
  _track_stats('terminal', 'connect', session_name=self.session_name)
package/init.sh CHANGED
@@ -26,6 +26,16 @@ print_error() {
26
26
  echo -e "${RED}✗${NC} $1"
27
27
  }
28
28
 
29
+ # 非交互 sudo:有免密 sudo 则执行(带 5 分钟超时),否则跳过
30
+ _sudo_or_skip() {
31
+ if sudo -n true 2>/dev/null; then
32
+ timeout 300 sudo "$@"
33
+ else
34
+ print_warning "sudo 需要密码,跳过: sudo $*"
35
+ return 1
36
+ fi
37
+ }
38
+
29
39
  print_header() {
30
40
  echo ""
31
41
  echo -e "${GREEN}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${NC}"
@@ -200,96 +210,22 @@ check_tmux() {
200
210
  brew install tmux 2>/dev/null || true
201
211
  elif [[ "$OS" == "Linux" ]]; then
202
212
  if command -v apt-get &> /dev/null; then
203
- sudo apt-get update && sudo apt-get install -y tmux || true
213
+ _sudo_or_skip apt-get update && _sudo_or_skip apt-get install -y tmux || true
204
214
  elif command -v yum &> /dev/null; then
205
- sudo yum install -y tmux || true
215
+ _sudo_or_skip yum install -y tmux || true
206
216
  elif command -v pacman &> /dev/null; then
207
- sudo pacman -Sy --noconfirm tmux || true
217
+ _sudo_or_skip pacman -Sy --noconfirm tmux || true
208
218
  elif command -v apk &> /dev/null; then
209
- sudo apk add --no-cache tmux || true
219
+ _sudo_or_skip apk add --no-cache tmux || true
210
220
  elif command -v zypper &> /dev/null; then
211
- sudo zypper install -y tmux || true
221
+ _sudo_or_skip zypper install -y tmux || true
212
222
  else
213
- print_warning "无法识别包管理器,尝试从源码编译 tmux..."
214
- install_tmux_from_source
215
- return
223
+ print_warning "无法识别包管理器,请手动安装 tmux 或运行 remote-claude deps"
216
224
  fi
217
225
  fi
218
226
  print_success "tmux 安装成功"
219
227
  }
220
228
 
221
- install_tmux_from_source() {
222
- local TMUX_VERSION_TAG="3.6a"
223
- local TMUX_URL="https://github.com/tmux/tmux/releases/download/${TMUX_VERSION_TAG}/tmux-${TMUX_VERSION_TAG}.tar.gz"
224
-
225
- print_warning "包管理器版本不满足要求,尝试从源码编译 tmux ${TMUX_VERSION_TAG}..."
226
-
227
- # 安装编译依赖
228
- if [[ "$OS" == "Darwin" ]]; then
229
- brew install libevent ncurses pkg-config bison 2>/dev/null || true
230
- elif command -v apt-get &> /dev/null; then
231
- sudo apt-get install -y build-essential libevent-dev libncurses5-dev libncursesw5-dev bison pkg-config || true
232
- elif command -v yum &> /dev/null; then
233
- sudo yum groupinstall -y "Development Tools" || true
234
- sudo yum install -y libevent-devel ncurses-devel bison || true
235
- fi
236
-
237
- # 确定安装前缀
238
- local PREFIX="/usr/local"
239
- if ! sudo -n true 2>/dev/null; then
240
- print_warning "无 sudo 权限,将安装到 \$HOME/.local"
241
- PREFIX="$HOME/.local"
242
- fi
243
-
244
- # 创建临时目录,编译完成后清理
245
- local TMPDIR
246
- TMPDIR=$(mktemp -d)
247
- trap "rm -rf '$TMPDIR'" RETURN
248
-
249
- print_warning "下载 tmux-${TMUX_VERSION_TAG}.tar.gz..."
250
- if ! curl -fsSL "$TMUX_URL" -o "$TMPDIR/tmux.tar.gz"; then
251
- print_warning "下载失败,请检查网络或手动安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
252
- WARNINGS+=("tmux 源码下载失败,请手动安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+")
253
- return
254
- fi
255
-
256
- tar -xzf "$TMPDIR/tmux.tar.gz" -C "$TMPDIR"
257
- local SRC_DIR
258
- SRC_DIR=$(find "$TMPDIR" -maxdepth 1 -type d -name "tmux-*" | head -1)
259
-
260
- print_warning "编译 tmux(可能需要几分钟)..."
261
- if ! (cd "$SRC_DIR" && ./configure --prefix="$PREFIX" && make -j"$(nproc 2>/dev/null || echo 2)"); then
262
- print_warning "编译失败,请手动安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
263
- WARNINGS+=("tmux 源码编译失败,请手动安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+")
264
- return
265
- fi
266
-
267
- if [[ "$PREFIX" == "/usr/local" ]]; then
268
- sudo make -C "$SRC_DIR" install || { WARNINGS+=("tmux make install 失败,请手动安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"); return; }
269
- else
270
- make -C "$SRC_DIR" install || { WARNINGS+=("tmux make install 失败,请手动安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"); return; }
271
- # 若 $HOME/.local/bin 不在 PATH 中,自动写入 shell 配置
272
- if [[ ":$PATH:" != *":$HOME/.local/bin:"* ]]; then
273
- export PATH="$HOME/.local/bin:$PATH"
274
- local _RC
275
- if [[ "$(basename "$SHELL")" == "zsh" ]]; then
276
- _RC="$HOME/.zshrc"
277
- else
278
- _RC="$HOME/.bashrc"
279
- fi
280
- local _PATH_LINE='export PATH="$HOME/.local/bin:$PATH"'
281
- if ! grep -qF "$HOME/.local/bin" "$_RC" 2>/dev/null; then
282
- echo "" >> "$_RC"
283
- echo "# remote-claude: tmux 路径" >> "$_RC"
284
- echo "$_PATH_LINE" >> "$_RC"
285
- print_success "已自动将 \$HOME/.local/bin 加入 PATH(写入 $_RC)"
286
- fi
287
- fi
288
- fi
289
-
290
- print_success "tmux ${TMUX_VERSION_TAG} 源码编译安装完成(前缀:${PREFIX})"
291
- }
292
-
293
229
  check_version() {
294
230
  # tmux -V 输出格式:tmux 3.6 或 tmux 3.4a
295
231
  local ver_str
@@ -310,30 +246,27 @@ check_tmux() {
310
246
  print_success "$TMUX_VERSION 已安装(满足 >= ${REQUIRED_MAJOR}.${REQUIRED_MINOR})"
311
247
  return
312
248
  else
313
- print_warning "$TMUX_VERSION 版本过低,需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR} 或更高,正在升级..."
249
+ print_warning "$TMUX_VERSION 版本过低,需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR} 或更高,尝试通过包管理器升级..."
314
250
  install_tmux
315
- # 升级后再次验证,版本仍不满足则走源码编译(跨平台)
316
- if ! check_version; then
317
- install_tmux_from_source
318
- if check_version; then
319
- print_success "tmux 已升级至 $(tmux -V)"
320
- else
321
- print_warning "源码编译后版本仍不满足要求($(tmux -V)),需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
322
- WARNINGS+=("tmux 版本不满足要求($(tmux -V)),需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+,请手动升级")
323
- fi
324
- else
251
+ if check_version; then
325
252
  print_success "tmux 已升级至 $(tmux -V)"
253
+ else
254
+ print_warning "包管理器安装后版本仍不满足要求($(tmux -V)),需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
255
+ print_info "请运行 'remote-claude deps' 从源码编译安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
256
+ WARNINGS+=("tmux 版本不满足要求($(tmux -V)),需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+,请运行 remote-claude deps 升级")
326
257
  fi
327
258
  fi
328
259
  else
329
260
  print_warning "未找到 tmux,正在安装..."
330
261
  install_tmux
331
- if ! check_version; then
332
- install_tmux_from_source
333
- if ! check_version; then
334
- print_warning "源码编译后版本仍不满足要求($(tmux -V)),需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
335
- WARNINGS+=("tmux 版本不满足要求($(tmux -V)),需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+,请手动升级")
336
- fi
262
+ if command -v tmux &> /dev/null && check_version; then
263
+ print_success "tmux 已安装: $(tmux -V)"
264
+ else
265
+ local _cur_ver=""
266
+ command -v tmux &> /dev/null && _cur_ver="(当前: $(tmux -V)"
267
+ print_warning "tmux 安装后版本仍不满足要求${_cur_ver},需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
268
+ print_info "请运行 'remote-claude deps' 从源码编译安装 tmux ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+"
269
+ WARNINGS+=("tmux 版本不满足要求,需要 ${REQUIRED_MAJOR}.${REQUIRED_MINOR}+,请运行 remote-claude deps 升级")
337
270
  fi
338
271
  fi
339
272
  }
@@ -427,6 +427,7 @@ def _build_buttons_v2(options: List[Dict[str, str]]) -> List[Dict[str, Any]]:
427
427
  "value": {
428
428
  "action": "select_option",
429
429
  "value": opt["value"],
430
+ "needs_input": opt.get("needs_input", False),
430
431
  "total": str(total),
431
432
  }
432
433
  }
@@ -770,6 +771,16 @@ def build_stream_card(
770
771
 
771
772
  # === 辅助卡片(保留不变)===
772
773
 
774
+ def _get_display_name(name: str, cwd: str = None) -> str:
775
+ """从会话名和 CWD 获取显示名。自定义会话名直接用,默认路径名取 CWD 末段。"""
776
+ is_default = bool(_re.search(r'_\d{4}_\d{6}$', name))
777
+ if not is_default:
778
+ return name
779
+ if cwd:
780
+ return cwd.rstrip("/").rsplit("/", 1)[-1] or name
781
+ return name
782
+
783
+
773
784
  def _build_session_list_elements(sessions: List[Dict], current_session: Optional[str], session_groups: Optional[Dict[str, str]], page: int = 0) -> List[Dict]:
774
785
  """构建会话列表元素(供 build_menu_card 复用)"""
775
786
  import os
@@ -793,10 +804,7 @@ def _build_session_list_elements(sessions: List[Dict], current_session: Optional
793
804
 
794
805
  status_icon = "🟢" if is_current else "⚪"
795
806
  current_label = "(当前)" if is_current else ""
796
- if cwd:
797
- short_name = cwd.rstrip("/").rsplit("/", 1)[-1] or name
798
- else:
799
- short_name = name
807
+ short_name = _get_display_name(name, cwd)
800
808
 
801
809
  # 构建4行内容:名字、cli类型、启动时间、目录
802
810
  lines = [f"{status_icon} **{short_name}**{current_label}"]
@@ -1022,22 +1030,50 @@ def build_dir_card(target, entries: List[Dict], sessions: List[Dict], tree: bool
1022
1030
 
1023
1031
  if is_dir and depth == 0:
1024
1032
  auto_session = _dir_session_name(full_path)
1025
- group_btn = {
1026
- "tag": "button",
1027
- "text": {"tag": "plain_text", "content": "进入群聊" if (session_groups and auto_session in session_groups) else "创建群聊"},
1028
- "type": "default",
1029
- "behaviors": [{"type": "open_url",
1030
- "default_url": f"https://applink.feishu.cn/client/chat/open?openChatId={session_groups[auto_session]}",
1031
- "android_url": f"https://applink.feishu.cn/client/chat/open?openChatId={session_groups[auto_session]}",
1032
- "ios_url": f"https://applink.feishu.cn/client/chat/open?openChatId={session_groups[auto_session]}",
1033
- "pc_url": f"https://applink.feishu.cn/client/chat/open?openChatId={session_groups[auto_session]}"}]
1034
- if (session_groups and auto_session in session_groups) else
1035
- [{"type": "callback", "value": {
1036
- "action": "dir_new_group",
1037
- "path": full_path,
1038
- "session_name": auto_session
1039
- }}]
1040
- }
1033
+ # 前缀匹配:找 session_groups 中精确匹配或以 auto_session + "_" 开头的条目(取最后一个,即最新的)
1034
+ matched_group_cid = None
1035
+ if session_groups:
1036
+ for sn, cid in session_groups.items():
1037
+ if sn == auto_session or sn.startswith(auto_session + "_"):
1038
+ matched_group_cid = cid
1039
+ if matched_group_cid:
1040
+ action_btns = [
1041
+ {
1042
+ "tag": "button",
1043
+ "text": {"tag": "plain_text", "content": "进入群聊"},
1044
+ "type": "default",
1045
+ "behaviors": [{"type": "open_url",
1046
+ "default_url": f"https://applink.feishu.cn/client/chat/open?openChatId={matched_group_cid}",
1047
+ "android_url": f"https://applink.feishu.cn/client/chat/open?openChatId={matched_group_cid}",
1048
+ "ios_url": f"https://applink.feishu.cn/client/chat/open?openChatId={matched_group_cid}",
1049
+ "pc_url": f"https://applink.feishu.cn/client/chat/open?openChatId={matched_group_cid}"}]
1050
+ }
1051
+ ]
1052
+ else:
1053
+ action_btns = [
1054
+ {
1055
+ "tag": "button",
1056
+ "text": {"tag": "plain_text", "content": "Claude群聊"},
1057
+ "type": "primary",
1058
+ "behaviors": [{"type": "callback", "value": {
1059
+ "action": "dir_new_group",
1060
+ "path": full_path,
1061
+ "session_name": auto_session,
1062
+ "cli_type": "claude"
1063
+ }}]
1064
+ },
1065
+ {
1066
+ "tag": "button",
1067
+ "text": {"tag": "plain_text", "content": "Codex群聊"},
1068
+ "type": "default",
1069
+ "behaviors": [{"type": "callback", "value": {
1070
+ "action": "dir_new_group",
1071
+ "path": full_path,
1072
+ "session_name": auto_session,
1073
+ "cli_type": "codex"
1074
+ }}]
1075
+ }
1076
+ ]
1041
1077
  elements.append({
1042
1078
  "tag": "column_set",
1043
1079
  "flex_mode": "none",
@@ -1060,19 +1096,7 @@ def build_dir_card(target, entries: List[Dict], sessions: List[Dict], tree: bool
1060
1096
  "tag": "column",
1061
1097
  "width": "weighted",
1062
1098
  "weight": 2,
1063
- "elements": [
1064
- {
1065
- "tag": "button",
1066
- "text": {"tag": "plain_text", "content": "Claude"},
1067
- "type": "primary",
1068
- "behaviors": [{"type": "callback", "value": {
1069
- "action": "dir_start",
1070
- "path": full_path,
1071
- "session_name": auto_session
1072
- }}]
1073
- },
1074
- group_btn
1075
- ]
1099
+ "elements": action_btns
1076
1100
  }
1077
1101
  ]
1078
1102
  })
@@ -1145,6 +1169,9 @@ def build_help_card() -> Dict[str, Any]:
1145
1169
  **群聊协作**
1146
1170
  • `/new-group <会话名>` - 创建专属群聊,多人共用同一 Claude
1147
1171
 
1172
+ **按键控制**
1173
+ • `/press <按键>` - 发送按键到会话(如 `/press ctrl+c`、`/press esc`、`/press ctrl+f`)
1174
+
1148
1175
  **其他**
1149
1176
  • `/help` - 显示此帮助
1150
1177
  • `/menu` - 快捷操作面板"""
@@ -48,3 +48,7 @@ LARK_LOG_LEVEL = {
48
48
  "WARNING": 30,
49
49
  "ERROR": 40,
50
50
  }.get(_LARK_LOG_LEVEL, 20) # 默认 INFO
51
+
52
+ # SOCKS 代理兼容(可选,默认 False)
53
+ # 系统有 SOCKS 代理但飞书可直连时,设为 1 绕过代理
54
+ LARK_NO_PROXY = os.getenv("LARK_NO_PROXY", "").strip() in ("1", "true", "yes")