Files
chatgpt-on-wechat/run.sh
2026-05-31 20:11:23 +08:00

1353 lines
54 KiB
Bash
Executable File
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
#!/bin/bash
set -e
# ============================
# CowAgent Management Script
# ============================
# ANSI colors
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[0;33m'
CYAN='\033[0;36m'
BOLD='\033[1m'
NC='\033[0m'
# Emojis
EMOJI_ROCKET="🚀"
EMOJI_COW="🐄"
EMOJI_CHECK="✅"
EMOJI_CROSS="❌"
EMOJI_WARN="⚠️"
EMOJI_STOP="🛑"
EMOJI_WRENCH="🔧"
# Check if using Bash
if [ -z "$BASH_VERSION" ]; then
echo -e "${RED}❌ Please run this script with Bash.${NC}"
exit 1
fi
# ============================
# i18n: install-flow language
# ============================
# UI_LANG controls the language of install prompts/menus. Detected on first run
# (or chosen by the user), defaults to auto-detection. "zh" or "en".
UI_LANG=""
# A terminal we can read from. When the script runs via `curl | bash`, stdin is
# the script pipe (EOF on read), so interactive prompts must read from the tty.
TTY_DEV="/dev/tty"
HAS_TTY=false
if [ -r /dev/tty ] && [ -w /dev/tty ]; then
HAS_TTY=true
fi
# Detect default UI language from environment (best-effort, mirrors common/i18n).
detect_ui_lang() {
local loc=""
# macOS: prefer AppleLocale, which reflects the real UI language
if [ "$(uname)" = "Darwin" ] && command -v defaults &> /dev/null; then
loc=$(defaults read -g AppleLocale 2>/dev/null || true)
fi
[ -z "$loc" ] && loc="${LC_ALL:-${LC_MESSAGES:-${LANG:-}}}"
case "$loc" in
zh* | *zh_* | *_CN* | *_TW* | *_HK* | *Hans* | *Hant*) echo "zh" ;;
*) echo "en" ;;
esac
}
# Translation helper: t <zh_text> <en_text>
t() {
if [ "$UI_LANG" = "en" ]; then
printf '%s' "$2"
else
printf '%s' "$1"
fi
}
# Read a line from the controlling terminal (works under `curl | bash`).
# Usage: tty_read VAR "prompt"
tty_read() {
local __var=$1 __prompt=$2 __input=""
if [ "$HAS_TTY" = true ]; then
# Ensure the tty is in normal line mode. A preceding arrow-key menu
# may have left it in cbreak/-echo mode; without this, `read` could
# return immediately or not echo typed characters.
stty sane < "$TTY_DEV" 2>/dev/null || true
# Print the prompt explicitly (not via read -p, whose prompt can be
# swallowed right after an arrow-key menu) and read from the tty.
# `|| true` so a non-zero read (EOF) does NOT trip `set -e`.
printf '%s' "$__prompt" > /dev/tty
read -r __input < "$TTY_DEV" || true
else
read -r -p "$__prompt" __input || true
fi
printf -v "$__var" '%s' "$__input"
}
# Arrow-key selectable menu with number fallback.
# Usage: select_menu OUT_VAR "Title" "opt1" "opt2" ...
# Result: OUT_VAR is set to the selected index (1-based).
select_menu() {
# Interactive function: never let a non-zero command (read EOF, arithmetic
# evaluating to 0, etc.) abort the caller under `set -e`.
set +e
local __out=$1; shift
local title=$1; shift
local options=("$@")
local count=${#options[@]}
# Initial highlight: MENU_DEFAULT (1-based) if set, else first option.
local cur=0
if [[ "${MENU_DEFAULT:-}" =~ ^[0-9]+$ ]] && (( MENU_DEFAULT >= 1 && MENU_DEFAULT <= count )); then
cur=$((MENU_DEFAULT - 1))
fi
MENU_DEFAULT=""
# Fallback to numbered input when no interactive terminal is available
# (e.g. CI, non-tty pipe). Arrow-key rendering needs a real tty.
if [ "$HAS_TTY" != true ] || [ ! -t 1 ]; then
local def=$((cur + 1))
echo -e "${CYAN}${BOLD}${title}${NC}"
local i=1
for opt in "${options[@]}"; do
echo -e " ${YELLOW}${i})${NC} ${opt}"
i=$((i + 1))
done
local choice=""
while true; do
tty_read choice "$(t "请输入序号" "Enter number") [1-${count}, $(t "默认" "default") ${def}]: "
choice=${choice:-$def}
if [[ "$choice" =~ ^[0-9]+$ ]] && (( choice >= 1 && choice <= count )); then
break
fi
echo -e "${RED}$(t "无效选择,请输入" "Invalid choice, enter") 1-${count}${NC}"
done
printf -v "$__out" '%s' "$choice"
return
fi
# Interactive arrow-key menu.
# Use literal escape characters (via $'...') and printf instead of
# `echo -e`, because `echo`'s backslash handling is not portable and
# leaks raw "\e[K" text on some shells/terminals.
local ESC=$'\033'
local UP="${ESC}[A" # move cursor up one line
local CLR="${ESC}[K" # clear to end of line
# fd 3 is a long-lived (read) handle to the controlling terminal, opened
# once by menu_session_begin() before the install flow. Reusing one fd
# across all menus avoids the bash 3.2 bug where re-opening /dev/tty per
# menu makes the second menu read EOF and auto-select the default.
# Detect whether fd 3 is already open using a READ redirection (fd 3 is
# read-only; testing with `>&3` would wrongly report it as closed).
local _own_fd3=false
if ! { : <&3; } 2>/dev/null; then
exec 3<"$TTY_DEV"
_own_fd3=true
fi
# Put the terminal into cbreak/raw input mode so single keystrokes arrive
# immediately and are not echoed.
# -echo : don't echo keystrokes (otherwise arrow keys leak as ^[[A)
# -icanon : disable line buffering
# min 1 time 0 : read returns as soon as 1 byte is available
local _restore="tput cnorm 2>/dev/null; stty echo icanon <${TTY_DEV} 2>/dev/null"
trap "$_restore" EXIT INT TERM
tput civis 2>/dev/null || true
stty -echo -icanon min 1 time 0 <&3 2>/dev/null || true
printf '%b\n' "${CYAN}${BOLD}${title}${NC}"
printf '%b\n' "${CYAN}$(t "↑/↓ 选择Enter 确认" "Use ↑/↓ to move, Enter to select")${NC}"
local first_draw=true
while true; do
# Move cursor up to the top of the option block to redraw it.
if [ "$first_draw" = false ]; then
local i=0
while [ $i -lt $count ]; do
printf '%s' "$UP"
i=$((i + 1))
done
fi
first_draw=false
local idx=0
for opt in "${options[@]}"; do
if [ $idx -eq $cur ]; then
printf '%s%b\n' "$CLR" " ${GREEN}${BOLD} ${opt}${NC}"
else
printf '%s%b\n' "$CLR" " ${opt}"
fi
idx=$((idx + 1))
done
# Read one key from the shared terminal fd 3.
local key=""
IFS= read -rsn1 key <&3
local rc=$?
if [ $rc -ne 0 ]; then
# No usable terminal: restore and fall back to numbered input.
eval "$_restore"; trap - EXIT INT TERM
[ "${_own_fd3:-}" = true ] && exec 3<&- 2>/dev/null
local choice=""
while true; do
tty_read choice "$(t "请输入序号" "Enter number") [1-${count}]: "
choice=${choice:-$((cur + 1))}
if [[ "$choice" =~ ^[0-9]+$ ]] && (( choice >= 1 && choice <= count )); then
break
fi
done
printf -v "$__out" '%s' "$choice"
return
fi
# Empty key means Enter/Return (read -n1 strips the newline delimiter).
if [ -z "$key" ]; then
break
fi
case "$key" in
"$ESC")
# Arrow key: ESC [ A/B (or ESC O A/B). Read the two trailing
# bytes one at a time, no timeout (bash 3.2 has no fractional
# read -t; in cbreak mode the bytes are already buffered).
local b2="" b3=""
IFS= read -rsn1 b2 <&3 2>/dev/null || b2=""
IFS= read -rsn1 b3 <&3 2>/dev/null || b3=""
case "${b2}${b3}" in
"[A" | "OA") cur=$(( (cur - 1 + count) % count )) ;; # up
"[B" | "OB") cur=$(( (cur + 1) % count )) ;; # down
esac
;;
$'\n' | $'\r')
break
;;
[0-9])
if (( key >= 1 && key <= count )); then
cur=$((key - 1))
break
fi
;;
$'\003')
# Ctrl-C: restore and abort.
eval "$_restore"; trap - EXIT INT TERM
[ "${_own_fd3:-}" = true ] && exec 3<&- 2>/dev/null
printf '\n%b\n' "${RED}$(t "已取消安装" "Installation cancelled")${NC}"
exit 130
;;
esac
done
eval "$_restore"
trap - EXIT INT TERM
[ "${_own_fd3:-}" = true ] && exec 3<&- 2>/dev/null
printf -v "$__out" '%s' "$((cur + 1))"
}
# Open/close a long-lived terminal handle (fd 3) shared by all menus in an
# install/config session. Opening fd 3 once avoids per-menu re-open issues on
# bash 3.2 (second menu reading EOF). Safe no-ops when there is no tty.
menu_session_begin() {
[ "$HAS_TTY" = true ] || return 0
exec 3<"$TTY_DEV" 2>/dev/null || true
}
menu_session_end() {
exec 3<&- 2>/dev/null || true
}
# Ask the user to choose the install/UI language (first step of install).
select_language() {
# Order is fixed (English first, Chinese second). The default highlight
# follows detection, but conservatively: only a confident "zh" signal
# (macOS AppleLocale / Linux zh_* locale) preselects Chinese; everything
# else (English, empty/C/POSIX locale, server images) defaults to English.
local detected
detected=$(detect_ui_lang)
if [ "$detected" = "zh" ]; then
MENU_DEFAULT=2
UI_LANG="zh"
else
MENU_DEFAULT=1
UI_LANG="en"
fi
local lang_choice
select_menu lang_choice "Select Language / 选择语言" "English" "中文 (Chinese)"
case "$lang_choice" in
1) UI_LANG="en" ;;
2) UI_LANG="zh" ;;
*) UI_LANG="en" ;;
esac
# Remember for the rest of the flow (config write happens later)
INSTALL_LANG="$UI_LANG"
}
# Cross-platform timeout: prefer GNU timeout/gtimeout, fallback to a pure-bash implementation
# that uses background process + sleep to enforce a hard time limit.
if command -v timeout &> /dev/null; then
_timeout() { timeout "$@"; }
elif command -v gtimeout &> /dev/null; then
_timeout() { gtimeout "$@"; }
else
_timeout() {
local secs=$1; shift
"$@" &
local cmd_pid=$!
( sleep "$secs"; kill $cmd_pid 2>/dev/null ) &
local watcher_pid=$!
wait $cmd_pid 2>/dev/null
local exit_code=$?
kill $watcher_pid 2>/dev/null
wait $watcher_pid 2>/dev/null
return $exit_code
}
fi
# Get current script directory.
# When launched via process substitution (`bash <(curl ...)`) or a pipe,
# $0 points at /dev/fd/* or "bash", so dirname is meaningless. Fall back to
# the current working directory in that case (remote install will cd into
# the cloned project dir and reset BASE_DIR afterwards).
_script_src="$0"
case "$_script_src" in
/dev/fd/* | /proc/self/fd/* | bash | sh | -* | "")
export BASE_DIR="$(pwd)"
;;
*)
export BASE_DIR=$(cd "$(dirname "$_script_src")" 2>/dev/null && pwd || pwd)
;;
esac
# Detect if in project directory
IS_PROJECT_DIR=false
if [ -f "${BASE_DIR}/config-template.json" ] && [ -f "${BASE_DIR}/app.py" ]; then
IS_PROJECT_DIR=true
fi
# Check and install tool
check_and_install_tool() {
local tool_name=$1
if ! command -v "$tool_name" &> /dev/null; then
echo -e "${YELLOW}⚙️ $tool_name not found, installing...${NC}"
if command -v yum &> /dev/null; then
sudo yum install "$tool_name" -y
elif command -v apt-get &> /dev/null; then
sudo apt-get update && sudo apt-get install "$tool_name" -y
elif command -v brew &> /dev/null; then
brew install "$tool_name"
else
echo -e "${RED}❌ Unsupported package manager. Please install $tool_name manually.${NC}"
return 1
fi
if ! command -v "$tool_name" &> /dev/null; then
echo -e "${RED}❌ Failed to install $tool_name.${NC}"
return 1
else
echo -e "${GREEN}$tool_name installed successfully.${NC}"
return 0
fi
else
echo -e "${GREEN}$tool_name is already installed.${NC}"
return 0
fi
}
# Detect and set Python command
detect_python_command() {
FOUND_NEWER_VERSION=""
# Try to find Python command in order of preference
for cmd in python3 python python3.12 python3.11 python3.10 python3.9 python3.8 python3.7; do
if command -v $cmd &> /dev/null; then
# Check Python version
major_version=$($cmd -c 'import sys; print(sys.version_info[0])' 2>/dev/null)
minor_version=$($cmd -c 'import sys; print(sys.version_info[1])' 2>/dev/null)
if [[ "$major_version" == "3" ]]; then
# Check if version is in supported range (3.7 - 3.12)
if (( minor_version >= 7 && minor_version <= 12 )); then
PYTHON_CMD=$cmd
PYTHON_VERSION="${major_version}.${minor_version}"
break
elif (( minor_version >= 13 )); then
# Found Python 3.13+, but not compatible
if [ -z "$FOUND_NEWER_VERSION" ]; then
FOUND_NEWER_VERSION="${major_version}.${minor_version}"
fi
fi
fi
fi
done
if [ -z "$PYTHON_CMD" ]; then
echo -e "${YELLOW}Tried: python3, python, python3.12, python3.11, python3.10, python3.9, python3.8, python3.7${NC}"
if [ -n "$FOUND_NEWER_VERSION" ]; then
echo -e "${RED}❌ Found Python $FOUND_NEWER_VERSION, but this project requires Python 3.7-3.12${NC}"
echo -e "${YELLOW}Python 3.13+ has compatibility issues with some dependencies (web.py, cgi module removed)${NC}"
echo -e "${YELLOW}Please install Python 3.7-3.12 (recommend Python 3.12)${NC}"
else
echo -e "${RED}❌ No suitable Python found. Please install Python 3.7-3.12${NC}"
fi
exit 1
fi
# Export for global use
export PYTHON_CMD
export PYTHON_VERSION
echo -e "${GREEN}✅ Found Python: $PYTHON_CMD (version $PYTHON_VERSION)${NC}"
}
# Check Python version (>= 3.7)
check_python_version() {
detect_python_command
# Verify pip is available
if ! $PYTHON_CMD -m pip --version &> /dev/null; then
echo -e "${RED}❌ pip not found for $PYTHON_CMD. Please install pip.${NC}"
exit 1
fi
echo -e "${GREEN}✅ pip is available for $PYTHON_CMD${NC}"
}
# Clone project
clone_project() {
echo -e "${GREEN}🔍 Cloning CowAgent project...${NC}"
if [ -d "CowAgent" ]; then
# An existing directory is automatically backed up (no prompt) so the
# installer stays one-shot / hands-off.
local backup_dir="CowAgent_backup_$(date +%s)"
echo -e "${YELLOW}⚠️ $(t "目录 'CowAgent' 已存在,自动备份到" "Directory 'CowAgent' exists, backing up to") '$backup_dir'...${NC}"
mv CowAgent "$backup_dir"
fi
check_and_install_tool git
if ! command -v git &> /dev/null; then
echo -e "${YELLOW}⚠️ Git not available. Trying wget/curl...${NC}"
local zip_url="https://gitee.com/zhayujie/CowAgent/repository/archive/master.zip"
if command -v wget &> /dev/null; then
wget "$zip_url" -O CowAgent.zip
elif command -v curl &> /dev/null; then
curl -L "$zip_url" -o CowAgent.zip
else
echo -e "${RED}❌ Cannot download project. Please install Git, wget, or curl.${NC}"
exit 1
fi
# Unzip: prefer `unzip`, otherwise fall back to Python's zipfile (no
# extra dependency) so minimal environments without unzip still work.
if command -v unzip &> /dev/null; then
unzip CowAgent.zip
elif command -v python3 &> /dev/null; then
python3 -m zipfile -e CowAgent.zip .
elif command -v python &> /dev/null; then
python -m zipfile -e CowAgent.zip .
else
echo -e "${RED}❌ Cannot extract archive. Please install 'unzip' or Python.${NC}"
exit 1
fi
# Archive top-level dir name may vary (CowAgent-master, etc.); detect it.
local _extracted="CowAgent-master"
if [ ! -d "$_extracted" ]; then
_extracted=$(ls -d CowAgent-*/ 2>/dev/null | head -1 | sed 's:/*$::')
fi
[ -n "$_extracted" ] && [ -d "$_extracted" ] && mv "$_extracted" CowAgent
rm -f CowAgent.zip
else
local clone_ok=false
# Detect and temporarily disable invalid git proxy settings
local _git_proxy_unset=false
local _http_proxy=$(git config --global http.proxy 2>/dev/null)
local _https_proxy=$(git config --global https.proxy 2>/dev/null)
if [ -n "$_http_proxy" ] && ! curl -s --connect-timeout 3 --max-time 5 --proxy "$_http_proxy" https://github.com > /dev/null 2>&1; then
echo -e "${YELLOW}⚠️ Invalid git proxy detected: $_http_proxy, temporarily disabling...${NC}"
git config --global --unset http.proxy
[ -n "$_https_proxy" ] && git config --global --unset https.proxy
_git_proxy_unset=true
fi
# Test GitHub connectivity before attempting clone
if curl -sI --connect-timeout 5 --max-time 10 https://github.com > /dev/null 2>&1; then
echo -e "${YELLOW}🌐 GitHub is reachable, cloning from GitHub...${NC}"
_timeout 60 git clone --depth 10 --progress https://github.com/zhayujie/CowAgent.git && clone_ok=true
fi
if [ "$clone_ok" = false ]; then
echo -e "${YELLOW}⚠️ GitHub clone failed or timed out, switching to Gitee mirror...${NC}"
_timeout 30 git clone --depth 10 --progress https://gitee.com/zhayujie/CowAgent.git && clone_ok=true
fi
if [ "$clone_ok" = false ]; then
echo -e "${RED}❌ Project clone failed. Please check network connection.${NC}"
if git config --global http.proxy &> /dev/null || git config --global https.proxy &> /dev/null || [ -n "$http_proxy" ] || [ -n "$https_proxy" ] || [ -n "$HTTP_PROXY" ] || [ -n "$HTTPS_PROXY" ]; then
echo -e "${YELLOW}💡 Detected proxy settings. If proxy is misconfigured, try removing it with:${NC}"
echo -e "${YELLOW} git config --global --unset http.proxy${NC}"
echo -e "${YELLOW} git config --global --unset https.proxy${NC}"
echo -e "${YELLOW} unset http_proxy https_proxy HTTP_PROXY HTTPS_PROXY${NC}"
fi
exit 1
fi
fi
cd CowAgent || { echo -e "${RED}❌ Failed to enter project directory.${NC}"; exit 1; }
export BASE_DIR=$(pwd)
echo -e "${GREEN}✅ Project cloned successfully: $BASE_DIR${NC}"
# Add execute permission to management script
if [ -f "${BASE_DIR}/run.sh" ]; then
chmod +x "${BASE_DIR}/run.sh" 2>/dev/null || true
echo -e "${GREEN}✅ Execute permission added to run.sh${NC}"
fi
sleep 1
}
# Install dependencies
install_dependencies() {
echo -e "${GREEN}📦 Installing dependencies...${NC}"
# Pick the pip index by install language, then fall back to the other if the
# preferred one is unreachable:
# - zh users: Tsinghua mirror first (fast in China), official PyPI fallback
# - others : official PyPI first, Tsinghua mirror fallback
local PIP_MIRROR=""
local _tuna="https://pypi.tuna.tsinghua.edu.cn/simple"
local _pypi="https://pypi.org/simple"
if [ "$UI_LANG" = "zh" ]; then
# Prefer Tsinghua; if it's down, fall back to official PyPI (pip default).
if curl -s --connect-timeout 5 "${_tuna}/" > /dev/null 2>&1; then
PIP_MIRROR="-i $_tuna"
fi
else
# Prefer official PyPI; only use Tsinghua if PyPI is unreachable.
if ! curl -s --connect-timeout 5 "${_pypi}/" > /dev/null 2>&1 \
&& curl -s --connect-timeout 5 "${_tuna}/" > /dev/null 2>&1; then
PIP_MIRROR="-i $_tuna"
fi
fi
if [ -n "$PIP_MIRROR" ]; then
echo -e "${YELLOW}Using pip mirror: ${_tuna}${NC}"
fi
# Only pass --break-system-packages if this pip actually supports it
# (pip >= 23.x). Older pip versions error out with "no such option",
# which previously dumped a confusing usage message and failed the install.
PIP_EXTRA_ARGS=""
if $PYTHON_CMD -c "import sys; exit(0 if sys.version_info >= (3, 11) else 1)" 2>/dev/null \
&& $PYTHON_CMD -m pip install --help 2>/dev/null | grep -q -- "--break-system-packages"; then
PIP_EXTRA_ARGS="--break-system-packages"
echo -e "${YELLOW}Python 3.11+ with break-system-packages support detected${NC}"
fi
echo -e "${YELLOW}Upgrading pip and basic tools...${NC}"
set +e
$PYTHON_CMD -m pip install --upgrade pip setuptools wheel importlib_metadata --ignore-installed $PIP_EXTRA_ARGS $PIP_MIRROR > /tmp/pip_upgrade.log 2>&1
[ $? -ne 0 ] && echo -e "${YELLOW}⚠️ Some tools failed to upgrade, but continuing...${NC}"
set -e
rm -f /tmp/pip_upgrade.log
echo -e "${YELLOW}Installing project dependencies...${NC}"
set +e
$PYTHON_CMD -m pip install -r requirements.txt $PIP_EXTRA_ARGS $PIP_MIRROR > /tmp/pip_install.log 2>&1
local exit_code=$?
set -e
cat /tmp/pip_install.log
if [ $exit_code -eq 0 ]; then
echo -e "${GREEN}✅ Dependencies installed successfully.${NC}"
elif grep -qE "distutils installed project|uninstall-no-record-file|installed by debian" /tmp/pip_install.log; then
echo -e "${YELLOW}⚠️ Detected system package conflict, retrying with workaround...${NC}"
local IGNORE_PACKAGES=""
for pkg in PyYAML setuptools wheel certifi charset-normalizer; do
IGNORE_PACKAGES="$IGNORE_PACKAGES --ignore-installed $pkg"
done
set +e
$PYTHON_CMD -m pip install -r requirements.txt $IGNORE_PACKAGES $PIP_EXTRA_ARGS $PIP_MIRROR \
&& echo -e "${GREEN}✅ Dependencies installed successfully (workaround applied).${NC}" \
|| echo -e "${YELLOW}⚠️ Some dependencies may have issues, but continuing...${NC}"
set -e
elif grep -q "externally-managed-environment" /tmp/pip_install.log; then
echo -e "${YELLOW}⚠️ Detected externally-managed environment, retrying with --break-system-packages...${NC}"
set +e
$PYTHON_CMD -m pip install -r requirements.txt --break-system-packages $PIP_MIRROR \
&& echo -e "${GREEN}✅ Dependencies installed successfully (system packages override applied).${NC}" \
|| echo -e "${YELLOW}⚠️ Some dependencies may have issues, but continuing...${NC}"
set -e
else
echo -e "${YELLOW}⚠️ Installation had errors, but continuing...${NC}"
fi
rm -f /tmp/pip_install.log
# Register `cow` CLI command via editable install
echo -e "${YELLOW}Registering cow CLI...${NC}"
set +e
$PYTHON_CMD -m pip install -e . $PIP_EXTRA_ARGS $PIP_MIRROR > /dev/null 2>&1
if command -v cow &> /dev/null; then
echo -e "${GREEN}✅ cow CLI registered.${NC}"
else
echo -e "${YELLOW}⚠️ cow CLI not in PATH, you can still use: $PYTHON_CMD -m cli.cli${NC}"
fi
set -e
}
# Select model
select_model() {
echo ""
local title sel
title="$(t "选择 AI 模型" "Select AI Model")"
# The 11th option is "skip" -> configure later in the web console.
select_menu sel "$title" \
"DeepSeek (deepseek-v4-flash, deepseek-v4-pro, etc.)" \
"Claude (claude-opus-4-8, claude-opus-4-7, claude-sonnet-4-6, etc.)" \
"Gemini (gemini-3.1-flash-lite-preview, gemini-3.1-pro-preview, etc.)" \
"OpenAI GPT (gpt-5.4, gpt-5.2, gpt-4.1, etc.)" \
"MiniMax (MiniMax-M2.7, MiniMax-M2.5, etc.)" \
"Zhipu AI (glm-5.1, glm-5-turbo, glm-5, etc.)" \
"Qwen (qwen3.6-plus, qwen3.5-plus, qwen3-max, qwq-plus, etc.)" \
"Doubao (doubao-seed-2-0-code-preview-260215, etc.)" \
"Kimi (kimi-k2.6, kimi-k2.5, kimi-k2, etc.)" \
"LinkAI ($(t "一个 Key 接入所有模型" "access all models via one API"))" \
"$(t "⏭ 跳过(稍后在 Web 控制台配置)" "⏭ Skip (configure later in the web console)")"
model_choice="$sel"
}
# Read model config: provider, default_model, key_variable_name
read_model_config() {
local provider=$1 default_model=$2 key_var=$3
echo -e "${GREEN}$(t "正在配置" "Configuring") ${provider}...${NC}"
# Only ask for the API key here; the model name and API base default to
# sensible values and can be changed later in the web console.
local _api_key
tty_read _api_key "$(t "请输入" "Enter") ${provider} API Key ($(t "回车跳过,稍后在 Web 控制台填写" "press Enter to skip, set later in web console")): "
MODEL_NAME="$default_model"
# printf -v (not eval) so keys containing quotes/backticks/$() are safe.
printf -v "${key_var}" '%s' "$_api_key"
}
# Configure model. The "skip" choice leaves the model empty so the user can
# finish configuration in the web console after first start.
configure_model() {
case "$model_choice" in
1) read_model_config "DeepSeek" "deepseek-v4-flash" "DEEPSEEK_KEY" ;;
2) read_model_config "Claude" "claude-opus-4-8" "CLAUDE_KEY" ;;
3) read_model_config "Gemini" "gemini-3.1-pro-preview" "GEMINI_KEY" ;;
4) read_model_config "OpenAI GPT" "gpt-5.4" "OPENAI_KEY" ;;
5) read_model_config "MiniMax" "MiniMax-M2.7" "MINIMAX_KEY" ;;
6) read_model_config "Zhipu AI" "glm-5.1" "ZHIPU_KEY" ;;
7) read_model_config "Qwen (DashScope)" "qwen3.6-plus" "DASHSCOPE_KEY" ;;
8) read_model_config "Doubao (Volcengine Ark)" "doubao-seed-2-0-code-preview-260215" "ARK_KEY" ;;
9) read_model_config "Kimi (Moonshot)" "kimi-k2.6" "MOONSHOT_KEY" ;;
10)
# Show where to obtain a LinkAI key (zh users -> console page).
echo -e "${CYAN}$(t "获取 LinkAI Key" "Get your LinkAI Key"): https://link-ai.tech/console/interface${NC}"
read_model_config "LinkAI" "deepseek-v4-flash" "LINKAI_KEY"
USE_LINKAI="true"
;;
11)
# Skip: leave model unset, will be configured in web console
MODEL_SKIPPED="true"
MODEL_NAME=""
echo -e "${YELLOW}$(t "已跳过模型配置,稍后可在 Web 控制台填写" "Model configuration skipped, you can set it later in the web console")${NC}"
;;
esac
}
# Channel label by stable key (independent of menu order).
channel_label() {
case "$1" in
web) t "Web 网页控制台(推荐,开箱即用)" "Web Console (recommended, ready to use)" ;;
weixin) t "微信" "WeChat (Weixin)" ;;
feishu) t "飞书" "Feishu / Lark" ;;
dingtalk) t "钉钉" "DingTalk" ;;
wecom_bot) t "企微智能机器人" "WeCom Bot" ;;
qq) printf '%s' "QQ" ;;
wechatcom_app) t "企微自建应用" "WeCom App" ;;
telegram) printf '%s' "Telegram" ;;
slack) printf '%s' "Slack" ;;
discord) printf '%s' "Discord" ;;
skip) t "⏭ 跳过(稍后在 Web 控制台配置)" "⏭ Skip (configure later in the web console)" ;;
esac
}
# Select channel. The display order depends on the install language:
# - English: Web first, then the global IM channels (Telegram/Discord/Slack),
# then the China-focused channels.
# - Chinese: Web first, then China-focused channels, then global ones.
# A stable key list (CHANNEL_KEYS) decouples the menu order from the config
# logic, so reordering the menu never breaks configure_channel().
select_channel() {
echo ""
local title sel
title="$(t "选择接入渠道" "Select Communication Channel")"
if [ "$UI_LANG" = "en" ]; then
CHANNEL_KEYS=(web telegram discord slack weixin feishu dingtalk wecom_bot qq wechatcom_app skip)
else
CHANNEL_KEYS=(web weixin feishu dingtalk wecom_bot qq wechatcom_app telegram slack discord skip)
fi
local labels=() k
for k in "${CHANNEL_KEYS[@]}"; do
labels+=("$(channel_label "$k")")
done
select_menu sel "$title" "${labels[@]}"
# Map the 1-based menu position back to the stable channel key.
channel_choice="${CHANNEL_KEYS[$((sel - 1))]}"
}
# Configure channel, dispatched by stable channel key (not menu position).
configure_channel() {
case "$channel_choice" in
web|skip)
# Web (also the default when skipped). Use the default port with
# no prompt; it can be changed later in the web console / config.
CHANNEL_TYPE="web"
WEB_PORT="9899"
ACCESS_INFO="$(t "Web 控制台地址" "Web console") : http://localhost:9899/chat"
;;
weixin)
# Weixin
CHANNEL_TYPE="weixin"
ACCESS_INFO="$(t "微信渠道已配置,请在终端或 Web 控制台扫码登录" "Weixin channel configured. Scan QR code in terminal or web console to login.")"
;;
feishu)
# Feishu (WebSocket mode)
CHANNEL_TYPE="feishu"
echo -e "${GREEN}$(t "配置飞书WebSocket 模式)" "Configure Feishu (WebSocket mode)")...${NC}"
local fs_app_id fs_app_secret
tty_read fs_app_id "$(t "请输入飞书 App ID" "Enter Feishu App ID"): "
tty_read fs_app_secret "$(t "请输入飞书 App Secret" "Enter Feishu App Secret"): "
FEISHU_APP_ID="$fs_app_id"
FEISHU_APP_SECRET="$fs_app_secret"
FEISHU_EVENT_MODE="websocket"
ACCESS_INFO="$(t "飞书渠道已配置WebSocket 模式)" "Feishu channel configured (WebSocket mode)")"
;;
dingtalk)
# DingTalk
CHANNEL_TYPE="dingtalk"
echo -e "${GREEN}$(t "配置钉钉" "Configure DingTalk")...${NC}"
local dt_client_id dt_client_secret
tty_read dt_client_id "$(t "请输入钉钉 Client ID" "Enter DingTalk Client ID"): "
tty_read dt_client_secret "$(t "请输入钉钉 Client Secret" "Enter DingTalk Client Secret"): "
DT_CLIENT_ID="$dt_client_id"
DT_CLIENT_SECRET="$dt_client_secret"
ACCESS_INFO="$(t "钉钉渠道已配置" "DingTalk channel configured")"
;;
wecom_bot)
# WeCom Bot
CHANNEL_TYPE="wecom_bot"
echo -e "${GREEN}$(t "配置企微智能机器人" "Configure WeCom Bot")...${NC}"
local wecom_bot_id wecom_bot_secret
tty_read wecom_bot_id "$(t "请输入 WeCom Bot ID" "Enter WeCom Bot ID"): "
tty_read wecom_bot_secret "$(t "请输入 WeCom Bot Secret" "Enter WeCom Bot Secret"): "
WECOM_BOT_ID="$wecom_bot_id"
WECOM_BOT_SECRET="$wecom_bot_secret"
ACCESS_INFO="$(t "企微智能机器人渠道已配置" "WeCom Bot channel configured")"
;;
qq)
# QQ
CHANNEL_TYPE="qq"
echo -e "${GREEN}$(t "配置 QQ 机器人" "Configure QQ Bot")...${NC}"
local qq_app_id qq_app_secret
tty_read qq_app_id "$(t "请输入 QQ App ID" "Enter QQ App ID"): "
tty_read qq_app_secret "$(t "请输入 QQ App Secret" "Enter QQ App Secret"): "
QQ_APP_ID="$qq_app_id"
QQ_APP_SECRET="$qq_app_secret"
ACCESS_INFO="$(t "QQ 机器人渠道已配置" "QQ Bot channel configured")"
;;
wechatcom_app)
# WeCom App
CHANNEL_TYPE="wechatcom_app"
echo -e "${GREEN}$(t "配置企微自建应用" "Configure WeCom App")...${NC}"
local corp_id com_token com_secret com_agent_id com_aes_key com_port
tty_read corp_id "$(t "请输入企业 Corp ID" "Enter WeChat Corp ID"): "
tty_read com_token "$(t "请输入应用 Token" "Enter WeChat Com App Token"): "
tty_read com_secret "$(t "请输入应用 Secret" "Enter WeChat Com App Secret"): "
tty_read com_agent_id "$(t "请输入应用 Agent ID" "Enter WeChat Com App Agent ID"): "
tty_read com_aes_key "$(t "请输入应用 AES Key" "Enter WeChat Com App AES Key"): "
tty_read com_port "$(t "请输入应用端口" "Enter WeChat Com App Port") [$(t "默认" "default"): 9898]: "
com_port=${com_port:-9898}
WECHATCOM_CORP_ID="$corp_id"
WECHATCOM_TOKEN="$com_token"
WECHATCOM_SECRET="$com_secret"
WECHATCOM_AGENT_ID="$com_agent_id"
WECHATCOM_AES_KEY="$com_aes_key"
WECHATCOM_PORT="$com_port"
ACCESS_INFO="$(t "企微自建应用渠道已配置,端口" "WeCom App channel configured on port") ${com_port}"
;;
telegram)
# Telegram
CHANNEL_TYPE="telegram"
echo -e "${GREEN}$(t "配置 Telegram" "Configure Telegram")...${NC}"
local tg_token
tty_read tg_token "$(t "请输入 Telegram Bot Token" "Enter Telegram Bot Token"): "
TELEGRAM_TOKEN="$tg_token"
ACCESS_INFO="$(t "Telegram 渠道已配置" "Telegram channel configured")"
;;
slack)
# Slack
CHANNEL_TYPE="slack"
echo -e "${GREEN}$(t "配置 Slack" "Configure Slack")...${NC}"
local slack_bot slack_app
tty_read slack_bot "$(t "请输入 Slack Bot Token" "Enter Slack Bot Token") (xoxb-...): "
tty_read slack_app "$(t "请输入 Slack App Token" "Enter Slack App Token") (xapp-...): "
SLACK_BOT_TOKEN="$slack_bot"
SLACK_APP_TOKEN="$slack_app"
ACCESS_INFO="$(t "Slack 渠道已配置" "Slack channel configured")"
;;
discord)
# Discord
CHANNEL_TYPE="discord"
echo -e "${GREEN}$(t "配置 Discord" "Configure Discord")...${NC}"
local discord_token
tty_read discord_token "$(t "请输入 Discord Bot Token" "Enter Discord Bot Token"): "
DISCORD_TOKEN="$discord_token"
ACCESS_INFO="$(t "Discord 渠道已配置" "Discord channel configured")"
;;
esac
}
# Generate config file
create_config_file() {
echo -e "${GREEN}📝 $(t "正在生成 config.json" "Generating config.json")...${NC}"
CHANNEL_TYPE="$CHANNEL_TYPE" \
MODEL_NAME="$MODEL_NAME" \
OPENAI_KEY="${OPENAI_KEY:-}" \
OPENAI_BASE="${OPENAI_BASE:-https://api.openai.com/v1}" \
CLAUDE_KEY="${CLAUDE_KEY:-}" \
CLAUDE_BASE="${CLAUDE_BASE:-https://api.anthropic.com/v1}" \
GEMINI_KEY="${GEMINI_KEY:-}" \
GEMINI_BASE="${GEMINI_BASE:-https://generativelanguage.googleapis.com}" \
ZHIPU_KEY="${ZHIPU_KEY:-}" \
MOONSHOT_KEY="${MOONSHOT_KEY:-}" \
ARK_KEY="${ARK_KEY:-}" \
DASHSCOPE_KEY="${DASHSCOPE_KEY:-}" \
MINIMAX_KEY="${MINIMAX_KEY:-}" \
DEEPSEEK_KEY="${DEEPSEEK_KEY:-}" \
DEEPSEEK_BASE="${DEEPSEEK_BASE:-https://api.deepseek.com/v1}" \
USE_LINKAI="${USE_LINKAI:-false}" \
LINKAI_KEY="${LINKAI_KEY:-}" \
FEISHU_APP_ID="${FEISHU_APP_ID:-}" \
FEISHU_APP_SECRET="${FEISHU_APP_SECRET:-}" \
WEB_PORT="${WEB_PORT:-}" \
DT_CLIENT_ID="${DT_CLIENT_ID:-}" \
DT_CLIENT_SECRET="${DT_CLIENT_SECRET:-}" \
WECOM_BOT_ID="${WECOM_BOT_ID:-}" \
WECOM_BOT_SECRET="${WECOM_BOT_SECRET:-}" \
QQ_APP_ID="${QQ_APP_ID:-}" \
QQ_APP_SECRET="${QQ_APP_SECRET:-}" \
WECHATCOM_CORP_ID="${WECHATCOM_CORP_ID:-}" \
WECHATCOM_TOKEN="${WECHATCOM_TOKEN:-}" \
WECHATCOM_SECRET="${WECHATCOM_SECRET:-}" \
WECHATCOM_AGENT_ID="${WECHATCOM_AGENT_ID:-}" \
WECHATCOM_AES_KEY="${WECHATCOM_AES_KEY:-}" \
WECHATCOM_PORT="${WECHATCOM_PORT:-}" \
TELEGRAM_TOKEN="${TELEGRAM_TOKEN:-}" \
SLACK_BOT_TOKEN="${SLACK_BOT_TOKEN:-}" \
SLACK_APP_TOKEN="${SLACK_APP_TOKEN:-}" \
DISCORD_TOKEN="${DISCORD_TOKEN:-}" \
COW_LANG="${INSTALL_LANG:-auto}" \
$PYTHON_CMD -c "
import json, os
e = os.environ.get
base = {
'channel_type': e('CHANNEL_TYPE') or 'web',
'model': e('MODEL_NAME') or '',
'cow_lang': e('COW_LANG', 'auto'),
'open_ai_api_key': e('OPENAI_KEY', ''),
'open_ai_api_base': e('OPENAI_BASE'),
'claude_api_key': e('CLAUDE_KEY', ''),
'claude_api_base': e('CLAUDE_BASE'),
'gemini_api_key': e('GEMINI_KEY', ''),
'gemini_api_base': e('GEMINI_BASE'),
'zhipu_ai_api_key': e('ZHIPU_KEY', ''),
'moonshot_api_key': e('MOONSHOT_KEY', ''),
'ark_api_key': e('ARK_KEY', ''),
'dashscope_api_key': e('DASHSCOPE_KEY', ''),
'minimax_api_key': e('MINIMAX_KEY', ''),
'deepseek_api_key': e('DEEPSEEK_KEY', ''),
'deepseek_api_base': e('DEEPSEEK_BASE'),
'voice_to_text': 'openai',
'text_to_voice': 'openai',
'voice_reply_voice': False,
'speech_recognition': True,
'group_speech_recognition': False,
'use_linkai': e('USE_LINKAI') == 'true',
'linkai_api_key': e('LINKAI_KEY', ''),
'linkai_app_code': '',
'agent': True,
'agent_max_context_tokens': 40000,
'agent_max_context_turns': 30,
'agent_max_steps': 15,
}
channel_map = {
'feishu': {'feishu_app_id': 'FEISHU_APP_ID', 'feishu_app_secret': 'FEISHU_APP_SECRET'},
'web': {'web_port': ('WEB_PORT', int)},
'dingtalk': {'dingtalk_client_id': 'DT_CLIENT_ID', 'dingtalk_client_secret': 'DT_CLIENT_SECRET'},
'wecom_bot': {'wecom_bot_id': 'WECOM_BOT_ID', 'wecom_bot_secret': 'WECOM_BOT_SECRET'},
'qq': {'qq_app_id': 'QQ_APP_ID', 'qq_app_secret': 'QQ_APP_SECRET'},
'wechatcom_app': {'wechatcom_corp_id': 'WECHATCOM_CORP_ID', 'wechatcomapp_token': 'WECHATCOM_TOKEN', 'wechatcomapp_secret': 'WECHATCOM_SECRET', 'wechatcomapp_agent_id': 'WECHATCOM_AGENT_ID', 'wechatcomapp_aes_key': 'WECHATCOM_AES_KEY', 'wechatcomapp_port': ('WECHATCOM_PORT', int)},
'telegram': {'telegram_token': 'TELEGRAM_TOKEN'},
'slack': {'slack_bot_token': 'SLACK_BOT_TOKEN', 'slack_app_token': 'SLACK_APP_TOKEN'},
'discord': {'discord_token': 'DISCORD_TOKEN'},
}
def _to_int(val, default):
try:
return int(val)
except (TypeError, ValueError):
return default
ch = e('CHANNEL_TYPE') or 'web'
for key, spec in channel_map.get(ch, {}).items():
if isinstance(spec, tuple):
env_name, conv = spec
# Guard int() against non-numeric input; fall back to a sane port.
base[key] = _to_int(e(env_name), 9899 if key == 'web_port' else 9898) if conv is int else conv(e(env_name))
else:
base[key] = e(spec, '')
with open('config.json', 'w') as f:
json.dump(base, f, indent=2, ensure_ascii=False)
"
echo -e "${GREEN}$(t "配置文件创建成功" "Configuration file created successfully").${NC}"
}
# Start project
start_project() {
echo ""
echo -e "${GREEN}${EMOJI_ROCKET} Starting CowAgent...${NC}"
sleep 1
local USE_COW=false
if command -v cow &> /dev/null; then
USE_COW=true
fi
if $USE_COW; then
cd "${BASE_DIR}"
cow start --no-logs
else
if [ ! -f "${BASE_DIR}/nohup.out" ]; then
touch "${BASE_DIR}/nohup.out"
fi
OS_TYPE=$(uname)
if [[ "$OS_TYPE" == "Linux" ]]; then
nohup setsid $PYTHON_CMD "${BASE_DIR}/app.py" > "${BASE_DIR}/nohup.out" 2>&1 &
echo -e "${GREEN}${EMOJI_COW} CowAgent started on Linux (using $PYTHON_CMD)${NC}"
elif [[ "$OS_TYPE" == "Darwin" ]]; then
nohup $PYTHON_CMD "${BASE_DIR}/app.py" > "${BASE_DIR}/nohup.out" 2>&1 &
echo -e "${GREEN}${EMOJI_COW} CowAgent started on macOS (using $PYTHON_CMD)${NC}"
else
echo -e "${RED}❌ Unsupported OS: ${OS_TYPE}${NC}"
exit 1
fi
fi
sleep 2
echo ""
echo -e "${CYAN}${BOLD}=========================================${NC}"
echo -e "${GREEN}${EMOJI_CHECK} $(t "CowAgent 已在后台运行" "CowAgent is now running in background")!${NC}"
echo -e "${GREEN}${EMOJI_CHECK} $(t "关闭终端后进程仍会继续运行" "Process will continue after closing terminal").${NC}"
echo -e "${CYAN}$ACCESS_INFO${NC}"
# If the model was skipped, guide the user to finish setup in the web console.
if [ "${MODEL_SKIPPED:-}" = "true" ]; then
local _port="${WEB_PORT:-9899}"
echo ""
echo -e "${YELLOW}${EMOJI_WARN} $(t "尚未配置模型,请在 Web 控制台完成配置" "Model not configured yet, please finish setup in the web console"):${NC}"
echo -e "${CYAN} http://localhost:${_port}/chat${NC}"
fi
echo ""
echo -e "${CYAN}${BOLD}$(t "管理命令" "Management Commands"):${NC}"
if $USE_COW; then
echo -e " ${GREEN}cow stop${NC} $(t "停止服务" "Stop the service")"
echo -e " ${GREEN}cow restart${NC} $(t "重启服务" "Restart the service")"
echo -e " ${GREEN}cow status${NC} $(t "查看状态" "Check status")"
echo -e " ${GREEN}cow logs${NC} $(t "查看日志" "View logs")"
echo -e " ${GREEN}cow update${NC} $(t "更新并重启" "Update and restart")"
echo -e " ${GREEN}cow install-browser${NC} $(t "安装浏览器工具" "Install browser tool")"
else
echo -e " ${GREEN}./run.sh stop${NC} $(t "停止服务" "Stop the service")"
echo -e " ${GREEN}./run.sh restart${NC} $(t "重启服务" "Restart the service")"
echo -e " ${GREEN}./run.sh status${NC} $(t "查看状态" "Check status")"
echo -e " ${GREEN}./run.sh logs${NC} $(t "查看日志" "View logs")"
echo -e " ${GREEN}./run.sh update${NC} $(t "更新并重启" "Update and restart")"
fi
echo -e "${CYAN}${BOLD}=========================================${NC}"
echo ""
echo -e "${YELLOW}$(t "显示最近日志Ctrl+C 退出Agent 继续运行)" "Showing recent logs (Ctrl+C to exit, agent keeps running)"):${NC}"
sleep 2
tail -n 30 -f "${BASE_DIR}/nohup.out"
}
# Show usage
show_usage() {
echo -e "${CYAN}${BOLD}=========================================${NC}"
echo -e "${CYAN}${BOLD} ${EMOJI_COW} CowAgent Management Script${NC}"
echo -e "${CYAN}${BOLD}=========================================${NC}"
echo ""
echo -e "${YELLOW}$(t "用法" "Usage"):${NC}"
echo -e " ${GREEN}./run.sh${NC} ${CYAN}# $(t "安装/配置项目" "Install/Configure project")${NC}"
echo -e " ${GREEN}./run.sh <command>${NC} ${CYAN}# $(t "执行管理命令" "Execute management command")${NC}"
echo ""
echo -e "${YELLOW}$(t "命令" "Commands"):${NC}"
echo -e " ${GREEN}start${NC} $(t "启动服务" "Start the service")"
echo -e " ${GREEN}stop${NC} $(t "停止服务" "Stop the service")"
echo -e " ${GREEN}restart${NC} $(t "重启服务" "Restart the service")"
echo -e " ${GREEN}status${NC} $(t "查看服务状态" "Check service status")"
echo -e " ${GREEN}logs${NC} $(t "查看日志 (tail -f)" "View logs (tail -f)")"
echo -e " ${GREEN}config${NC} $(t "重新配置项目" "Reconfigure project")"
echo -e " ${GREEN}update${NC} $(t "更新并重启" "Update and restart")"
echo ""
echo -e "${YELLOW}$(t "示例" "Examples"):${NC}"
echo -e " ${GREEN}./run.sh start${NC}"
echo -e " ${GREEN}./run.sh logs${NC}"
echo -e " ${GREEN}./run.sh status${NC}"
echo -e "${CYAN}${BOLD}=========================================${NC}"
}
# Ensure PYTHON_CMD is set
ensure_python_cmd() {
if [ -z "$PYTHON_CMD" ]; then
detect_python_command > /dev/null 2>&1 || PYTHON_CMD="python3"
fi
}
# Get service PID (empty string if not running)
get_pid() {
ensure_python_cmd > /dev/null 2>&1
ps ax | grep -i app.py | grep "${BASE_DIR}" | grep "$PYTHON_CMD" | grep -v grep | awk '{print $1}' | grep -E '^[0-9]+$' | head -1
}
# Check if service is running
is_running() {
[ -n "$(get_pid)" ]
}
# Check if cow CLI is available
has_cow() {
command -v cow &> /dev/null
}
# Start service
cmd_start() {
if [ ! -f "${BASE_DIR}/config.json" ]; then
echo -e "${RED}${EMOJI_CROSS} $(t "未找到 config.json" "config.json not found")${NC}"
echo -e "${YELLOW}$(t "请先运行 './run.sh' 进行配置" "Please run './run.sh' to configure first")${NC}"
exit 1
fi
if has_cow; then
cd "${BASE_DIR}"
cow start
else
if is_running; then
echo -e "${YELLOW}${EMOJI_WARN} $(t "CowAgent 已在运行中" "CowAgent is already running") (PID: $(get_pid))${NC}"
echo -e "${YELLOW}$(t "使用 './run.sh restart' 重启" "Use './run.sh restart' to restart")${NC}"
return
fi
check_python_version
start_project
fi
}
# Stop service
cmd_stop() {
# Don't let kill/return non-zero (e.g. process already gone) abort the
# caller (cmd_restart) under `set -e`.
set +e
if has_cow; then
cd "${BASE_DIR}"
cow stop
else
echo -e "${GREEN}${EMOJI_STOP} $(t "正在停止 CowAgent" "Stopping CowAgent")...${NC}"
if ! is_running; then
echo -e "${YELLOW}${EMOJI_WARN} $(t "CowAgent 未在运行" "CowAgent is not running")${NC}"
return 0
fi
pid=$(get_pid)
if [ -z "$pid" ] || ! echo "$pid" | grep -qE '^[0-9]+$'; then
echo -e "${RED}$(t "获取有效 PID 失败" "Failed to get valid PID") (${pid})${NC}"
return 0
fi
echo -e "${GREEN}$(t "找到运行中的进程" "Found running process") (PID: ${pid})${NC}"
kill ${pid} 2>/dev/null || true
sleep 3
if ps -p ${pid} > /dev/null 2>&1; then
echo -e "${YELLOW}⚠️ $(t "进程未停止,强制终止" "Process not stopped, forcing termination")...${NC}"
kill -9 ${pid} 2>/dev/null || true
fi
echo -e "${GREEN}${EMOJI_CHECK} $(t "CowAgent 已停止" "CowAgent stopped")${NC}"
fi
}
# Restart service
cmd_restart() {
if has_cow; then
cd "${BASE_DIR}"
cow restart
else
cmd_stop
sleep 1
cmd_start
fi
}
# Check status
cmd_status() {
if has_cow; then
cd "${BASE_DIR}"
cow status
else
echo -e "${CYAN}${BOLD}=========================================${NC}"
echo -e "${CYAN}${BOLD} ${EMOJI_COW} CowAgent Status${NC}"
echo -e "${CYAN}${BOLD}=========================================${NC}"
if is_running; then
pid=$(get_pid)
echo -e "${GREEN}$(t "状态" "Status"):${NC}$(t "运行中" "Running")"
echo -e "${GREEN}PID:${NC} ${pid}"
if [ -f "${BASE_DIR}/nohup.out" ]; then
echo -e "${GREEN}$(t "日志" "Logs"):${NC} ${BASE_DIR}/nohup.out"
fi
else
echo -e "${YELLOW}$(t "状态" "Status"):${NC}$(t "已停止" "Stopped")"
fi
if [ -f "${BASE_DIR}/config.json" ]; then
# `|| true`: grep returns 1 when the key is absent (set -e safe).
model=$(grep -o '"model"[[:space:]]*:[[:space:]]*"[^"]*"' "${BASE_DIR}/config.json" 2>/dev/null | cut -d'"' -f4 || true)
channel=$(grep -o '"channel_type"[[:space:]]*:[[:space:]]*"[^"]*"' "${BASE_DIR}/config.json" 2>/dev/null | cut -d'"' -f4 || true)
echo -e "${GREEN}$(t "模型" "Model"):${NC} ${model:-$(t "(未配置)" "(not set)")}"
echo -e "${GREEN}$(t "渠道" "Channel"):${NC} ${channel:-$(t "(未配置)" "(not set)")}"
fi
echo -e "${CYAN}${BOLD}=========================================${NC}"
fi
}
# View logs
cmd_logs() {
if has_cow; then
cd "${BASE_DIR}"
cow logs -f
else
if [ -f "${BASE_DIR}/nohup.out" ]; then
echo -e "${YELLOW}$(t "查看日志Ctrl+C 退出)" "Viewing logs (Ctrl+C to exit)"):${NC}"
tail -f "${BASE_DIR}/nohup.out"
else
echo -e "${RED}$(t "日志文件未找到" "Log file not found"): ${BASE_DIR}/nohup.out${NC}"
fi
fi
}
# Reconfigure
cmd_config() {
# Interactive flow: disable `set -e` (see install_mode for rationale).
set +e
# One shared terminal handle for all menus in this session.
menu_session_begin
# Choose language first so the rest of the flow is localized.
select_language
echo ""
echo -e "${YELLOW}${EMOJI_WRENCH} $(t "正在重新配置 CowAgent" "Reconfiguring CowAgent")...${NC}"
if [ -f "${BASE_DIR}/config.json" ]; then
backup_file="${BASE_DIR}/config.json.backup.$(date +%s)"
cp "${BASE_DIR}/config.json" "${backup_file}"
echo -e "${GREEN}$(t "已备份配置到" "Backed up config to"): ${backup_file}${NC}"
fi
check_python_version
install_dependencies
select_model
configure_model
select_channel
configure_channel
menu_session_end
create_config_file
echo ""
local restart_now
tty_read restart_now "$(t "现在重启服务" "Restart service now")? [Y/n]: "
if [[ ! $restart_now == [Nn]* ]]; then
cmd_restart
fi
}
# Update project
cmd_update() {
echo -e "${GREEN}${EMOJI_WRENCH} $(t "正在更新 CowAgent" "Updating CowAgent")...${NC}"
cd "${BASE_DIR}"
# Pull latest code first (service still running)
local pull_ok=false
if [ -d .git ]; then
echo -e "${GREEN}🔄 $(t "正在拉取最新代码" "Pulling latest code")...${NC}"
if git pull; then
pull_ok=true
else
echo -e "${YELLOW}⚠️ $(t "git pull 失败,尝试 Gitee 镜像" "git pull failed, trying Gitee mirror")...${NC}"
git remote set-url origin https://gitee.com/zhayujie/CowAgent.git
if git pull; then
pull_ok=true
else
echo -e "${RED}$(t "拉取代码失败,更新已中止" "Failed to pull code. Update aborted").${NC}"
exit 1
fi
fi
else
echo -e "${YELLOW}⚠️ $(t "非 git 仓库,跳过代码更新" "Not a git repository, skipping code update")${NC}"
fi
# Re-exec with the updated run.sh to pick up new logic
exec "$0" _post_update
}
# Post-update: called by cmd_update after git pull to run with new code
cmd_post_update() {
cd "${BASE_DIR}"
# Stop service
if is_running; then
cmd_stop
fi
# Reinstall dependencies
check_python_version
install_dependencies
# Restart service
cmd_start
}
# Installation mode
install_mode() {
# Interactive flow: disable `set -e` so a single non-zero command (e.g. an
# arithmetic `(( ))` evaluating to 0, a `read` hitting EOF, or an optional
# step failing) does not silently abort the whole installer.
set +e
clear
echo -e "${CYAN}${BOLD}=========================================${NC}"
echo -e "${CYAN}${BOLD} ${EMOJI_COW} CowAgent Installation${NC}"
echo -e "${CYAN}${BOLD}=========================================${NC}"
echo ""
# Open one shared terminal handle for ALL menus in this session (language,
# model, channel). One long-lived fd 3 avoids per-menu re-open issues on
# bash 3.2. Closed on early return and before config generation.
menu_session_begin
# Step 0: choose the install/UI language. Everything after this is localized.
select_language
echo ""
sleep 1
if [ "$IS_PROJECT_DIR" = true ]; then
echo -e "${GREEN}$(t "检测到已有项目目录" "Detected existing project directory").${NC}"
if [ -f "${BASE_DIR}/config.json" ]; then
menu_session_end
echo -e "${GREEN}$(t "项目已配置" "Project already configured")${NC}"
echo ""
show_usage
return
fi
echo -e "${YELLOW}📝 $(t "未找到 config.json开始配置项目" "No config.json found. Let's configure your project")!${NC}"
echo ""
# Project directory already exists, skip clone
check_python_version
else
# Remote install mode, need to clone project
check_python_version
clone_project
fi
# Install dependencies and configure
install_dependencies
select_model
configure_model
select_channel
configure_channel
menu_session_end
create_config_file
# Auto-start after configuration for a true out-of-the-box experience.
echo ""
start_project
}
# Require running inside the project directory
require_project_dir() {
if [ "$IS_PROJECT_DIR" = false ]; then
echo -e "${RED}${EMOJI_CROSS} $(t "必须在项目目录下运行" "Must run in project directory")${NC}"
exit 1
fi
}
# Initialize UI_LANG for management commands: prefer cow_lang from an existing
# config.json, otherwise fall back to environment detection. The install flow
# overrides this later via select_language().
init_ui_lang() {
[ -n "$UI_LANG" ] && return
local cfg_lang=""
if [ -f "${BASE_DIR}/config.json" ]; then
# `|| true`: grep returns 1 when cow_lang is absent, which would abort
# the whole script under `set -e` at the very first management command.
cfg_lang=$(grep -o '"cow_lang"[[:space:]]*:[[:space:]]*"[^"]*"' "${BASE_DIR}/config.json" 2>/dev/null | cut -d'"' -f4 || true)
fi
case "$cfg_lang" in
zh) UI_LANG="zh" ;;
en) UI_LANG="en" ;;
*) UI_LANG=$(detect_ui_lang) ;;
esac
}
# Main function
main() {
init_ui_lang
case "$1" in
start|stop|restart|status|logs|config|update|_post_update)
require_project_dir
;;
esac
case "$1" in
start) cmd_start ;;
stop) cmd_stop ;;
restart) cmd_restart ;;
status) cmd_status ;;
logs) cmd_logs ;;
config) cmd_config ;;
update) cmd_update ;;
_post_update) cmd_post_update ;;
help|--help|-h)
show_usage
;;
"")
install_mode
;;
*)
echo -e "${RED}${EMOJI_CROSS} $(t "未知命令" "Unknown command"): $1${NC}"
echo ""
show_usage
exit 1
;;
esac
}
# Execute main function
main "$@"