mirror of
https://github.com/anten-ka/gotelegram_pro.git
synced 2026-05-21 14:16:03 +00:00
Compare commits
5 Commits
v2.4.3
...
release-ma
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b10ea54ce9 | ||
|
|
2f3045bcc0 | ||
|
|
eb5175ccab | ||
|
|
3403975636 | ||
|
|
3919f201f5 |
2
bootstrap.sh
Normal file → Executable file
2
bootstrap.sh
Normal file → Executable file
@@ -6,7 +6,7 @@ set -euo pipefail
|
|||||||
|
|
||||||
REPO="anten-ka/gotelegram_pro"
|
REPO="anten-ka/gotelegram_pro"
|
||||||
BRANCH="${GOTELEGRAM_BRANCH:-test}"
|
BRANCH="${GOTELEGRAM_BRANCH:-test}"
|
||||||
PAT="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"
|
PAT="${GOTELEGRAM_PAT:-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}"
|
||||||
INSTALL_DIR="/opt/gotelegram"
|
INSTALL_DIR="/opt/gotelegram"
|
||||||
# Use raw.githubusercontent.com (CDN) — faster and avoids Contents API caching
|
# Use raw.githubusercontent.com (CDN) — faster and avoids Contents API caching
|
||||||
# issues that occasionally return 404 for recently added files on non-default branches.
|
# issues that occasionally return 404 for recently added files on non-default branches.
|
||||||
|
|||||||
@@ -100,7 +100,7 @@ logger = logging.getLogger(__name__)
|
|||||||
# CONFIGURATION
|
# CONFIGURATION
|
||||||
# ============================================================================
|
# ============================================================================
|
||||||
|
|
||||||
GOTELEGRAM_VERSION = "2.4.3"
|
GOTELEGRAM_VERSION = "2.4.6"
|
||||||
GOTELEGRAM_CONFIG = "/opt/gotelegram/config.json"
|
GOTELEGRAM_CONFIG = "/opt/gotelegram/config.json"
|
||||||
TELEMT_CONFIG = "/etc/telemt/config.toml"
|
TELEMT_CONFIG = "/etc/telemt/config.toml"
|
||||||
TELEMT_SERVICE = "telemt"
|
TELEMT_SERVICE = "telemt"
|
||||||
|
|||||||
122
install.sh
122
install.sh
@@ -751,11 +751,36 @@ menu_bot() {
|
|||||||
bot_install() {
|
bot_install() {
|
||||||
log_step "$(t bot_install_step)"
|
log_step "$(t bot_install_step)"
|
||||||
|
|
||||||
# Python
|
# Python + venv + pip (always ensure — python3 can be present without venv/pip)
|
||||||
if ! command -v python3 &>/dev/null; then
|
local need_py=0
|
||||||
|
command -v python3 &>/dev/null || need_py=1
|
||||||
|
# python3-venv not having its own command; probe by trying 'python3 -m venv --help'
|
||||||
|
if ! python3 -m venv --help &>/dev/null; then need_py=1; fi
|
||||||
|
# pip check
|
||||||
|
if ! python3 -m pip --version &>/dev/null; then need_py=1; fi
|
||||||
|
|
||||||
|
if [ "$need_py" = "1" ]; then
|
||||||
log_info "$(t bot_install_python)"
|
log_info "$(t bot_install_python)"
|
||||||
if command -v apt-get &>/dev/null; then
|
if command -v apt-get &>/dev/null; then
|
||||||
apt-get update -qq && apt-get install -y -qq python3 python3-pip python3-venv
|
# Detect Python version for versioned venv package (Debian 12 / Ubuntu 24.04 need python3.12-venv)
|
||||||
|
local py_ver=""
|
||||||
|
if command -v python3 &>/dev/null; then
|
||||||
|
py_ver=$(python3 -c 'import sys; print(f"{sys.version_info.major}.{sys.version_info.minor}")' 2>/dev/null)
|
||||||
|
fi
|
||||||
|
|
||||||
|
apt_update
|
||||||
|
|
||||||
|
# Build package list with versioned venv fallback
|
||||||
|
local pkg_list=(python3 python3-venv python3-pip)
|
||||||
|
[ -n "$py_ver" ] && pkg_list+=("python${py_ver}-venv")
|
||||||
|
# python3-full optional
|
||||||
|
if ! apt_install "${pkg_list[@]}" python3-full; then
|
||||||
|
log_warning "python3-full unavailable, installing core packages only..."
|
||||||
|
apt_install "${pkg_list[@]}" || {
|
||||||
|
log_error "Failed to install Python packages. Run manually: apt install ${pkg_list[*]}"
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
fi
|
||||||
elif command -v dnf &>/dev/null; then
|
elif command -v dnf &>/dev/null; then
|
||||||
dnf install -y -q python3 python3-pip
|
dnf install -y -q python3 python3-pip
|
||||||
elif command -v yum &>/dev/null; then
|
elif command -v yum &>/dev/null; then
|
||||||
@@ -782,17 +807,60 @@ bot_install() {
|
|||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Templates catalog
|
# Templates catalog — skip if source and dest are the same file (symlink install case)
|
||||||
[ -f "$SCRIPT_DIR/templates_catalog.json" ] && \
|
if [ -f "$SCRIPT_DIR/templates_catalog.json" ]; then
|
||||||
cp "$SCRIPT_DIR/templates_catalog.json" "$GOTELEGRAM_DIR/"
|
local src_tc="$SCRIPT_DIR/templates_catalog.json"
|
||||||
|
local dst_tc="$GOTELEGRAM_DIR/templates_catalog.json"
|
||||||
# Venv
|
if [ "$(readlink -f "$src_tc" 2>/dev/null)" != "$(readlink -f "$dst_tc" 2>/dev/null)" ]; then
|
||||||
if [ ! -d "$BOT_DIR/venv" ]; then
|
cp "$src_tc" "$dst_tc"
|
||||||
log_info "$(t bot_create_venv)"
|
fi
|
||||||
python3 -m venv "$BOT_DIR/venv"
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Venv — create, and verify pip exists (python3-venv can silently create broken venv)
|
||||||
|
if [ ! -d "$BOT_DIR/venv" ] || [ ! -x "$BOT_DIR/venv/bin/pip" ]; then
|
||||||
|
log_info "$(t bot_create_venv)"
|
||||||
|
rm -rf "$BOT_DIR/venv"
|
||||||
|
if ! python3 -m venv "$BOT_DIR/venv" 2>/tmp/venv_err; then
|
||||||
|
log_error "venv creation failed:"
|
||||||
|
cat /tmp/venv_err >&2 2>/dev/null
|
||||||
|
# Try to fix by installing versioned python3.X-venv package
|
||||||
|
if command -v apt-get &>/dev/null; then
|
||||||
|
local py_ver
|
||||||
|
py_ver=$(python3 -c 'import sys; print(f"{sys.version_info.major}.{sys.version_info.minor}")' 2>/dev/null)
|
||||||
|
log_info "reinstalling python${py_ver}-venv..."
|
||||||
|
apt_install python3-venv python3-pip "python${py_ver}-venv" python3-full || \
|
||||||
|
apt_install python3-venv python3-pip "python${py_ver}-venv" || true
|
||||||
|
rm -rf "$BOT_DIR/venv"
|
||||||
|
python3 -m venv "$BOT_DIR/venv" || { log_error "venv still broken, aborting. Manual fix: apt install python${py_ver}-venv python3-pip"; return 1; }
|
||||||
|
else
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -x "$BOT_DIR/venv/bin/pip" ]; then
|
||||||
|
log_info "bootstrapping pip via ensurepip..."
|
||||||
|
"$BOT_DIR/venv/bin/python" -m ensurepip --upgrade 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -x "$BOT_DIR/venv/bin/pip" ]; then
|
||||||
|
log_error "pip missing in venv — install python3-venv manually: apt install python3-venv python3-pip"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
log_info "$(t bot_install_deps)"
|
log_info "$(t bot_install_deps)"
|
||||||
"$BOT_DIR/venv/bin/pip" install -r "$BOT_DIR/requirements.txt" -q
|
if ! "$BOT_DIR/venv/bin/pip" install -r "$BOT_DIR/requirements.txt" -q 2>/tmp/pip_err; then
|
||||||
|
log_error "pip install failed:"
|
||||||
|
tail -n 5 /tmp/pip_err >&2
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Sanity check: verify critical imports succeed
|
||||||
|
if ! "$BOT_DIR/venv/bin/python" -c "import telegram, toml, dotenv" 2>/tmp/imp_err; then
|
||||||
|
log_error "dependency import check failed:"
|
||||||
|
cat /tmp/imp_err >&2
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
# Configuration
|
# Configuration
|
||||||
if [ ! -f "$BOT_DIR/.env" ]; then
|
if [ ! -f "$BOT_DIR/.env" ]; then
|
||||||
@@ -862,14 +930,13 @@ SVCEOF
|
|||||||
has_ids=$(grep "^ALLOWED_IDS=" "$BOT_DIR/.env" 2>/dev/null | cut -d= -f2)
|
has_ids=$(grep "^ALLOWED_IDS=" "$BOT_DIR/.env" 2>/dev/null | cut -d= -f2)
|
||||||
if [ -z "$has_ids" ]; then
|
if [ -z "$has_ids" ]; then
|
||||||
echo ""
|
echo ""
|
||||||
echo -e " ${YELLOW}╔══════════════════════════════════════════════════════╗${NC}"
|
# Simple bullet-style block (no box — printf %-Ns breaks on UTF-8 multibyte chars)
|
||||||
printf " ${YELLOW}║${NC} ${BOLD}%-52s${NC} ${YELLOW}║${NC}\n" "$(t bot_wait_admin_title)"
|
echo -e " ${YELLOW}▸${NC} ${BOLD}$(t bot_wait_admin_title)${NC}"
|
||||||
echo -e " ${YELLOW}║${NC} ${YELLOW}║${NC}"
|
echo ""
|
||||||
printf " ${YELLOW}║${NC} %s ${CYAN}/start${NC}%*s${YELLOW}║${NC}\n" "$(t bot_wait_admin_msg1)" 0 ""
|
echo -e " $(t bot_wait_admin_msg1) ${CYAN}/start${NC}"
|
||||||
printf " ${YELLOW}║${NC} %-52s ${YELLOW}║${NC}\n" "$(t bot_wait_admin_msg2)"
|
echo -e " $(t bot_wait_admin_msg2)"
|
||||||
echo -e " ${YELLOW}║${NC} ${YELLOW}║${NC}"
|
echo ""
|
||||||
printf " ${YELLOW}║${NC} ${DIM}%-52s${NC} ${YELLOW}║${NC}\n" "$(t bot_wait_admin_skip)"
|
echo -e " ${DIM}$(t bot_wait_admin_skip)${NC}"
|
||||||
echo -e " ${YELLOW}╚══════════════════════════════════════════════════════╝${NC}"
|
|
||||||
echo ""
|
echo ""
|
||||||
|
|
||||||
local frames=('⠋' '⠙' '⠹' '⠸' '⠼' '⠴' '⠦' '⠧' '⠇' '⠏')
|
local frames=('⠋' '⠙' '⠹' '⠸' '⠼' '⠴' '⠦' '⠧' '⠇' '⠏')
|
||||||
@@ -1071,21 +1138,14 @@ mark_promo_shown() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# ── Promo with QR + delay (on install + once per day) ───────────────────
|
# ── Promo with QR + delay (on install + once per day) ───────────────────
|
||||||
|
# QR показываем ТОЛЬКО для чаевых/донатов. Для хостеров оставлены только
|
||||||
|
# текстовые ссылки и промокоды (см. _promo_block) — QR-коды хостеров
|
||||||
|
# визуально конкурировали с чаевыми и перегружали экран.
|
||||||
show_promo_with_qr() {
|
show_promo_with_qr() {
|
||||||
_promo_block
|
_promo_block
|
||||||
|
|
||||||
# QR codes
|
# QR только для чаевых
|
||||||
if command -v qrencode &>/dev/null; then
|
if command -v qrencode &>/dev/null; then
|
||||||
echo -e " ${DIM}$(t promo_qr_host1)${NC}"
|
|
||||||
qrencode -t UTF8 -m 1 "https://vk.cc/ct29NQ" 2>/dev/null | while IFS= read -r qr_line; do
|
|
||||||
echo " $qr_line"
|
|
||||||
done
|
|
||||||
echo ""
|
|
||||||
echo -e " ${DIM}$(t promo_qr_host2)${NC}"
|
|
||||||
qrencode -t UTF8 -m 1 "https://vk.cc/cUxAhj" 2>/dev/null | while IFS= read -r qr_line; do
|
|
||||||
echo " $qr_line"
|
|
||||||
done
|
|
||||||
echo ""
|
|
||||||
echo -e " ${DIM}$(t promo_qr_tips)${NC}"
|
echo -e " ${DIM}$(t promo_qr_tips)${NC}"
|
||||||
qrencode -t UTF8 -m 1 "https://pay.cloudtips.ru/p/7410814f" 2>/dev/null | while IFS= read -r qr_line; do
|
qrencode -t UTF8 -m 1 "https://pay.cloudtips.ru/p/7410814f" 2>/dev/null | while IFS= read -r qr_line; do
|
||||||
echo " $qr_line"
|
echo " $qr_line"
|
||||||
|
|||||||
66
lib/common.sh
Normal file → Executable file
66
lib/common.sh
Normal file → Executable file
@@ -3,7 +3,7 @@
|
|||||||
# Colors, logging, spinner, system helpers, v1 compat, i18n-aware
|
# Colors, logging, spinner, system helpers, v1 compat, i18n-aware
|
||||||
|
|
||||||
# ── Version ───────────────────────────────────────────────────────────────────
|
# ── Version ───────────────────────────────────────────────────────────────────
|
||||||
GOTELEGRAM_VERSION="2.4.3"
|
GOTELEGRAM_VERSION="2.4.6"
|
||||||
GOTELEGRAM_NAME="GoTelegram"
|
GOTELEGRAM_NAME="GoTelegram"
|
||||||
|
|
||||||
# ── Пути ──────────────────────────────────────────────────────────────────────
|
# ── Пути ──────────────────────────────────────────────────────────────────────
|
||||||
@@ -240,13 +240,68 @@ get_pkg_manager() {
|
|||||||
install_pkg() {
|
install_pkg() {
|
||||||
local pkg="$1"
|
local pkg="$1"
|
||||||
case "$(get_pkg_manager)" in
|
case "$(get_pkg_manager)" in
|
||||||
apt) apt-get install -y -qq "$pkg" ;;
|
apt) apt_install "$pkg" ;;
|
||||||
dnf) dnf install -y -q "$pkg" ;;
|
dnf) dnf install -y -q "$pkg" ;;
|
||||||
yum) yum install -y -q "$pkg" ;;
|
yum) yum install -y -q "$pkg" ;;
|
||||||
*) log_error "$(_t_or err_bad_pkg_mgr 'Unknown package manager')"; return 1 ;;
|
*) log_error "$(_t_or err_bad_pkg_mgr 'Unknown package manager')"; return 1 ;;
|
||||||
esac
|
esac
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# ── apt lock wait + install ─────────────────────────────────────────────────
|
||||||
|
# На свежих Ubuntu/Debian unattended-upgrades часто держит dpkg lock на старте
|
||||||
|
# → любой apt-get install падает с "Could not get lock /var/lib/dpkg/lock-frontend".
|
||||||
|
# Эти функции ждут освобождения лока до 300с, потом запускают apt с нативным
|
||||||
|
# таймаутом DPkg::Lock::Timeout. Использовать везде, где раньше был
|
||||||
|
# "apt-get install ...".
|
||||||
|
apt_lock_wait() {
|
||||||
|
local max_wait="${1:-300}"
|
||||||
|
local waited=0
|
||||||
|
local warned=0
|
||||||
|
while fuser /var/lib/dpkg/lock-frontend &>/dev/null \
|
||||||
|
|| fuser /var/lib/dpkg/lock &>/dev/null \
|
||||||
|
|| fuser /var/lib/apt/lists/lock &>/dev/null \
|
||||||
|
|| pgrep -f '^/usr/bin/unattended-upgrade' &>/dev/null; do
|
||||||
|
if [ "$warned" = "0" ]; then
|
||||||
|
log_warning "apt/dpkg locked by unattended-upgrades, waiting up to ${max_wait}s..."
|
||||||
|
warned=1
|
||||||
|
fi
|
||||||
|
sleep 3
|
||||||
|
waited=$((waited + 3))
|
||||||
|
if [ "$waited" -ge "$max_wait" ]; then
|
||||||
|
log_error "apt lock not released after ${max_wait}s"
|
||||||
|
log_dim "Manual fix: systemctl stop unattended-upgrades && killall -9 unattended-upgr 2>/dev/null; dpkg --configure -a"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
[ "$warned" = "1" ] && log_success "apt lock released (waited ${waited}s)"
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# apt_install <pkg> [pkg2 ...] — ждёт lock + ставит пакеты + показывает ошибку
|
||||||
|
apt_install() {
|
||||||
|
[ $# -eq 0 ] && return 0
|
||||||
|
apt_lock_wait || return 1
|
||||||
|
export DEBIAN_FRONTEND=noninteractive
|
||||||
|
local opts="-o DPkg::Lock::Timeout=120"
|
||||||
|
local err_file; err_file=$(mktemp 2>/dev/null || echo /tmp/apt_err.$$)
|
||||||
|
if ! apt-get $opts install -y -qq "$@" 2>"$err_file"; then
|
||||||
|
log_error "apt-get install failed: $*"
|
||||||
|
[ -s "$err_file" ] && tail -n 5 "$err_file" | sed 's/^/ /' >&2
|
||||||
|
rm -f "$err_file"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
rm -f "$err_file"
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# apt_update — тихий update с ожиданием лока
|
||||||
|
apt_update() {
|
||||||
|
apt_lock_wait || return 1
|
||||||
|
export DEBIAN_FRONTEND=noninteractive
|
||||||
|
apt-get -o DPkg::Lock::Timeout=120 update -qq 2>/dev/null || true
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
# ── Зависимости GoTelegram ──────────────────────────────────────────────────
|
# ── Зависимости GoTelegram ──────────────────────────────────────────────────
|
||||||
# Полный список внешних команд, которые скрипт использует. Для каждой команды
|
# Полный список внешних команд, которые скрипт использует. Для каждой команды
|
||||||
# указан пакет на apt и dnf/yum (имена различаются: например dig = dnsutils на
|
# указан пакет на apt и dnf/yum (имена различаются: например dig = dnsutils на
|
||||||
@@ -344,9 +399,8 @@ ensure_deps() {
|
|||||||
|
|
||||||
case "$pkg_mgr" in
|
case "$pkg_mgr" in
|
||||||
apt)
|
apt)
|
||||||
export DEBIAN_FRONTEND=noninteractive
|
apt_update
|
||||||
apt-get update -qq 2>/dev/null
|
apt_install "${uniq_pkgs[@]}" || true
|
||||||
apt-get install -y -qq "${uniq_pkgs[@]}" 2>/dev/null
|
|
||||||
;;
|
;;
|
||||||
dnf) dnf install -y -q "${uniq_pkgs[@]}" 2>/dev/null ;;
|
dnf) dnf install -y -q "${uniq_pkgs[@]}" 2>/dev/null ;;
|
||||||
yum) yum install -y -q "${uniq_pkgs[@]}" 2>/dev/null ;;
|
yum) yum install -y -q "${uniq_pkgs[@]}" 2>/dev/null ;;
|
||||||
@@ -358,7 +412,7 @@ ensure_deps() {
|
|||||||
|
|
||||||
# Фолбэки для xxd: на некоторых системах нужен vim-common вместо xxd
|
# Фолбэки для xxd: на некоторых системах нужен vim-common вместо xxd
|
||||||
if ! command -v xxd &>/dev/null && [ "$pkg_mgr" = "apt" ]; then
|
if ! command -v xxd &>/dev/null && [ "$pkg_mgr" = "apt" ]; then
|
||||||
apt-get install -y -qq vim-common 2>/dev/null
|
apt_install vim-common || true
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Повторная проверка критических команд
|
# Повторная проверка критических команд
|
||||||
|
|||||||
@@ -9,9 +9,9 @@ install_nginx() {
|
|||||||
fi
|
fi
|
||||||
log_info "Установка nginx..."
|
log_info "Установка nginx..."
|
||||||
case "$(get_pkg_manager)" in
|
case "$(get_pkg_manager)" in
|
||||||
apt) apt-get update -qq && apt-get install -y -qq nginx ;;
|
apt) apt_update && apt_install nginx || return 1 ;;
|
||||||
dnf) dnf install -y -q nginx ;;
|
dnf) dnf install -y -q nginx || return 1 ;;
|
||||||
yum) yum install -y -q nginx ;;
|
yum) yum install -y -q nginx || return 1 ;;
|
||||||
esac
|
esac
|
||||||
systemctl enable nginx 2>/dev/null
|
systemctl enable nginx 2>/dev/null
|
||||||
}
|
}
|
||||||
@@ -24,9 +24,9 @@ install_certbot() {
|
|||||||
fi
|
fi
|
||||||
log_info "Установка certbot..."
|
log_info "Установка certbot..."
|
||||||
case "$(get_pkg_manager)" in
|
case "$(get_pkg_manager)" in
|
||||||
apt) apt-get install -y -qq certbot python3-certbot-nginx ;;
|
apt) apt_install certbot python3-certbot-nginx || return 1 ;;
|
||||||
dnf) dnf install -y -q certbot python3-certbot-nginx ;;
|
dnf) dnf install -y -q certbot python3-certbot-nginx || return 1 ;;
|
||||||
yum) yum install -y -q certbot python3-certbot-nginx ;;
|
yum) yum install -y -q certbot python3-certbot-nginx || return 1 ;;
|
||||||
esac
|
esac
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user