From d7a7821cca4fa847a9a7e73965b585d4800841ca Mon Sep 17 00:00:00 2001 From: Beginner <70857188+Beginner-Go@users.noreply.github.com> Date: Mon, 6 Sep 2021 22:17:00 +0800 Subject: [PATCH] luci-app-serverchan: add package (#7809) Co-authored-by: tty228 --- package/lean/luci-app-serverchan/Makefile | 23 + package/lean/luci-app-serverchan/README.md | 50 + .../luasrc/controller/serverchan.lua | 33 + .../luasrc/model/cbi/serverchan/advanced.lua | 111 ++ .../luasrc/model/cbi/serverchan/client.lua | 6 + .../luasrc/model/cbi/serverchan/log.lua | 5 + .../luasrc/model/cbi/serverchan/setting.lua | 503 +++++++ .../luasrc/view/serverchan/log.htm | 33 + .../view/serverchan/serverchan_status.htm | 22 + .../root/etc/config/serverchan | 11 + .../root/etc/init.d/serverchan | 26 + .../root/etc/uci-defaults/luci-serverchan | 11 + .../root/usr/bin/serverchan/api/diy.json | 40 + .../root/usr/bin/serverchan/api/ip_blacklist | 1 + .../root/usr/bin/serverchan/api/ipv4.list | 6 + .../root/usr/bin/serverchan/api/ipv6.list | 5 + .../root/usr/bin/serverchan/api/logo.jpg | Bin 0 -> 33685 bytes .../root/usr/bin/serverchan/api/pushplus.json | 21 + .../usr/bin/serverchan/api/qywx_markdown.json | 23 + .../usr/bin/serverchan/api/qywx_mpnews.json | 33 + .../usr/bin/serverchan/api/serverchan.json | 17 + .../root/usr/bin/serverchan/api/telegram.json | 20 + .../root/usr/bin/serverchan/api/wxpusher.json | 23 + .../root/usr/bin/serverchan/serverchan | 1246 +++++++++++++++++ .../share/rpcd/acl.d/luci-app-serverchan.json | 11 + 25 files changed, 2280 insertions(+) create mode 100644 package/lean/luci-app-serverchan/Makefile create mode 100644 package/lean/luci-app-serverchan/README.md create mode 100644 package/lean/luci-app-serverchan/luasrc/controller/serverchan.lua create mode 100644 package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/advanced.lua create mode 100644 package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/client.lua create mode 100644 package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/log.lua create mode 100644 package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/setting.lua create mode 100644 package/lean/luci-app-serverchan/luasrc/view/serverchan/log.htm create mode 100644 package/lean/luci-app-serverchan/luasrc/view/serverchan/serverchan_status.htm create mode 100644 package/lean/luci-app-serverchan/root/etc/config/serverchan create mode 100755 package/lean/luci-app-serverchan/root/etc/init.d/serverchan create mode 100755 package/lean/luci-app-serverchan/root/etc/uci-defaults/luci-serverchan create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/diy.json create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ip_blacklist create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv4.list create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv6.list create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/logo.jpg create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/pushplus.json create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/qywx_markdown.json create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/qywx_mpnews.json create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/serverchan.json create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/telegram.json create mode 100644 package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/wxpusher.json create mode 100755 package/lean/luci-app-serverchan/root/usr/bin/serverchan/serverchan create mode 100644 package/lean/luci-app-serverchan/root/usr/share/rpcd/acl.d/luci-app-serverchan.json diff --git a/package/lean/luci-app-serverchan/Makefile b/package/lean/luci-app-serverchan/Makefile new file mode 100644 index 000000000..861d55413 --- /dev/null +++ b/package/lean/luci-app-serverchan/Makefile @@ -0,0 +1,23 @@ +include $(TOPDIR)/rules.mk + +PKG_NAME:=luci-app-serverchan +PKG_VERSION:=2.01.3 +PKG_RELEASE:=9 + +PKG_MAINTAINER:=tty228 + +LUCI_TITLE:=LuCI support for serverchan +LUCI_PKGARCH:=all +LUCI_DEPENDS:=+iputils-arping +curl +jq + +define Package/$(PKG_NAME)/conffiles +/etc/config/serverchan +/usr/bin/serverchan/api/diy.json +/usr/bin/serverchan/api/logo.jpg +/usr/bin/serverchan/api/ipv4.list +/usr/bin/serverchan/api/ipv6.list +endef + +include $(TOPDIR)/feeds/luci/luci.mk + +# call BuildPackage - OpenWrt buildroot signature diff --git a/package/lean/luci-app-serverchan/README.md b/package/lean/luci-app-serverchan/README.md new file mode 100644 index 000000000..4d60e2d0d --- /dev/null +++ b/package/lean/luci-app-serverchan/README.md @@ -0,0 +1,50 @@ +# 简介 +- 用于 OpenWRT/LEDE 路由器上进行 微信/Telegram 推送的插件 +- 支持列表: +- 微信推送/Server酱 https://sct.ftqq.com/ +- 企业微信/应用推送 https://work.weixin.qq.com/api/doc/90000/90135/90248 +- 微信推送/WxPusher https://wxpusher.zjiecode.com/docs +- 微信推送/推送加 http://www.pushplus.plus/ +- Telegram/BotFather https://t.me/BotFather +- 精力有限,如需要钉钉推送、飞书推送、Bark推送等请尝试 https://github.com/zzsj0928/luci-app-pushbot +- +- **基于X86 OpenWrt 19.07.8 制作,不同系统不同设备,请自行修改部分代码,无测试条件无法重现的 bug 不考虑修复** +- 依赖 iputils-arping + curl + jq 命令,安装前请 `opkg update`,小内存路由谨慎安装 +- 使用主动探测设备连接的方式检测设备在线状态,以避免WiFi休眠机制,主动探测较为耗时,**如遇设备休眠频繁,请自行调整超时设置** +- 流量统计功能依赖 wrtbwmon ,自行选装或编译,该插件与 Routing/NAT 、Flow Offloading 冲突,开启无法获取流量,自行选择 + +#### 主要功能 +- 路由 ip/ipv6 变动推送 +- 设备别名 +- 设备上线推送 +- 设备离线推送及流量使用情况 +- CPU 负载、温度监视 +- 定时推送设备运行状态 +- MAC 白名单、黑名单、按接口检测设备 +- 免打扰 +- 无人值守任务 + +#### 已知问题 +- 直接关闭接口时,该接口的离线设备会忽略检测 +- 部分设备无法读取到设备名,脚本使用 `cat /var/dhcp.leases` 命令读取设备名,如果 dhcp 中不存在设备名,则无法读取设备名(如二级路由设备、静态ip设备),请使用设备名备注. +- 基于X86 OpenWrt 19.07.8 制作,不同系统不同设备,可能会遇到各种问题 +- 潘多拉系统、或不支持 sh 的系统,请将脚本开头 `#!/bin/sh` 改为 `#!/bin/bash`,或手动安装 `sh` + +# Download +- [luci-app-serverchan](https://github.com/tty228/luci-app-serverchan/releases) +- [wrtbwmon](https://github.com/brvphoenix/wrtbwmon) +- [luci-app-wrtbwmon](https://github.com/brvphoenix/luci-app-wrtbwmon) +- **L大版本直接编译 luci-app-wrtbwmon ,非原版 luci 如使用以上 wrtbwmon,请注意安装版本号** + +#### ps +- 新功能看情况开发,忙得头晕眼花 +- 欢迎各种代码提交 +- 提交bug时请尽量带上设备信息,日志与描述 +(如执行 `/usr/bin/serverchan/serverchan` 后的提示、日志信息、/tmp/serverchan/ipAddress 文件信息、或尝试使用 sh -x /usr/bin/serverchan/serverchan 查看详细运行信息 ) +- 三言两句恕我无能为力 + +# Donate +如果你觉得此项目对你有帮助,请捐助我们,以使项目能持续发展,更加完善。 + +![image](https://github.com/tty228/Python-100-Days/blob/master/res/WX.jpg) + diff --git a/package/lean/luci-app-serverchan/luasrc/controller/serverchan.lua b/package/lean/luci-app-serverchan/luasrc/controller/serverchan.lua new file mode 100644 index 000000000..abf76d01b --- /dev/null +++ b/package/lean/luci-app-serverchan/luasrc/controller/serverchan.lua @@ -0,0 +1,33 @@ +module("luci.controller.serverchan",package.seeall) + +function index() + + if not nixio.fs.access("/etc/config/serverchan")then + return + end + + entry({"admin", "services", "serverchan"}, alias("admin", "services", "serverchan", "setting"),_("微信推送"), 30).dependent = true + entry({"admin", "services", "serverchan", "setting"}, cbi("serverchan/setting"),_("配置"), 40).leaf = true + entry({"admin", "services", "serverchan", "advanced"}, cbi("serverchan/advanced"),_("高级设置"), 50).leaf = true + entry({"admin", "services", "serverchan", "client"}, form("serverchan/client"), "在线设备", 80) + entry({"admin", "services", "serverchan", "log"}, form("serverchan/log"),_("日志"), 99).leaf = true + entry({"admin", "services", "serverchan", "get_log"}, call("get_log")).leaf = true + entry({"admin", "services", "serverchan", "clear_log"}, call("clear_log")).leaf = true + entry({"admin", "services", "serverchan", "status"}, call("act_status")).leaf=true +end + +function act_status() + local e={} + e.running=luci.sys.call("busybox ps|grep -v grep|grep -c serverchan >/dev/null")==0 + luci.http.prepare_content("application/json") + luci.http.write_json(e) +end + +function get_log() + luci.http.write(luci.sys.exec( + "[ -f '/tmp/serverchan/serverchan.log' ] && cat /tmp/serverchan/serverchan.log")) +end + +function clear_log() + luci.sys.call("echo '' > /tmp/serverchan/serverchan.log") +end diff --git a/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/advanced.lua b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/advanced.lua new file mode 100644 index 000000000..c593739de --- /dev/null +++ b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/advanced.lua @@ -0,0 +1,111 @@ +local nt = require "luci.sys".net +local fs=require"nixio.fs" + +m=Map("serverchan",translate("提示:"), +translate("如果你不了解这些选项的含义,请不要修改这些选项")) + +s = m:section(TypedSection, "serverchan", "高级设置") +s.anonymous = true +s.addremove = false + +a=s:option(Value,"up_timeout",translate('设备上线检测超时(s)')) +a.default = "2" +a.optional=false +a.datatype="uinteger" + +a=s:option(Value,"down_timeout",translate('设备离线检测超时(s)')) +a.default = "20" +a.optional=false +a.datatype="uinteger" + +a=s:option(Value,"timeout_retry_count",translate('离线检测次数')) +a.default = "2" +a.optional=false +a.datatype="uinteger" +a.description = translate("若无二级路由设备,信号强度良好,可以减少以上数值
因夜间 wifi 休眠较为玄学,遇到设备频繁推送断开,烦请自行调整参数
..╮(╯_╰)╭..") + +a=s:option(Value,"thread_num",translate('最大并发进程数')) +a.default = "3" +a.datatype="uinteger" + +a=s:option(Value, "soc_code", "自定义温度读取命令") +a.rmempty = true +a:value("",translate("默认")) +a.description = translate("请尽量避免使用特殊符号,如双引号、$、!等,执行结果需为数字,用于温度对比") + +a=s:option(Button,"soc",translate("测试温度命令")) +a.inputtitle = translate("输出信息") +a.write = function() + luci.sys.call("/usr/bin/serverchan/serverchan soc") + luci.http.redirect(luci.dispatcher.build_url("admin","services","serverchan","advanced")) +end + +if nixio.fs.access("/tmp/serverchan/soc_tmp") then +e=s:option(TextValue,"soc_tmp") +e.rows=2 +e.readonly=true +e.cfgvalue = function() + return luci.sys.exec("cat /tmp/serverchan/soc_tmp && rm -f /tmp/serverchan/soc_tmp") +end +end + +a=s:option(Flag,"err_enable",translate("无人值守任务")) +a.default=0 +a.rmempty=true +a.description = translate("请确认脚本可以正常运行,否则可能造成频繁重启等错误!") + +a=s:option(Flag,"err_sheep_enable",translate("仅在免打扰时段重拨")) +a.default=0 +a.rmempty=true +a.description = translate("避免白天重拨 ddns 域名等待解析,此功能不影响断网检测
因夜间跑流量问题,该功能可能不稳定") +a:depends({err_enable="1"}) + +a= s:option(DynamicList, "err_device_aliases", translate("关注列表")) +a.rmempty = true +a.description = translate("只会在列表中设备都不在线时才会执行
免打扰时段一小时后,关注设备五分钟低流量(约100kb/m)将视为离线") +nt.mac_hints(function(mac, name) a :value(mac, "%s (%s)" %{ mac, name }) end) +a:depends({err_enable="1"}) + +a=s:option(ListValue,"network_err_event",translate("网络断开时")) +a.default="" +a:depends({err_enable="1"}) +a:value("",translate("无操作")) +a:value("1",translate("重启路由器")) +a:value("2",translate("重新拨号")) +a:value("3",translate("修改相关设置项,尝试自动修复网络")) +a.description = translate("选项 1 选项 2 不会修改设置,并最多尝试 2 次。
选项 3 会将设置项备份于 /usr/bin/serverchan/configbak 目录,并在失败后还原。
【!!无法保证兼容性!!】不熟悉系统设置项,不会救砖请勿使用") + +a=s:option(ListValue,"system_time_event",translate("定时重启")) +a.default="" +a:depends({err_enable="1"}) +a:value("",translate("无操作")) +a:value("1",translate("重启路由器")) +a:value("2",translate("重新拨号")) + +a= s:option(Value, "autoreboot_time", "系统运行时间大于") +a.rmempty = true +a.default = "24" +a.datatype="uinteger" +a:depends({system_time_event="1"}) +a.description = translate("单位为小时") + +a=s:option(Value, "network_restart_time", "网络在线时间大于") +a.rmempty = true +a.default = "24" +a.datatype="uinteger" +a:depends({system_time_event="2"}) +a.description = translate("单位为小时") + +a=s:option(Flag,"public_ip_event",translate("重拨尝试获取公网 ip")) +a.default=0 +a.rmempty=true +a:depends({err_enable="1"}) +a.description = translate("重拨时不会推送 ip 变动通知,并会导致你的域名无法及时更新 ip 地址
请确认你可以通过重拨获取公网 ip,否则这不仅徒劳无功还会引起频繁断网
移动等大内网你就别挣扎了!!") + +a= s:option(Value, "public_ip_retry_count", "当天最大重试次数") +a.rmempty = true +a.default = "10" +a.datatype="uinteger" +a:depends({public_ip_event="1"}) + +return m diff --git a/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/client.lua b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/client.lua new file mode 100644 index 000000000..9c2dafd4f --- /dev/null +++ b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/client.lua @@ -0,0 +1,6 @@ +f = SimpleForm("serverchan") +luci.sys.call("/usr/bin/serverchan/serverchan client") +f.reset = false +f.submit = false +f:append(Template("serverchan/client")) +return f diff --git a/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/log.lua b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/log.lua new file mode 100644 index 000000000..9063c55c2 --- /dev/null +++ b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/log.lua @@ -0,0 +1,5 @@ +f = SimpleForm("serverchan") +f.reset = false +f.submit = false +f:append(Template("serverchan/log")) +return f diff --git a/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/setting.lua b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/setting.lua new file mode 100644 index 000000000..74ffc7a42 --- /dev/null +++ b/package/lean/luci-app-serverchan/luasrc/model/cbi/serverchan/setting.lua @@ -0,0 +1,503 @@ + +local nt = require "luci.sys".net +local fs=require"nixio.fs" +local e=luci.model.uci.cursor() +local net = require "luci.model.network".init() +local sys = require "luci.sys" +local ifaces = sys.net:devices() + +m=Map("serverchan",translate("ServerChan"), +translate("「Server酱」,英文名「ServerChan」,是一款从服务器推送报警信息和日志到微信的工具。

如果你在使用中遇到问题,请到这里提交:") +.. [[]] +.. translate("github 项目地址") +.. [[]] +) + +m:section(SimpleSection).template = "serverchan/serverchan_status" + +s=m:section(NamedSection,"serverchan","serverchan",translate("")) +s:tab("basic", translate("基本设置")) +s:tab("content", translate("推送内容")) +s:tab("crontab", translate("定时推送")) +s:tab("disturb", translate("免打扰")) +s.addremove = false +s.anonymous = true + +--基本设置 +a=s:taboption("basic", Flag,"serverchan_enable",translate("启用")) +a.rmempty = true + +a = s:taboption("basic", MultiValue, "lite_enable", translate("精简模式")) +a:value("device", translate("精简当前设备列表")) +a:value("nowtime", translate("精简当前时间")) +a:value("content", translate("只推送标题")) +a.widget = "checkbox" +a.default = nil +a.optional = true + +a=s:taboption("basic", ListValue,"jsonpath",translate("推送模式")) +a.default="/usr/bin/serverchan/api/serverchan.json" +a.rmempty = true +a:value("/usr/bin/serverchan/api/serverchan.json",translate("微信 Server酱")) +a:value("/usr/bin/serverchan/api/qywx_mpnews.json",translate("企业微信 图文消息")) +a:value("/usr/bin/serverchan/api/qywx_markdown.json",translate("企业微信 markdown版(不支持公众号)")) +a:value("/usr/bin/serverchan/api/wxpusher.json",translate("微信 wxpusher")) +a:value("/usr/bin/serverchan/api/pushplus.json",translate("微信 pushplus")) +a:value("/usr/bin/serverchan/api/telegram.json",translate("Telegram")) +a:value("/usr/bin/serverchan/api/diy.json",translate("自定义推送")) + +a=s:taboption("basic", Value,"sckey",translate('微信推送/新旧共用'), translate("").."Server酱 sendkey 点击这里
") +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/serverchan.json") + +a=s:taboption("basic", Value,"corpid",translate('企业ID(corpid)'),translate("").."获取说明 点击这里") +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_mpnews.json") +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_markdown.json") +a=s:taboption("basic", Value,"userid",translate('帐号(userid)')) +a.rmempty = true +a.description = translate("群发到应用请填入 @all ") +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_mpnews.json") +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_markdown.json") +a=s:taboption("basic", Value,"agentid",translate('应用id(agentid)')) +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_mpnews.json") +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_markdown.json") +a=s:taboption("basic", Value,"corpsecret",translate('应用密钥(Secret)')) +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_mpnews.json") +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_markdown.json") +a=s:taboption("basic", Value,"mediapath",translate('图片缩略图文件路径')) +a.rmempty = true +a.default = "/usr/bin/serverchan/api/logo.jpg" +a:depends("jsonpath","/usr/bin/serverchan/api/qywx_mpnews.json") +a.description = translate("只支持 2MB 以内 JPG,PNG 格式
900*383 或 2.35:1 为佳 ") + +a=s:taboption("basic",Value,"wxpusher_apptoken",translate('appToken'),translate("").."获取 appToken 点击这里
") +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/wxpusher.json") +a=s:taboption("basic", Value,"wxpusher_uids",translate('uids')) +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/wxpusher.json") +a=s:taboption("basic",Value,"wxpusher_topicIds",translate('topicIds(群发)'),translate("").."接口说明 点击这里
") +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/wxpusher.json") + +a=s:taboption("basic",Value,"pushplus_token",translate('pushplus_token'),translate("").."获取pushplus_token 点击这里
") +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/pushplus.json") + +a=s:taboption("basic", Value, "tg_token", translate("TG_token"),translate("").."获取机器人点击这里
与创建的机器人发一条消息,开启对话
") +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/telegram.json") +a=s:taboption("basic", Value,"chat_id",translate('TG_chatid'),translate("").."获取 chat_id 点击这里") +a.rmempty = true +a:depends("jsonpath","/usr/bin/serverchan/api/telegram.json") + +a=s:taboption("basic", TextValue, "diy_json", translate("自定义推送")) +a.optional = false +a.rows = 28 +a.wrap = "soft" +a.cfgvalue = function(self, section) + return fs.readfile("/usr/bin/serverchan/api/diy.json") +end +a.write = function(self, section, value) + fs.writefile("/usr/bin/serverchan/api/diy.json", value:gsub("\r\n", "\n")) +end +a:depends("jsonpath","/usr/bin/serverchan/api/diy.json") + +a=s:taboption("basic", Button,"__add",translate("发送测试")) +a.inputtitle=translate("发送") +a.inputstyle = "apply" +function a.write(self, section) + luci.sys.call("cbi.apply") + luci.sys.call("/usr/bin/serverchan/serverchan test &") +end + +a=s:taboption("basic", Value,"device_name",translate('本设备名称')) +a.rmempty = true +a.description = translate("在推送信息标题中会标识本设备名称,用于区分推送信息的来源设备") + +a=s:taboption("basic", Value,"sleeptime",translate('检测时间间隔(s)')) +a.rmempty = true +a.optional = false +a.default = "60" +a.datatype="and(uinteger,min(10))" +a.description = translate("越短的时间时间响应越及时,但会占用更多的系统资源") + +a=s:taboption("basic", ListValue,"oui_data",translate("MAC设备信息数据库")) +a.rmempty = true +a.default="" +a:value("",translate("关闭")) +a:value("1",translate("简化版")) +a:value("2",translate("完整版")) +a:value("3",translate("网络查询")) +a.description = translate("需下载 4.36m 原始数据,处理后完整版约 1.2M,简化版约 250kb
若无梯子,请勿使用网络查询") + +a=s:taboption("basic", Flag,"oui_dir",translate("下载到内存")) +a.rmempty = true +a:depends("oui_data","1") +a:depends("oui_data","2") +a.description = translate("懒得做自动更新了,下载到内存中,重启会重新下载
若无梯子,还是下到机身吧") + +a=s:taboption("basic", Flag,"reset_regularly",translate("每天零点重置流量数据")) +a.rmempty = true + +a=s:taboption("basic", Flag,"debuglevel",translate("开启日志")) +a.rmempty = true + +a= s:taboption("basic", DynamicList, "device_aliases", translate("设备别名")) +a.rmempty = true +a.description = translate("
请输入设备 MAC 和设备别名,用“-”隔开,如:
XX:XX:XX:XX:XX:XX-我的手机") + +--设备状态 +a=s:taboption("content", ListValue,"serverchan_ipv4",translate("ipv4 变动通知")) +a.rmempty = true +a.default="" +a:value("",translate("关闭")) +a:value("1",translate("通过接口获取")) +a:value("2",translate("通过URL获取")) + +a = s:taboption("content", ListValue, "ipv4_interface", translate("接口名称")) +a.rmempty = true +a:depends({serverchan_ipv4="1"}) +for _, iface in ipairs(ifaces) do + if not (iface == "lo" or iface:match("^ifb.*")) then + local nets = net:get_interface(iface) + nets = nets and nets:get_networks() or {} + for k, v in pairs(nets) do + nets[k] = nets[k].sid + end + nets = table.concat(nets, ",") + a:value(iface, ((#nets > 0) and "%s (%s)" % {iface, nets} or iface)) + end +end +a.description = translate("
一般选择 wan 接口,多拨环境请自行选择") + +a=s:taboption("content", TextValue, "ipv4_list", translate("ipv4 api列表")) +a.optional = false +a.rows = 8 +a.wrap = "soft" +a.cfgvalue = function(self, section) + return fs.readfile("/usr/bin/serverchan/api/ipv4.list") +end +a.write = function(self, section, value) + fs.writefile("/usr/bin/serverchan/api/ipv4.list", value:gsub("\r\n", "\n")) +end +a.description = translate("
会因服务器稳定性、连接频繁等原因导致获取失败
如接口可以正常获取 IP,不推荐使用
从以上列表中随机地址访问") +a:depends({serverchan_ipv4="2"}) + +a=s:taboption("content", ListValue,"serverchan_ipv6",translate("ipv6 变动通知")) +a.rmempty = true +a.default="disable" +a:value("0",translate("关闭")) +a:value("1",translate("通过接口获取")) +a:value("2",translate("通过URL获取")) + +a = s:taboption("content", ListValue, "ipv6_interface", translate("接口名称")) +a.rmempty = true +a:depends({serverchan_ipv6="1"}) +for _, iface in ipairs(ifaces) do + if not (iface == "lo" or iface:match("^ifb.*")) then + local nets = net:get_interface(iface) + nets = nets and nets:get_networks() or {} + for k, v in pairs(nets) do + nets[k] = nets[k].sid + end + nets = table.concat(nets, ",") + a:value(iface, ((#nets > 0) and "%s (%s)" % {iface, nets} or iface)) + end +end +a.description = translate("
一般选择 wan 接口,多拨环境请自行选择") + +a=s:taboption("content", TextValue, "ipv6_list", translate("ipv6 api列表")) +a.optional = false +a.rows = 8 +a.wrap = "soft" +a.cfgvalue = function(self, section) + return fs.readfile("/usr/bin/serverchan/api/ipv6.list") +end +a.write = function(self, section, value) + fs.writefile("/usr/bin/serverchan/api/ipv6.list", value:gsub("\r\n", "\n")) +end +a.description = translate("
会因服务器稳定性、连接频繁等原因导致获取失败
如接口可以正常获取 IP,不推荐使用
从以上列表中随机地址访问") +a:depends({serverchan_ipv6="2"}) + + +a=s:taboption("content", Flag,"serverchan_up",translate("设备上线通知")) +a.default=1 +a.rmempty = true + +a=s:taboption("content", Flag,"serverchan_down",translate("设备下线通知")) +a.default=1 +a.rmempty = true + +a=s:taboption("content", Flag,"cpuload_enable",translate("CPU 负载报警")) +a.default=1 +a.rmempty = true + +a= s:taboption("content", Value, "cpuload", "负载报警阈值") +a.default = 2 +a.rmempty = true +a:depends({cpuload_enable="1"}) + +a=s:taboption("content", Flag,"temperature_enable",translate("CPU 温度报警")) +a.default=1 +a.rmempty = true +a.description = translate("请确认设备可以获取温度,如需修改命令,请移步高级设置") + +a= s:taboption("content", Value, "temperature", "温度报警阈值") +a.rmempty = true +a.default = "80" +a.datatype="and(uinteger,min(1))" +a:depends({temperature_enable="1"}) +a.description = translate("
设备报警只会在连续五分钟超过设定值时才会推送
而且一个小时内不会再提醒第二次") + +a=s:taboption("content", Flag,"client_usage",translate("设备异常流量")) +a.default=0 +a.rmempty = true + +a= s:taboption("content", Value, "client_usage_max", "每分钟流量限制") +a.default = "10M" +a.rmempty = true +a:depends({client_usage="1"}) +a.description = translate("设备异常流量警报(byte),你可以追加 K 或者 M") + +a=s:taboption("content", Flag,"client_usage_disturb",translate("异常流量免打扰")) +a.default=1 +a.rmempty = true +a:depends({client_usage="1"}) + +a = s:taboption("content", DynamicList, "client_usage_whitelist", translate("异常流量关注列表")) +nt.mac_hints(function(mac, name) a:value(mac, "%s (%s)" %{ mac, name }) end) +a.rmempty = true +a:depends({client_usage_disturb="1"}) +a.description = translate("请输入设备 MAC") + +a=s:taboption("content", Flag,"web_logged",translate("web 登录提醒")) +a.default=0 +a.rmempty = true + +a=s:taboption("content", Flag,"ssh_logged",translate("ssh 登录提醒")) +a.default=0 +a.rmempty = true + +a=s:taboption("content", Flag,"web_login_failed",translate("web 错误尝试提醒")) +a.default=0 +a.rmempty = true + +a=s:taboption("content", Flag,"ssh_login_failed",translate("ssh 错误尝试提醒")) +a.default=0 +a.rmempty = true + +a= s:taboption("content", Value, "login_max_num", "错误尝试次数") +a.default = "3" +a.datatype="and(uinteger,min(1))" +a:depends("web_login_failed","1") +a:depends("ssh_login_failed","1") +a.description = translate("超过次数后推送提醒") + +a=s:taboption("content", Flag,"web_login_black",translate("自动拉黑")) +a.default=0 +a.rmempty = true +a:depends("web_login_failed","1") +a:depends("ssh_login_failed","1") +a.description = translate("直到重启前都不会重置次数,请先添加白名单") + +a= s:taboption("content", Value, "ip_black_timeout", "拉黑时间(秒)") +a.default = "86400" +a.datatype="and(uinteger,min(0))" +a:depends("web_login_black","1") +a.description = translate("0 为永久拉黑,慎用
如不幸误操作,请更改设备 IP 进入 LUCI 界面清空规则") + +a=s:taboption("content", DynamicList, "ip_white_list", translate("白名单 IP 列表")) +a.datatype = "ipaddr" +a.rmempty = true +luci.ip.neighbors({family = 4}, function(entry) + if entry.reachable then + a:value(entry.dest:string()) + end +end) +a:depends("web_logged","1") +a:depends("ssh_logged","1") +a:depends("web_login_failed","1") +a:depends("ssh_login_failed","1") +a.description = translate("忽略白名单登陆提醒和拉黑操作,暂不支持掩码位表示") + +a=s:taboption("content", TextValue, "ip_black_list", translate("IP 黑名单列表")) +a.optional = false +a.rows = 8 +a.wrap = "soft" +a.cfgvalue = function(self, section) + return fs.readfile("/usr/bin/serverchan/api/ip_blacklist") +end +a.write = function(self, section, value) + fs.writefile("/usr/bin/serverchan/api/ip_blacklist", value:gsub("\r\n", "\n")) +end +a:depends("web_login_black","1") + +--定时推送 +a=s:taboption("crontab", ListValue,"crontab",translate("定时任务设定")) +a.rmempty = true +a.default="" +a:value("",translate("关闭")) +a:value("1",translate("定时发送")) +a:value("2",translate("间隔发送")) + +a=s:taboption("crontab", ListValue,"regular_time",translate("发送时间")) +a.rmempty = true +for t=0,23 do +a:value(t,translate("每天"..t.."点")) +end +a.default=8 +a.datatype=uinteger +a:depends("crontab","1") + +a=s:taboption("crontab", ListValue,"regular_time_2",translate("发送时间")) +a.rmempty = true +a:value("",translate("关闭")) +for t=0,23 do +a:value(t,translate("每天"..t.."点")) +end +a.default="关闭" +a.datatype=uinteger +a:depends("crontab","1") + +a=s:taboption("crontab", ListValue,"regular_time_3",translate("发送时间")) +a.rmempty = true + +a:value("",translate("关闭")) +for t=0,23 do +a:value(t,translate("每天"..t.."点")) +end +a.default="关闭" +a.datatype=uinteger +a:depends("crontab","1") + +a=s:taboption("crontab", ListValue,"interval_time",translate("发送间隔")) +a.rmempty = true +for t=1,23 do +a:value(t,translate(t.."小时")) +end +a.default=6 +a.datatype=uinteger +a:depends("crontab","2") +a.description = translate("
从 00:00 开始,每 * 小时发送一次") + +a= s:taboption("crontab", Value, "send_title", translate("微信推送标题")) +a:depends("crontab","1") +a:depends("crontab","2") +a.placeholder = "OpenWrt By tty228 路由状态:" +a.description = translate("
使用特殊符号可能会造成发送失败") + +a=s:taboption("crontab", Flag,"router_status",translate("系统运行情况")) +a.default=1 +a:depends("crontab","1") +a:depends("crontab","2") + +a=s:taboption("crontab", Flag,"router_temp",translate("设备温度")) +a.default=1 +a:depends("crontab","1") +a:depends("crontab","2") + +a=s:taboption("crontab", Flag,"router_wan",translate("WAN信息")) +a.default=1 +a:depends("crontab","1") +a:depends("crontab","2") + +a=s:taboption("crontab", Flag,"client_list",translate("客户端列表")) +a.default=1 +a:depends("crontab","1") +a:depends("crontab","2") + +e=s:taboption("crontab", Button,"_add",translate("手动发送")) +e.inputtitle=translate("发送") +e:depends("crontab","1") +e:depends("crontab","2") +e.inputstyle = "apply" +function e.write(self, section) +luci.sys.call("cbi.apply") + luci.sys.call("/usr/bin/serverchan/serverchan send &") +end + +--免打扰 +a=s:taboption("disturb", ListValue,"serverchan_sheep",translate("免打扰时段设置"),translate("在指定整点时间段内,暂停推送消息
免打扰时间中,定时推送也会被阻止。")) +a.rmempty = true + +a:value("",translate("关闭")) +a:value("1",translate("模式一:脚本挂起")) +a:value("2",translate("模式二:静默模式")) +a.description = translate("模式一停止一切检测,包括无人值守。") +a=s:taboption("disturb", ListValue,"starttime",translate("免打扰开始时间")) +a.rmempty = true + +for t=0,23 do +a:value(t,translate("每天"..t.."点")) +end +a.default=0 +a.datatype=uinteger +a:depends({serverchan_sheep="1"}) +a:depends({serverchan_sheep="2"}) +a=s:taboption("disturb", ListValue,"endtime",translate("免打扰结束时间")) +a.rmempty = true + +for t=0,23 do +a:value(t,translate("每天"..t.."点")) +end +a.default=8 +a.datatype=uinteger +a:depends({serverchan_sheep="1"}) +a:depends({serverchan_sheep="2"}) + +a=s:taboption("disturb", ListValue,"macmechanism",translate("MAC过滤")) +a:value("",translate("disable")) +a:value("allow",translate("忽略列表内设备")) +a:value("block",translate("仅通知列表内设备")) +a:value("interface",translate("仅通知此接口设备")) +a.rmempty = true + +a = s:taboption("disturb", DynamicList, "serverchan_whitelist", translate("忽略列表")) +nt.mac_hints(function(mac, name) a :value(mac, "%s (%s)" %{ mac, name }) end) +a.rmempty = true +a:depends({macmechanism="allow"}) +a.description = translate("AA:AA:AA:AA:AA:AA\\|BB:BB:BB:BB:BB:B 可以将多个 MAC 视为同一用户
任一设备在线后不再推送,设备全部离线时才会推送,避免双 wifi 频繁推送") + +a = s:taboption("disturb", DynamicList, "serverchan_blacklist", translate("关注列表")) +nt.mac_hints(function(mac, name) a:value(mac, "%s (%s)" %{ mac, name }) end) +a.rmempty = true +a:depends({macmechanism="block"}) +a.description = translate("AA:AA:AA:AA:AA:AA\\|BB:BB:BB:BB:BB:B 可以将多个 MAC 视为同一用户
任一设备在线后不再推送,设备全部离线时才会推送,避免双 wifi 频繁推送") + +a = s:taboption("disturb", ListValue, "serverchan_interface", translate("接口名称")) +a:depends({macmechanism="interface"}) +a.rmempty = true + +for _, iface in ipairs(ifaces) do + if not (iface == "lo" or iface:match("^ifb.*")) then + local nets = net:get_interface(iface) + nets = nets and nets:get_networks() or {} + for k, v in pairs(nets) do + nets[k] = nets[k].sid + end + nets = table.concat(nets, ",") + a:value(iface, ((#nets > 0) and "%s (%s)" % {iface, nets} or iface)) + end +end + +a=s:taboption("disturb", ListValue,"macmechanism2",translate("MAC过滤2")) +a:value("",translate("disable")) +a:value("MAC_online",translate("列表内任意设备在线时免打扰")) +a:value("MAC_offline",translate("列表内设备都离线后免打扰")) +a.rmempty = true + +a = s:taboption("disturb", DynamicList, "MAC_online_list", translate("在线免打扰列表")) +nt.mac_hints(function(mac, name) a:value(mac, "%s (%s)" %{ mac, name }) end) +a.rmempty = true +a:depends({macmechanism2="MAC_online"}) + +a = s:taboption("disturb", DynamicList, "MAC_offline_list", translate("任意离线免打扰列表")) +nt.mac_hints(function(mac, name) a:value(mac, "%s (%s)" %{ mac, name }) end) +a.rmempty = true +a:depends({macmechanism2="MAC_offline"}) + +return m diff --git a/package/lean/luci-app-serverchan/luasrc/view/serverchan/log.htm b/package/lean/luci-app-serverchan/luasrc/view/serverchan/log.htm new file mode 100644 index 000000000..56fe70d65 --- /dev/null +++ b/package/lean/luci-app-serverchan/luasrc/view/serverchan/log.htm @@ -0,0 +1,33 @@ +<% +local dsp = require "luci.dispatcher" +-%> + + +
+ <%:自动刷新%> + + +
diff --git a/package/lean/luci-app-serverchan/luasrc/view/serverchan/serverchan_status.htm b/package/lean/luci-app-serverchan/luasrc/view/serverchan/serverchan_status.htm new file mode 100644 index 000000000..e36561dd0 --- /dev/null +++ b/package/lean/luci-app-serverchan/luasrc/view/serverchan/serverchan_status.htm @@ -0,0 +1,22 @@ + + +
+

+ <%:Collecting data...%> +

+
diff --git a/package/lean/luci-app-serverchan/root/etc/config/serverchan b/package/lean/luci-app-serverchan/root/etc/config/serverchan new file mode 100644 index 000000000..a9a53d9be --- /dev/null +++ b/package/lean/luci-app-serverchan/root/etc/config/serverchan @@ -0,0 +1,11 @@ + +config serverchan 'serverchan' + option serverchan_enable '0' + option sleeptime '60' + option serverchan_ipv6 '0' + option serverchan_up '1' + option serverchan_down '1' + option cpuload_enable '1' + option cpuload '2' + option temperature_enable '0' + diff --git a/package/lean/luci-app-serverchan/root/etc/init.d/serverchan b/package/lean/luci-app-serverchan/root/etc/init.d/serverchan new file mode 100755 index 000000000..2631e4f61 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/etc/init.d/serverchan @@ -0,0 +1,26 @@ +#!/bin/sh /etc/rc.common + +START=99 +STOP=10 + +start() { + state=`pgrep -f "/usr/bin/serverchan/serverchan"` + if [ ! -z "$state" ]; then + restart + else + /usr/bin/serverchan/serverchan & + fi + echo "serverchan is starting now ..." +} + +stop() { + kill -9 `pgrep -f "/usr/bin/serverchan/serverchan"` 2>/dev/null + echo "serverchan exit ..." +} + +restart(){ + stop + sleep 1 + start + echo "restarted." +} diff --git a/package/lean/luci-app-serverchan/root/etc/uci-defaults/luci-serverchan b/package/lean/luci-app-serverchan/root/etc/uci-defaults/luci-serverchan new file mode 100755 index 000000000..72ce46dd6 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/etc/uci-defaults/luci-serverchan @@ -0,0 +1,11 @@ +#!/bin/sh + +uci -q batch <<-EOF >/dev/null + delete ucitrack.@serverchan[-1] + add ucitrack serverchan + set ucitrack.@serverchan[-1].init=serverchan + commit ucitrack +EOF + +rm -rf /tmp/luci-* +exit 0 diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/diy.json b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/diy.json new file mode 100644 index 000000000..04339ef58 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/diy.json @@ -0,0 +1,40 @@ +{ + "_//": "-------------------------------------------------------------------------------", + "_readme": "这是 自定义 api 文件,这里以 telegram 为例", + "_readme": "特殊符号请使用斜杠转义,变量使用 ${var} 表示", + "_//": "-------------------------------------------------------------------------------", + "_api": "【DIY 推送】", + "_url": "api 地址", + "_data": "生成的 json 文件路径,一般不需要改,如 api 不支持 json,请参考 serverchan 推送接口", + "_content_type": "post 内容类型,这里为 json", + "_//": "-------------------------------------------------------------------------------", + "_str_title_start": "标题粗体字开始符号", + "_str_title_end": "标题粗体字结束符号", + "_str_linefeed": "换行符号", + "_str_splitline": "换行+分隔符", + "_str_space": "空格", + "_str_tab": "TAB(用在行首,生成文字区块)", + "_//": "-------------------------------------------------------------------------------", + "_type": + { + "_readme": "type 对象因为需要转义变量,前后必须使用 斜杠+双引号 转义", + "_readme": "参照上文说明,填写下文相关参数" + }, + "_//": "-------------------------------------------------------------------------------", + + "url": "https://api.telegram.org/bot${tg_token}/sendMessage", + "data": "@${tempjsonpath}", + "content_type": "Content-Type: application/json", + "str_title_start": "", + "str_title_end": "", + "str_linefeed": "\\n", + "str_splitline": "\\n----\\n", + "str_space": " ", + "str_tab": " ", + "type": + { + "text":"\"${str_title_start}${1}${str_title_end}${str_splitline}${nowtime}${2}\"", + "chat_id":"\"${chat_id}\"", + "parse_mode":"\"HTML\"" + } +} diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ip_blacklist b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ip_blacklist new file mode 100644 index 000000000..8b1378917 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ip_blacklist @@ -0,0 +1 @@ + diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv4.list b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv4.list new file mode 100644 index 000000000..72081532b --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv4.list @@ -0,0 +1,6 @@ +www.cip.cc +ipv4.ddnspod.com +ifcfg.cn +speed.neu.edu.cn/getIP.php +ddns.oray.com/checkip +www.net.cn/static/customercare/yourip.asp diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv6.list b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv6.list new file mode 100644 index 000000000..5f1ca230c --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/ipv6.list @@ -0,0 +1,5 @@ +ip.sb +ipv6.ddnspod.com +api-ipv6.ip.sb/ip +speed.neu6.edu.cn/getIP.php +v6.myip.la/json diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/logo.jpg b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/logo.jpg new file mode 100644 index 0000000000000000000000000000000000000000..a89fa5cc291f3834ef76eb1e33c814ddabca6e14 GIT binary patch literal 33685 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

HCLuD)v~ZL!KsVcG~Iw$offG#-N6MWfLp#h1h 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 ", + "type": + { + "touser": "\"${userid}\"", + "msgtype": "\"markdown\"", + "agentid": "\"${agentid}\"", + "markdown": { + "content": "\"${1}${str_linefeed}${nowtime}${2}\"" + } + } +} diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/qywx_mpnews.json b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/qywx_mpnews.json new file mode 100644 index 000000000..4d452a4c6 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/qywx_mpnews.json @@ -0,0 +1,33 @@ +{ + "_api": "这是企业微信图文信息 api 文件", + "_api": "【企业微信】", + + "url": "\"https://qyapi.weixin.qq.com/cgi-bin/message/send?access_token=$(curl -s \"https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid=${corpid}&corpsecret=${corpsecret}\"|/usr/bin/jq '.access_token'|sed 's/\"//g')\"", + "data": "@${tempjsonpath}", + "content_type": "Content-Type: application/json", + "str_title_start": "

