mirror of
https://github.com/hotyue/IP-Sentinel.git
synced 2026-05-11 23:29:48 +08:00
Compare commits
13 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
be75e5b65c | ||
|
|
c7ece6620c | ||
|
|
7e9da4b82a | ||
|
|
c657c92b27 | ||
|
|
d43163703e | ||
|
|
1150450718 | ||
|
|
df6483afa8 | ||
|
|
29de2eadf8 | ||
|
|
318689f163 | ||
|
|
6b9563b858 | ||
|
|
2c7491449c | ||
|
|
f698bc4b92 | ||
|
|
c18a10dbd5 |
44
.github/workflows/daily_keywords.yml
vendored
Normal file
44
.github/workflows/daily_keywords.yml
vendored
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
name: Daily Trends Factory
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
# 每天 UTC 18:00 运行 (北京时间凌晨 02:00)
|
||||||
|
- cron: '0 18 * * *'
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
update-trends:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout Code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Set up Python
|
||||||
|
uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: '3.10'
|
||||||
|
|
||||||
|
- name: Execute Trends Engine
|
||||||
|
run: python scripts/fetch_trends.py
|
||||||
|
|
||||||
|
- name: Commit and Push
|
||||||
|
run: |
|
||||||
|
git config --global user.name "github-actions[bot]"
|
||||||
|
git config --global user.email "github-actions[bot]@users.noreply.github.com"
|
||||||
|
|
||||||
|
git add data/keywords/
|
||||||
|
|
||||||
|
# 防御机制:如果没有新数据,就静默退出,不产生空提交
|
||||||
|
if git diff --staged --quiet; then
|
||||||
|
echo "No changes, skipping."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 策略:放弃危险的 amend 强制覆盖,采用带日期的标准安全提交
|
||||||
|
git commit -m "chore(data): 🤖 自动机兵:刷新全战区热点词库 [$(date +'%Y-%m-%d')]"
|
||||||
|
git push origin main
|
||||||
23
README.md
23
README.md
@@ -18,8 +18,16 @@
|
|||||||
|
|
||||||
- 🏭 **自动化指纹兵工厂 (Automated UA Factory)**:依托 GitHub Actions CI/CD 流水线,每月 1 日无人值守全自动生成 4000+ 带绝对物理分区的真实终端设备数据。配合边缘节点的守护进程静默拉取,实现千万级指纹资产的“自动驾驶”级演进。
|
- 🏭 **自动化指纹兵工厂 (Automated UA Factory)**:依托 GitHub Actions CI/CD 流水线,每月 1 日无人值守全自动生成 4000+ 带绝对物理分区的真实终端设备数据。配合边缘节点的守护进程静默拉取,实现千万级指纹资产的“自动驾驶”级演进。
|
||||||
|
|
||||||
|
- 📡 **OTA 动态活体词库 (Dynamic Trends)**:v3.3.0 跨时代跃升。彻底废弃静态搜索词,引入 GitHub Actions 云端流水线。每天自动抓取美、日、德、英等全战区当日 Google 热搜榜单,并通过边缘节点每日静默同步,让您的 IP 搜索行为永远贴合当地当天的真实网络脉搏。
|
||||||
|
|
||||||
|
- 🔀 **智能错峰调度 (Thundering Herd Mitigation)**:v3.3.0 架构升级。首创节点部署时间戳锚定逻辑。边缘节点按需智能分频(每日拉取几百行轻量词库,每月按 30 天周期错峰拉取千万级指纹库),完美化解“惊群效应”,彻底抹平统一升级时的数据并发特征,隐匿于无形。
|
||||||
|
|
||||||
- 🖧 **底层路由死锁 (Hard-Bind Routing)**:v3.2.1 热修复升级。底层探测引擎强力接管 curl 核心参数 (--interface),强制将发出的每一滴伪装流量死死绑定在您设定的物理网卡或隧道 IP 上,彻底杜绝双栈或多网卡环境下的流量溢出漏洞。
|
- 🖧 **底层路由死锁 (Hard-Bind Routing)**:v3.2.1 热修复升级。底层探测引擎强力接管 curl 核心参数 (--interface),强制将发出的每一滴伪装流量死死绑定在您设定的物理网卡或隧道 IP 上,彻底杜绝双栈或多网卡环境下的流量溢出漏洞。
|
||||||
|
|
||||||
|
- 🎯 **多级容灾与高精度探针 (High-Precision Probe)**:v3.2.2 底层重构。重写战报模块与底层协议自适应逻辑,植入多级 ISP 容灾探针链路,并按“底层数据共识原则”智能清洗冗余 AS 号。确保在纯 V6、隧道或弱网环境下,数据获取依然 100% 精准畅通。
|
||||||
|
|
||||||
|
- 🔄 **平滑热更新装甲 (Smooth Upgrade Engine)**:v3.2.2 体验进化。全系植入状态机嗅探逻辑。无论是 Master 司令部还是 Agent 边缘节点,再次执行部署脚本时将自动识别并继承历史配置、SQLite 数据库与锚定 IP,一键回车即可瞬间完成无损换代,告别繁琐的重复配置。
|
||||||
|
|
||||||
- ☁️ **云端中枢 (Public Master)**:引入官方公共机器人 @OmniBeacon_bot,新手无需部署 Master 司令部,部署 Agent 时一键回车即可调用官方加密网关,30 秒极速入伍!
|
- ☁️ **云端中枢 (Public Master)**:引入官方公共机器人 @OmniBeacon_bot,新手无需部署 Master 司令部,部署 Agent 时一键回车即可调用官方加密网关,30 秒极速入伍!
|
||||||
|
|
||||||
- 🧠 **分布式中枢 (Master-Agent)**:对于硬核极客,支持私有化部署。一台 Master 主控集成 SQLite 数据库,统管无数台 Agent 边缘节点,确保数据绝对私有。
|
- 🧠 **分布式中枢 (Master-Agent)**:对于硬核极客,支持私有化部署。一台 Master 主控集成 SQLite 数据库,统管无数台 Agent 边缘节点,确保数据绝对私有。
|
||||||
@@ -54,7 +62,7 @@
|
|||||||
|
|
||||||
## 🚀 极速部署 (Quick Start)
|
## 🚀 极速部署 (Quick Start)
|
||||||
|
|
||||||
v3.2.x 提供了两种接入模式,请根据您的战术需求选择:
|
v3.3.x 提供了两种接入模式,请根据您的战术需求选择:
|
||||||
|
|
||||||
### 🔹 模式 A:官方公共模式 (最简、推荐)
|
### 🔹 模式 A:官方公共模式 (最简、推荐)
|
||||||
**适合不想折腾、只想快速养护 IP 的新兵。**
|
**适合不想折腾、只想快速养护 IP 的新兵。**
|
||||||
@@ -74,7 +82,6 @@ bash <(curl -sL https://raw.githubusercontent.com/hotyue/IP-Sentinel/main/core/i
|
|||||||
```Bash
|
```Bash
|
||||||
bash <(curl -sL https://raw.githubusercontent.com/hotyue/IP-Sentinel/main/master/install_master.sh)
|
bash <(curl -sL https://raw.githubusercontent.com/hotyue/IP-Sentinel/main/master/install_master.sh)
|
||||||
|
|
||||||
|
|
||||||
```
|
```
|
||||||
2. **部署 Agent**:在需要养护的机器上执行 Agent 脚本,输入您自建机器人的 Token 以及与 Master 一致的配置。
|
2. **部署 Agent**:在需要养护的机器上执行 Agent 脚本,输入您自建机器人的 Token 以及与 Master 一致的配置。
|
||||||
```Bash
|
```Bash
|
||||||
@@ -83,15 +90,13 @@ bash <(curl -sL https://raw.githubusercontent.com/hotyue/IP-Sentinel/main/core/i
|
|||||||
```
|
```
|
||||||
3. **激活节点**:同上,将暗号转发给您自己的机器人即可。
|
3. **激活节点**:同上,将暗号转发给您自己的机器人即可。
|
||||||
|
|
||||||
### ⚠️ 存量节点升级指引 (Upgrade to v3.2.x)
|
### ⚠️ 平滑升级指引 (Upgrade to v3.3.0)
|
||||||
从 `v3.1.x` 升级至 `v3.2.x` 涉及**核心哈希锚定引擎**与**底层路由死锁机制**的深层 Bash 逻辑重构。边缘节点原有的后台守护进程无法自行完成这种级别的“换脑手术”。
|
|
||||||
|
|
||||||
为了彻底根除僵尸网络特征并修复流量溢出问题,**存量节点必须手动执行覆盖安装**。
|
得益于 **v3.2.2 全新引入的平滑热更新引擎 (Smooth Upgrade Engine)**,系统升级现已变得极其优雅与安全。
|
||||||
无需卸载,直接在您的所有 Agent 节点上再次运行官方部署指令即可(系统将自动覆盖旧版核心引擎,您的 Token 与绑定身份将完美保留):
|
|
||||||
```Bash
|
|
||||||
bash <(curl -sL https://raw.githubusercontent.com/hotyue/IP-Sentinel/main/core/install.sh)
|
|
||||||
|
|
||||||
```
|
无需卸载旧版本,无论您是要升级 Agent 边缘节点还是 Master 控制中枢,只需在您的终端中**再次运行上方对应的官方部署指令**。
|
||||||
|
|
||||||
|
安装雷达会自动嗅探您的历史部署状态(包括您的 Token、区域设定、SQLite 数据库及物理网卡锚点)。当询问是否平滑升级时,您只需**一路回车 (默认选 y)**,脚本将在短短 3 秒内瞬间完成核心装甲的无损换脑手术,您的所有战术资产将得到 100% 保留!
|
||||||
|
|
||||||
🗑️ 一键无痕卸载
|
🗑️ 一键无痕卸载
|
||||||
如果你需要清理某个边缘节点,只需重新运行 `core/install.sh` 并选择 **[2]**,或直接在节点终端执行:
|
如果你需要清理某个边缘节点,只需重新运行 `core/install.sh` 并选择 **[2]**,或直接在节点终端执行:
|
||||||
|
|||||||
100
core/install.sh
100
core/install.sh
@@ -1,8 +1,8 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
# ==========================================================
|
# ==========================================================
|
||||||
# 脚本名称: install.sh (IP-Sentinel 分布式边缘节点部署脚本 v3.0.3 - Global Nexus)
|
# 脚本名称: install.sh (IP-Sentinel 分布式边缘节点部署脚本 v3.3.0 - OTA 活体引擎)
|
||||||
# 核心功能: 区域选择、模块按需开启、官方机器人一键配置
|
# 核心功能: 区域选择、模块按需开启、官方机器人一键配置、平滑热更新、分频错峰调度
|
||||||
# ==========================================================
|
# ==========================================================
|
||||||
|
|
||||||
# 你的 GitHub 仓库 Raw 数据直链前缀
|
# 你的 GitHub 仓库 Raw 数据直链前缀
|
||||||
@@ -51,8 +51,31 @@ if [ "$ACTION_CHOICE" == "2" ]; then
|
|||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# ================== [v3.1.1 新增: 安装前环境纯净度清理 (严格保留日志)] ==================
|
# ================== [v3.2.2 新增: 平滑升级模式嗅探] ==================
|
||||||
echo -e "\n⏳ 正在清理旧版守护进程与冗余任务 (保留历史日志)..."
|
UPGRADE_MODE="false"
|
||||||
|
KEEP_LOGS="true"
|
||||||
|
|
||||||
|
if [ "$ACTION_CHOICE" == "1" ] && [ -f "$CONFIG_FILE" ]; then
|
||||||
|
echo -e "\n\033[33m💡 哨兵雷达提示:检测到本机已部署过 IP-Sentinel。\033[0m"
|
||||||
|
read -p "👉 是否按原配置直接进行平滑升级?(y/n, 默认y): " UPGRADE_CHOICE
|
||||||
|
if [[ -z "$UPGRADE_CHOICE" || "$UPGRADE_CHOICE" =~ ^[Yy]$ ]]; then
|
||||||
|
UPGRADE_MODE="true"
|
||||||
|
read -p "👉 是否保留历史运行日志?(y/n, 默认y): " LOG_CHOICE
|
||||||
|
if [[ "$LOG_CHOICE" =~ ^[Nn]$ ]]; then
|
||||||
|
KEEP_LOGS="false"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 将原配置读入环境变量,为后续跳过配置步骤提供燃料
|
||||||
|
source "$CONFIG_FILE"
|
||||||
|
echo -e "\033[32m✅ 已激活 [平滑升级模式],即将跳过基础配置,直接更新核心装甲...\033[0m"
|
||||||
|
else
|
||||||
|
echo -e "\033[33m🔄 您选择了重新配置,旧的哨兵数据将被彻底抹除。\033[0m"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
# ====================================================================
|
||||||
|
|
||||||
|
# ================== [v3.1.1/v3.2.2 优化: 安装前环境纯净度清理] ==================
|
||||||
|
echo -e "\n⏳ 正在清理旧版守护进程与冗余任务..."
|
||||||
# 1. 强制超度可能存活的 Webhook 及各类看门狗进程,释放端口
|
# 1. 强制超度可能存活的 Webhook 及各类看门狗进程,释放端口
|
||||||
pkill -9 -f "webhook.py" >/dev/null 2>&1 || true
|
pkill -9 -f "webhook.py" >/dev/null 2>&1 || true
|
||||||
pkill -9 -f "agent_daemon.sh" >/dev/null 2>&1 || true
|
pkill -9 -f "agent_daemon.sh" >/dev/null 2>&1 || true
|
||||||
@@ -65,13 +88,30 @@ if crontab -l >/dev/null 2>&1; then
|
|||||||
rm -f /tmp/cron_clean
|
rm -f /tmp/cron_clean
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# 3. 抹除旧版核心代码与配置文件,杜绝代码冲突 (精准避开 logs 目录)
|
# 3. 抹除旧版核心代码,杜绝代码冲突 (根据模式分流)
|
||||||
|
if [ "$UPGRADE_MODE" == "true" ]; then
|
||||||
|
# 升级模式:仅销毁核心引擎,严格保留 config 与 data
|
||||||
|
rm -rf "${INSTALL_DIR}/core" 2>/dev/null
|
||||||
|
if [ "$KEEP_LOGS" == "false" ]; then
|
||||||
|
rm -rf "${INSTALL_DIR}/logs" 2>/dev/null
|
||||||
|
echo -e "🗑️ 历史日志已按指令清空。"
|
||||||
|
else
|
||||||
|
echo -e "📦 历史配置与战地日志已妥善保留。"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
# 全新安装模式:焦土政策,彻底抹除
|
||||||
if [ -d "$INSTALL_DIR" ]; then
|
if [ -d "$INSTALL_DIR" ]; then
|
||||||
rm -rf "${INSTALL_DIR}/core" "${INSTALL_DIR}/data" "${INSTALL_DIR}/config.conf" "${INSTALL_DIR}/.last_ip" 2>/dev/null
|
rm -rf "${INSTALL_DIR}/core" "${INSTALL_DIR}/data" "${INSTALL_DIR}/config.conf" "${INSTALL_DIR}/.last_ip" 2>/dev/null
|
||||||
fi
|
fi
|
||||||
|
fi
|
||||||
echo -e "\033[32m✅ 环境清理完毕,幽灵进程已肃清!\033[0m"
|
echo -e "\033[32m✅ 环境清理完毕,幽灵进程已肃清!\033[0m"
|
||||||
# ========================================================================================
|
# ========================================================================================
|
||||||
|
|
||||||
|
# ==========================================================
|
||||||
|
# 🛑 如果是全新部署,才执行以下所有交互逻辑;否则直接跳过
|
||||||
|
# ==========================================================
|
||||||
|
if [ "$UPGRADE_MODE" == "false" ]; then
|
||||||
|
|
||||||
# 📍 动态一级菜单:国家选择
|
# 📍 动态一级菜单:国家选择
|
||||||
echo -e "\n\033[36m📍 【第一级】请选择目标国家/地区:\033[0m"
|
echo -e "\n\033[36m📍 【第一级】请选择目标国家/地区:\033[0m"
|
||||||
jq -r '.countries[] | "\(.id)|\(.name)|\(.keyword_file)"' /tmp/map.json > /tmp/countries.txt
|
jq -r '.countries[] | "\(.id)|\(.name)|\(.keyword_file)"' /tmp/map.json > /tmp/countries.txt
|
||||||
@@ -323,8 +363,15 @@ EOF
|
|||||||
chmod 600 "$CONFIG_FILE"
|
chmod 600 "$CONFIG_FILE"
|
||||||
# ====================================================================
|
# ====================================================================
|
||||||
|
|
||||||
|
fi
|
||||||
|
# 🛑 拦截块结束 (全套交互配置跳过完毕)
|
||||||
|
|
||||||
# 6. 拉取全套组件 (按需下载,绝不浪费空间)
|
# 6. 拉取全套组件 (按需下载,绝不浪费空间)
|
||||||
echo -e "\n[6/7] 正在根据模块开关部署核心引擎与热数据..."
|
echo -e "\n[6/7] 正在根据模块开关部署核心引擎与热数据..."
|
||||||
|
# 确保目录在升级模式下也能被正确建立
|
||||||
|
mkdir -p "${INSTALL_DIR}/core"
|
||||||
|
mkdir -p "${INSTALL_DIR}/data/keywords"
|
||||||
|
|
||||||
# 基础公共组件
|
# 基础公共组件
|
||||||
curl -sL "${REPO_RAW_URL}/core/runner.sh" -o "${INSTALL_DIR}/core/runner.sh"
|
curl -sL "${REPO_RAW_URL}/core/runner.sh" -o "${INSTALL_DIR}/core/runner.sh"
|
||||||
curl -sL "${REPO_RAW_URL}/core/updater.sh" -o "${INSTALL_DIR}/core/updater.sh"
|
curl -sL "${REPO_RAW_URL}/core/updater.sh" -o "${INSTALL_DIR}/core/updater.sh"
|
||||||
@@ -336,8 +383,13 @@ curl -sL "${REPO_RAW_URL}/data/user_agents.txt" -o "${INSTALL_DIR}/data/user_age
|
|||||||
# 动态按需组件
|
# 动态按需组件
|
||||||
if [ "$ENABLE_GOOGLE" == "true" ]; then
|
if [ "$ENABLE_GOOGLE" == "true" ]; then
|
||||||
curl -sL "${REPO_RAW_URL}/core/mod_google.sh" -o "${INSTALL_DIR}/core/mod_google.sh"
|
curl -sL "${REPO_RAW_URL}/core/mod_google.sh" -o "${INSTALL_DIR}/core/mod_google.sh"
|
||||||
# 根据 map.json 动态匹配的词库文件进行下载
|
# [v3.2.2 修复] 动态匹配词库下载逻辑
|
||||||
|
if [ "$UPGRADE_MODE" == "false" ]; then
|
||||||
curl -sL "${REPO_RAW_URL}/data/keywords/${KEYWORD_FILE}" -o "${INSTALL_DIR}/data/keywords/${KEYWORD_FILE}"
|
curl -sL "${REPO_RAW_URL}/data/keywords/${KEYWORD_FILE}" -o "${INSTALL_DIR}/data/keywords/${KEYWORD_FILE}"
|
||||||
|
else
|
||||||
|
# 升级模式:利用已有的 REGION_CODE 更新通用词库
|
||||||
|
curl -sL "${REPO_RAW_URL}/data/keywords/kw_${REGION_CODE}.txt" -o "${INSTALL_DIR}/data/keywords/kw_${REGION_CODE}.txt" 2>/dev/null || true
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ "$ENABLE_TRUST" == "true" ]; then
|
if [ "$ENABLE_TRUST" == "true" ]; then
|
||||||
@@ -352,8 +404,11 @@ crontab -l 2>/dev/null | grep -v "ip_sentinel" > /tmp/cron_backup
|
|||||||
|
|
||||||
# 核心养护模块: 每 30 分钟触发一次
|
# 核心养护模块: 每 30 分钟触发一次
|
||||||
echo "*/30 * * * * ${INSTALL_DIR}/core/runner.sh >/dev/null 2>&1" >> /tmp/cron_backup
|
echo "*/30 * * * * ${INSTALL_DIR}/core/runner.sh >/dev/null 2>&1" >> /tmp/cron_backup
|
||||||
# 养料更新模块: 每周日凌晨 3 点静默去云端更新热数据
|
# 养料更新模块: (v3.3.0升级) 每天凌晨 3 点触发,由中枢自动进行分频调度
|
||||||
echo "0 3 * * 0 ${INSTALL_DIR}/core/updater.sh >/dev/null 2>&1" >> /tmp/cron_backup
|
echo "0 3 * * * ${INSTALL_DIR}/core/updater.sh >/dev/null 2>&1" >> /tmp/cron_backup
|
||||||
|
|
||||||
|
# [v3.3.0 新增] 初始化 UA 指纹库更新时间戳,确立 30 天滚动周期的计算锚点
|
||||||
|
echo $(date +%s) > "${INSTALL_DIR}/core/.ua_last_update"
|
||||||
|
|
||||||
# 如果配置了联控,启动 Webhook 与战报任务
|
# 如果配置了联控,启动 Webhook 与战报任务
|
||||||
if [[ -n "$TG_TOKEN" ]] && [[ -n "$CHAT_ID" ]]; then
|
if [[ -n "$TG_TOKEN" ]] && [[ -n "$CHAT_ID" ]]; then
|
||||||
@@ -376,11 +431,23 @@ fi
|
|||||||
crontab /tmp/cron_backup
|
crontab /tmp/cron_backup
|
||||||
rm -f /tmp/cron_backup
|
rm -f /tmp/cron_backup
|
||||||
|
|
||||||
|
# ================== [v3.2.2 优化: 战报通知分流 (注册/升级)] ==================
|
||||||
if [[ -n "$TG_TOKEN" ]] && [[ -n "$CHAT_ID" ]]; then
|
if [[ -n "$TG_TOKEN" ]] && [[ -n "$CHAT_ID" ]]; then
|
||||||
echo -e "\n📡 正在向指挥部发送注册暗号..."
|
|
||||||
|
|
||||||
# 构造注册暗号 (V3.1.3 协议升级: 携带 REGION_CODE 大区标识)
|
|
||||||
NODE_NAME=$(hostname | cut -c 1-15)
|
NODE_NAME=$(hostname | cut -c 1-15)
|
||||||
|
|
||||||
|
if [ "$UPGRADE_MODE" == "true" ]; then
|
||||||
|
echo -e "\n📡 正在向指挥部发送升级成功战报..."
|
||||||
|
curl -s -X POST "${TG_API_URL}" \
|
||||||
|
-d "chat_id=${CHAT_ID}" \
|
||||||
|
-d "parse_mode=Markdown" \
|
||||||
|
-d "text=✨ *IP-Sentinel 引擎热更新完成!*
|
||||||
|
📍 节点:\`${NODE_NAME}\`
|
||||||
|
🌐 IP:\`${BIND_IP}\`
|
||||||
|
🚀 状态:v3.3.0 OTA 动态活体养护引擎已部署" >/dev/null 2>&1
|
||||||
|
echo -e "\033[32m✅ 升级成功通知已推送到您的 Telegram!\033[0m"
|
||||||
|
else
|
||||||
|
echo -e "\n📡 正在向指挥部发送注册暗号..."
|
||||||
|
# 构造注册暗号 (V3.1.3 协议升级: 携带 REGION_CODE 大区标识)
|
||||||
REG_MSG="#REGISTER#|${REGION_CODE}|${NODE_NAME}|${BIND_IP}|${AGENT_PORT}"
|
REG_MSG="#REGISTER#|${REGION_CODE}|${NODE_NAME}|${BIND_IP}|${AGENT_PORT}"
|
||||||
|
|
||||||
# 执行主动推送
|
# 执行主动推送
|
||||||
@@ -401,9 +468,15 @@ if [[ -n "$TG_TOKEN" ]] && [[ -n "$CHAT_ID" ]]; then
|
|||||||
echo -e "\033[31m❌ 消息推送失败,请检查 Chat ID 是否正确或是否已关注机器人。\033[0m"
|
echo -e "\033[31m❌ 消息推送失败,请检查 Chat ID 是否正确或是否已关注机器人。\033[0m"
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
fi
|
||||||
|
# =========================================================================
|
||||||
|
|
||||||
echo "========================================================"
|
echo "========================================================"
|
||||||
|
if [ "$UPGRADE_MODE" == "true" ]; then
|
||||||
|
echo "🎉 边缘节点 (Agent) 平滑热更新已彻底完成!"
|
||||||
|
else
|
||||||
echo "🎉 边缘节点 (Agent) 部署流程彻底完成!"
|
echo "🎉 边缘节点 (Agent) 部署流程彻底完成!"
|
||||||
|
fi
|
||||||
echo "📍 你的本地守护区域已锁定为: $REGION_NAME"
|
echo "📍 你的本地守护区域已锁定为: $REGION_NAME"
|
||||||
echo "⚙️ 哨兵现已开启 [每30分钟] 的高频高拟真养护循环。"
|
echo "⚙️ 哨兵现已开启 [每30分钟] 的高频高拟真养护循环。"
|
||||||
if [[ -n "$TG_TOKEN" ]]; then
|
if [[ -n "$TG_TOKEN" ]]; then
|
||||||
@@ -416,8 +489,13 @@ if [[ -n "$TG_TOKEN" ]]; then
|
|||||||
elif command -v firewall-cmd >/dev/null 2>&1 && systemctl is-active firewalld | grep -qw active; then
|
elif command -v firewall-cmd >/dev/null 2>&1 && systemctl is-active firewalld | grep -qw active; then
|
||||||
FW_MSG="firewall-cmd --zone=public --add-port=$AGENT_PORT/tcp --permanent && firewall-cmd --reload"
|
FW_MSG="firewall-cmd --zone=public --add-port=$AGENT_PORT/tcp --permanent && firewall-cmd --reload"
|
||||||
elif command -v iptables >/dev/null 2>&1; then
|
elif command -v iptables >/dev/null 2>&1; then
|
||||||
|
# 智能双栈雷达:根据绑定的 IP 属性,动态下发对应的防火墙放行指令
|
||||||
|
if [[ "$BIND_IP" == *":"* ]]; then
|
||||||
|
FW_MSG="ip6tables -I INPUT -p tcp --dport $AGENT_PORT -j ACCEPT"
|
||||||
|
else
|
||||||
FW_MSG="iptables -I INPUT -p tcp --dport $AGENT_PORT -j ACCEPT"
|
FW_MSG="iptables -I INPUT -p tcp --dport $AGENT_PORT -j ACCEPT"
|
||||||
fi
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
echo -e "\033[33m⚠️ 警告:请务必确保本机及云服务商安全组放行了 TCP $AGENT_PORT 端口!\033[0m"
|
echo -e "\033[33m⚠️ 警告:请务必确保本机及云服务商安全组放行了 TCP $AGENT_PORT 端口!\033[0m"
|
||||||
if [ -n "$FW_MSG" ]; then
|
if [ -n "$FW_MSG" ]; then
|
||||||
|
|||||||
@@ -18,19 +18,53 @@ if [ -z "$TG_TOKEN" ] || [ -z "$CHAT_ID" ]; then
|
|||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# 2. 节点元数据抓取 (v3.0.1修复: 严格使用配置中的协议探测出口与多节点容灾)
|
# 2. 节点元数据抓取 (v3.2.2 协议自适应与多级容灾版)
|
||||||
NODE_NAME=$(hostname | cut -c 1-15)
|
NODE_NAME=$(hostname | cut -c 1-15)
|
||||||
|
|
||||||
# 多节点容灾探测
|
# --- [防线 1: 底层路由锁定与协议自适应] ---
|
||||||
CURRENT_IP=$( (curl -${IP_PREF:-4} -s -m 5 api.ip.sb/ip || curl -${IP_PREF:-4} -s -m 5 ifconfig.me) 2>/dev/null | tr -d '[:space:]' )
|
CURL_BIND_OPT=""
|
||||||
|
DYNAMIC_IP_PREF="-${IP_PREF:-4}"
|
||||||
|
|
||||||
|
if [[ -n "$BIND_IP" && "$BIND_IP" =~ ^[0-9a-fA-F:\.]+$ ]]; then
|
||||||
|
CURL_BIND_OPT="--interface $BIND_IP"
|
||||||
|
if [[ "$BIND_IP" == *":"* ]]; then
|
||||||
|
DYNAMIC_IP_PREF="-6"
|
||||||
|
elif [[ "$BIND_IP" == *"."* ]]; then
|
||||||
|
DYNAMIC_IP_PREF="-4"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 多节点容灾探测出口 IP (注入协议自适应)
|
||||||
|
CURRENT_IP=$( (curl $CURL_BIND_OPT $DYNAMIC_IP_PREF -s -m 5 api.ip.sb/ip || curl $CURL_BIND_OPT $DYNAMIC_IP_PREF -s -m 5 ifconfig.me) 2>/dev/null | tr -d '[:space:]' )
|
||||||
# 强制兜底:如果所有外部 API 都挂了,直接使用本地强行锁定的 BIND_IP
|
# 强制兜底:如果所有外部 API 都挂了,直接使用本地强行锁定的 BIND_IP
|
||||||
[ -z "$CURRENT_IP" ] && CURRENT_IP="$BIND_IP"
|
[ -z "$CURRENT_IP" ] && CURRENT_IP="$BIND_IP"
|
||||||
|
|
||||||
# 为可能获取到的 IPv6 自动添加方括号护甲
|
# 为可能获取到的 IPv6 自动添加方括号护甲
|
||||||
[[ "$CURRENT_IP" == *":"* ]] && [[ "$CURRENT_IP" != *"["* ]] && CURRENT_IP="[${CURRENT_IP}]"
|
[[ "$CURRENT_IP" == *":"* ]] && [[ "$CURRENT_IP" != *"["* ]] && CURRENT_IP="[${CURRENT_IP}]"
|
||||||
|
|
||||||
# 智能判断 IP 属性
|
# --- [防线 2: 多级 ISP 容灾探针链路] ---
|
||||||
ISP_INFO=$(curl -${IP_PREF:-4} -s -m 5 api.ip.sb/geoip | jq -r '.organization' 2>/dev/null)
|
ISP_INFO=""
|
||||||
|
|
||||||
|
# 探针 A: 纯文本 API (免 jq,极速稳定)
|
||||||
|
ISP_INFO=$(curl $CURL_BIND_OPT $DYNAMIC_IP_PREF -s -m 5 ipinfo.io/org 2>/dev/null)
|
||||||
|
|
||||||
|
# 探针 B: 备用纯文本 API
|
||||||
|
if [ -z "$ISP_INFO" ] || [[ "$ISP_INFO" == *"error"* ]]; then
|
||||||
|
ISP_INFO=$(curl $CURL_BIND_OPT $DYNAMIC_IP_PREF -s -m 5 ip-api.com/line/?fields=isp 2>/dev/null)
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 探针 C: 原版的 JSON API (需要 jq 兜底)
|
||||||
|
if [ -z "$ISP_INFO" ] || [[ "$ISP_INFO" == *"error"* ]]; then
|
||||||
|
if command -v jq &> /dev/null; then
|
||||||
|
ISP_INFO=$(curl $CURL_BIND_OPT $DYNAMIC_IP_PREF -s -m 5 api.ip.sb/geoip | jq -r '.organization' 2>/dev/null)
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# --- [防线 3: 数据清洗 (遵循底层共识原则)] ---
|
||||||
|
# 剔除 ipinfo 返回的开头 AS 号 (例如 "AS137535 JT TELECOM" -> "JT TELECOM")
|
||||||
|
ISP_INFO=$(echo "$ISP_INFO" | sed -E 's/^AS[0-9]+ //')
|
||||||
|
|
||||||
|
# 最终兜底判断
|
||||||
[ -z "$ISP_INFO" ] || [ "$ISP_INFO" == "null" ] && ISP_INFO="未知 ISP"
|
[ -z "$ISP_INFO" ] || [ "$ISP_INFO" == "null" ] && ISP_INFO="未知 ISP"
|
||||||
|
|
||||||
if [[ "$ISP_INFO" == *"Cloudflare"* ]]; then
|
if [[ "$ISP_INFO" == *"Cloudflare"* ]]; then
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
# ==========================================================
|
# ==========================================================
|
||||||
# 脚本名称: updater.sh (IP-Sentinel V3.1.4 养料注入与维护中枢)
|
# 脚本名称: updater.sh (IP-Sentinel V3.3.0 养料注入与分频调度中枢)
|
||||||
# 核心功能: 静默更新热数据(指纹/词库/LBS规则)、清理瘦身日志
|
# 核心功能: 静默更新热搜词/LBS、指纹库错峰调度、强制出站死锁
|
||||||
# ==========================================================
|
# ==========================================================
|
||||||
|
|
||||||
INSTALL_DIR="/opt/ip_sentinel"
|
INSTALL_DIR="/opt/ip_sentinel"
|
||||||
CONFIG_FILE="${INSTALL_DIR}/config.conf"
|
CONFIG_FILE="${INSTALL_DIR}/config.conf"
|
||||||
# 你的 GitHub 仓库 Raw 数据直链前缀 (统一标准)
|
UA_TIME_FILE="${INSTALL_DIR}/core/.ua_last_update"
|
||||||
|
|
||||||
|
# GitHub 仓库 Raw 数据直链前缀
|
||||||
REPO_RAW_URL="https://raw.githubusercontent.com/hotyue/IP-Sentinel/main"
|
REPO_RAW_URL="https://raw.githubusercontent.com/hotyue/IP-Sentinel/main"
|
||||||
# 临时改为私库地址用于测试
|
|
||||||
# REPO_RAW_URL="https://git.94211762.xyz/hotyue/IP-Sentinel/raw/branch/main"
|
|
||||||
|
|
||||||
# 1. 加载本地冷数据配置
|
# 1. 加载本地冷数据配置
|
||||||
if [ ! -f "$CONFIG_FILE" ]; then
|
if [ ! -f "$CONFIG_FILE" ]; then
|
||||||
@@ -26,49 +26,94 @@ log() {
|
|||||||
|
|
||||||
log "Updater" "INFO " "========== 触发后台静默 OTA 热数据更新 =========="
|
log "Updater" "INFO " "========== 触发后台静默 OTA 热数据更新 =========="
|
||||||
|
|
||||||
# 3. 容灾机制拉取 UA 指纹池 (强制遵循锚点协议)
|
# ==========================================================
|
||||||
|
# 🛡️ 终极护城河:构建强锚定出站的 curl 请求引擎
|
||||||
|
# ==========================================================
|
||||||
|
# 基础参数:跟随 install.sh 锁定的协议偏好 (4 或 6)
|
||||||
|
CURL_CMD="curl -${IP_PREF:-4} -sL"
|
||||||
|
|
||||||
|
# 【防坑核心】如果用户配置了死锁锚点,必须强制绑定网卡,杜绝流量溢出!
|
||||||
|
if [ -n "$BIND_IP" ]; then
|
||||||
|
# curl 的 --interface 参数不支持带方括号的 IPv6 地址,必须强行脱壳
|
||||||
|
RAW_BIND_IP=$(echo "$BIND_IP" | tr -d '[]')
|
||||||
|
CURL_CMD="$CURL_CMD --interface $RAW_BIND_IP"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# ==========================================================
|
||||||
|
# 3. 容灾机制拉取 UA 指纹池 (V3.3.0 引入 30 天错峰防惊群逻辑)
|
||||||
|
# ==========================================================
|
||||||
|
NOW=$(date +%s)
|
||||||
|
LAST_UPDATE=0
|
||||||
|
|
||||||
|
# 读取上一次更新的时间戳
|
||||||
|
if [ -f "$UA_TIME_FILE" ]; then
|
||||||
|
# tr -d 清除可能存在的换行或回车符,防止算术崩溃
|
||||||
|
LAST_UPDATE=$(cat "$UA_TIME_FILE" | tr -d '\r\n')
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 校验数据合法性,防崩溃
|
||||||
|
if ! [[ "$LAST_UPDATE" =~ ^[0-9]+$ ]]; then
|
||||||
|
LAST_UPDATE=0
|
||||||
|
fi
|
||||||
|
|
||||||
|
DIFF=$((NOW - LAST_UPDATE))
|
||||||
|
|
||||||
|
# 距离上次拉取超过 30 天 (2592000 秒),才执行下载
|
||||||
|
if [ "$DIFF" -ge 2592000 ] || [ "$LAST_UPDATE" -eq 0 ]; then
|
||||||
TMP_UA="/tmp/ip_sentinel_ua.txt"
|
TMP_UA="/tmp/ip_sentinel_ua.txt"
|
||||||
curl -${IP_PREF:-4} -sL "${REPO_RAW_URL}/data/user_agents.txt" -o "$TMP_UA"
|
# 使用重装升级后的 CURL_CMD
|
||||||
|
$CURL_CMD "${REPO_RAW_URL}/data/user_agents.txt" -o "$TMP_UA"
|
||||||
|
|
||||||
if [ -s "$TMP_UA" ]; then
|
if [ -s "$TMP_UA" ]; then
|
||||||
mv "$TMP_UA" "${INSTALL_DIR}/data/user_agents.txt"
|
mv "$TMP_UA" "${INSTALL_DIR}/data/user_agents.txt"
|
||||||
log "Updater" "INFO " "✅ 设备指纹池 (User-Agents) 更新成功"
|
echo "$NOW" > "$UA_TIME_FILE"
|
||||||
|
log "Updater" "INFO " "✅ 设备指纹池 (User-Agents) 30天错峰滚动更新成功"
|
||||||
else
|
else
|
||||||
log "Updater" "WARN " "❌ UA 池拉取失败,保留本地旧数据防崩溃"
|
log "Updater" "WARN " "❌ UA 池拉取失败,保留本地旧数据防崩溃"
|
||||||
rm -f "$TMP_UA"
|
rm -f "$TMP_UA"
|
||||||
fi
|
fi
|
||||||
|
else
|
||||||
|
DAYS_LEFT=$(((2592000 - DIFF) / 86400))
|
||||||
|
log "Updater" "INFO " "⏳ 设备指纹池处于 30 天静默期 (剩余约 ${DAYS_LEFT} 天),跳过拉取"
|
||||||
|
fi
|
||||||
|
|
||||||
# 4. 容灾机制拉取当地最新搜索词库
|
# ==========================================================
|
||||||
|
# 4. 容灾机制拉取当地最新搜索词库 (每日高频拉取,保证活体新鲜度)
|
||||||
|
# ==========================================================
|
||||||
TMP_KW="/tmp/ip_sentinel_kw.txt"
|
TMP_KW="/tmp/ip_sentinel_kw.txt"
|
||||||
curl -${IP_PREF:-4} -sL "${REPO_RAW_URL}/data/keywords/kw_${REGION_CODE}.txt" -o "$TMP_KW"
|
$CURL_CMD "${REPO_RAW_URL}/data/keywords/kw_${REGION_CODE}.txt" -o "$TMP_KW"
|
||||||
|
|
||||||
if [ -s "$TMP_KW" ]; then
|
if [ -s "$TMP_KW" ]; then
|
||||||
mv "$TMP_KW" "${INSTALL_DIR}/data/keywords/kw_${REGION_CODE}.txt"
|
mv "$TMP_KW" "${INSTALL_DIR}/data/keywords/kw_${REGION_CODE}.txt"
|
||||||
log "Updater" "INFO " "✅ 区域搜索词库 (kw_${REGION_CODE}) 更新成功"
|
log "Updater" "INFO " "✅ 区域搜索词库 (kw_${REGION_CODE}) 每日同步成功"
|
||||||
else
|
else
|
||||||
log "Updater" "WARN " "❌ 搜索词库拉取失败,保留本地旧数据防崩溃"
|
log "Updater" "WARN " "❌ 搜索词库拉取失败,保留本地旧数据防崩溃"
|
||||||
rm -f "$TMP_KW"
|
rm -f "$TMP_KW"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# 5. 【V3.1.4 核心升级】自适应拉取本地 LBS 专属 JSON 规则库
|
# ==========================================================
|
||||||
# 利用 find 穿透寻找本地唯一的 json
|
# 5. 自适应拉取本地 LBS 专属 JSON 规则库 (每日同步)
|
||||||
|
# ==========================================================
|
||||||
REGION_JSON_FILE=$(find "${INSTALL_DIR}/data/regions" -name "*.json" 2>/dev/null | head -n 1)
|
REGION_JSON_FILE=$(find "${INSTALL_DIR}/data/regions" -name "*.json" 2>/dev/null | head -n 1)
|
||||||
|
|
||||||
if [ -n "$REGION_JSON_FILE" ] && [ -f "$REGION_JSON_FILE" ]; then
|
if [ -n "$REGION_JSON_FILE" ] && [ -f "$REGION_JSON_FILE" ]; then
|
||||||
# 提取本地文件的相对路径 (例如: data/regions/US/CA/San_Jose.json)
|
|
||||||
REL_PATH=${REGION_JSON_FILE#*${INSTALL_DIR}/}
|
REL_PATH=${REGION_JSON_FILE#*${INSTALL_DIR}/}
|
||||||
TMP_JSON="/tmp/ip_sentinel_region.json"
|
TMP_JSON="/tmp/ip_sentinel_region.json"
|
||||||
|
|
||||||
# 按照相同的相对路径去云端拉取更新
|
$CURL_CMD "${REPO_RAW_URL}/${REL_PATH}" -o "$TMP_JSON"
|
||||||
curl -${IP_PREF:-4} -sL "${REPO_RAW_URL}/${REL_PATH}" -o "$TMP_JSON"
|
|
||||||
if [ -s "$TMP_JSON" ]; then
|
if [ -s "$TMP_JSON" ]; then
|
||||||
mv "$TMP_JSON" "$REGION_JSON_FILE"
|
mv "$TMP_JSON" "$REGION_JSON_FILE"
|
||||||
log "Updater" "INFO " "✅ 核心战区规则库 ($REL_PATH) 更新成功"
|
log "Updater" "INFO " "✅ 核心战区规则库 ($REL_PATH) 每日同步成功"
|
||||||
else
|
else
|
||||||
log "Updater" "WARN " "❌ 战区规则库拉取失败,保留本地旧数据"
|
log "Updater" "WARN " "❌ 战区规则库拉取失败,保留本地旧数据"
|
||||||
rm -f "$TMP_JSON"
|
rm -f "$TMP_JSON"
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# ==========================================================
|
||||||
# 6. 日志防满瘦身机制 (保留最近 2000 行)
|
# 6. 日志防满瘦身机制 (保留最近 2000 行)
|
||||||
|
# ==========================================================
|
||||||
if [ -f "$LOG_FILE" ]; then
|
if [ -f "$LOG_FILE" ]; then
|
||||||
tail -n 2000 "$LOG_FILE" > "${LOG_FILE}.tmp"
|
tail -n 2000 "$LOG_FILE" > "${LOG_FILE}.tmp"
|
||||||
mv "${LOG_FILE}.tmp" "$LOG_FILE"
|
mv "${LOG_FILE}.tmp" "$LOG_FILE"
|
||||||
|
|||||||
@@ -3,5 +3,4 @@
|
|||||||
最新のニュース 速報
|
最新のニュース 速報
|
||||||
ゴールド 相場 チャート
|
ゴールド 相場 チャート
|
||||||
近くの静かなカフェ
|
近くの静かなカフェ
|
||||||
地震 速報
|
|
||||||
円安 影響 生活
|
円安 影響 生活
|
||||||
@@ -1,5 +1,10 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
|
# ==========================================================
|
||||||
|
# 脚本名称: install_master.sh (IP-Sentinel 控制中枢部署脚本 v3.3.0)
|
||||||
|
# 核心功能: 部署/卸载调度中枢、SQLite 资产管理、平滑热更新引擎
|
||||||
|
# ==========================================================
|
||||||
|
|
||||||
# [新增] 提取仓库直链前缀变量,方便后续在官方库和私库间一键切换
|
# [新增] 提取仓库直链前缀变量,方便后续在官方库和私库间一键切换
|
||||||
REPO_RAW_URL="https://raw.githubusercontent.com/hotyue/IP-Sentinel/main"
|
REPO_RAW_URL="https://raw.githubusercontent.com/hotyue/IP-Sentinel/main"
|
||||||
# 临时改为私库地址用于测试
|
# 临时改为私库地址用于测试
|
||||||
@@ -10,7 +15,7 @@ DB_FILE="${MASTER_DIR}/sentinel.db"
|
|||||||
|
|
||||||
echo "========================================================"
|
echo "========================================================"
|
||||||
# [修改] 将欢迎语改为更通用的文案,因为现在不仅能部署,还能卸载
|
# [修改] 将欢迎语改为更通用的文案,因为现在不仅能部署,还能卸载
|
||||||
echo " 🧠 欢迎使用 IP-Sentinel Master (控制中枢)"
|
echo " 🧠 欢迎使用 IP-Sentinel Master (控制中枢) v3.3.0"
|
||||||
echo "========================================================"
|
echo "========================================================"
|
||||||
|
|
||||||
# [新增] 交互式操作菜单:支持选择部署或调用卸载程序
|
# [新增] 交互式操作菜单:支持选择部署或调用卸载程序
|
||||||
@@ -29,13 +34,51 @@ if [ "$ACTION_CHOICE" == "2" ]; then
|
|||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# ================== [v3.1.1 延续: 安装前环境纯净度清理] ==================
|
# ================== [v3.2.2 新增: 平滑升级模式嗅探] ==================
|
||||||
echo -e "\n⏳ 正在清理旧版 Master 守护进程 (绝对安全保留 SQLite 数据库)..."
|
UPGRADE_MODE="false"
|
||||||
|
KEEP_DB="true"
|
||||||
|
|
||||||
|
if [ "$ACTION_CHOICE" == "1" ] && [ -f "${MASTER_DIR}/master.conf" ]; then
|
||||||
|
echo -e "\n\033[33m💡 司令部雷达提示:检测到本机已部署过 Master 中枢。\033[0m"
|
||||||
|
read -p "👉 是否按原配置直接进行平滑升级?(y/n, 默认y): " UPGRADE_CHOICE
|
||||||
|
if [[ -z "$UPGRADE_CHOICE" || "$UPGRADE_CHOICE" =~ ^[Yy]$ ]]; then
|
||||||
|
UPGRADE_MODE="true"
|
||||||
|
read -p "👉 是否保留历史节点数据库 (SQLite)?(y/n, 默认y): " DB_CHOICE
|
||||||
|
if [[ "$DB_CHOICE" =~ ^[Nn]$ ]]; then
|
||||||
|
KEEP_DB="false"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 汲取原配置进入内存
|
||||||
|
source "${MASTER_DIR}/master.conf"
|
||||||
|
echo -e "\033[32m✅ 已激活 [平滑升级模式],即将跳过基础配置,直接更新核心中枢...\033[0m"
|
||||||
|
else
|
||||||
|
echo -e "\033[33m🔄 您选择了重新配置,旧的中枢数据将被彻底抹除。\033[0m"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
# ====================================================================
|
||||||
|
|
||||||
|
# ================== [v3.2.2 优化: 安装前环境纯净度清理与数据保护] ==================
|
||||||
|
echo -e "\n⏳ 正在清理旧版 Master 守护进程..."
|
||||||
pkill -9 -f "tg_master.sh" >/dev/null 2>&1 || true
|
pkill -9 -f "tg_master.sh" >/dev/null 2>&1 || true
|
||||||
|
|
||||||
|
if [ "$UPGRADE_MODE" == "true" ]; then
|
||||||
|
if [ "$KEEP_DB" == "false" ]; then
|
||||||
|
rm -f "$DB_FILE" 2>/dev/null
|
||||||
|
echo -e "🗑️ 历史节点数据库已按指令清空。"
|
||||||
|
else
|
||||||
|
echo -e "📦 历史节点数据库 (SQLite) 已绝密保留。"
|
||||||
|
fi
|
||||||
|
# 删除旧的核心脚本,准备拉取新的
|
||||||
|
rm -f "${MASTER_DIR}/tg_master.sh" 2>/dev/null
|
||||||
|
else
|
||||||
|
# 焦土政策:如果不是升级模式,直接扬了整个司令部目录
|
||||||
|
rm -rf "$MASTER_DIR" 2>/dev/null
|
||||||
|
fi
|
||||||
|
echo -e "\033[32m✅ 旧进程已肃清!\033[0m"
|
||||||
# =======================================================================
|
# =======================================================================
|
||||||
|
|
||||||
# 1. 环境依赖安装
|
# 1. 环境依赖安装
|
||||||
echo "[1/4] 安装核心依赖 (curl, jq, sqlite3)..."
|
echo -e "\n[1/4] 安装核心依赖 (curl, jq, sqlite3)..."
|
||||||
if [ -f /etc/debian_version ]; then
|
if [ -f /etc/debian_version ]; then
|
||||||
apt-get update -y >/dev/null 2>&1
|
apt-get update -y >/dev/null 2>&1
|
||||||
apt-get install -y curl jq sqlite3 procps >/dev/null 2>&1
|
apt-get install -y curl jq sqlite3 procps >/dev/null 2>&1
|
||||||
@@ -45,6 +88,10 @@ fi
|
|||||||
|
|
||||||
mkdir -p "$MASTER_DIR"
|
mkdir -p "$MASTER_DIR"
|
||||||
|
|
||||||
|
# ==========================================================
|
||||||
|
# 🛑 如果是全新部署,才询问 Token 并写入配置
|
||||||
|
# ==========================================================
|
||||||
|
if [ "$UPGRADE_MODE" == "false" ]; then
|
||||||
# 2. 交互配置机器人
|
# 2. 交互配置机器人
|
||||||
echo -e "\n[2/4] 配置控制中枢机器人:"
|
echo -e "\n[2/4] 配置控制中枢机器人:"
|
||||||
read -p "请输入 Telegram Bot Token: " TG_TOKEN
|
read -p "请输入 Telegram Bot Token: " TG_TOKEN
|
||||||
@@ -54,8 +101,10 @@ TG_TOKEN="$TG_TOKEN"
|
|||||||
DB_FILE="$DB_FILE"
|
DB_FILE="$DB_FILE"
|
||||||
MASTER_DIR="$MASTER_DIR"
|
MASTER_DIR="$MASTER_DIR"
|
||||||
EOF
|
EOF
|
||||||
|
fi
|
||||||
|
# 🛑 拦截块结束
|
||||||
|
|
||||||
# 3. 初始化 SQLite 数据库
|
# 3. 初始化 SQLite 数据库 (幂等操作,升级模式下可安全修补表结构)
|
||||||
echo -e "\n[3/4] 正在初始化 SQLite 数据库表结构..."
|
echo -e "\n[3/4] 正在初始化 SQLite 数据库表结构..."
|
||||||
sqlite3 "$DB_FILE" <<EOF
|
sqlite3 "$DB_FILE" <<EOF
|
||||||
CREATE TABLE IF NOT EXISTS nodes (
|
CREATE TABLE IF NOT EXISTS nodes (
|
||||||
@@ -89,10 +138,17 @@ rm -f /tmp/cron_master
|
|||||||
# 立刻启动
|
# 立刻启动
|
||||||
pgrep -f tg_master.sh >/dev/null || nohup bash "${MASTER_DIR}/tg_master.sh" >/dev/null 2>&1 &
|
pgrep -f tg_master.sh >/dev/null || nohup bash "${MASTER_DIR}/tg_master.sh" >/dev/null 2>&1 &
|
||||||
|
|
||||||
|
# ================== [v3.2.2 优化: 战报文案分流] ==================
|
||||||
echo "========================================================"
|
echo "========================================================"
|
||||||
|
if [ "$UPGRADE_MODE" == "true" ]; then
|
||||||
|
echo "🎉 Master 控制中枢平滑热更新完成!"
|
||||||
|
echo "🤖 新版中枢引擎已接管数据库,继续等待边缘节点汇报。"
|
||||||
|
else
|
||||||
echo "🎉 Master 控制中枢部署完成!"
|
echo "🎉 Master 控制中枢部署完成!"
|
||||||
echo "🤖 机器人现已开始全局接客,等待边缘节点注册。"
|
echo "🤖 机器人现已开始全局接客,等待边缘节点注册。"
|
||||||
|
fi
|
||||||
echo "========================================================"
|
echo "========================================================"
|
||||||
|
# =================================================================
|
||||||
|
|
||||||
# ================== [v3.1.2 新增: 玻璃房透明装机统计] ==================
|
# ================== [v3.1.2 新增: 玻璃房透明装机统计] ==================
|
||||||
echo -e "\n📡 正在向开源社区汇报装机量 (完全匿名,不收集IP)..."
|
echo -e "\n📡 正在向开源社区汇报装机量 (完全匿名,不收集IP)..."
|
||||||
|
|||||||
81
scripts/fetch_trends.py
Normal file
81
scripts/fetch_trends.py
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
import urllib.request
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
import os
|
||||||
|
import json
|
||||||
|
import re
|
||||||
|
|
||||||
|
# ================== [路径防弹装甲] ==================
|
||||||
|
# 无论在哪里执行该脚本,都能精准反推项目根目录
|
||||||
|
SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
PROJECT_ROOT = os.path.dirname(SCRIPT_DIR)
|
||||||
|
|
||||||
|
MAP_JSON_PATH = os.path.join(PROJECT_ROOT, "data", "map.json")
|
||||||
|
DATA_DIR = os.path.join(PROJECT_ROOT, "data", "keywords")
|
||||||
|
# ====================================================
|
||||||
|
|
||||||
|
# 特殊战区代码映射 (Google Trends RSS 要求)
|
||||||
|
GEO_FIX = {'UK': 'GB'}
|
||||||
|
|
||||||
|
HEADERS = {
|
||||||
|
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36'
|
||||||
|
}
|
||||||
|
|
||||||
|
def get_active_regions():
|
||||||
|
"""动态提取 map.json 中的战区 (修正:精准穿透 JSON 数组结构)"""
|
||||||
|
try:
|
||||||
|
with open(MAP_JSON_PATH, 'r', encoding='utf-8') as f:
|
||||||
|
data = json.load(f)
|
||||||
|
# 穿透到 countries 列表,提取所有的国家代码 (id)
|
||||||
|
regions = []
|
||||||
|
for country in data.get('countries', []):
|
||||||
|
if 'id' in country:
|
||||||
|
regions.append(country['id'])
|
||||||
|
return regions
|
||||||
|
except Exception as e:
|
||||||
|
print(f"❌ [读取地图失败]: {e}")
|
||||||
|
return []
|
||||||
|
|
||||||
|
def fetch_trends(region_code):
|
||||||
|
"""从 Google Trends 抓取当日热搜"""
|
||||||
|
geo = GEO_FIX.get(region_code, region_code)
|
||||||
|
url = f"https://trends.google.com/trending/rss?geo={geo}"
|
||||||
|
try:
|
||||||
|
req = urllib.request.Request(url, headers=HEADERS)
|
||||||
|
with urllib.request.urlopen(req, timeout=10) as response:
|
||||||
|
xml_data = response.read()
|
||||||
|
root = ET.fromstring(xml_data)
|
||||||
|
return [re.sub(r'[\n\r\t]', ' ', item.find('title').text).strip()
|
||||||
|
for item in root.findall('./channel/item')
|
||||||
|
if item.find('title') is not None]
|
||||||
|
except Exception as e:
|
||||||
|
print(f"⚠️ {region_code} 抓取异常: {e}")
|
||||||
|
return []
|
||||||
|
|
||||||
|
def update_file(region, new_words):
|
||||||
|
"""滑动窗口更新,保留 200 条最热记录"""
|
||||||
|
os.makedirs(DATA_DIR, exist_ok=True)
|
||||||
|
file_path = os.path.join(DATA_DIR, f"kw_{region}.txt")
|
||||||
|
old_words = []
|
||||||
|
if os.path.exists(file_path):
|
||||||
|
with open(file_path, 'r', encoding='utf-8') as f:
|
||||||
|
old_words = [l.strip() for l in f if l.strip()]
|
||||||
|
|
||||||
|
# 新词排在最前面,去重
|
||||||
|
combined = new_words + [w for w in old_words if w not in new_words]
|
||||||
|
final_list = combined[:200]
|
||||||
|
|
||||||
|
with open(file_path, 'w', encoding='utf-8') as f:
|
||||||
|
f.write('\n'.join(final_list) + '\n')
|
||||||
|
print(f"✅ [同步完成] {region}: 注入 {len(new_words)} 条新热点")
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
regions = get_active_regions()
|
||||||
|
if not regions:
|
||||||
|
print("🛑 未发现活跃战区,请检查 map.json")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
for r in regions:
|
||||||
|
print(f"📡 正在拉取 {r} 战区情报...")
|
||||||
|
words = fetch_trends(r)
|
||||||
|
if words:
|
||||||
|
update_file(r, words)
|
||||||
Reference in New Issue
Block a user