Update 实时 history 监控
This commit is contained in:
740
实时 history 监控
740
实时 history 监控
@@ -1,269 +1,83 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
# 极简版命令监控脚本 - 前后台同时记录
|
||||||
# 交互式实时命令监控脚本 - 修复版
|
|
||||||
# 版本: 4.4
|
|
||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
### 配置区域 ###
|
### 配置 ###
|
||||||
LOG_DIR="/root/command_monitor_logs"
|
INSTALL_DIR="/root/安装"
|
||||||
SCRIPT_NAME="command_monitor.sh"
|
SCRIPT_NAME="cmd_monitor.sh"
|
||||||
SCRIPT_PATH="/usr/local/bin/$SCRIPT_NAME"
|
SCRIPT_PATH="$INSTALL_DIR/$SCRIPT_NAME"
|
||||||
|
LOG_DIR="/root/command_logs"
|
||||||
|
PID_FILE="/tmp/cmd_monitor.pid"
|
||||||
|
|
||||||
### 颜色定义 ###
|
### 颜色定义 ###
|
||||||
RED='\033[0;31m'
|
RED='\033[0;31m'
|
||||||
GREEN='\033[0;32m'
|
GREEN='\033[0;32m'
|
||||||
YELLOW='\033[1;33m'
|
YELLOW='\033[1;33m'
|
||||||
BLUE='\033[0;34m'
|
BLUE='\033[0;34m'
|
||||||
CYAN='\033[0;36m'
|
|
||||||
NC='\033[0m'
|
NC='\033[0m'
|
||||||
|
|
||||||
### 全局变量 ###
|
# 创建安装目录
|
||||||
CURRENT_LOG=""
|
create_install_dir() {
|
||||||
LATEST_LOG=""
|
if [ ! -d "$INSTALL_DIR" ]; then
|
||||||
DAEMON_MODE=false
|
echo -e "${YELLOW}创建安装目录: $INSTALL_DIR${NC}"
|
||||||
|
mkdir -p "$INSTALL_DIR"
|
||||||
# 检查TO命令
|
|
||||||
check_to_command() {
|
|
||||||
if [ "$1" = "TO" ] || [ "$1" = "to" ]; then
|
|
||||||
echo -e "${GREEN}切换到监控控制界面...${NC}"
|
|
||||||
exec bash <(curl -sSL https://raw.githubusercontent.com/xzx3344521/dock/refs/heads/main/%E5%AE%9E%E6%97%B6%20history%20%E7%9B%91%E6%8E%A7)
|
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# 清理所有残留文件和进程
|
|
||||||
cleanup_all() {
|
|
||||||
echo -e "${YELLOW}正在清理所有残留...${NC}"
|
|
||||||
|
|
||||||
# 停止所有监控进程
|
|
||||||
echo -e "${BLUE}停止监控进程...${NC}"
|
|
||||||
pkill -f "monitor_history_files" 2>/dev/null || true
|
|
||||||
pkill -f "command_monitor" 2>/dev/null || true
|
|
||||||
|
|
||||||
# 删除PID文件
|
|
||||||
rm -f "/tmp/command_monitor.pid" 2>/dev/null || true
|
|
||||||
rm -f "/tmp/last_size_"* 2>/dev/null || true
|
|
||||||
|
|
||||||
# 删除缓存文件
|
|
||||||
rm -f "/tmp/ip_geo_cache.txt" 2>/dev/null || true
|
|
||||||
|
|
||||||
# 删除旧的自启动配置
|
|
||||||
echo -e "${BLUE}清理自启动配置...${NC}"
|
|
||||||
rm -f /etc/profile.d/command_monitor_auto.sh 2>/dev/null || true
|
|
||||||
rm -f /etc/systemd/system/command-monitor.service 2>/dev/null || true
|
|
||||||
systemctl daemon-reload 2>/dev/null || true
|
|
||||||
|
|
||||||
# 清理rc.local中的配置
|
|
||||||
if [ -f "/etc/rc.local" ]; then
|
|
||||||
sed -i '/command_monitor.sh/d' "/etc/rc.local" 2>/dev/null || true
|
|
||||||
fi
|
|
||||||
if [ -f "/etc/rc.d/rc.local" ]; then
|
|
||||||
sed -i '/command_monitor.sh/d' "/etc/rc.d/rc.local" 2>/dev/null || true
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 清理crontab
|
|
||||||
crontab -l 2>/dev/null | grep -v "command_monitor.sh" | crontab -
|
|
||||||
|
|
||||||
echo -e "${GREEN}✓ 所有残留已清理完成${NC}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检查磁盘空间
|
|
||||||
check_disk_space() {
|
|
||||||
local available_space=$(df / | awk 'NR==2{print $4}')
|
|
||||||
if [ "$available_space" -lt 10240 ]; then # 小于10MB
|
|
||||||
echo -e "${RED}磁盘空间不足,可用空间: ${available_space}KB${NC}"
|
|
||||||
echo -e "${YELLOW}建议清理磁盘空间后再继续${NC}"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
echo -e "${GREEN}磁盘空间充足: ${available_space}KB${NC}"
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取时间戳
|
|
||||||
timestamp() {
|
|
||||||
date '+%Y-%m-%d %H:%M:%S'
|
|
||||||
}
|
|
||||||
|
|
||||||
# 日志函数
|
|
||||||
log_message() {
|
|
||||||
local level="$1"
|
|
||||||
local message="$2"
|
|
||||||
local color="$GREEN"
|
|
||||||
|
|
||||||
case "$level" in
|
|
||||||
"ERROR") color="$RED" ;;
|
|
||||||
"WARN") color="$YELLOW" ;;
|
|
||||||
"INFO") color="$BLUE" ;;
|
|
||||||
"SUCCESS") color="$GREEN" ;;
|
|
||||||
"COMMAND") color="$CYAN" ;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
if [ "$DAEMON_MODE" = true ]; then
|
|
||||||
echo -e "${color}[$(timestamp)] [$level] $message${NC}" >> "$CURRENT_LOG"
|
|
||||||
else
|
|
||||||
echo -e "${color}[$(timestamp)] [$level] $message${NC}" | tee -a "$CURRENT_LOG"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取客户端IP(修复版)
|
|
||||||
get_client_ip() {
|
|
||||||
local ip="unknown"
|
|
||||||
|
|
||||||
# 方法1: 从SSH连接获取
|
|
||||||
if [ -n "$SSH_CLIENT" ]; then
|
|
||||||
ip=$(echo "$SSH_CLIENT" | awk '{print $1}')
|
|
||||||
elif [ -n "$SSH_CONNECTION" ]; then
|
|
||||||
ip=$(echo "$SSH_CONNECTION" | awk '{print $1}')
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 方法2: 从who命令获取
|
|
||||||
if [ "$ip" = "unknown" ]; then
|
|
||||||
ip=$(who -m 2>/dev/null | awk '{print $5}' | sed 's/[()]//g' | head -1)
|
|
||||||
if [[ "$ip" == ":0" ]] || [[ "$ip" == ":1" ]] || [[ -z "$ip" ]]; then
|
|
||||||
ip="localhost"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 方法3: 从网络连接获取
|
|
||||||
if [ "$ip" = "unknown" ] || [ "$ip" = "localhost" ]; then
|
|
||||||
ip=$(netstat -tn 2>/dev/null | grep ESTABLISHED | awk '{print $5}' | cut -d: -f1 | head -1)
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "$ip"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取IP地理位置(修复版)
|
|
||||||
get_ip_location() {
|
|
||||||
local ip="$1"
|
|
||||||
|
|
||||||
if [[ "$ip" == "192.168."* ]] || [[ "$ip" == "10."* ]] || [[ "$ip" == "172."* ]]; then
|
|
||||||
echo "内网IP"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "$ip" == "127.0.0.1" ]] || [[ "$ip" == "localhost" ]] || [[ "$ip" == "unknown" ]]; then
|
|
||||||
echo "本机"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
local location_info=$(curl -s -m 3 "http://ip-api.com/json/$ip?fields=status,country,regionName,city,isp" 2>/dev/null || true)
|
|
||||||
if [ -n "$location_info" ] && echo "$location_info" | grep -q '"status":"success"'; then
|
|
||||||
local country=$(echo "$location_info" | grep -o '"country":"[^"]*"' | cut -d'"' -f4)
|
|
||||||
local region=$(echo "$location_info" | grep -o '"regionName":"[^"]*"' | cut -d'"' -f4)
|
|
||||||
local city=$(echo "$location_info" | grep -o '"city":"[^"]*"' | cut -d'"' -f4)
|
|
||||||
local isp=$(echo "$location_info" | grep -o '"isp":"[^"]*"' | cut -d'"' -f4)
|
|
||||||
|
|
||||||
if [ -n "$country" ]; then
|
|
||||||
local result=""
|
|
||||||
[ -n "$country" ] && result="$country"
|
|
||||||
[ -n "$region" ] && result="$result-$region"
|
|
||||||
[ -n "$city" ] && result="$result-$city"
|
|
||||||
[ -n "$isp" ] && result="$result($isp)"
|
|
||||||
echo "$result"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "未知位置"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 初始化日志系统
|
|
||||||
init_log_system() {
|
|
||||||
mkdir -p "$LOG_DIR"
|
|
||||||
|
|
||||||
local client_ip=$(get_client_ip | sed 's/\./_/g')
|
|
||||||
local log_date=$(date '+%Y%m%d_%H%M%S')
|
|
||||||
CURRENT_LOG="$LOG_DIR/monitor_${client_ip}_${log_date}.log"
|
|
||||||
LATEST_LOG="$LOG_DIR/latest.log"
|
|
||||||
|
|
||||||
ln -sf "$CURRENT_LOG" "$LATEST_LOG" 2>/dev/null || true
|
|
||||||
}
|
|
||||||
|
|
||||||
# 配置实时history记录
|
|
||||||
configure_realtime_history() {
|
|
||||||
for user_dir in /home/* /root; do
|
|
||||||
if [ -d "$user_dir" ]; then
|
|
||||||
local user=$(basename "$user_dir")
|
|
||||||
local bashrc="$user_dir/.bashrc"
|
|
||||||
|
|
||||||
if [ -f "$bashrc" ] && ! grep -q "REAL_TIME_HISTORY" "$bashrc"; then
|
|
||||||
cat >> "$bashrc" << 'EOF'
|
|
||||||
|
|
||||||
# REAL_TIME_HISTORY - 实时命令记录配置
|
|
||||||
export PROMPT_COMMAND='history -a; history -c; history -r'
|
|
||||||
export HISTTIMEFORMAT='%F %T '
|
|
||||||
export HISTSIZE=10000
|
|
||||||
export HISTFILESIZE=20000
|
|
||||||
shopt -s histappend
|
|
||||||
export HISTCONTROL=ignoredups:erasedups
|
|
||||||
|
|
||||||
EOF
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
# 安装脚本到系统(修复版)
|
|
||||||
install_script() {
|
|
||||||
echo -e "${GREEN}正在安装脚本到系统...${NC}"
|
|
||||||
|
|
||||||
# 检查磁盘空间
|
|
||||||
if ! check_disk_space; then
|
|
||||||
echo -e "${RED}磁盘空间不足,安装中止${NC}"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 先清理残留
|
|
||||||
cleanup_all
|
|
||||||
|
|
||||||
# 创建修复版脚本
|
|
||||||
cat > "$SCRIPT_PATH" << 'EOF'
|
|
||||||
#!/bin/bash
|
|
||||||
# 修复版命令监控脚本
|
|
||||||
|
|
||||||
LOG_DIR="/root/command_monitor_logs"
|
|
||||||
PID_FILE="/tmp/command_monitor.pid"
|
|
||||||
LOG_FILE="$LOG_DIR/command_monitor.log"
|
|
||||||
|
|
||||||
# 获取客户端IP
|
# 获取客户端IP
|
||||||
get_client_ip() {
|
get_client_ip() {
|
||||||
local ip="unknown"
|
local ip="unknown"
|
||||||
|
|
||||||
if [ -n "$SSH_CLIENT" ]; then
|
if [ -n "$SSH_CLIENT" ]; then
|
||||||
ip=$(echo "$SSH_CLIENT" | awk '{print $1}')
|
ip=$(echo "$SSH_CLIENT" | awk '{print $1}')
|
||||||
elif [ -n "$SSH_CONNECTION" ]; then
|
elif [ -n "$SSH_CONNECTION" ]; then
|
||||||
ip=$(echo "$SSH_CONNECTION" | awk '{print $1}')
|
ip=$(echo "$SSH_CONNECTION" | awk '{print $1}')
|
||||||
else
|
else
|
||||||
ip=$(who -m 2>/dev/null | awk '{print $5}' | sed 's/[()]//g' | head -1)
|
ip=$(who -m 2>/dev/null | awk '{print $NF}' | sed 's/[()]//g' | head -1)
|
||||||
|
[[ "$ip" =~ ^:[0-9]+$ ]] && ip="localhost"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo "$ip"
|
echo "$ip"
|
||||||
}
|
}
|
||||||
|
|
||||||
start_daemon() {
|
# 配置实时history记录
|
||||||
mkdir -p "$LOG_DIR"
|
setup_realtime_history() {
|
||||||
|
|
||||||
# 后台监控进程
|
|
||||||
(
|
|
||||||
# 配置实时history
|
|
||||||
for user_dir in /home/* /root; do
|
for user_dir in /home/* /root; do
|
||||||
[ -d "$user_dir" ] || continue
|
[ -d "$user_dir" ] || continue
|
||||||
user=$(basename "$user_dir")
|
local bashrc="$user_dir/.bashrc"
|
||||||
bashrc="$user_dir/.bashrc"
|
|
||||||
[ -f "$bashrc" ] || continue
|
[ -f "$bashrc" ] || continue
|
||||||
|
|
||||||
if ! grep -q "REAL_TIME_HISTORY" "$bashrc"; then
|
if ! grep -q "AUTO_HISTORY_SETUP" "$bashrc"; then
|
||||||
cat >> "$bashrc" << 'EOS'
|
cat >> "$bashrc" << 'EOF'
|
||||||
|
|
||||||
# REAL_TIME_HISTORY - 实时命令记录配置
|
# AUTO_HISTORY_SETUP - 实时命令记录
|
||||||
export PROMPT_COMMAND='history -a; history -c; history -r'
|
export PROMPT_COMMAND='history -a; history -c; history -r'
|
||||||
export HISTTIMEFORMAT='%F %T '
|
export HISTTIMEFORMAT='%F %T '
|
||||||
export HISTSIZE=10000
|
export HISTSIZE=10000
|
||||||
export HISTFILESIZE=20000
|
export HISTFILESIZE=20000
|
||||||
shopt -s histappend
|
shopt -s histappend
|
||||||
export HISTCONTROL=ignoredups:erasedups
|
EOF
|
||||||
|
|
||||||
EOS
|
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
# 后台监控进程
|
||||||
|
start_background_monitor() {
|
||||||
|
# 如果已经在运行,先停止
|
||||||
|
stop_background_monitor 2>/dev/null || true
|
||||||
|
|
||||||
|
echo -e "${GREEN}启动后台监控...${NC}"
|
||||||
|
|
||||||
|
(
|
||||||
|
# 设置实时history
|
||||||
|
setup_realtime_history
|
||||||
|
|
||||||
|
# 创建日志目录
|
||||||
|
mkdir -p "$LOG_DIR"
|
||||||
|
local log_file="$LOG_DIR/background_$(date +%Y%m%d).log"
|
||||||
|
|
||||||
|
echo "=== 后台监控启动于: $(date) ===" >> "$log_file"
|
||||||
|
|
||||||
# 主监控循环
|
# 主监控循环
|
||||||
declare -A last_sizes
|
declare -A last_sizes
|
||||||
@@ -272,24 +86,29 @@ EOS
|
|||||||
while true; do
|
while true; do
|
||||||
for user_dir in /home/* /root; do
|
for user_dir in /home/* /root; do
|
||||||
[ -d "$user_dir" ] || continue
|
[ -d "$user_dir" ] || continue
|
||||||
user=$(basename "$user_dir")
|
local user=$(basename "$user_dir")
|
||||||
history_file="$user_dir/.bash_history"
|
local history_file="$user_dir/.bash_history"
|
||||||
[ -f "$history_file" ] || continue
|
[ -f "$history_file" ] || continue
|
||||||
|
|
||||||
current_size=$(stat -c%s "$history_file" 2>/dev/null || echo 0)
|
local current_size=$(stat -c%s "$history_file" 2>/dev/null || echo 0)
|
||||||
last_size=${last_sizes["$user"]:-0}
|
local last_size=${last_sizes["$user"]:-0}
|
||||||
|
|
||||||
if [ "$current_size" -gt "$last_size" ]; then
|
if [ "$current_size" -gt "$last_size" ]; then
|
||||||
new_command=$(tail -n 1 "$history_file" 2>/dev/null | sed 's/^[ \t]*//;s/[ \t]*$//')
|
local new_command=$(tail -n 1 "$history_file" 2>/dev/null | sed 's/^[ \t]*//;s/[ \t]*$//')
|
||||||
if [ -n "$new_command" ] && [ ${#new_command} -gt 2 ] && \
|
|
||||||
[[ ! "$new_command" =~ ^(ls|cd|pwd|history|exit|clear|TO|to)$ ]] && \
|
if [ -n "$new_command" ] &&
|
||||||
[[ ! "$new_command" =~ ^#[0-9]+$ ]] && \
|
[ ${#new_command} -gt 1 ] &&
|
||||||
|
[[ ! "$new_command" =~ ^(ls|cd|pwd|ll|la|history|exit|clear|to|TO)$ ]] &&
|
||||||
[ "$new_command" != "${last_commands["$user"]}" ]; then
|
[ "$new_command" != "${last_commands["$user"]}" ]; then
|
||||||
|
|
||||||
client_ip=$(get_client_ip)
|
local client_ip=$(get_client_ip)
|
||||||
echo "[$(date "+%Y-%m-%d %H:%M:%S")] 用户: $user | 命令: $new_command | 来源: $client_ip" >> "$LOG_FILE"
|
local timestamp=$(date '+%Y-%m-%d %H:%M:%S')
|
||||||
|
local log_entry="[BG][$timestamp] 用户:$user | 命令:$new_command | 来源:$client_ip"
|
||||||
|
|
||||||
|
echo "$log_entry" >> "$log_file"
|
||||||
last_commands["$user"]="$new_command"
|
last_commands["$user"]="$new_command"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
last_sizes["$user"]=$current_size
|
last_sizes["$user"]=$current_size
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
@@ -298,165 +117,40 @@ EOS
|
|||||||
) &
|
) &
|
||||||
|
|
||||||
echo $! > "$PID_FILE"
|
echo $! > "$PID_FILE"
|
||||||
echo "监控已启动 (PID: $!)"
|
echo -e "${GREEN}✓ 后台监控已启动 (PID: $!)${NC}"
|
||||||
echo "日志文件: $LOG_FILE"
|
echo -e "日志文件: $LOG_DIR/background_*.log"
|
||||||
echo "输入 'TO' 进入控制界面"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
stop_daemon() {
|
# 停止后台监控
|
||||||
|
stop_background_monitor() {
|
||||||
if [ -f "$PID_FILE" ]; then
|
if [ -f "$PID_FILE" ]; then
|
||||||
pid=$(cat "$PID_FILE")
|
local pid=$(cat "$PID_FILE")
|
||||||
kill "$pid" 2>/dev/null && echo "监控已停止" || echo "停止失败"
|
kill "$pid" 2>/dev/null && echo -e "${GREEN}✓ 后台监控已停止${NC}" || echo -e "${YELLOW}后台监控未运行${NC}"
|
||||||
rm -f "$PID_FILE"
|
rm -f "$PID_FILE"
|
||||||
else
|
else
|
||||||
echo "监控未运行"
|
echo -e "${YELLOW}后台监控未运行${NC}"
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# 检查TO命令
|
# 前台监控模式
|
||||||
if [ "$1" = "TO" ] || [ "$1" = "to" ]; then
|
|
||||||
echo "切换到监控控制界面..."
|
|
||||||
exec bash <(curl -sSL https://raw.githubusercontent.com/xzx3344521/dock/refs/heads/main/%E5%AE%9E%E6%97%B6%20history%20%E7%9B%91%E6%8E%A7)
|
|
||||||
fi
|
|
||||||
|
|
||||||
case "$1" in
|
|
||||||
start) start_daemon ;;
|
|
||||||
stop) stop_daemon ;;
|
|
||||||
status)
|
|
||||||
if [ -f "$PID_FILE" ] && ps -p $(cat "$PID_FILE") >/dev/null 2>&1; then
|
|
||||||
echo "监控运行中 (PID: $(cat "$PID_FILE"))"
|
|
||||||
if [ -f "$LOG_FILE" ]; then
|
|
||||||
echo "最新日志:"
|
|
||||||
tail -1 "$LOG_FILE" 2>/dev/null || echo "无日志记录"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
echo "监控未运行"
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
echo "使用: $0 {start|stop|status}"
|
|
||||||
echo "或者: $0 TO 进入控制界面"
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
EOF
|
|
||||||
|
|
||||||
chmod +x "$SCRIPT_PATH"
|
|
||||||
echo -e "${GREEN}✓ 脚本已安装到: $SCRIPT_PATH${NC}"
|
|
||||||
echo -e "${YELLOW}使用方法:${NC}"
|
|
||||||
echo -e " ${CYAN}command_monitor.sh start${NC} # 启动监控"
|
|
||||||
echo -e " ${CYAN}command_monitor.sh stop${NC} # 停止监控"
|
|
||||||
echo -e " ${CYAN}command_monitor.sh status${NC} # 查看状态"
|
|
||||||
echo -e " ${CYAN}command_monitor.sh TO${NC} # 进入控制界面"
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
# 设置开机自启动
|
|
||||||
setup_auto_start() {
|
|
||||||
if [ ! -f "$SCRIPT_PATH" ]; then
|
|
||||||
install_script
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 使用crontab实现自启动
|
|
||||||
(crontab -l 2>/dev/null | grep -v "command_monitor.sh"; echo "@reboot $SCRIPT_PATH start > /dev/null 2>&1") | crontab -
|
|
||||||
|
|
||||||
echo -e "${GREEN}✓ 已设置开机自启动${NC}"
|
|
||||||
echo -e "${YELLOW}重启后会自动启动监控服务${NC}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 取消开机自启动
|
|
||||||
remove_auto_start() {
|
|
||||||
crontab -l 2>/dev/null | grep -v "command_monitor.sh" | crontab -
|
|
||||||
echo -e "${YELLOW}✓ 已取消开机自动启动${NC}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 后台运行监控
|
|
||||||
start_background_monitor() {
|
|
||||||
echo -e "${GREEN}启动后台监控服务...${NC}"
|
|
||||||
|
|
||||||
# 如果已安装系统脚本,使用系统脚本
|
|
||||||
if [ -f "$SCRIPT_PATH" ]; then
|
|
||||||
$SCRIPT_PATH start
|
|
||||||
return $?
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 否则使用内置功能
|
|
||||||
if [ -f "/tmp/command_monitor.pid" ]; then
|
|
||||||
local old_pid=$(cat "/tmp/command_monitor.pid" 2>/dev/null)
|
|
||||||
if ps -p "$old_pid" >/dev/null 2>&1; then
|
|
||||||
echo -e "${YELLOW}监控服务已经在运行 (PID: $old_pid)${NC}"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
(
|
|
||||||
DAEMON_MODE=true
|
|
||||||
init_log_system
|
|
||||||
configure_realtime_history
|
|
||||||
|
|
||||||
log_message "INFO" "后台监控进程启动 - PID: $$"
|
|
||||||
log_message "INFO" "客户端IP: $(get_client_ip)"
|
|
||||||
log_message "INFO" "地理位置: $(get_ip_location $(get_client_ip))"
|
|
||||||
|
|
||||||
echo $$ > "/tmp/command_monitor.pid"
|
|
||||||
monitor_history_files
|
|
||||||
) &
|
|
||||||
|
|
||||||
local main_pid=$!
|
|
||||||
sleep 2
|
|
||||||
|
|
||||||
if ps -p $main_pid >/dev/null 2>&1; then
|
|
||||||
echo -e "${GREEN}✓ 后台监控已启动!${NC}"
|
|
||||||
echo -e "主进程PID: $main_pid"
|
|
||||||
echo -e "日志文件: $LATEST_LOG"
|
|
||||||
echo -e "查看日志: ${GREEN}tail -f $LATEST_LOG${NC}"
|
|
||||||
return 0
|
|
||||||
else
|
|
||||||
echo -e "${RED}✗ 后台监控启动失败${NC}"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 前台运行监控
|
|
||||||
start_foreground_monitor() {
|
start_foreground_monitor() {
|
||||||
echo -e "${YELLOW}前台运行模式 (输入 TO 切换后台,Ctrl+C 停止)${NC}"
|
echo -e "${YELLOW}=== 前台监控模式 ===${NC}"
|
||||||
|
echo -e "输入 ${GREEN}to${NC} 切换到后台模式"
|
||||||
|
echo -e "按 ${RED}Ctrl+C${NC} 停止监控"
|
||||||
|
echo -e "${YELLOW}===================${NC}"
|
||||||
|
|
||||||
DAEMON_MODE=false
|
# 前台也记录日志(与后台独立的日志文件)
|
||||||
init_log_system
|
local fg_log="$LOG_DIR/foreground_$(date +%Y%m%d_%H%M%S).log"
|
||||||
configure_realtime_history
|
mkdir -p "$LOG_DIR"
|
||||||
|
echo "=== 前台监控启动于: $(date) ===" > "$fg_log"
|
||||||
|
|
||||||
local client_ip=$(get_client_ip)
|
declare -A fg_last_sizes
|
||||||
local location_info=$(get_ip_location "$client_ip")
|
declare -A fg_last_commands
|
||||||
|
|
||||||
log_message "INFO" "前台监控启动 - PID: $$"
|
|
||||||
log_message "INFO" "客户端IP: $client_ip"
|
|
||||||
log_message "INFO" "地理位置: $location_info"
|
|
||||||
|
|
||||||
echo -e "${GREEN}开始监控...${NC}"
|
|
||||||
echo -e "客户端IP: $client_ip"
|
|
||||||
echo -e "地理位置: $location_info"
|
|
||||||
echo -e "输入 ${CYAN}TO${NC} 切换到后台模式"
|
|
||||||
|
|
||||||
# 监控命令历史文件变化
|
|
||||||
declare -A file_sizes
|
|
||||||
declare -A last_commands
|
|
||||||
|
|
||||||
# 初始化文件大小
|
|
||||||
for user_dir in /home/* /root; do
|
|
||||||
if [ -d "$user_dir" ]; then
|
|
||||||
local user=$(basename "$user_dir")
|
|
||||||
local history_file="$user_dir/.bash_history"
|
|
||||||
if [ -f "$history_file" ]; then
|
|
||||||
file_sizes["$user"]=$(stat -c%s "$history_file" 2>/dev/null || echo 0)
|
|
||||||
last_commands["$user"]=""
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
# 监控循环
|
|
||||||
while true; do
|
while true; do
|
||||||
# 检查用户输入
|
# 检查是否输入to
|
||||||
if read -t 0.1 -n 2 user_input 2>/dev/null; then
|
if read -t 0.1 -n 2 user_input 2>/dev/null; then
|
||||||
if [ "$user_input" = "TO" ] || [ "$user_input" = "to" ]; then
|
if [ "$user_input" = "to" ] || [ "$user_input" = "TO" ]; then
|
||||||
echo -e "\n${GREEN}切换到后台模式...${NC}"
|
echo -e "\n${GREEN}切换到后台模式...${NC}"
|
||||||
start_background_monitor
|
start_background_monitor
|
||||||
exit 0
|
exit 0
|
||||||
@@ -465,157 +159,193 @@ start_foreground_monitor() {
|
|||||||
|
|
||||||
# 监控命令历史
|
# 监控命令历史
|
||||||
for user_dir in /home/* /root; do
|
for user_dir in /home/* /root; do
|
||||||
if [ -d "$user_dir" ]; then
|
[ -d "$user_dir" ] || continue
|
||||||
local user=$(basename "$user_dir")
|
local user=$(basename "$user_dir")
|
||||||
local history_file="$user_dir/.bash_history"
|
local history_file="$user_dir/.bash_history"
|
||||||
|
[ -f "$history_file" ] || continue
|
||||||
|
|
||||||
if [ -f "$history_file" ]; then
|
|
||||||
local current_size=$(stat -c%s "$history_file" 2>/dev/null || echo 0)
|
local current_size=$(stat -c%s "$history_file" 2>/dev/null || echo 0)
|
||||||
local last_size=${file_sizes["$user"]}
|
local last_size=${fg_last_sizes["$user"]:-0}
|
||||||
|
|
||||||
if [ "$current_size" -gt "$last_size" ]; then
|
if [ "$current_size" -gt "$last_size" ]; then
|
||||||
local new_content=$(tail -n 1 "$history_file" 2>/dev/null | tr -d '\0')
|
local new_command=$(tail -n 1 "$history_file" 2>/dev/null | sed 's/^[ \t]*//;s/[ \t]*$//')
|
||||||
|
|
||||||
if [ -n "$new_content" ]; then
|
if [ -n "$new_command" ] &&
|
||||||
local line=$(echo "$new_content" | sed 's/^[ \t]*//;s/[ \t]*$//')
|
[ ${#new_command} -gt 1 ] &&
|
||||||
|
[[ ! "$new_command" =~ ^(ls|cd|pwd|ll|la|history|exit|clear|to|TO)$ ]] &&
|
||||||
# 过滤无用命令
|
[ "$new_command" != "${fg_last_commands["$user"]}" ]; then
|
||||||
if [ -n "$line" ] && [ "${#line}" -gt 1 ] && \
|
|
||||||
[[ ! "$line" =~ ^(ls|cd|pwd|ll|la|history|exit|clear|echo|date|whoami|TO|to)$ ]] && \
|
|
||||||
[[ ! "$line" =~ ^#[0-9]+$ ]] && \
|
|
||||||
[ "$line" != "${last_commands["$user"]}" ]; then
|
|
||||||
|
|
||||||
local client_ip=$(get_client_ip)
|
local client_ip=$(get_client_ip)
|
||||||
local location_info=$(get_ip_location "$client_ip")
|
local timestamp=$(date '+%Y-%m-%d %H:%M:%S')
|
||||||
|
local log_entry="[FG][$timestamp] 用户:$user | 命令:$new_command | 来源:$client_ip"
|
||||||
|
|
||||||
log_message "COMMAND" "用户: $user | 命令: $line | 来源: $client_ip [$location_info]"
|
# 前台显示并记录
|
||||||
last_commands["$user"]="$line"
|
echo -e "${BLUE}$log_entry${NC}"
|
||||||
fi
|
echo "$log_entry" >> "$fg_log"
|
||||||
|
fg_last_commands["$user"]="$new_command"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
file_sizes["$user"]=$current_size
|
fg_last_sizes["$user"]=$current_size
|
||||||
fi
|
|
||||||
fi
|
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
sleep 1
|
sleep 1
|
||||||
done
|
done
|
||||||
}
|
}
|
||||||
|
|
||||||
# 查看监控状态
|
# 安装脚本
|
||||||
check_monitor_status() {
|
install_script() {
|
||||||
# 优先使用系统脚本检查
|
create_install_dir
|
||||||
if [ -f "$SCRIPT_PATH" ]; then
|
|
||||||
$SCRIPT_PATH status
|
|
||||||
return $?
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -f "/tmp/command_monitor.pid" ]; then
|
echo -e "${GREEN}正在安装脚本到: $SCRIPT_PATH${NC}"
|
||||||
local main_pid=$(cat "/tmp/command_monitor.pid")
|
|
||||||
if ps -p "$main_pid" >/dev/null 2>&1; then
|
|
||||||
echo -e "${GREEN}✓ 监控服务运行中${NC}"
|
|
||||||
echo "主进程PID: $main_pid"
|
|
||||||
[ -f "$LATEST_LOG" ] && echo "最新日志:" && tail -1 "$LATEST_LOG" 2>/dev/null || true
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo -e "${RED}✗ 监控服务未运行${NC}"
|
# 创建安装脚本
|
||||||
return 1
|
cat > "$SCRIPT_PATH" << 'EOF'
|
||||||
}
|
#!/bin/bash
|
||||||
|
# 安装版命令监控脚本
|
||||||
|
|
||||||
# 停止监控进程
|
INSTALL_DIR="/root/安装"
|
||||||
stop_monitor() {
|
SCRIPT_PATH="$INSTALL_DIR/cmd_monitor.sh"
|
||||||
# 优先使用系统脚本停止
|
LOG_DIR="/root/command_logs"
|
||||||
if [ -f "$SCRIPT_PATH" ]; then
|
PID_FILE="/tmp/cmd_monitor.pid"
|
||||||
$SCRIPT_PATH stop
|
|
||||||
return $?
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -f "/tmp/command_monitor.pid" ]; then
|
# 检查to命令
|
||||||
local main_pid=$(cat "/tmp/command_monitor.pid")
|
if [ "$1" = "to" ] || [ "$1" = "TO" ]; then
|
||||||
echo -e "${YELLOW}正在停止监控进程 (PID: $main_pid)...${NC}"
|
if [ -f "$PID_FILE" ]; then
|
||||||
kill "$main_pid" 2>/dev/null || true
|
echo "切换到前台模式..."
|
||||||
sleep 1
|
kill $(cat "$PID_FILE") 2>/dev/null
|
||||||
rm -f "/tmp/command_monitor.pid"
|
rm -f "$PID_FILE"
|
||||||
echo -e "${GREEN}✓ 监控进程已停止${NC}"
|
exec $SCRIPT_PATH foreground
|
||||||
else
|
else
|
||||||
pkill -f "monitor_history_files" 2>/dev/null && echo -e "${GREEN}✓ 监控进程已停止${NC}" || echo -e "${YELLOW}没有找到运行的监控进程${NC}"
|
echo "切换到后台模式..."
|
||||||
|
exec $SCRIPT_PATH background
|
||||||
fi
|
fi
|
||||||
}
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
# 显示交互式菜单
|
case "${1:-}" in
|
||||||
show_interactive_menu() {
|
background|start)
|
||||||
clear
|
# 后台启动
|
||||||
echo -e "${GREEN}========================================${NC}"
|
(
|
||||||
echo -e "${GREEN} 实时命令监控系统 v4.4${NC}"
|
mkdir -p "$LOG_DIR"
|
||||||
echo -e "${GREEN}========================================${NC}"
|
while true; do
|
||||||
echo
|
for user_dir in /home/* /root; do
|
||||||
echo -e "${YELLOW}请选择操作:${NC}"
|
[ -d "$user_dir" ] || continue
|
||||||
echo
|
user=$(basename "$user_dir")
|
||||||
echo -e " ${CYAN}1${NC}. 启动后台监控模式"
|
history_file="$user_dir/.bash_history"
|
||||||
echo -e " ${CYAN}2${NC}. 启动前台监控模式"
|
[ -f "$history_file" ] || continue
|
||||||
echo -e " ${CYAN}3${NC}. 查看监控状态"
|
|
||||||
echo -e " ${CYAN}4${NC}. 停止监控服务"
|
|
||||||
echo -e " ${CYAN}5${NC}. 设置开机自启动"
|
|
||||||
echo -e " ${CYAN}6${NC}. 取消开机自启动"
|
|
||||||
echo -e " ${CYAN}7${NC}. 安装脚本到系统"
|
|
||||||
echo -e " ${CYAN}8${NC}. 清理所有残留"
|
|
||||||
echo -e " ${CYAN}0${NC}. 退出"
|
|
||||||
echo
|
|
||||||
echo -e "${GREEN}========================================${NC}"
|
|
||||||
echo -e "${YELLOW}提示:${NC}"
|
|
||||||
echo -e " - 前台模式下输入 ${CYAN}TO${NC} 可切换到后台"
|
|
||||||
echo -e " - 任何时候输入 ${CYAN}command_monitor.sh TO${NC} 进入控制界面"
|
|
||||||
echo -e "${GREEN}========================================${NC}"
|
|
||||||
echo
|
|
||||||
}
|
|
||||||
|
|
||||||
# 处理用户选择
|
current_size=$(stat -c%s "$history_file" 2>/dev/null || echo 0)
|
||||||
handle_user_choice() {
|
last_size=0
|
||||||
local choice
|
[ -f "/tmp/last_size_$user" ] && last_size=$(cat "/tmp/last_size_$user")
|
||||||
read -p "请输入选择 [0-8]: " choice
|
|
||||||
|
|
||||||
case $choice in
|
if [ "$current_size" -gt "$last_size" ]; then
|
||||||
1) start_background_monitor ;;
|
new_cmd=$(tail -n 1 "$history_file" 2>/dev/null | sed 's/^[ \t]*//;s/[ \t]*$//')
|
||||||
2) start_foreground_monitor ;;
|
if [ -n "$new_cmd" ] && [ ${#new_cmd} -gt 1 ]; then
|
||||||
3) check_monitor_status ;;
|
client_ip="unknown"
|
||||||
4) stop_monitor ;;
|
[ -n "$SSH_CLIENT" ] && client_ip=$(echo "$SSH_CLIENT" | awk '{print $1}')
|
||||||
5) setup_auto_start ;;
|
echo "[$(date '+%Y-%m-%d %H:%M:%S')] 用户:$user | 命令:$new_cmd | 来源:$client_ip" >> "$LOG_DIR/auto_monitor.log"
|
||||||
6) remove_auto_start ;;
|
fi
|
||||||
7) install_script ;;
|
echo "$current_size" > "/tmp/last_size_$user"
|
||||||
8) cleanup_all ;;
|
fi
|
||||||
0) echo -e "${GREEN}再见!${NC}"; exit 0 ;;
|
done
|
||||||
*) echo -e "${RED}无效选择,请重新输入${NC}" ;;
|
sleep 1
|
||||||
esac
|
done
|
||||||
|
) &
|
||||||
|
echo $! > "$PID_FILE"
|
||||||
|
echo "后台监控已启动 (PID: $!)"
|
||||||
|
;;
|
||||||
|
foreground)
|
||||||
|
# 前台模式
|
||||||
|
echo "前台监控模式 - 输入 'to' 切换到后台"
|
||||||
|
while true; do
|
||||||
|
for user_dir in /home/* /root; do
|
||||||
|
[ -d "$user_dir" ] || continue
|
||||||
|
user=$(basename "$user_dir")
|
||||||
|
history_file="$user_dir/.bash_history"
|
||||||
|
[ -f "$history_file" ] || continue
|
||||||
|
|
||||||
echo
|
current_size=$(stat -c%s "$history_file" 2>/dev/null || echo 0)
|
||||||
read -p "按回车键返回菜单..."
|
last_size=0
|
||||||
|
[ -f "/tmp/last_size_$user" ] && last_size=$(cat "/tmp/last_size_$user")
|
||||||
|
|
||||||
|
if [ "$current_size" -gt "$last_size" ]; then
|
||||||
|
new_cmd=$(tail -n 1 "$history_file" 2>/dev/null | sed 's/^[ \t]*//;s/[ \t]*$//')
|
||||||
|
if [ -n "$new_cmd" ] && [ ${#new_cmd} -gt 1 ]; then
|
||||||
|
client_ip="unknown"
|
||||||
|
[ -n "$SSH_CLIENT" ] && client_ip=$(echo "$SSH_CLIENT" | awk '{print $1}')
|
||||||
|
echo "[$(date '+%Y-%m-%d %H:%M:%S')] 用户:$user | 命令:$new_cmd | 来源:$client_ip"
|
||||||
|
fi
|
||||||
|
echo "$current_size" > "/tmp/last_size_$user"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# 检查to命令输入
|
||||||
|
if read -t 0.1 -n 2 input 2>/dev/null; then
|
||||||
|
if [ "$input" = "to" ] || [ "$input" = "TO" ]; then
|
||||||
|
echo "切换到后台模式..."
|
||||||
|
$SCRIPT_PATH background
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
sleep 1
|
||||||
|
done
|
||||||
|
;;
|
||||||
|
stop)
|
||||||
|
[ -f "$PID_FILE" ] && kill $(cat "$PID_FILE") 2>/dev/null && rm -f "$PID_FILE"
|
||||||
|
echo "监控已停止"
|
||||||
|
;;
|
||||||
|
install)
|
||||||
|
# 设置开机自启动
|
||||||
|
(crontab -l 2>/dev/null | grep -v "$SCRIPT_PATH"; echo "@reboot $SCRIPT_PATH background >/dev/null 2>&1") | crontab -
|
||||||
|
|
||||||
|
# 设置实时history
|
||||||
|
for user_dir in /home/* /root; do
|
||||||
|
[ -d "$user_dir" ] || continue
|
||||||
|
bashrc="$user_dir/.bashrc"
|
||||||
|
[ -f "$bashrc" ] || continue
|
||||||
|
grep -q "PROMPT_COMMAND" "$bashrc" || echo "export PROMPT_COMMAND='history -a; history -c; history -r'" >> "$bashrc"
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "安装完成!已设置开机自启动"
|
||||||
|
echo "使用方法:"
|
||||||
|
echo " to # 切换前后台模式"
|
||||||
|
echo " $SCRIPT_PATH foreground # 前台模式"
|
||||||
|
echo " $SCRIPT_PATH background # 后台模式"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "使用方法: $0 {foreground|background|stop|install|to}"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
EOF
|
||||||
|
|
||||||
|
chmod +x "$SCRIPT_PATH"
|
||||||
|
echo -e "${GREEN}✓ 脚本已安装到: $SCRIPT_PATH${NC}"
|
||||||
|
|
||||||
|
# 执行安装配置
|
||||||
|
$SCRIPT_PATH install
|
||||||
}
|
}
|
||||||
|
|
||||||
### 主程序 ###
|
### 主程序 ###
|
||||||
main() {
|
case "${1:-}" in
|
||||||
local command="${1:-}"
|
"install")
|
||||||
|
install_script
|
||||||
# 检查TO命令
|
;;
|
||||||
check_to_command "$command"
|
"background")
|
||||||
|
start_background_monitor
|
||||||
case "$command" in
|
;;
|
||||||
-d|--daemon) start_background_monitor ;;
|
"foreground")
|
||||||
-s|--status) check_monitor_status ;;
|
start_foreground_monitor
|
||||||
-k|--kill) stop_monitor ;;
|
;;
|
||||||
-c|--clean) cleanup_all ;;
|
"stop")
|
||||||
-h|--help|"") show_interactive_menu; handle_user_choice ;;
|
stop_background_monitor
|
||||||
*) echo -e "${RED}未知选项: $command${NC}"; exit 1 ;;
|
;;
|
||||||
esac
|
*)
|
||||||
}
|
echo -e "${YELLOW}使用方法:${NC}"
|
||||||
|
echo -e " $0 install # 安装脚本并设置自启动"
|
||||||
# 启动
|
echo -e " $0 foreground # 启动前台监控"
|
||||||
if [ $# -eq 0 ]; then
|
echo -e " $0 background # 启动后台监控"
|
||||||
while true; do
|
echo -e " $0 stop # 停止监控"
|
||||||
show_interactive_menu
|
echo -e ""
|
||||||
handle_user_choice
|
echo -e "${GREEN}安装后直接输入 'to' 切换前后台模式${NC}"
|
||||||
done
|
;;
|
||||||
else
|
esac
|
||||||
main "$@"
|
|
||||||
fi
|
|
||||||
|
|||||||
Reference in New Issue
Block a user