123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168 |
- -- Copyright (C) 2017 yushi studio <[email protected]>
- -- Licensed to the public under the GNU General Public License v3.
- module("luci.controller.shadowsocksr", package.seeall)
- function index()
- if not nixio.fs.access("/etc/config/shadowsocksr") then
- call("act_reset")
- end
- local page
- page = entry({"admin", "services", "shadowsocksr"}, alias("admin", "services", "shadowsocksr", "client"), _("ShadowSocksR Plus+"), 10)
- page.dependent = true
- page.acl_depends = { "luci-app-ssr-plus" }
- entry({"admin", "services", "shadowsocksr", "client"}, cbi("shadowsocksr/client"), _("SSR Client"), 10).leaf = true
- entry({"admin", "services", "shadowsocksr", "servers"}, arcombine(cbi("shadowsocksr/servers", {autoapply = true}), cbi("shadowsocksr/client-config")), _("Servers Nodes"), 20).leaf = true
- entry({"admin", "services", "shadowsocksr", "control"}, cbi("shadowsocksr/control"), _("Access Control"), 30).leaf = true
- entry({"admin", "services", "shadowsocksr", "advanced"}, cbi("shadowsocksr/advanced"), _("Advanced Settings"), 50).leaf = true
- entry({"admin", "services", "shadowsocksr", "server"}, arcombine(cbi("shadowsocksr/server"), cbi("shadowsocksr/server-config")), _("SSR Server"), 60).leaf = true
- entry({"admin", "services", "shadowsocksr", "status"}, form("shadowsocksr/status"), _("Status"), 70).leaf = true
- entry({"admin", "services", "shadowsocksr", "check"}, call("check_status"))
- entry({"admin", "services", "shadowsocksr", "refresh"}, call("refresh_data"))
- entry({"admin", "services", "shadowsocksr", "subscribe"}, call("subscribe"))
- entry({"admin", "services", "shadowsocksr", "checkport"}, call("check_port"))
- entry({"admin", "services", "shadowsocksr", "log"}, form("shadowsocksr/log"), _("Log"), 80).leaf = true
- entry({"admin", "services", "shadowsocksr", "get_log"}, call("get_log")).leaf = true
- entry({"admin", "services", "shadowsocksr", "clear_log"}, call("clear_log")).leaf = true
- entry({"admin", "services", "shadowsocksr", "run"}, call("act_status"))
- entry({"admin", "services", "shadowsocksr", "ping"}, call("act_ping"))
- entry({"admin", "services", "shadowsocksr", "reset"}, call("act_reset"))
- entry({"admin", "services", "shadowsocksr", "restart"}, call("act_restart"))
- entry({"admin", "services", "shadowsocksr", "delete"}, call("act_delete"))
- --[[Backup]]
- entry({"admin", "services", "shadowsocksr", "backup"}, call("create_backup")).leaf = true
- end
- function subscribe()
- luci.sys.call("/usr/bin/lua /usr/share/shadowsocksr/subscribe.lua >>/var/log/ssrplus.log")
- luci.http.prepare_content("application/json")
- luci.http.write_json({ret = 1})
- end
- function act_status()
- local e = {}
- e.running = luci.sys.call("busybox ps -w | grep ssr-retcp | grep -v grep >/dev/null") == 0
- luci.http.prepare_content("application/json")
- luci.http.write_json(e)
- end
- function act_ping()
- local e = {}
- local domain = luci.http.formvalue("domain")
- local port = luci.http.formvalue("port")
- local transport = luci.http.formvalue("transport")
- local wsPath = luci.http.formvalue("wsPath")
- local tls = luci.http.formvalue("tls")
- e.index = luci.http.formvalue("index")
- local iret = luci.sys.call("ipset add ss_spec_wan_ac " .. domain .. " 2>/dev/null")
- if transport == "ws" then
- local prefix = tls=='1' and "https://" or "http://"
- local address = prefix..domain..':'..port..wsPath
- local result = luci.sys.exec("curl --http1.1 -m 2 -ksN -o /dev/null -w 'time_connect=%{time_connect}\nhttp_code=%{http_code}' -H 'Connection: Upgrade' -H 'Upgrade: websocket' -H 'Sec-WebSocket-Key: SGVsbG8sIHdvcmxkIQ==' -H 'Sec-WebSocket-Version: 13' "..address)
- e.socket = string.match(result,"http_code=(%d+)")=="101"
- e.ping = tonumber(string.match(result, "time_connect=(%d+.%d%d%d)"))*1000
- else
- local socket = nixio.socket("inet", "stream")
- socket:setopt("socket", "rcvtimeo", 3)
- socket:setopt("socket", "sndtimeo", 3)
- e.socket = socket:connect(domain, port)
- socket:close()
- e.ping = luci.sys.exec(string.format("echo -n $(tcping -q -c 1 -i 1 -t 2 -p %s %s 2>&1 | grep -o 'time=[0-9]*' | awk -F '=' '{print $2}') 2>/dev/null", port, domain))
- if (e.ping == "") then
- e.ping = luci.sys.exec("echo -n $(ping -c 1 -W 1 %q 2>&1 | grep -o 'time=[0-9]*' | awk -F '=' '{print $2}') 2>/dev/null" % domain)
- if (e.ping == "") then
- -- UDP ping test using nping
- e.ping = luci.sys.exec(string.format("echo -n $(nping --udp -c 1 -p %s %s 2>/dev/null | grep -o 'Avg rtt: [0-9.]*ms' | awk '{print $3}' | sed 's/ms//' | head -1) 2>/dev/null", port, domain))
- end
- end
- end
- if (iret == 0) then
- luci.sys.call(" ipset del ss_spec_wan_ac " .. domain)
- end
- luci.http.prepare_content("application/json")
- luci.http.write_json(e)
- end
- function check_status()
- local e = {}
- e.ret = luci.sys.call("/usr/bin/ssr-check www." .. luci.http.formvalue("set") .. ".com 80 3 1")
- luci.http.prepare_content("application/json")
- luci.http.write_json(e)
- end
- function refresh_data()
- local set = luci.http.formvalue("set")
- local retstring = loadstring("return " .. luci.sys.exec("/usr/bin/lua /usr/share/shadowsocksr/update.lua " .. set))()
- luci.http.prepare_content("application/json")
- luci.http.write_json(retstring)
- end
- function check_port()
- local retstring = "<br /><br />"
- local s
- local server_name = ""
- local uci = require "luci.model.uci".cursor()
- local iret = 1
- uci:foreach("shadowsocksr", "servers", function(s)
- if s.alias then
- server_name = s.alias
- elseif s.server and s.server_port then
- server_name = "%s:%s" % {s.server, s.server_port}
- end
- iret = luci.sys.call("ipset add ss_spec_wan_ac " .. s.server .. " 2>/dev/null")
- socket = nixio.socket("inet", "stream")
- socket:setopt("socket", "rcvtimeo", 3)
- socket:setopt("socket", "sndtimeo", 3)
- ret = socket:connect(s.server, s.server_port)
- if tostring(ret) == "true" then
- socket:close()
- retstring = retstring .. "<font><b style='color:green'>[" .. server_name .. "] OK.</b></font><br />"
- else
- retstring = retstring .. "<font><b style='color:red'>[" .. server_name .. "] Error.</b></font><br />"
- end
- if iret == 0 then
- luci.sys.call("ipset del ss_spec_wan_ac " .. s.server)
- end
- end)
- luci.http.prepare_content("application/json")
- luci.http.write_json({ret = retstring})
- end
- function act_reset()
- luci.sys.call("/etc/init.d/shadowsocksr reset >/dev/null 2>&1")
- luci.http.redirect(luci.dispatcher.build_url("admin", "services", "shadowsocksr"))
- end
- function act_restart()
- luci.sys.call("/etc/init.d/shadowsocksr restart &")
- luci.http.redirect(luci.dispatcher.build_url("admin", "services", "shadowsocksr"))
- end
- function act_delete()
- luci.sys.call("/etc/init.d/shadowsocksr restart &")
- luci.http.redirect(luci.dispatcher.build_url("admin", "services", "shadowsocksr", "servers"))
- end
- function get_log()
- luci.http.write(luci.sys.exec("[ -f '/var/log/ssrplus.log' ] && cat /var/log/ssrplus.log"))
- end
-
- function clear_log()
- luci.sys.call("echo '' > /var/log/ssrplus.log")
- end
- function create_backup()
- local backup_files = {
- "/etc/config/shadowsocksr",
- "/etc/ssrplus/*"
- }
- local date = os.date("%Y-%m-%d-%H-%M-%S")
- local tar_file = "/tmp/shadowsocksr-" .. date .. "-backup.tar.gz"
- nixio.fs.remove(tar_file)
- local cmd = "tar -czf " .. tar_file .. " " .. table.concat(backup_files, " ")
- luci.sys.call(cmd)
- luci.http.header("Content-Disposition", "attachment; filename=shadowsocksr-" .. date .. "-backup.tar.gz")
- luci.http.header("X-Backup-Filename", "shadowsocksr-" .. date .. "-backup.tar.gz")
- luci.http.prepare_content("application/octet-stream")
- luci.http.write(nixio.fs.readfile(tar_file))
- nixio.fs.remove(tar_file)
- end
|