", + "str_title_end": "

", + "str_linefeed": "\\n", + "str_splitline": "
", + "str_space": " ", + "str_tab": "
  • ", + "type": + { + "touser": "\"${userid}\"", + "msgtype": "\"mpnews\"", + "agentid": "\"${agentid}\"", + "mpnews":{ + "articles":[ + { + "title": "\"${nowtime}${str_linefeed}${1}\"", + "thumb_media_id": "\"`curl \"https://qyapi.weixin.qq.com/cgi-bin/media/upload?access_token=$(curl -s \"https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid=${corpid}&corpsecret=${corpsecret}\"|/usr/bin/jq '.access_token'|sed 's/\"//g')&type=image\" -F \"file=@${mediapath}\"|/usr/bin/jq '.media_id'|sed 's/\"//g'`\"", + "author": "\"\"", + "content_source_url": "\"\"", + "content": "\"${2}\"", + "digest": "\"\"" + } + ] + }, + "safe":0 + } +} diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/serverchan.json b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/serverchan.json new file mode 100644 index 000000000..116e3f208 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/serverchan.json @@ -0,0 +1,17 @@ +{ + "_api": "这是 serverchan api 文件", + "_api": "【serverchan】", + + "url": "\"https://sctapi.ftqq.com/${sckey}.send\"", + "data": "\"text=${1}&desp=${nowtime}${str_linefeed}${2}\"", + "content_type": "Content-Type:application/x-www-form-urlencoded", + "str_title_start": "#### ", + "str_title_end": "", + "str_linefeed": "%0D%0A%0D%0A", + "str_splitline": "%0D%0A%0D%0A----%0D%0A%0D%0A", + "str_space": " ", + "str_tab": " ", + "type": + { + } +} diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/telegram.json b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/telegram.json new file mode 100644 index 000000000..a8c57b040 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/telegram.json @@ -0,0 +1,20 @@ +{ + "_api": "这是 telegram api 文件", + "_api": "【telegram】", + + "url": "https://api.telegram.org/bot${tg_token}/sendMessage", + "data": "@${tempjsonpath}", + "content_type": "Content-Type: application/json", + "str_title_start": "", + "str_title_end": "", + "str_linefeed": "\\n", + "str_splitline": "\\n----\\n", + "str_space": " ", + "str_tab": " ", + "type": + { + "text":"\"${str_title_start}${1}${str_title_end}${str_splitline}${nowtime}${2}\"", + "chat_id":"\"${chat_id}\"", + "parse_mode":"\"HTML\"" + } +} diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/wxpusher.json b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/wxpusher.json new file mode 100644 index 000000000..83e1a78ee --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/api/wxpusher.json @@ -0,0 +1,23 @@ +{ + "_api": "这是 wxpusher api 文件", + "_api": "【wxpusher】", + + "url": "http://wxpusher.zjiecode.com/api/send/message", + "data": "@${tempjsonpath}", + "content_type": "Content-Type: application/json", + "str_title_start": "#### ", + "str_title_end": "", + "str_linefeed": "\\n", + "str_splitline": "\\n----\\n", + "str_space": " ", + "str_tab": " ", + "type": + { + "summary":"\"${1}\"", + "content":"\"${2}\"", + "appToken":"\"${wxpusher_apptoken}\"", + "topicIds":"[\"${wxpusher_topicIds}\"]", + "uids":"[\"${wxpusher_uids}\"]", + "contentType":3 + } +} diff --git a/package/lean/luci-app-serverchan/root/usr/bin/serverchan/serverchan b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/serverchan new file mode 100755 index 000000000..f0e9a60a4 --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/bin/serverchan/serverchan @@ -0,0 +1,1246 @@ +#!/bin/sh + +# 读取设置文件 +function get_config(){ + while [[ "$*" != "" ]]; do + eval ${1}='`uci get serverchan.serverchan.$1`' 2>/dev/null + shift + done +} + +# 初始化设置信息 +function read_config(){ + get_config "serverchan_enable" "lite_enable" "device_name" "sleeptime" "oui_data" "oui_dir" "reset_regularly" "debuglevel" "device_aliases" \ + "serverchan_ipv4" "ipv4_interface" "serverchan_ipv6" "ipv6_interface" "serverchan_up" "serverchan_down" "cpuload_enable" "cpuload" "temperature_enable" "temperature" "client_usage" "client_usage_max" "client_usage_disturb" "client_usage_whitelist" "web_logged" "ssh_logged" "web_login_failed" "ssh_login_failed" "login_max_num" "web_login_black" "ip_white_list" "ip_black_timeout"\ + "regular_time" "regular_time_2" "regular_time_3" "interval_time" \ + "serverchan_sheep" "starttime" "endtime" "serverchan_whitelist" "serverchan_blacklist" "serverchan_interface" "MAC_online_list" "MAC_offline_list" \ + "up_timeout" "down_timeout" "timeout_retry_count" "thread_num" "soc_code" "err_enable" "err_sheep_enable" "err_device_aliases" "network_err_event" "system_time_event" "autoreboot_time" "network_restart_time" "public_ip_event" "public_ip_retry_count" \ + "jsonpath" "sckey" "corpid" "userid" "agentid" "corpsecret" "mediapath" "wxpusher_apptoken" "wxpusher_uids" "wxpusher_topicIds" "pushplus_token" "tg_token" "chat_id" + + for str_version in "wrtbwmon" "iputils-arping" "curl" "iw"; do + eval `echo ${str_version:0:2}"_version"`=`opkg list-installed|grep -w ^${str_version}|awk '{print $3}'` 2>/dev/null + done + dir="/tmp/serverchan/" && mkdir -p ${dir} + tempjsonpath="/tmp/serverchan/temp.json" + ip_blacklist_path="/usr/bin/serverchan/api/ip_blacklist" + [ ! -z "$oui_dir" ] && [ "$oui_dir" -eq "1" ] && oui_base="${dir}oui_base.txt" || oui_base="/usr/bin/serverchan/oui_base.txt" + debuglevel=`echo "$debuglevel"` && [ -z "$debuglevel" ] && logfile="/dev/null" || logfile="${dir}serverchan.log" + serverchan_blacklist=`echo "$serverchan_blacklist"|sed 's/ /\n/g'` 2>/dev/null + serverchan_whitelist=`echo "$serverchan_whitelist"|sed 's/ /\n/g'` 2>/dev/null + device_aliases=`echo "$device_aliases"|sed 's/ /\n/g'|sed 's/-/ /'` 2>/dev/null + err_device_aliases=`echo "$err_device_aliases"|sed 's/ /\n/g'` 2>/dev/null + client_usage_whitelist=`echo "$client_usage_whitelist"|sed 's/ /\n/g'` 2>/dev/null + ip_white_list=`echo "$ip_white_list"|sed 's/ /\n/g'` 2>/dev/null + mark_mac_list="${MAC_online_list} ${MAC_offline_list}" + mark_mac_list=`echo "$mark_mac_list"|sed 's/ /\n/g'|sed 's/-/ /'` 2>/dev/null + ipv4_urllist=`cat /usr/bin/serverchan/api/ipv4.list` 2>/dev/null + ipv6_urllist=`cat /usr/bin/serverchan/api/ipv6.list` 2>/dev/null + [ -z "$serverchan_ipv4" ] && serverchan_ipv4=0 + [ -z "$serverchan_ipv6" ] && serverchan_ipv6=0 + [ "$iw_version" ] && wlan_interface=`iw dev|grep Interface|awk '{print $2}'` >/dev/null 2>&1 + [ -z "$up_timeout" ] || [ "$up_timeout" -eq "0" ] && up_timeout="2" + [ -z "$down_timeout" ] || [ "$down_timeout" -eq "0" ] && down_timeout="20";down_timeout=`expr ${down_timeout} / 2 + 1` + [ -z "$timeout_retry_count" ] && timeout_retry_count="2";[ "$timeout_retry_count" -eq "0" ] && timeout_retry_count="1" + str_title_start=`/usr/bin/jq -r '.str_title_start' ${jsonpath}` + str_title_end=`/usr/bin/jq -r '.str_title_end' ${jsonpath}` + str_linefeed=`/usr/bin/jq -r '.str_linefeed' ${jsonpath}` + str_splitline=`/usr/bin/jq -r '.str_splitline' ${jsonpath}` + str_space=`/usr/bin/jq -r '.str_space' ${jsonpath}` + str_tab=`/usr/bin/jq -r '.str_tab' ${jsonpath}` + ( echo "$lite_enable"|grep -q "content" ) && str_title_start="" && str_title_end="" && str_splitline="" && str_linefeed="" && str_tab="" +} + +# 初始化 +function serverchan_init(){ + enable_detection + if [ -f "/usr/bin/serverchan/errlog" ]; then + cat /usr/bin/serverchan/errlog > ${logfile} + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】载入上次重启前日志" >> ${logfile} + echo "--------------------------------------------------------" >> ${logfile} + fi + down_oui & + deltemp + get_syslog + add_ip_black + + rm -f ${dir}fd1 ${dir}sheep_usage ${dir}old_sheep_usage ${dir}client_usage_aliases ${dir}old_client_usage_aliases /usr/bin/serverchan/errlog >/dev/null 2>&1 + [ ! -f "/usr/sbin/wrtbwmon" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】未安装 wrtbwmon ,流量统计不可用" >> ${logfile} + [ -z "$ip_version" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】无法获取依赖项 iputils-arping 版本号,请确认插件是否正常运行" >> ${logfile} + [ -z "$cu_version" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】无法获取依赖项 curl 版本号,请确认插件是否正常运行" >> ${logfile} + [ -z "${sckey}${tg_token}${pushplus_token}${corpid}${wxpusher_apptoken}${wxpusher_uids}${wxpusher_topicIds}" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】请填写正确的 key " >> ${logfile} && return 1 + local interfacelist=`getinterfacelist` && [ -z "$interfacelist" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】无法正确获取接口信息,请确认插件是否正常运行" >> ${logfile} + return 0 +} + +# 推送 +function diy_send(){ + ( ! echo "$lite_enable"|grep -q "content" ) && ( ! echo "$lite_enable"|grep -q "nowtime" ) && local nowtime=`date "+%Y-%m-%d %H:%M:%S"` + local diyurl=`/usr/bin/jq -r .url ${3}` && local diyurl=`eval echo ${diyurl}` + local type=`/usr/bin/jq -r '.type' ${3}` && local type=`eval echo ${type}` + local data=`/usr/bin/jq -r '.data' ${3}` && local data=`eval echo ${data}` + local content_type=`/usr/bin/jq -r '.content_type' ${3}` + /usr/bin/jq ".type + $type" ${jsonpath} > ${tempjsonpath} + + /usr/bin/jq -r '.[]' ${tempjsonpath}|grep -w "null" && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】参数值错误,请检查设置项 `/usr/bin/jq -r '.' ${tempjsonpath}|grep "null"`" >> ${logfile} && return 1 + [ -f ${tempjsonpath} ] && local logrow=$(grep -c "" ${tempjsonpath}) || local logrow="0" + [ $logrow -eq "0" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】json 文件生成失败,请检查文件格式" >> ${logfile} && return 1 + /usr/bin/jq -r '.[]' ${tempjsonpath}|grep "null" && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】参数变量生成失败,请检查设置项 `/usr/bin/jq -r '.' ${tempjsonpath}|grep "null"`" >> ${logfile} + + curl -X POST -H "$content_type" -d "${data}" "${diyurl}" +} + +# 下载设备MAC厂商信息 +function down_oui(){ + [ -f ${oui_base} ] && local logrow=$(grep -c "" ${oui_base}) || local logrow="0" + [ $logrow -lt "10" ] && rm -f ${oui_base} >/dev/null 2>&1 + if [ ! -z "$oui_data" ] && [ "$oui_data" -ne "3" ] && [ ! -f ${oui_base} ]; then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【初始化】设备MAC厂商信息不存在,重新下载" >> ${logfile} + wget --no-check-certificate -t 3 -T 15 -O ${dir}oui.txt https://linuxnet.ca/ieee/oui.txt >/dev/null 2>&1 + if [ -f ${dir}oui.txt ] && [ "$oui_data" -eq "1" ]; then + cat ${dir}oui.txt|grep "base 16"|grep -i "apple\|aruba\|asus\|autelan\|belkin\|bhu\|buffalo\|cctf\|cisco\|comba\|datang\|dell\|dlink\|dowell\|ericsson\|fast\|feixun\|\ +fiberhome\|fujitsu\|grentech\|h3c\|hisense\|hiwifi\|honghai\|honghao\|hp\|htc\|huawei\|intel\|jinli\|jse\|lenovo\|lg\|liteon\|malata\|meizu\|mercury\|meru\|moto\|netcore\|\ +netgear\|nokia\|omron\|oneplus\|oppo\|philips\|router_unkown\|samsung\|shanzhai\|sony\|start_net\|sunyuanda\|tcl\|tenda\|texas\|tianyu\|tp-link\|ubq\|undefine\|VMware\|\ +utstarcom\|volans\|xerox\|xiaomi\|zdc\|zhongxing\|smartisan" > ${oui_base} && echo "`date "+%Y-%m-%d %H:%M:%S"` 【初始化】设备MAC厂商信息下载成功" >> ${logfile} || echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】设备MAC厂商信息下载失败" >> ${logfile} + fi + if [ -f ${dir}oui.txt ] && [ "$oui_data" -eq "2" ]; then + cat ${dir}oui.txt|grep "base 16" > ${oui_base} && echo "`date "+%Y-%m-%d %H:%M:%S"` 【初始化】设备MAC厂商信息下载成功" >> ${logfile} || echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】设备MAC厂商信息下载失败" >> ${logfile} + fi + rm -f ${dir}oui.txt >/dev/null 2>&1 + fi +} + +# 清理临时文件 +function deltemp(){ + unset title content ipAddress_logrow online_list online_mac mac_online_status + rm -f ${dir}title ${dir}content ${dir}tmp_downlist ${dir}send_enable.lock ${tempjsonpath} >/dev/null 2>&1 + LockFile unlock + [ -f ${logfile} ] && local logrow=$(grep -c "" ${logfile}) || local logrow="0" + [ $logrow -gt 500 ] && sed -i '1,100d' ${logfile} && echo "`date "+%Y-%m-%d %H:%M:%S"` 【清理】日志超出上限,删除前 100 条" >> ${logfile} +} + +# 检测程序开关 +function enable_detection(){ + [ ! "$1" ] && local time_n=1 + for i in `seq 1 $time_n`; do + get_config serverchan_enable;[ -z "$serverchan_enable" ] || [ "$serverchan_enable" -eq "0" ] && `/etc/init.d/serverchan stop` || sleep 1 + done +} + +# 获取 ip +function getip(){ + [ ! "$1" ] && return + if [ $1 == "wanipv4" ] ;then + [ ! -z "$ipv4_interface" ] && local wanIP=$(/sbin/ifconfig ${ipv4_interface}|awk '/inet addr/ {print $2}'|awk -F: '{print $2}'|grep -oE '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}') + [ -z "$ipv4_interface" ] && local wanIP=$(getinterfacelist|grep '\"address\"'|grep -oE '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}') + echo "$wanIP" + elif [ $1 == "hostipv4" ] ;then + function get_hostipv4() + { + local url_number=`echo "$ipv4_urllist"|wc -l` + local ipv4_URL=`echo "$ipv4_urllist"| sed -n "$(rand 1 $url_number)p"|sed -e 's/\r//g'` + [ ! -z "$ipv4_interface" ] && local hostIP=$(curl -k -s -4 --interface ${ipv4_interface} -m 5 ${ipv4_URL}) || local hostIP=$(curl -k -s -4 -m 5 ${ipv4_URL}) + echo $hostIP|grep -oE '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}'|head -n1 + } + local hostIP=`get_hostipv4` + [ -z "$hostIP" ] && local hostIP=`get_hostipv4` + [ -z "$hostIP" ] && local hostIP=`get_hostipv4` + echo $hostIP # 重试,偷懒,有空再优化 + elif [ $1 == "wanipv6" ] ;then + [ ! -z "$ipv6_interface" ] && local wanIPv6=$(ip addr show ${ipv6_interface}|grep -v deprecated|grep -A1 'inet6 [^f:]'|sed -nr ':a;N;s#^ +inet6 ([a-f0-9:]+)/.+? scope global .*? valid_lft ([0-9]+sec) .*#\2 \1#p;ta'|sort -nr|head -n1|awk '{print $2}') + [ -z "$ipv6_interface" ] && local wanIPv6=$(ip addr show|grep -v deprecated|grep -A1 'inet6 [^f:]'|sed -nr ':a;N;s#^ +inet6 ([a-f0-9:]+)/.+? scope global .*? valid_lft ([0-9]+sec) .*#\2 \1#p;ta'|sort -nr|head -n1|awk '{print $2}') + echo "$wanIPv6" + elif [ $1 == "hostipv6" ] ;then + function get_hostipv6() + { + local urlv6_number=`echo "$ipv6_urllist"|wc -l` + local ipv6_URL=`echo "$ipv6_urllist"| sed -n "$(rand 1 $urlv6_number)p"|sed -e 's/\r//g'` + [ ! -z "$ipv6_interface" ] && local hostIPv6=$(curl -k -s -6 --interface ${ipv6_interface} -m 5 ${ipv6_URL}) || local hostIPv6=$(curl -k -s -6 -m 5 ${ipv6_URL}) + echo $hostIPv6|grep -oE '([\da-fA-F0-9]{1,4}(:{1,2})){1,15}[\da-fA-F0-9]{1,4}'|head -n1 + } + local hostIPv6=`get_hostipv6` + [ -z "$hostIPv6" ] && local hostIPv6=`get_hostipv6` + [ -z "$hostIPv6" ] && local hostIPv6=`get_hostipv6` + echo $hostIPv6 # 重试,偷懒,有空再优化 + fi +} + +# 获取接口信息 +function getinterfacelist(){ + [ `ubus list|grep -w -i "network.interface.wan"|wc -l` -ge "1" ] && ubus call network.interface.wan status && return + [ `ubus list|grep -i "network.interface."|grep -v "loopback"|grep -v "wan6"|wc -l` -eq "1" ] && ubus call `ubus list|grep "network.interface."|grep -v "loopback"` status && return +} + +# 获取接口在线时间 +function getinterfaceuptime(){ + getinterfacelist|grep \"uptime\"|sed $'s/\"uptime": //g'|sed $'s/\,//g' +} + +# 查询 mac 地址 +function getmac(){ + ( echo "$tmp_mac"|grep -q "unknown" ) && unset tmp_mac # 为unknown时重新读取 + [ -f "${dir}ipAddress" ] && [ -z "$tmp_mac" ] && local tmp_mac=`cat ${dir}ipAddress|grep -w ${1}|awk '{print $2}'|grep -v "^$"|sort -u|head -n1` + [ -f "${dir}tmp_downlist" ] && [ -z "$tmp_mac" ] && local tmp_mac=`cat ${dir}tmp_downlist|grep -w ${1}|awk '{print $2}'|grep -v "^$"|sort -u|head -n1` + [ -f "/var/dhcp.leases" ] && [ -z "$tmp_mac" ] && local tmp_mac=`cat /var/dhcp.leases|grep -w ${1}|awk '{print $2}'|grep -v "^$"|sort -u|head -n1` + [ -z "$tmp_mac" ] && local tmp_mac=`cat /proc/net/arp|grep "0x2\|0x6"|grep -w ${1}|awk '{print $4}'|grep -v "^$"|sort -u|head -n1` + [ -z "$tmp_mac" ] && local tmp_mac="unknown" + echo "$tmp_mac" +} + +# 查询主机名 +function getname(){ + [ -z "$tmp_name" ] && local tmp_name=`echo "$device_aliases"|grep -i $2|awk '{print $2}'|grep -v "^$"|sort -u|head -n1` + [ -f "${dir}ipAddress" ] && [ -z "$tmp_name" ] && local tmp_name=`cat ${dir}ipAddress|grep -w ${1}|awk '{print $3}'|grep -v "^$"|sort -u|head -n1` + [ -f "${dir}tmp_downlist" ] && [ -z "$tmp_name" ] && local tmp_name=`cat ${dir}tmp_downlist|grep -w ${1}|awk '{print $3}'|grep -v "^$"|sort -u|head -n1` + ( ! echo "$tmp_name"|grep -q -w "unknown\|*" ) && [ ! -z "$tmp_name" ] && echo "$tmp_name" && return || unset tmp_name # 为unknown时重新读取 + [ -f "/var/dhcp.leases" ] && [ -z "$tmp_name" ] && local tmp_name=`cat /var/dhcp.leases|grep -w ${1}|awk '{print $4}'|grep -v "^$"|sort -u|head -n1` + ( ! echo "$tmp_name"|grep -q -w "unknown\|*" ) && [ ! -z "$tmp_name" ] && echo "$tmp_name" && return || unset tmp_name # 为unknown时重新读取 + [ -z "$dhcp_config" ] && dhcp_config=`uci show dhcp|grep "ip\|mac\|name"` + for dhcp_config_str in "host" "domain"; do + local dhcp_ip_n=`echo "$dhcp_config"|grep -w ^dhcp.@${dhcp_config_str}.*ip=.${1}|sed -nr 's#^dhcp.(.*).ip.*#\1#gp'` 2>/dev/null + [ ! -z "$dhcp_ip_n" ] && [ -z "$tmp_name" ] && local tmp_name=`uci get dhcp.${dhcp_ip_n}.name` 2>/dev/null + local dhcp_mac_n=`echo "$dhcp_config"|grep -i ^dhcp.@${dhcp_config_str}.*mac=.${2}|sed -nr 's#^dhcp.(.*).mac.*#\1#gp'` 2>/dev/null + [ ! -z "$dhcp_mac_n" ] && [ -z "$tmp_name" ] && local tmp_name=`uci get dhcp.${dhcp_ip_n}.name` 2>/dev/null + [ ! -z "$tmp_name" ] && break + done + ( ! echo "$tmp_name"|grep -q -w "unknown\|*" ) && [ ! -z "$tmp_name" ] && echo "$tmp_name" && return || unset tmp_name # 为unknown时重新读取 + [ -f "$oui_base" ] && local tmp_name=$(cat $oui_base|grep -i $(echo "$2"|cut -c 1,2,4,5,7,8)|sed -nr 's#^.*16)..(.*)#\1#gp'|sed 's/ /_/g') + [ ! -z "$oui_data" ] && [ "$oui_data" -eq "4" ] && local tmp_name=$(curl -sS "http://standards-oui.ieee.org/oui.txt"|grep -i $(echo "$2"|cut -c 1,2,4,5,7,8)|sed -nr 's#^.*16)..(.*)#\1#gp'|sed 's/ /_/g') + [ -z "$tmp_name" ] && local tmp_name="unknown" + echo "$tmp_name" +} + +# 查询设备接口 +function getinterface(){ + [ -f "${dir}ipAddress" ] && local ip_interface=`cat ${dir}ipAddress|grep -w ${1}|awk '{print $5}'|grep -v "^$"|sort -u|head -n1` + [ -f "${dir}tmp_downlist" ] && [ -z "$ip_interface" ] && local ip_interface=`cat ${dir}tmp_downlist|grep -w ${1}|awk '{print $5}'|grep -v "^$"|sort -u|head -n1` + if [ -z "$ip_interface" ] && [ ! -z "$wlan_interface" ]; then + for interface in $wlan_interface; do + local ip_interface=`iw dev $interface station dump 2>/dev/null|grep Station|grep -i -w ${1}|sed -nr 's#^.*on (.*))#\1#gp'` >/dev/null 2>&1 + [ ! -z "$ip_interface" ] && echo "$ip_interface" && return + done + fi + [ -z "$ip_interface" ] && local ip_interface=`cat /proc/net/arp|grep "0x2\|0x6"|grep -i -w ${1}|awk '{print $6}'|grep -v "^$"|sort -u|head -n1` + echo "$ip_interface" +} + +# ping +function getping(){ + [ "$iw_version" ] && local wlan_online=`iw dev ${ip_interface} station dump|grep -i -w ${ip_mac}|grep Station` >/dev/null 2>&1 + [ "$wlan_online" ] && return 0 + for i in `seq 1 ${3}`; do + ( ! echo "$ip_ms"|grep -q "ms" ) && local ip_ms=$( arping -I `cat /proc/net/arp|grep -w ${1}|awk '{print $6}'|grep -v "^$"|sort -u|head -n1` -c 20 -f -w ${2} $1 ) 2>/dev/null + ( ! echo "$ip_ms"|grep -q "ms" ) && local ip_ms=`ping -c 5 -w ${2} ${1}|grep -v '100% packet loss'` 2>/dev/null + ( ! echo "$ip_ms"|grep -q "ms" ) && sleep 1 + done + ( echo "$ip_ms"|grep -q "ms" ) +} + +# CPU 占用率 +function getcpu(){ + local AT=$(cat /proc/stat|grep "^cpu "|awk '{print $2+$3+$4+$5+$6+$7+$8 " " $2+$3+$4+$7+$8}') + sleep 3 + local BT=$(cat /proc/stat|grep "^cpu "|awk '{print $2+$3+$4+$5+$6+$7+$8 " " $2+$3+$4+$7+$8}') + printf "%.01f%%" $(echo ${AT} ${BT}|awk '{print (($4-$2)/($3-$1))*100}') +} + +# 获取SOC温度 (取所有传感器温度最大值) +function soc_temp(){ + [ -z "$soc_code" ] && local soctemp=`sensors 2>/dev/null|grep °C|sed -nr 's#^.*:.*\+(.*)°C .*#\1#gp'|sort -nr|head -n1` + [ -z "$soc_code" ] && [ -z "$soctemp" ] && local soctemp=`cat /sys/class/thermal/thermal_zone*/temp 2>/dev/null|sort -nr|head -n1|cut -c-2` + [ ! -z "$soctemp" ] && echo "$soctemp" && return + [ ! -z "$soc_code" ] && echo "$soc_code"|awk '{run=$0;system(run)}' 2>/dev/null +} + +# 流量数据 +function usage(){ + [ ! -f "/usr/sbin/wrtbwmon" ] || [ ! "$1" ] && return + if [ $1 == "update" ] ;then + function version_le() { test "$(echo "$@"|tr " " "\n"|sort -n|head -n 1)" == "$1"; } + function version_ge() { test "$(echo "$@"|tr " " "\n"|sort -r|head -n 1)" == "$1"; } + [ ! -z "$wr_version" ] && ( version_ge "${wr_version}" "1.2.0" ) && wrtbwmon -f ${dir}usage.db 2>/dev/null && return + [ ! -z "$wr_version" ] && ( version_le "${wr_version}" "1.0.0" ) || [ -z "$wr_version" ] && wrtbwmon update ${dir}usage.db 2>/dev/null && return + elif [ $1 == "get" ] ;then + [ ! -f "${dir}usage.db" ] && [ ! "$3" ] && echo `bytes_for_humans 0` && return + [ ! -f "${dir}usage.db" ] && [ "$3" ] && echo 0 && return + [ -z "$total_n" ] && total_n=`cat ${dir}usage.db|head -n1|grep "total"|sed 's/,/\n/g'|awk '/total/{print NR}'` 2>/dev/null + [ -z "$total_n" ] && total_n="6" + [ "$2" ] && local tmptotal=`cat ${dir}usage.db|sed 's/,,,/,0,0,/g'|sed 's/,,/,0,/g'|sed 's/,/ /g'|grep -i -w ${2}|awk "{print "'$'$total_n"}"|grep -v "^$"|sort -u|head -n1` 2>/dev/null + [ -z "$tmptotal" ] && local tmptotal="0" + [ ! "$3" ] && echo `bytes_for_humans ${tmptotal}` || echo "$tmptotal" + elif [ $1 == "down" ] ;then + [ "$2" ] && sed -i "/,${2},/d" ${dir}usage.db 2>/dev/null + fi +} + +# 流量数据单位换算 +function bytes_for_humans { + [ ! "$1" ] && return + [ "$1" -gt 1073741824 ] && echo "`awk 'BEGIN{printf "%.2f\n",'$1'/'1073741824'}'` GB" && return + [ "$1" -gt 1048576 ] && echo "`awk 'BEGIN{printf "%.2f\n",'$1'/'1048576'}'` MB" && return + [ "$1" -gt 1024 ] && echo "`awk 'BEGIN{printf "%.2f\n",'$1'/'1024'}'` KB" && return + echo "${1} bytes" +} + +# 设备异常流量检测 +function get_client_usage(){ + [ -z "$client_usage" ] && return + [ "$client_usage" -ne "1" ] && return + [ -z "$client_usage_max" ] && return + [ -z "$get_client_usage_time" ] && get_client_usage_time=`date +%s` + ( echo $client_usage_max|sed -r 's/.*(.)$/\1/'|grep -q "K\|k" ) && client_usage_max=`expr ${client_usage_max%?} \* 1024` + ( echo $client_usage_max|sed -r 's/.*(.)$/\1/'|grep -q "M\|m" ) && client_usage_max=`expr ${client_usage_max%?} \* 1048576` + ( echo $client_usage_max|sed -r 's/.*(.)$/\1/'|grep -q "G\|g" ) && client_usage_max=`expr ${client_usage_max%?} \* 1073741824` + [ -z "$client_usage_disturb" ] && client_usage_disturb="0" + [ "$client_usage_disturb" -eq "0" ] && [ -f "${dir}ipAddress" ] && local MACLIST=`cat ${dir}ipAddress|awk '{print $2}'|grep -v "^$"|sort -u` + [ "$client_usage_disturb" -eq "1" ] && [ ! -z "$client_usage_whitelist" ] && local MACLIST=`echo "$client_usage_whitelist"` + [ -z "$MACLIST" ] && return + + if [ "$((`date +%s`-$get_client_usage_time))" -ge "60" ]; then + > ${dir}client_usage_aliases + for mac in $MACLIST; do + ( ! cat ${dir}ipAddress|grep -q -i -w $mac|grep -v "^$"|sort -u|head -n1 ) && continue + echo "$mac" `usage get ${mac} bytes` >> ${dir}client_usage_aliases + [ -f "${dir}old_client_usage_aliases" ] && get_client_usage_bytes=`cat ${dir}old_client_usage_aliases|grep -i -w $mac|awk '{print $2}'|grep -v "^$"|sort -u|head -n1` || continue + [ -z "$get_client_usage_bytes" ] && get_client_usage_bytes="0" + if [ "$((`usage get ${mac} bytes`-$get_client_usage_bytes))" -ge "$client_usage_max" ]; then + local ip=`cat ${dir}ipAddress|grep -i -w $mac|awk '{print $1}'|grep -v "^$"|sort -u|head -n1` + local ip_name=`getname ${ip} ${mac}` + local tmp_usage=$(bytes_for_humans $(expr `usage get ${mac} bytes` - ${get_client_usage_bytes})) + local time_up=`cat ${dir}ipAddress|grep -w ${ip}|awk '{print $4}'|grep -v "^$"|sort -u|head -n1` + local ip_total=`usage get $mac` && [ ! -z "$ip_total" ] && local ip_total="${str_linefeed}${str_tab}总计流量: ${str_space}${str_space}${str_space}${str_space}${ip_total}" + local time1=`date +%s` + local time1=$(time_for_humans `expr ${time1} - ${time_up}`) + if [ -z "$title" ]; then + title="${ip_name} 流量异常" + content="${content}${str_splitline}${str_title_start} 设备流量异常${str_title_end}${str_linefeed}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${ip}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${mac}$ip_total${str_linefeed}${str_tab}一分钟内流量: ${str_space}${str_space}${tmp_usage}${str_linefeed}${str_tab}在线时间: ${str_space}${str_space}${str_space}${str_space}${time1}" + elif ( echo "$title"|grep -q "流量异常" ); then + title="${ip_name} ${title}" + content="${content}${str_splitline}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${ip}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${mac}$ip_total${str_linefeed}${str_tab}一分钟内流量: ${str_space}${str_space}${str_space}${tmp_usage}${str_linefeed}${str_tab}在线时间: ${str_space}${str_space}${str_space}${str_space}${time1}" + else + title="设备状态变化" + content="${content}${str_splitline}${str_title_start} 设备流量异常${str_title_end}${str_linefeed}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${ip}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${mac}$ip_total${str_linefeed}${str_tab}一分钟内流量: ${str_space}${str_space}${str_space}${tmp_usage}${str_linefeed}${str_tab}在线时间: ${str_space}${str_space}${str_space}${str_space}${time1}" + fi + fi + done + cat ${dir}client_usage_aliases > ${dir}old_client_usage_aliases + get_client_usage_time=`date +%s` + fi +} + +# 时间单位换算 +function time_for_humans { + [ ! "$1" ] && return + if [ "$1" -lt 60 ]; then + echo "${1} 秒" + elif [ "$1" -lt 3600 ]; then + local usetime_min=`expr $1 / 60` + local usetime_sec=`expr $usetime_min \* 60` + local usetime_sec=`expr $1 - $usetime_sec` + echo "${usetime_min} 分 ${usetime_sec} 秒" + elif [ "$1" -lt 86400 ]; then + local usetime_hour=`expr $1 / 3600` + local usetime_min=`expr $usetime_hour \* 3600` + local usetime_min=`expr $1 - $usetime_min` + local usetime_min=`expr $usetime_min / 60` + echo "${usetime_hour} 小时 ${usetime_min} 分" + else + local usetime_day=`expr $1 / 86400` + local usetime_hour=`expr $usetime_day \* 86400` + local usetime_hour=`expr $1 - $usetime_hour` + local usetime_hour=`expr $usetime_hour / 3600` + echo "${usetime_day} 天 ${usetime_hour} 小时" + fi +} + +# 计算字符真实长度 +function length_str { + [ ! "$1" ] && return + local length_zh=`echo "$1"|awk '{print gensub(/[\u4e00-\u9FA5A-Za-z0-9_]/,"","g",$0)}'|awk -F "" '{print NF}'` + local length_en=`echo "$1"|awk '{print gensub(/[^\u4e00-\u9FA5A-Za-z0-9_]/,"","g",$0)}'|awk -F "" '{print NF}'` + echo `expr $length_zh / 3 \* 2 + $length_en` +} + +# 截取字符,避免中文乱码 +function cut_str { + [ ! "$1" ] && return + [ ! "$2" ] && return + [ `length_str $1` -le "$2" ] && echo "$1" && return + local temp_length=$2 + while [ $(length_str `echo "$1"|cut -c -$temp_length`) -lt "$2" ]; do + temp_length=`expr $temp_length + 1` + done + while [ $(printf "%d" \'`echo "$1"|cut -c $temp_length`) -ge "128" ] && [ $(printf "%d" \'`echo "$1"|cut -c $temp_length`) -lt "224" ]; do + temp_length=`expr $temp_length + 1` + done + temp_length=`expr $temp_length - 1` + echo $(echo "$1"|cut -c -$temp_length)"*" +} + +# 随机数 +function rand(){ + local min=$1 + local max=$(($2- $min + 1)) + local num=$(date +%s%N) + echo $(($num % $max + $min)) +} + +# 在线设备列表 +function serverchan_first(){ + [ -f "${dir}ipAddress" ] && local IPLIST=`cat ${dir}ipAddress|awk '{print $1}'|grep -v "^$"|sort -u` + for ip in $IPLIST; do + read -u 5 + { + down $ip + echo "" >&5 + }& + done + wait + unset ip IPLIST + local IPLIST=`cat /proc/net/arp|grep "0x2\|0x6"|awk '{print $1}'|grep -v "^169.254."|grep -v "^$"|sort -u|grep -oE '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}'` + for ip in $IPLIST; do + read -u 5 + { + up $ip + echo "" >&5 + }& + done + wait +} + +# 创建计划任务 +function serverchan_cron(){ + function del_cron(){ + ( echo `crontab -l 2>/dev/null`|grep -q "serverchan" ) && crontab -l > conf && sed -i "/serverchan/d" conf && crontab conf && rm -f conf >/dev/null 2>&1 + } + function re_cron(){ + /etc/init.d/cron stop + /etc/init.d/cron start + } + del_cron + if [ -z "$serverchan_enable" ]; then + re_cron + return + fi + + # 重置流量 + if [ ! -z "$reset_regularly" ] && [ "$reset_regularly" -eq "1" ]; then + crontab -l 2>/dev/null > conf && echo -e "0 0 * * * rm /tmp/serverchan/usage.db >/dev/null 2>&1" >> conf && crontab conf && rm -f conf >/dev/null 2>&1 + crontab -l 2>/dev/null > conf && echo -e "0 0 * * * rm /tmp/serverchan/usage6.db >/dev/null 2>&1" >> conf && crontab conf && rm -f conf >/dev/null 2>&1 + fi + [ ! -z "$regular_time_2" ] && local regular_time_2=",${regular_time_2}" + [ ! -z "$regular_time_3" ] && local regular_time_3=",${regular_time_3}" + # 定时发送 + if [ ! -z "$regular_time" ] || [ ! -z "$regular_time_2" ] || [ ! -z "$regular_time_3" ]; then + crontab -l 2>/dev/null > conf && echo -e "0 $regular_time$regular_time_2$regular_time_3 * * * /usr/bin/serverchan/serverchan send &" >> conf && crontab conf && rm -f conf >/dev/null 2>&1 + # 间隔发送 + elif [ ! -z "$interval_time" ]; then + crontab -l 2>/dev/null > conf && echo -e "0 */$interval_time * * * /usr/bin/serverchan/serverchan send &" >> conf && crontab conf && rm -f conf >/dev/null 2>&1 + fi + re_cron +} + +# 免打扰检测 +function serverchan_disturb(){ + [ -z "$serverchan_sheep" ] || [ -z "$starttime" ] || [ -z "$endtime" ] && return 0 + if [ `date +%H` -ge $endtime -a $starttime -lt $endtime ] || [ `date +%H` -lt $starttime -a $starttime -lt $endtime ] || [ `date +%H` -lt $starttime -a `date +%H` -ge $endtime -a $starttime -gt $endtime ]; then + unset sheep_starttime + rm -f ${dir}sheep_usage ${dir}old_sheep_usage 2>/dev/null + disturb_text=`/usr/bin/jq -r '._api' ${jsonpath}` + return 0 + else + [ -z "$sheep_starttime" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【免打扰】夜深了,该休息了" >> ${logfile} && sheep_starttime=`date +%s` + if [ "$serverchan_sheep" -eq "1" ] ;then + while [ `date +%H` -lt "$endtime" ]; do + enable_detection + sleep $sleeptime + done + elif [ "$serverchan_sheep" -eq "2" ] ;then + disturb_text="【免打扰】" + return 1 + fi + fi +} + +# 文件锁 +function LockFile(){ + if [ $1 = "lock" ] ;then + [ ! -f "${dir}serverchan.lock" ] && > ${dir}serverchan.lock && return + while [ -f "${dir}serverchan.lock" ]; do + enable_detection 1 + done + LockFile lock + fi + [ $1 = "unlock" ] && rm -f ${dir}serverchan.lock >/dev/null 2>&1 + return 0 +} + +# 检测黑白名单 +function blackwhitelist(){ + [ ! "$1" ] && return 1 + [ -z "$serverchan_whitelist" ] && [ -z "$serverchan_blacklist" ] && [ -z "$serverchan_interface" ] && [ -z "$MAC_online_list" ] && [ -z "$MAC_offline_list" ] && return 0 + [ ! -z "$serverchan_whitelist" ] && ( echo "$serverchan_whitelist"|grep -q -i -w $1 ) && return 1 + [ ! -z "$serverchan_blacklist" ] && ( ! echo "$serverchan_blacklist"|grep -q -i -w $1 ) && return 1 + [ ! -z "$serverchan_interface" ] && ( ! echo `getinterface ${1}`|grep -q -i -w $serverchan_interface ) && return 1 + [ ! -z "$MAC_online_list" ] && [ ! -z "$mac_online_status" ] && return 1 + [ ! -z "$MAC_online_list" ] && ( echo "$MAC_online_list"|grep -q -i -w $1 ) && return 1 + [ ! -z "$MAC_offline_list" ] && [ -z "$mac_online_status" ] && return 1 + return 0 +} + +function get_client(){ + if [ -f "${dir}ipAddress" ]; then + while read line; do + local js_str="${js_str}
    " + local js_str="${js_str}
    <%:`echo "$line"|awk '{print $3}'`%>
    " + local tmp_mac=`echo "$line"|awk '{print $2}'` + local js_str="${js_str}
    <%:${tmp_mac}%>
    " + local js_str="${js_str}
    <%:`echo "$line"|awk '{print $1}'`%>
    " + local tmp_usage=`usage get ${tmp_mac}` + local js_str="${js_str}
    <%:${tmp_usage}%>
    " + local tmp_uptime=`echo "$line"|awk '{print $4}'` + local tmp_timenow=`date +%s` + local tmp_uptime=$(time_for_humans `expr ${tmp_timenow} - ${tmp_uptime}`) + local js_str="${js_str}
    <%:${tmp_uptime}%>
    " + done < ${dir}ipAddress + fi +cat>/usr/lib/lua/luci/view/serverchan/client.htm<<%:在线设备列表%>
    <%:客户端名%>
    <%:MAC%>
    <%:IP%>
    <%:总计流量%>
    <%: 在线时间%>
    +$js_str +
    +EOF +} + +# 重启网络服务 +function network_restart(){ +cat>${dir}network_restart</dev/null 2>&1 & +/etc/init.d/firewall restart >/dev/null 2>&1 & +/etc/init.d/dnsmasq restart >/dev/null 2>&1 & +EOF + chmod 0755 ${dir}network_restart && ${dir}network_restart + rm -f ${dir}network_restart >/dev/null 2>&1 +} + +# 查看无人值守任务设备是否在线 +function geterrdevicealiases(){ + [ -z "$err_device_aliases" ] && return + [ -f ${dir}ipAddress ] && local logrow=$(grep -c "" ${dir}ipAddress) || local logrow="0";[ $logrow -eq "0" ] && return + local MACLIST=`cat ${dir}ipAddress|awk '{print $2}'|grep -v "^$"|sort -u` + for mac in $MACLIST; do + [ -z "$err_mac" ] && [ ! -z "$mac" ] && local err_mac=`echo "$err_device_aliases"|grep -i $mac|grep -v "^$"|sort -u|head -n1` + done + # 进入免打扰时间已经超过一小时 + if [ ! -z "$sheep_starttime" ] && [ "$((`date +%s`-$sheep_starttime))" -ge "3600" ]; then + > ${dir}sheep_usage + local MACLIST=`echo "$err_device_aliases"|grep -v "^$"|sort -u` + for mac in $MACLIST; do + [ ! -z "$mac" ] && local tmptotal=`usage get ${mac} bytes` + [ ! -z "$tmptotal" ] && awk 'BEGIN{printf "%.0f\n",'$tmptotal'/'204800'}' 2>/dev/null >> ${dir}sheep_usage + done + old_sheep_usage=`cat ${dir}old_sheep_usage` 2>/dev/null + sheep_usage=`cat ${dir}sheep_usage` 2>/dev/null + [ "$old_sheep_usage" == "$sheep_usage" ] && [ -z "$sheep_nousage_starttime" ] && sheep_nousage_starttime=`date +%s` + [ "$old_sheep_usage" != "$sheep_usage" ] && unset sheep_nousage_starttime && cat ${dir}sheep_usage 2>/dev/null > ${dir}old_sheep_usage + [ ! -z "$sheep_nousage_starttime" ] && [ "$((`date +%s`-$sheep_nousage_starttime))" -ge "300" ] && unset err_mac + fi + [ -z "$err_mac" ] +} + +# 无人值守任务 +function unattended(){ + [ -z "$err_enable" ] || [ "$err_enable" -ne "1" ] && return + [ ! -z "$err_sheep_enable" ] && [ "$err_sheep_enable" -eq "1" ] && [ -z "$sheep_starttime" ] && return + geterrdevicealiases;[ $? -eq "1" ] && return + + if [ ! -z "$system_time_event" ]; then + local interfaceuptime=`getinterfaceuptime` + if [ ! -z "$autoreboot_time" ] && [ `cat /proc/uptime|awk -F. '{run_hour=$1/3600;printf("%d",run_hour)}'` -ge "$autoreboot_time" ] && [ "$system_time_event" -eq "1" ]; then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【无人值守任务】重启路由器咯" >> ${logfile} + cat ${logfile} > /usr/bin/serverchan/errlog + sleep 2 && reboot && exit + elif [ ! -z "$network_restart_time" ] && [ ! -z "$interfaceuptime" ] && [ `echo "$interfaceuptime"|awk -F. '{run_hour=$1/3600;printf("%d",run_hour)}'` -ge "$network_restart_time" ] && [ "$system_time_event" -eq "2" ]; then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【无人值守任务】重新拨号咯" >> ${logfile} + ifup wan >/dev/null 2>&1 + sleep 60 + fi + fi + + [ -z "$public_ip_today" ] && public_ip_today=`date +"%d"` + [ -z "$public_ip_count" ] && public_ip_count="0" + [ $public_ip_today -ne `date +"%d"` ] && public_ip_today=`date +"%d"` && public_ip_count=1 + if [ ! -z "$public_ip_event" ] && [ ! -z "$public_ip_retry_count" ] && [ "$public_ip_count" -le "$public_ip_retry_count" ]; then + public_ip_count=`expr $public_ip_count + 1` + local wanIP=`getip wanipv4` + local hostIP=`getip hostipv4` + if [ ! -z "$wanIP" ] && [ ! -z "$hostIP" ] && ( ! echo "$wanIP"|grep -q -w ${hostIP} );then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【无人值守任务】重拨尝试获取公网 ip,当前第 $public_ip_count 次 " >> ${logfile} + ifup wan >/dev/null 2>&1 + sleep 60 + local wanIP=`getip wanipv4` && local hostIP=`getip hostipv4` + [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -eq "1" ] && local IPv4=${wanIP} + [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -eq "2" ] && local IPv4=${hostIP} + [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -eq "1" ] && local IPv6=`getip wanipv6` + [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -eq "2" ] && local IPv6=`getip hostipv6` + [ ! -z "$wanIP" ] && [ ! -z "$hostIP" ] && ( ! echo "$wanIP"|grep -q -w ${hostIP} ) && echo IPv4 $IPv4 > ${dir}ip && echo -e IPv6 $last_IPv6 >> ${dir}ip + fi + fi +} + +# 检测网络状态 +function rand_geturl(){ + function getcheck(){ + local urllist="https://www.163.com https://www.qq.com https://www.baidu.com https://www.qidian.com https://www.douban.com" + local url_number=`expr $(echo "$urllist"|grep -o ' '|wc -l) + 1` + local url_str=`echo "$urllist"|awk -v i=$(rand 1 $url_number) '{print $i}'` + echo `curl -k -s -w "%{http_code}" -m 5 ${url_str} -A "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/86.0.4240.198 Safari/537.36" -o /dev/null` + } + local check=`getcheck` + while [ -z "$check" ] || [ "$check" -ne "200" ]; do + local check=`getcheck` + if [ ! -z "$check" ] && [ "$check" -eq "200" ]; then + [ ! -z "$network_enable" ] && [ "$network_enable" -eq "404" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【网络状态】网络恢复正常.." >> ${logfile} + local network_enable="200" + else + [ -z "$network_enable" ] || [ "$network_enable" -eq "200" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】当前网络不通!停止检测! " >> ${logfile} + local network_enable="404" + [ -z "$network_err_time" ] && network_err_time=`date +%s` + if [ ! -z "$network_err_event" ] && [ "$((`date +%s`-$network_err_time))" -ge "600" ]; then + > ${dir}send_enable.lock && serverchan_first && deltemp + geterrdevicealiases + if [ "$?" -eq "0" ]; then + [ -f /usr/bin/serverchan/autoreboot_count ] && retry_count=`cat /usr/bin/serverchan/autoreboot_count` && rm -f /usr/bin/serverchan/autoreboot_count >/dev/null 2>&1 + [ ! -z ${retry_count} ] && retry_count=0;retry_count=`expr $retry_count + 1` + if [ "$network_err_event" -eq "1" ] ;then + if [ "$retry_count" -lt "3" ] ;then + echo "$retry_count" > /usr/bin/serverchan/autoreboot_count + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】正在尝试重启路由,当前第 $retry_count 次 " >> ${logfile} + cat ${logfile} > /usr/bin/serverchan/errlog + sleep 2 && reboot && exit + fi + [ "$retry_count" -eq "3" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】已经重启路由2次,修复失败,请主人自行修复哦" >> ${logfile} + elif [ "$network_err_event" -eq "2" ] ;then + [ "$retry_count" -lt "3" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】正在尝试重启网络,当前第 $retry_count 次 " >> ${logfile} && ifup wan >/dev/null 2>&1 + [ "$retry_count" -eq "3" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】已经重启网络2次,修复失败,请主人自行修复哦 " >> ${logfile} + elif [ "$network_err_event" -eq "3" ] ;then + if [ "$retry_count" -eq "1" ] ;then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】正在尝试修复网络,当前第 1 次,重启网络服务中 " >> ${logfile} && network_restart + elif [ "$retry_count" -eq "2" ] ;then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】正在尝试修复网络,当前第 2 次,关闭可能造成网络断开的软件" >> ${logfile} + [ `uci get koolproxy.@global[0].enabled 2>/dev/null` -eq "1" ] && [ `uci get koolproxy.@global[0].koolproxy_mode 2>/dev/null` -eq "1" ] && /etc/init.d/koolproxy stop >/dev/null 2>&1 + [ `uci get adbyby.@adbyby[0].enable 2>/dev/null` -eq "1" ] && [ `uci get adbyby.@adbyby[0].wan_mode 2>/dev/null` -eq "0" ] && /etc/init.d/adbyby stop >/dev/null 2>&1 + [ `uci get passwall.@global[0].enabled 2>/dev/null` -eq "1" ] && [ `uci get passwall.@global[0].proxy_mode 2>/dev/null|grep global` ] && /etc/init.d/koolproxy stop >/dev/null 2>&1 + local shadowsocksr_enabled=`uci get shadowsocksr.@global[0].global_server 2>/dev/null|grep nil` + local shadowsocksr_run_mode=`uci get shadowsocksr.@global[0].run_mode 2>/dev/null|grep all` + [ -z "$shadowsocksr_enabled" ] && [ ! -z "$shadowsocksr_run_mode" ] && /etc/init.d/shadowsocksr stop >/dev/null 2>&1 + sleep 60 && network_restart + elif [ "$retry_count" -eq "3" ] ;then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】正在尝试修复网络,当前第 3 次,备份设置项,并修改相关设置" >> ${logfile} + mkdir -p /usr/bin/serverchan/configbak + cp -p -f /etc/config/network /usr/bin/serverchan/configbak/network + cp -p -f /etc/config/dhcp /usr/bin/serverchan/configbak/dhcp + cp -p -f /etc/config/firewall /usr/bin/serverchan/configbak/firewall + cp -p -f /etc/firewall.user /usr/bin/serverchan/configbak/firewall.user + uci set network.wan.peerdns='0' + uci delete network.wan.dns + uci add_list network.wan.dns='223.5.5.5' + uci add_list network.wan.dns='119.29.29.29' + uci delete network.wan.mtu + uci commit network + uci set dhcp.@dnsmasq[0].port='53' + uci set dhcp.@dnsmasq[0].resolvfile='/tmp/resolv.conf.auto' + uci delete dhcp.@dnsmasq[0].server + uci delete dhcp.@dnsmasq[0].noresolv + uci commit dhcp + uci delete firewall.redirect + >/etc/firewall.user + uci commit firewall + sleep 60 && network_restart + elif [ "$retry_count" -eq "4" ] ;then + echo "$retry_count" > /usr/bin/serverchan/autoreboot_count + cat ${logfile} > /usr/bin/serverchan/errlog + sleep 2 && reboot && exit + elif [ "$retry_count" -eq "5" ] ;then + echo "$retry_count" > /usr/bin/serverchan/autoreboot_count + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!!】修复失败,还原设置中,请自行检查网络设置" >> ${logfile} + cp -p -f /usr/bin/serverchan/configbak/network /etc/config/network + cp -p -f /usr/bin/serverchan/configbak/dhcp /etc/config/dhcp + cp -p -f /usr/bin/serverchan/configbak/firewall /etc/config/firewall + cp -p -f /usr/bin/serverchan/configbak/firewall.user /etc/firewall.user + cat ${logfile} > /usr/bin/serverchan/errlog + sleep 2 && reboot && exit + fi + fi + fi + elif [ -f /usr/bin/serverchan/autoreboot_count ]; then + network_err_time=`expr $network_err_time - 600` && sleep 60 + fi + enable_detection + sleep $sleeptime + fi + continue + done + rm -f /usr/bin/serverchan/autoreboot_count >/dev/null 2>&1 +} + +# 检测 ip 状况 +function ip_changes(){ + [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -eq "1" ] && local IPv4=`getip wanipv4` + [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -eq "2" ] && local IPv4=`getip hostipv4` + [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -eq "1" ] && local IPv6=`getip wanipv6` + [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -eq "2" ] && local IPv6=`getip hostipv6` + + if [ -f ${dir}ip ]; then + local last_IPv4=$(cat "${dir}ip"|grep IPv4|awk '{print $2}'|grep -v "^$"|sort -u|head -n1) + local last_IPv6=$(cat "${dir}ip"|grep IPv6|awk '{print $2}'|grep -v "^$"|sort -u|head -n1) + if [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -ne "0" ] && [ ! -z "$IPv4" ] && ( ! echo ${IPv4}|grep -w -q ${last_IPv4} ); then + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text}当前IP:${IPv4}" >> ${logfile} + echo IPv4 $IPv4 > ${dir}ip && echo -e IPv6 $last_IPv6 >> ${dir}ip + title="IP 地址变化" + content="${content}${str_splitline}${str_title_start} IP 地址变化${str_title_end}${str_linefeed}${str_tab}当前 IP:${IPv4}" + elif [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -ne "0" ] && [ -z "$IPv4" ]; then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】获取 IPv4 地址失败" >> ${logfile} + fi + + if [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -ne "0" ] && [ ! -z "$IPv6" ] && ( ! echo "$IPv6"|grep -w -q ${last_IPv6} ); then + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text}当前IPv6:${IPv6}" >> ${logfile} + echo IPv4 $IPv4 > ${dir}ip && echo -e IPv6 $IPv6 >> ${dir}ip + [ -z "$title" ] && title="IPv6 地址变化" + [ ! -z "$title" ] && title="IP 地址变化" + content="${content}${str_splitline}${str_title_start} IPv6 地址变化${str_title_end}${str_linefeed}${str_tab}当前 IPv6:${IPv6}" + elif [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -ne "0" ] && [ -z "$IPv6" ]; then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】获取 IPv6 地址失败" >> ${logfile} + fi + + else + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text}路由器已经重启!" >> ${logfile} + [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -ne "0" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 当前IP: ${IPv4}" >> ${logfile} + [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -ne "0" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 当前IPv6: ${IPv6}" >> ${logfile} + echo IPv4 $IPv4 > ${dir}ip && echo -e IPv6 $IPv6 >> ${dir}ip + title="路由器重新启动" + content="${content}${str_splitline}${str_title_start} 路由器重新启动"${str_title_end} + [ ! -z "$serverchan_ipv4" ] && [ "$serverchan_ipv4" -ne "0" ] && content="${content}${str_linefeed}${str_tab}当前IP:${IPv4}" + [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -ne "0" ] && content="${content}${str_linefeed}${str_tab}当前IPv6:${IPv6}" + fi + + if [ ! -z "$content" ] ;then + [ -z "$ddns_enabled" ] && ddns_enabled=$(uci show ddns|grep "enabled"|grep "1") + [ -z "$ddns_enabled" ] && ddns_logrow=0 || ddns_logrow=$(echo "$ddns_enabled"|wc -l) + if [ $ddns_logrow -ge 1 ]; then + /etc/init.d/ddns restart >/dev/null 2>&1 + fi + [ -z "$zerotier_enabled" ] && zerotier_enabled=$(uci get zerotier.sample_config.enabled) + if [ ! -z "$zerotier_enabled" ] && [ $zerotier_enabled -eq "1" ] ; then + /etc/init.d/zerotier restart >/dev/null 2>&1 + fi + fi +} + +# 检测设备上线 +function up(){ + [ -f ${dir}ipAddress ] && ( cat ${dir}ipAddress|grep -q -w $1 ) && return + local ip_mac=`getmac $1` + local ip_name=`getname ${1} ${ip_mac}` + local ip_interface=`getinterface ${ip_mac}` + getping ${1} ${up_timeout} "1";local ping_online=$? + if [ "$ping_online" -eq "0" ]; then + LockFile lock + [ ! -z "$serverchan_blacklist" ] && local tmp_mac=`echo "${serverchan_blacklist}"|grep -w -i ${ip_mac}` + [ ! -z "$serverchan_whitelist" ] && local tmp_mac=`echo "${serverchan_whitelist}"|grep -w -i ${ip_mac}` + if [ ! -z "$tmp_mac" ] && ( cat ${dir}ipAddress|grep -q -w -i ${tmp_mac} ); then + usage down $1 + echo "$1 ${ip_mac} ${ip_name} `date +%s` ${ip_interface}" >> ${dir}ipAddress + LockFile unlock && return + elif [ ! -z "$tmp_mac" ] && [ -f "${dir}tmp_downlist" ] && ( cat ${dir}tmp_downip|grep -q -w -i ${tmp_mac} ); then + local tmp_downip=`cat ${dir}tmp_downlist|grep -w -i ${tmp_mac}|awk '{print $1}'|grep -v "^$"|sort -u|head -n1` + usage down $tmp_downip + sed -i "/^${tmp_downip} /d" ${dir}tmp_downlist + LockFile unlock && return + fi + [ -f "${dir}tmp_downlist" ] && local tmp_downip=`cat ${dir}tmp_downlist|grep -w ${1}|grep -v "^$"|sort -u|head -n1` + if [ ! -z "$tmp_downip" ]; then + cat ${dir}tmp_downlist|grep -w ${1}|grep -v "^$"|sort -u|head -n1 >> ${dir}ipAddress + sed -i "/^${1} /d" ${dir}tmp_downlist + else + usage down $1 + echo "$1 ${ip_mac} ${ip_name} `date +%s` ${ip_interface}" >> ${dir}ipAddress + blackwhitelist ${ip_mac};local ip_blackwhite=$? + [ -f "${dir}send_enable.lock" ] || [ -z "$serverchan_up" ] || [ -z "$ip_blackwhite" ] && LockFile unlock && return + [ ! -z "$serverchan_up" ] && [ "$serverchan_up" -ne "1" ] && LockFile unlock && return + [ -z "$ip_blackwhite" ] || [ "$ip_blackwhite" -ne "0" ] && LockFile unlock && return + [ -f "${dir}title" ] && local title=`cat ${dir}title` + [ -f "${dir}content" ] && local content=`cat ${dir}content` + if [ -z "$title" ]; then + local title="$ip_name 连接了你的路由器" + local content="${str_splitline}${str_title_start} 新设备连接${str_title_end}${str_linefeed}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${1}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${ip_mac}${str_linefeed}${str_tab}网络接口:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_interface}" + elif ( echo ${title}|grep -q "连接了你的路由器" ); then + local title="${ip_name} ${title}" + local content="${str_splitline}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${1}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${ip_mac}${str_linefeed}${str_tab}网络接口:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_interface}" + else + local title="设备状态变化" + local content="${str_splitline}${str_title_start} 新设备连接${str_title_end}${str_linefeed}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${1}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${ip_mac}${str_linefeed}${str_tab}网络接口:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_interface}" + fi + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text}新设备 ${ip_name} ${1} 连接了">> ${logfile} + #[ ! -z "$serverchan_blacklist" ] && local title="你偷偷关注的设备上线了" + [ ! -z "$title" ] && echo "$title" >${dir}title + [ ! -z "$content" ] && echo -n "$content" >>${dir}content + fi + fi + LockFile unlock +} + +# 检测设备离线 +function down(){ + local ip_mac=`getmac $1` + local ip_name=`getname ${1} ${ip_mac}` + local ip_interface=`getinterface ${ip_mac}` + getping ${1} ${down_timeout} ${timeout_retry_count};local ping_online=$? + if [ "$ping_online" -eq "1" ]; then + LockFile lock + [ ! -f "${dir}send_enable.lock" ] && cat ${dir}ipAddress|grep -w ${1}|grep -v "^$"|sort -u|head -n1 >> ${dir}tmp_downlist + sed -i "/^${1} /d" ${dir}ipAddress + LockFile unlock + else + local tmp_name=`cat ${dir}ipAddress|grep -w ${1}|awk '{print $3}'|grep -v "^$"|sort -u|head -n1` + if [ "$ip_name" != "$tmp_name" ]; then + LockFile lock + local tmp_str=$(echo "$1 ${ip_mac} ${ip_name} `cat ${dir}ipAddress|grep -w ${1}|awk '{print $4}'|grep -v "^$"|sort -u|head -n1` ${ip_interface}") + sed -i "/^${1} /d" ${dir}ipAddress + echo "$tmp_str" >> ${dir}ipAddress + LockFile unlock + fi + fi +} + +# 设备离线通知 +function down_send(){ + [ ! -f "${dir}tmp_downlist" ] && return + local IPLIST=`cat ${dir}tmp_downlist|awk '{print $1}'` + for ip in $IPLIST; do + local ip_mac=`getmac ${ip}` + blackwhitelist ${ip_mac};local ip_blackwhite=$? + [ -z "$serverchan_down" ] || [ -z "$ip_blackwhite" ] && continue + [ ! -z "$serverchan_down" ] && [ "$serverchan_down" -ne "1" ] && continue + [ -z "$ip_blackwhite" ] || [ "$ip_blackwhite" -ne "0" ] && continue + [ ! -z "$serverchan_blacklist" ] && local tmp_mac=`echo "${serverchan_blacklist}"|grep -w -i ${ip_mac}` + [ ! -z "$serverchan_whitelist" ] && local tmp_mac=`echo "${serverchan_whitelist}"|grep -w -i ${ip_mac}` + [ ! -z "$tmp_mac" ] && ( cat ${dir}ipAddress|grep -q -w -i ${tmp_mac} ) && continue + local ip_name=`getname ${ip} ${ip_mac}` + local time_up=`cat ${dir}tmp_downlist|grep -w ${ip}|awk '{print $4}'|grep -v "^$"|sort -u|head -n1` + local ip_total=`usage get $ip_mac` && [ ! -z "$ip_total" ] && local ip_total="${str_linefeed}${str_tab}总计流量: ${str_space}${str_space}${str_space}${str_space}${ip_total}" + local time1=`date +%s` + local time1=$(time_for_humans `expr ${time1} - ${time_up}`) + if [ -z "$title" ]; then + title="${ip_name} 断开连接" + content="${content}${str_splitline}${str_title_start} 设备断开连接${str_title_end}${str_linefeed}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${ip}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${ip_mac}$ip_total${str_linefeed}${str_tab}在线时间: ${str_space}${str_space}${str_space}${str_space}${time1}" + elif ( echo "$title"|grep -q "断开连接" ); then + title="${ip_name} ${title}" + content="${content}${str_splitline}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${ip}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${ip_mac}$ip_total${str_linefeed}${str_tab}在线时间: ${str_space}${str_space}${str_space}${str_space}${time1}" + else + title="设备状态变化" + content="${content}${str_splitline}${str_title_start} 设备断开连接${str_title_end}${str_linefeed}${str_tab}客户端名:${str_space}${str_space}${str_space}${str_space}${str_space}${ip_name}${str_linefeed}${str_tab}客户端IP: ${str_space}${str_space}${str_space}${str_space}${ip}${str_linefeed}${str_tab}客户端MAC:${str_space}${str_space}${str_space}${str_space}${ip_mac}$ip_total${str_linefeed}${str_tab}在线时间: ${str_space}${str_space}${str_space}${str_space}${time1}" + fi + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text}设备 ${ip_name} ${ip} 断开连接 " >> ${logfile} + done + rm -f ${dir}tmp_downlist >/dev/null 2>&1 +} + +# 当前设备列表 +function current_device(){ + ( echo "$lite_enable"|grep -q "content" ) || ( echo "$lite_enable"|grep -q "device" ) && return + [ -f ${dir}ipAddress ] && local logrow=$(grep -c "" ${dir}ipAddress) || local logrow="0";[ $logrow -eq "0" ] && return + [ -f ${dir}usage.db ] && local ip_total_db="总计流量${str_space}${str_space}${str_space}${str_space}" + content="${content}${str_splitline}${str_title_start} 现有在线设备 ${logrow} 台,具体如下${str_title_end}${str_linefeed}${str_tab}IP 地址${str_space}${str_space}${str_space}${str_space}${str_space}${str_space}${str_space}${str_space}${str_space}${ip_total_db}客户端名" + local IPLIST=`cat ${dir}ipAddress|awk '{print $1}'` + for ip in $IPLIST; do + local ip_mac=`getmac ${ip}` + local ip_total=`usage get ${ip_mac}` + local ip_name=`getname ${ip} ${ip_mac}` + local ip_name=`cut_str $ip_name 15` + if [ "${#ip}" -lt "15" ]; then + local n=`expr 15 - ${#ip}` + for i in `seq 1 $n`; do + local ip="${ip}${str_space}" + done + unset i n + fi + if [ ! -z "$ip_total" ]; then + local n=`expr 11 - ${#ip_total}` + for i in `seq 1 $n`; do + local ip_total="${ip_total}${str_space}" + done + fi + content="${content}${str_linefeed}${str_tab}${ip}${ip_total}${ip_name}" + unset i n ip_total ip_mac ip_name + done +} + +# 检测 cpu 状态 +function cpu_load(){ + if [ ! -z "$temperature_enable" ] && [ "$temperature_enable" -eq "1" ] && [ ! -z "$temperature" ]; then + [ -z "$temperature_time" ] && temperature_time=`date +%s` + local cpu_wendu=`soc_temp`; + [ -z "$cpu_wendu" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】无法读取设备温度,请检查命令" >> ${logfile} + + if [ "$cpu_wendu" -gt "$temperature" ]; then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!警报!!】 CPU 温度过高: ${cpu_wendu}" >> ${logfile} + else + temperature_time=`date +%s` + fi + + if [ "$((`date +%s`-$temperature_time))" -ge "300" ] && [ -z "$temperaturecd_time" ]; then + title="CPU 温度过高!" + temperaturecd_time=`date +%s` + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text} CPU 温 度过高: ${cpu_wendu}" >> ${logfile} + content="${content}${str_splitline}${str_title_start} CPU 温度过高${str_title_end}${str_linefeed}${str_tab}CPU 温度已连续五分钟超过预设${str_linefeed}${str_tab}接下来一小 时不再提示${str_linefeed}${str_tab}当前温度:${cpu_wendu}℃" + elif [ ! -z "$temperaturecd_time" ] && [ "$((`date +%s`-$temperaturecd_time))" -ge "3300" ] ;then + unset temperaturecd_time + fi + fi + + if [ ! -z "$cpuload_enable" ] && [ "$cpuload_enable" -eq "1" ] && [ ! -z "$cpuload" ]; then + [ -z "$cpuload_time" ] && cpuload_time=`date +%s` + local cpu_fuzai=`cat /proc/loadavg|awk '{print $1}'` 2>/dev/null + [ -z "$cpu_fuzai" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】无法读取设备负载,请检查命令" >> ${logfile} + + if [ `expr $cpu_fuzai \> $cpuload` -eq "1" ]; then + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!警报!!】 CPU 负载过高: ${cpu_fuzai}" >> ${logfile} + cputop log + else + cpuload_time=`date +%s` + fi + + if [ "$((`date +%s`-$cpuload_time))" -ge "300" ] && [ -z "$cpucd_time" ]; then + unset getlogtop + if [ ! -z "$title" ] && ( echo "$title"|grep -q "过高" ); then + title="设备报警!" + else + title="CPU 负载过高!" + fi + cpucd_time=`date +%s` + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text} CPU 负 载过高: ${cpu_fuzai}" >> ${logfile} + content="${content}${str_splitline}${str_title_start} CPU 负载过高${str_title_end}${str_linefeed}${str_tab}CPU 负载已连续五分钟超过预设${str_linefeed}${str_tab}接下来一小 时不再提示${str_linefeed}${str_tab}当前负载:${cpu_fuzai}" + cputop + elif [ ! -z "$cpucd_time" ] && [ "$((`date +%s`-$cpucd_time))" -ge "3300" ] ;then + unset cpucd_time + fi + fi +} + +function cputop(){ + [ -z "$1" ] && content="${content}${str_splitline}${str_title_start} 当前 CPU 占用前三的进程${str_title_end}" + local gettop=`top -bn 1|grep -v "top -bn 1"` + for i in `seq 5 7`; do + local top_name=`echo "${gettop}"|awk 'NR=='${i}|awk '{print ($8 ~ /\/bin\/sh|\/bin\/bash/) ? $9 : $8}'` + local top_load=`echo "${gettop}"|awk 'NR=='${i}|awk '{print $7}'` + local temp_top="${top_name} ${top_load}" + [ ! -z "$1" ] && local logtop="$logtop $temp_top" + [ -z "$1" ] && content="${content}${str_linefeed}${str_tab}${temp_top}" + done + unset i + [ ! -z "$1" ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!警报!!】 CPU 占用前三: ${logtop}" >> ${logfile} +} + +# 生成日志监控文件,避免后台影响 wait 语句 +function get_syslog(){ + kill -9 `pgrep -f "logread -f -p notice"` 2>/dev/null + [ -z "$web_logged" ] && [ -z "$ssh_logged" ] && [ -z "$web_login_failed" ] && [ -z "$ssh_login_failed" ] && return + rm -f ${dir}login_monitor >/dev/null 2>&1 + +cat>${dir}get_syslog<> ${dir}login_monitor & +EOF + chmod 0755 ${dir}get_syslog && ${dir}get_syslog + rm -f ${dir}get_syslog >/dev/null 2>&1 +} + +# 登陆提醒通知 +function login_send(){ + [ -z "$web_logged" ] && [ -z "$ssh_logged" ] && [ -z "$web_login_failed" ] && [ -z "$ssh_login_failed" ] && return + [ ! -f ${dir}login_monitor ] && return + cat ${dir}login_monitor|grep -i "accepted login"|awk '{print $4" "$NF}' >> ${dir}web_login + cat ${dir}login_monitor|grep -i "Password auth succeeded\|Pubkey auth succeeded"|grep -Eo "[0-9]{2}:[0-9]{2}:[0-9]{2}.*[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}"|awk '{print $1" "$NF" "$5}' >> ${dir}ssh_login + cat ${dir}login_monitor|grep -i "failed login"|grep -Eo "[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}" >> ${dir}web_failed + cat ${dir}login_monitor|grep -i "Bad password attempt\|Login attempt for nonexistent user from"|grep -Eo "[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}" >> ${dir}ssh_failed + echo "" > ${dir}login_monitor + add_ip_black + + local login_ip_list=`cat ${dir}web_login|awk '{print $2}'|grep -v "^$"|sort -u|head -n1` + for login_ip in $login_ip_list; do + [ -z "$login_ip" ] && continue + echo "$ip_white_list"|grep -w -q "$login_ip" && continue + local web_login_time=`cat ${dir}web_login|grep -w ${login_ip}|awk '{print $1}'|grep -v "^$"|sort -u|head -n1` + local web_login_mode=`cat ${dir}web_login|grep -w ${login_ip}|awk '{print $3}'|grep -v "^$"|sort -u|head -n1` + if [ ! -z "$web_logged" ] && [ "$web_logged" -eq "1" ]; then + if [ -z "$title" ]; then + title="${login_ip} 通过 web 登陆了路由器" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}时间:${str_space}${str_space}${str_space}${str_space}${str_space}${web_login_time}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}${content_mode}" + elif ( echo "$title"|grep -q "登陆了路由器" ); then + title="${login_ip} ${title}" + content="${content}${str_splitline}${str_tab}时间:${str_space}${str_space}${str_space}${str_space}${str_space}${web_login_time}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}${content_mode}" + else + title="设备状态变化" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}时间:${str_space}${str_space}${str_space}${str_space}${str_space}${web_login_time}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}${content_mode}" + fi + fi + echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text}设备 ${login_ip} 通过 web ${web_login_mode} 登陆了路由器 " >> ${logfile} + done + echo "" > ${dir}web_login + unset login_ip login_ip_list + + local login_ip_list=`cat ${dir}ssh_login|awk '{print $2}'|grep -v "^$"|sort -u|head -n1` + for login_ip in $login_ip_list; do + [ -z "$login_ip" ] && continue + echo "$ip_white_list"|grep -w -q "$login_ip" && continue + local ssh_login_time=`cat ${dir}ssh_login|grep -w ${login_ip}|awk '{print $1}'|grep -v "^$"|sort -u|head -n1` + local ssh_login_mode=`cat ${dir}ssh_login|grep -w ${login_ip}|awk '{print $3}'|grep -v "^$"|sort -u|head -n1` + [ ! -z "$ssh_login_mode" ] && local content_mode="${str_linefeed}${str_tab}登录方式: ${str_space}${str_space}${str_space}${str_space}${ssh_login_mode}" + if [ ! -z "$ssh_logged" ] && [ "$ssh_logged" -eq "1" ]; then + if [ -z "$title" ]; then + title="${login_ip} 通过 ssh 登陆了路由器" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}时间:${str_space}${str_space}${str_space}${str_space}${str_space}${ssh_login_time}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}${content_mode}" + elif ( echo "$title"|grep -q "登陆了路由器" ); then + title="${login_ip} ${title}" + content="${content}${str_splitline}${str_tab}时间:${str_space}${str_space}${str_space}${str_space}${str_space}${ssh_login_time}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}${content_mode}" + else + title="设备状态变化" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}时间:${str_space}${str_space}${str_space}${str_space}${str_space}${ssh_login_time}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}${content_mode}" + fi + fi + echo "`date "+%Y-%m-%d %H:%M:%S"` 【info】设备 ${login_ip} 通过 ssh ${ssh_login_mode} 登陆了路由器 " >> ${logfile} + done + echo "" > ${dir}ssh_login + unset login_ip login_ip_list + + local login_ip_list=`cat ${dir}web_failed|awk '{print $1}'|grep -v "^$"|sort -u|head -n1` + for login_ip in $login_ip_list; do + [ -z "$login_ip" ] && continue + echo "$ip_white_list"|grep -w -q "$login_ip" && continue + local web_login_sum=`cat ${dir}web_failed|grep -w "${login_ip}"|wc -l` + if [ "$web_login_sum" -ge "$login_max_num" ] ;then + if [ ! -z "$web_login_failed" ] && [ "$web_login_failed" -eq "1" ]; then + if [ -z "$title" ]; then + title="${login_ip} 通过 web 频繁尝试登陆" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}" + elif ( echo "$title"|grep -q "频繁尝试登陆" ); then + title="${login_ip} ${title}" + content="${content}${str_splitline}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}" + else + title="设备状态变化" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}" + fi + fi + sed -i "/^${login_ip}$/d" ${dir}web_failed + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】设备 ${login_ip} 通过 web 频繁尝试登陆" >> ${logfile} + add_ip_black $login_ip + fi + done + unset login_ip + + local login_ip_list=`cat ${dir}ssh_failed|awk '{print $1}'|grep -v "^$"|sort -u|head -n1` + for login_ip in $login_ip_list; do + [ -z "$login_ip" ] && continue + echo "$ip_white_list"|grep -w -q "$login_ip" && continue + local ssh_login_sum=`cat ${dir}ssh_failed|grep -w "${login_ip}"|wc -l` + if [ "$ssh_login_sum" -ge "$login_max_num" ] ;then + if [ ! -z "$ssh_login_failed" ] && [ "$ssh_login_failed" -eq "1" ]; then + if [ -z "$title" ]; then + title="${login_ip} 通过 ssh 频繁尝试登陆" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}" + elif ( echo "$title"|grep -q "频繁尝试登陆" ); then + title="${login_ip} ${title}" + content="${content}${str_splitline}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}" + else + title="设备状态变化" + content="${content}${str_splitline}${str_title_start} 登陆信息${str_title_end}${str_linefeed}${str_tab}设备 IP: ${str_space}${str_space}${str_space}${str_space}${login_ip}" + fi + fi + sed -i "/^${login_ip}$/d" ${dir}ssh_failed + echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】设备 ${login_ip} 通过 ssh 频繁尝试登陆" >> ${logfile} + add_ip_black $login_ip + fi + done + unset login_ip + +} + +# 添加黑名单 +function add_ip_black(){ + [ -f "${ip_blacklist_path}" ] && local logrow=$(grep -c "" ${ip_blacklist_path}) || local logrow="0" + [ ! -f "${ip_blacklist_path}" ] && local logrow="0" + [ ! -z "$web_login_black" ] && [ "$web_login_black" -eq "0" ] || [ -z "$web_login_black" ] && local logrow="0" + ipset flush ip_blacklist >/dev/null 2>&1 + + if [ $logrow -le "0" ]; then + iptables -D INPUT -m set --match-set ip_blacklist src -j DROP >/dev/null 2>&1 + ipset destroy ip_blacklist >/dev/null 2>&1 + return + fi + + ipset list ip_blacklist >/dev/null 2>&1 || ipset create ip_blacklist hash:ip timeout ${ip_black_timeout} >/dev/null 2>&1 + iptables -C INPUT -m set --match-set ip_blacklist src -j DROP >/dev/null 2>&1 || iptables -I INPUT -m set --match-set ip_blacklist src -j DROP >/dev/null 2>&1 + echo "$1" >> ${ip_blacklist_path} + for ip_black in `cat ${ip_blacklist_path}`; do + ip_black=`echo "$ip_black"|grep -Eo "[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}"` + ipset -! add ip_blacklist $ip_black >/dev/null 2>&1 + done + ipset list ip_blacklist|grep -Eo "[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}" > ${ip_blacklist_path} +} + + +# 发送定时数据 +function send(){ + echo "`date "+%Y-%m-%d %H:%M:%S"` 【定时数据】创建定时任务" >> ${logfile} + serverchan_disturb;local send_disturb=$? + get_config "send_title" "router_status" "router_temp" "router_wan" "client_list" + + [ -z "$send_title" ] && local send_title="路由状态:" + [ ! -z "$1" ] && local send_title="发送测试:" && local send_content="${str_splitline}${str_title_start}内容1${str_title_end}${str_linefeed}${str_tab}设备1${str_linefeed}${str_tab}设备2${str_splitline}${str_title_start}内容2${str_title_end}${str_linefeed}${str_tab}设备3${str_linefeed}${str_tab}设备4" + [ -z "$1" ] && [ ! -z "$client_list" ] && [ "$client_list" -eq "1" ] && > ${dir}send_enable.lock && serverchan_first & + + if [ -z "$1" ] && [ ! -z "$router_status" ] && [ "$router_status" -eq "1" ]; then + local systemload=`cat /proc/loadavg|awk '{print $1" "$2" "$3}'` + local cpuload=`getcpu` + local ramload=`free -m|sed -n '2p'|awk '{printf "%.2f%%\n",($3/$2)*100}'` + local systemstatustime=`cat /proc/uptime|awk -F. '{run_days=$1 / 86400;run_hour=($1 % 86400)/3600;run_minute=($1 % 3600)/60;run_second=$1 % 60;printf("运行时间:%d天%d时%d分%d秒",run_days,run_hour,run_minute,run_second)}'`;unset run_days run_hour run_minute run_second + local send_content="${send_content}${str_splitline}${str_title_start} 系统运行状态"${str_title_end} + local send_content="${send_content}${str_linefeed}${str_tab}平均负载:${systemload}" + local send_content="${send_content}${str_linefeed}${str_tab}CPU占用:${cpuload}" + local send_content="${send_content}${str_linefeed}${str_tab}内存占用:${ramload}" + local send_content="${send_content}${str_linefeed}${str_tab}${systemstatustime}" + fi + + if [ -z "$1" ] && [ ! -z "$router_temp" ] && [ "$router_temp" -eq "1" ]; then + local cputemp=`soc_temp` + [ ! -z "$cputemp" ] && local send_content="${send_content}${str_splitline}${str_title_start} 设备温度${str_title_end}${str_linefeed}${str_tab}CPU:${cputemp}℃" + [ -z "$cputemp" ] && local send_content="${send_content}${str_splitline}${str_title_start} 设备温度${str_title_end}${str_linefeed}${str_tab}无法获取设备温度" + fi + + if [ -z "$1" ] && [ ! -z "$router_wan" ] && [ "$router_wan" -eq "1" ]; then + local send_wanIP=`getip wanipv4`;local send_hostIP=`getip hostipv4` + local send_content="${send_content}${str_splitline}${str_title_start} WAN 口信息${str_title_end}${str_linefeed}${str_tab}接口ip:${send_wanIP}" + local send_content="${send_content}${str_linefeed}${str_tab}外网ip:${send_hostIP}" + if [ ! -z "$serverchan_ipv6" ] && [ "$serverchan_ipv6" -ne "0" ]; then + local send_wanIPv6=`getip wanipv6`;local send_hostIPv6=`getip hostipv6` + local send_content="${send_content}${str_linefeed}${str_tab}ipv6 :${send_wanIPv6}" + local send_content="${send_content}${str_linefeed}${str_tab}外网v6:${send_hostIPv6}" + fi + ( ! echo "$send_wanIP"|grep -q -w ${send_hostIP} ) && local send_content="${send_content}${str_linefeed}${str_tab}外网 ip 与接口 ip 不一致,你的 ip 不是公网 ip" + local interfaceuptime=`getinterfaceuptime` + [ ! -z "$interfaceuptime" ] && local wanstatustime=`getinterfaceuptime|awk -F. '{run_days=$1 / 86400;run_hour=($1 % 86400)/3600;run_minute=($1 % 3600)/60;run_second=$1 % 60;printf("在线时间:%d天%d时%d分%d秒",run_days,run_hour,run_minute,run_second)}'` && unset run_days run_hour run_minute run_second + local send_content="${send_content}${str_linefeed}${str_tab}${wanstatustime}" + fi + + if [ -z "$1" ] && [ ! -z "$client_list" ] && [ "$client_list" -eq "1" ]; then + wait + local IPLIST=`cat ${dir}ipAddress 2>/dev/null|awk '{print $1}'` + [ -f ${dir}ipAddress ] && local logrow=$(grep -c "" ${dir}ipAddress) || local logrow="0" + [ "$logrow" -eq "0" ] && local send_content="${send_content}${str_splitline}${str_title_start} 当前无在线设备${str_title_end}" || local send_content="${send_content}${str_splitline}${str_title_start} 现有在线设备 ${logrow} 台${str_title_end}" + for ip in $IPLIST; do + local time_up=`cat ${dir}ipAddress|grep -w ${ip}|awk '{print $4}'|grep -v "^$"|sort -u|head -n1` + local time1=`date +%s` + local time1=$(time_for_humans `expr ${time1} - ${time_up}`) + local ip_mac=`getmac ${ip}` + local ip_name=`getname ${ip} ${ip_mac}` + local ip_total=`usage get ${ip_mac}`;[ ! -z "$ip_total" ] && local ip_total="总计流量:${ip_total} " + local ip_name=`cut_str $ip_name 18` + local send_content="${send_content}${str_linefeed}${str_tab}【${ip_name}】 ${ip}${str_linefeed}${str_tab}${ip_total}在线 ${time1}" + unset ip_total time_down time_up time1 ip_mac ip_name + done + fi + [ ! -z "$device_name" ] && local send_title="【$device_name】${send_title}" + [ -z "$send_content" ] && local send_content="${str_splitline}${str_title_start} 我遇到了一个难题${str_title_end}${str_linefeed}${str_tab}定时发送选项错误,你没有选择需要发送的项目,该怎 么办呢${str_splitline}" + [ "$send_disturb" -eq "0" ] && diy_send "${send_title}" "${send_content}" "${jsonpath}" >/dev/null 2>&1 + [ $? -eq 1 ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】定时推送失败,请检查网络或设置信息" >> ${logfile} || echo "`date "+%Y-%m-%d %H:%M:%S"` ${disturb_text}定时推送任务完成" >> ${logfile} + deltemp +} + +# 初始化 +read_config +deltemp +serverchan_cron + +# 限制并发进程 +[ -z "$thread_num" ] || [ "$thread_num" -eq "0" ] && thread_num=5 +[ -e ${dir}fd1 ] || mkfifo ${dir}fd1 +exec 5<>${dir}fd1 +rm -f ${dir}fd1 >/dev/null 2>&1 +for i in `seq 1 $thread_num`; do + echo >&5 +done +unset i + +# 启动参数 +if [ "$1" ] ;then + [ $1 == "send" ] && send + [ $1 == "soc" ] && echo `soc_temp` > ${dir}soc_tmp + [ $1 == "client" ] && get_client + [ $1 == "test" ] && send test + exit +fi + +# 载入在线设备 +serverchan_init;[ $? -eq 1 ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】读取设置出错,请检查设置项 " >> ${logfile} && exit +echo "`date "+%Y-%m-%d %H:%M:%S"` 【初始化】载入在线设备" >> ${logfile} +> ${dir}send_enable.lock && serverchan_first && deltemp +echo "`date "+%Y-%m-%d %H:%M:%S"` 【初始化】初始化完成" >> ${logfile} + +# 循环 +while [ "$serverchan_enable" -eq "1" ]; do + deltemp + usage update + serverchan_disturb;disturb=$? + + [ -f ${dir}ipAddress ] && ipAddress_logrow=$(grep -c "" ${dir}ipAddress) || ipAddress_logrow="0"; + if [ $ipAddress_logrow -ne "0" ]; then + online_list=`cat ${dir}ipAddress|awk '{print $2}'|grep -v "^$"|sort -u` + for online_mac in $online_list; do + [ ! -z "$online_mac" ] && mac_online_status="`echo "$mark_mac_list"|grep -i $online_mac|grep -v "^$"|sort -u|head -n1`${mac_online_status}" + done + fi + + if [ "$serverchan_ipv4" -ne "0" ] || [ "$serverchan_ipv6" -ne "0" ]; then + rand_geturl + ip_changes + fi + + # 设备列表 + if [ ! -f "${dir}send_enable.lock" ]; then + [ ! -z "$title" ] && echo "$title" > ${dir}title + [ ! -z "$content" ] && echo "$content" > ${dir}content + serverchan_first + [ -f "${dir}title" ] && title=`cat ${dir}title` && rm -f ${dir}title >/dev/null 2>&1 + [ -f "${dir}content" ] && content=`cat ${dir}content` && rm -f ${dir}content >/dev/null 2>&1 + fi + + # 离线缓存区推送 + [ ! -f "${dir}send_enable.lock" ] && down_send + + # 当前设备列表 + [ ! -z "$content" ] && [ ! -f "${dir}send_enable.lock" ] && current_device + + # 无人值守任务 + [ ! -f "${dir}send_enable.lock" ] && unattended + + # CPU 检测 + [ ! -f "${dir}send_enable.lock" ] && cpu_load + + # 异常流量检测 + [ ! -f "${dir}send_enable.lock" ] && get_client_usage + + # 登陆提醒通知 + [ ! -f "${dir}send_enable.lock" ] && login_send + + if [ ! -f "${dir}send_enable.lock" ] && [ ! -z "$title" ] && [ ! -z "$content" ]; then + [ ! -z "$device_name" ] && title="【$device_name】$title" + ( echo "$lite_enable"|grep -q "content" ) && content="$title" + [ "$disturb" -eq "0" ] && diy_send "${title}" "${content}" "${jsonpath}" >/dev/null 2>&1 + [ $? -eq 1 ] && echo "`date "+%Y-%m-%d %H:%M:%S"` 【!!!】推送失败,请检查网络或设置信息 " >> ${logfile} + fi + + while [ -f "${dir}send_enable.lock" ]; do + sleep $sleeptime + done + sleep $sleeptime +done diff --git a/package/lean/luci-app-serverchan/root/usr/share/rpcd/acl.d/luci-app-serverchan.json b/package/lean/luci-app-serverchan/root/usr/share/rpcd/acl.d/luci-app-serverchan.json new file mode 100644 index 000000000..200ccd56f --- /dev/null +++ b/package/lean/luci-app-serverchan/root/usr/share/rpcd/acl.d/luci-app-serverchan.json @@ -0,0 +1,11 @@ +{ + "luci-app-serverchan": { + "description": "Grant UCI access for luci-app-serverchan", + "read": { + "uci": [ "serverchan" ] + }, + "write": { + "uci": [ "serverchan" ] + } + } +}