| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274 | 
							- #!/usr/bin/lua
 
- ------------------------------------------------
 
- -- This file is part of the luci-app-ssr-plus update.lua
 
- -- By Mattraks
 
- ------------------------------------------------
 
- require "luci.sys"
 
- require "luci.model.uci"
 
- local icount = 0
 
- local args = arg[1]
 
- local uci = luci.model.uci.cursor()
 
- -- 以下设置更新数据库至 DNSMASQ 配置路径
 
- -- 获取 DNSMASQ 配置 ID
 
- local DNSMASQ_UCI_CONFIG = uci:get_first("dhcp", "dnsmasq", ".name")
 
- -- 获取 DNSMASQ 默认配置文件
 
- local DNSMASQ_CONF_PATH = "/tmp/etc/dnsmasq.conf." .. DNSMASQ_UCI_CONFIG
 
- -- 检查 DNSMASQ 配置文件是否存在,如果存在则提取 conf-dir
 
- for line in io.lines(DNSMASQ_CONF_PATH) do
 
-     local conf_dir = line:match("^conf%-dir=(.+)")
 
-     if conf_dir then
 
-         DNSMASQ_CONF_DIR = conf_dir:gsub("%s+", "") -- 去除空白字符
 
-         break
 
-     end
 
- end
 
- -- 设置 dnsmasq-ssrplus.d 目录路径,并去除路径末尾的斜杠
 
- local TMP_DNSMASQ_PATH = DNSMASQ_CONF_DIR:match("^(.-)/?$") .. "/dnsmasq-ssrplus.d"
 
- local TMP_PATH = "/var/etc/ssrplus"
 
- -- match comments/title/whitelist/ip address/excluded_domain
 
- local comment_pattern = "^[!\\[@]+"
 
- local ip_pattern = "^%d+%.%d+%.%d+%.%d+"
 
- local domain_pattern = "([%w%-%_]+%.[%w%.%-%_]+)[%/%*]*"
 
- local excluded_domain = {
 
-     "apple.com", "sina.cn", "sina.com.cn", "baidu.com", "byr.cn", "jlike.com", 
 
-     "weibo.com", "zhongsou.com", "youdao.com", "sogou.com", "so.com", "soso.com", 
 
-     "aliyun.com", "taobao.com", "jd.com", "qq.com"
 
- }
 
- -- gfwlist parameter
 
- local mydnsip = '127.0.0.1'
 
- local mydnsport = '5335'
 
- local ipsetname = 'gfwlist'
 
- local new_appledns = uci:get_first("shadowsocksr", "global", "apple_dns")
 
- local bc = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'
 
- -- base64decoding
 
- local function base64_dec(data)
 
- 	data = string.gsub(data, '[^' .. bc .. '=]', '')
 
- 	return (data:gsub('.', function(x)
 
- 		if (x == '=') then
 
- 			return ''
 
- 		end
 
- 		local r, f = '', (bc:find(x) - 1)
 
- 		for i = 6, 1, -1 do
 
- 			r = r .. (f % 2 ^ i - f % 2 ^ (i - 1) > 0 and '1' or '0')
 
- 		end
 
- 		return r;
 
- 	end):gsub('%d%d%d?%d?%d?%d?%d?%d?', function(x)
 
- 		if (#x ~= 8) then
 
- 			return ''
 
- 		end
 
- 		local c = 0
 
- 		for i = 1, 8 do
 
- 			c = c + (x:sub(i, i) == '1' and 2 ^ (8 - i) or 0)
 
- 		end
 
- 		return string.char(c)
 
- 	end))
 
- end
 
- -- check if domain is excluded
 
- local function check_excluded_domain(value)
 
- 	for _, domain in ipairs(excluded_domain) do
 
- 		if value:find(domain) then
 
- 			return true
 
- 		end
 
- 	end
 
- end
 
- -- gfwlist转码至dnsmasq格式
 
- local function generate_gfwlist(type)
 
-     local domains, domains_map = {}, {}
 
-     local out = io.open("/tmp/ssr-update." .. type, "w")
 
-     for line in io.lines("/tmp/ssr-update.tmp") do
 
-         if not (string.find(line, comment_pattern) or string.find(line, ip_pattern) or check_excluded_domain(line)) then
 
-             local start, finish, match = string.find(line, domain_pattern)
 
-             if start and not domains_map[match] then
 
-                 domains_map[match] = true
 
-                 table.insert(domains, match)
 
-             end
 
-         end
 
-     end
 
-     for _, domain in ipairs(domains) do
 
-         out:write(string.format("server=/%s/%s#%s\n", domain, mydnsip, mydnsport))
 
-         out:write(string.format("ipset=/%s/%s\n", domain, ipsetname))
 
-     end
 
-     out:close()
 
-     os.remove("/tmp/ssr-update.tmp")
 
- end
 
- -- 更换 Apple dns
 
- local function generate_apple(type)
 
- 	local domains, domains_map = {}, {}
 
- 	local out = io.open("/tmp/ssr-update." .. type, "w")
 
- 	for line in io.lines("/tmp/ssr-update.tmp") do
 
- 		if not (string.find(line, comment_pattern)) then
 
- 			local start, finish, match = string.find(line, domain_pattern)
 
- 			if start and not domains_map[match] then
 
- 				domains_map[match] = true
 
- 				match = string.gsub(match, "%s", "") --从域名中去除所有空白字符
 
- 				table.insert(domains, match)
 
- 			end
 
- 		end
 
- 	end
 
- 	for _, domain in ipairs(domains) do
 
-         if new_appledns and new_appledns ~= "" then
 
-             out:write(string.format("server=/%s/%s\n", domain, new_appledns))
 
-         end
 
- 	end
 
- 	out:close()
 
- 	os.remove("/tmp/ssr-update.tmp")
 
- end
 
- -- adblock转码至dnsmasq格式
 
- local function generate_adblock(type)
 
- 	local domains, domains_map = {}, {}
 
- 	local out = io.open("/tmp/ssr-update." .. type, "w")
 
- 	for line in io.lines("/tmp/ssr-update.tmp") do
 
- 		if not (string.find(line, comment_pattern)) then
 
- 			local start, finish, match = string.find(line, domain_pattern)
 
- 			if start and not domains_map[match] then
 
- 				domains_map[match] = true
 
- 				table.insert(domains, match)
 
- 			end
 
- 		end
 
- 	end
 
- 	for _, domain in ipairs(domains) do
 
- 		out:write(string.format("address=/%s/\n", domain))
 
- 	end
 
- 	out:close()
 
- 	os.remove("/tmp/ssr-update.tmp")
 
- end
 
- local log = function(...)
 
- 	if args then
 
- 		print("{ret=" .. table.concat({...}, ",retcount=") .. "}")
 
- 	else
 
- 		print(os.date("%Y-%m-%d %H:%M:%S ") .. table.concat({...}, " "))
 
- 	end
 
- end
 
- local function update(url, file, type, file2)
 
- 	local Num = 1
 
- 	local refresh_cmd = "wget --no-check-certificate -q -O /tmp/ssr-update." .. type .. " " .. url
 
- 	local sret = luci.sys.call(refresh_cmd)
 
- 	if sret == 0 then
 
- 		if type == "gfw_data" then
 
- 			local gfwlist = io.open("/tmp/ssr-update." .. type, "r")
 
- 			local decode = gfwlist:read("*a")
 
- 			if not decode:find("google") then
 
- 				decode = base64_dec(decode)
 
- 			end
 
- 			gfwlist:close()
 
- 			-- 写回gfwlist
 
- 			gfwlist = io.open("/tmp/ssr-update.tmp", "w")
 
- 			gfwlist:write(decode)
 
- 			gfwlist:close()
 
- 			generate_gfwlist(type)
 
- 			Num = 2
 
- 		end
 
- 		if type == "apple_data" then
 
- 			local apple = io.open("/tmp/ssr-update." .. type, "r")
 
- 			local decode = apple:read("*a")
 
- 			if not decode:find("apple") then
 
- 				decode = base64_dec(decode)
 
- 			end
 
- 			apple:close()
 
- 			-- 写回applechina
 
- 			apple = io.open("/tmp/ssr-update.tmp", "w")
 
- 			apple:write(decode)
 
- 			apple:close()
 
- 			if new_appledns and new_appledns ~= "" then
 
- 				generate_apple(type)
 
- 			end
 
- 		end
 
- 		if type == "ad_data" then
 
- 			local adblock = io.open("/tmp/ssr-update." .. type, "r")
 
- 			local decode = adblock:read("*a")
 
- 			if decode:find("address=") then
 
- 				adblock:close()
 
- 			else
 
- 				adblock:close()
 
- 				-- 写回adblock
 
- 				adblock = io.open("/tmp/ssr-update.tmp", "w")
 
- 				adblock:write(decode)
 
- 				adblock:close()
 
- 				generate_adblock(type)
 
- 			end
 
- 		end
 
- 		local new_md5 = luci.sys.exec("echo -n $([ -f '/tmp/ssr-update." .. type .. "' ] && md5sum /tmp/ssr-update." .. type .. " | awk '{print $1}')")
 
- 		local old_md5 = luci.sys.exec("echo -n $([ -f '" .. file .. "' ] && md5sum " .. file .. " | awk '{print $1}')")
 
- 		if new_md5 == old_md5 then
 
- 			if args then
 
- 				log(1)
 
- 			else
 
- 				log("你已经是最新数据,无需更新!")
 
- 			end
 
- 		else
 
- 			icount = luci.sys.exec("cat /tmp/ssr-update." .. type .. " | wc -l")
 
- 			luci.sys.exec("cp -f /tmp/ssr-update." .. type .. " " .. file)
 
- 			if file2 then
 
- 				luci.sys.exec("cp -f /tmp/ssr-update." .. type .. " " .. file2)
 
- 			end
 
- 			if type == "gfw_data" or type == "ad_data" then
 
- 				luci.sys.call("/usr/share/shadowsocksr/gfw2ipset.sh")
 
- 			else
 
- 				luci.sys.call("/usr/share/shadowsocksr/chinaipset.sh " .. TMP_PATH .. "/china_ssr.txt")
 
- 			end
 
- 			if args then
 
- 				log(0, tonumber(icount) / Num)
 
- 			else
 
- 				log("更新成功! 新的总记录数:" .. tostring(tonumber(icount) / Num))
 
- 			end
 
- 		end
 
- 	else
 
- 		if args then
 
- 			log(-1)
 
- 		else
 
- 			log("更新失败!")
 
- 		end
 
- 	end
 
- 	os.remove("/tmp/ssr-update." .. type)
 
- end
 
- if args then
 
- 	if args == "gfw_data" then
 
- 		update(uci:get_first("shadowsocksr", "global", "gfwlist_url"), "/etc/ssrplus/gfw_list.conf", args, TMP_DNSMASQ_PATH .. "/gfw_list.conf")
 
- 		os.exit(0)
 
- 	end
 
- 	if args == "ip_data" then
 
- 		update(uci:get_first("shadowsocksr", "global", "chnroute_url"), "/etc/ssrplus/china_ssr.txt", args, TMP_PATH .. "/china_ssr.txt")
 
- 		os.exit(0)
 
- 	end
 
- 	if args == "apple_data" then
 
- 		update(uci:get_first("shadowsocksr", "global", "apple_url"), "/etc/ssrplus/applechina.conf", args, TMP_DNSMASQ_PATH .. "/applechina.conf")
 
- 		os.exit(0)
 
- 	end
 
- 	if args == "ad_data" then
 
- 		update(uci:get_first("shadowsocksr", "global", "adblock_url"), "/etc/ssrplus/ad.conf", args, TMP_DNSMASQ_PATH .. "/ad.conf")
 
- 		os.exit(0)
 
- 	end
 
- 	if args == "nfip_data" then
 
- 		update(uci:get_first("shadowsocksr", "global", "nfip_url"), "/etc/ssrplus/netflixip.list", args, TMP_DNSMASQ_PATH .. "/netflixip.list")
 
- 		os.exit(0)
 
- 	end
 
- else
 
- 	log("正在更新【GFW列表】数据库")
 
- 	update(uci:get_first("shadowsocksr", "global", "gfwlist_url"), "/etc/ssrplus/gfw_list.conf", "gfw_data", TMP_DNSMASQ_PATH .. "/gfw_list.conf")
 
- 	log("正在更新【国内IP段】数据库")
 
- 	update(uci:get_first("shadowsocksr", "global", "chnroute_url"), "/etc/ssrplus/china_ssr.txt", "ip_data", TMP_PATH .. "/china_ssr.txt")
 
- 	if uci:get_first("shadowsocksr", "global", "apple_optimization", "0") == "1" then
 
- 		log("正在更新【Apple域名】数据库")
 
- 		update(uci:get_first("shadowsocksr", "global", "apple_url"), "/etc/ssrplus/applechina.conf", "apple_data", TMP_DNSMASQ_PATH .. "/applechina.conf")
 
- 	end
 
- 	if uci:get_first("shadowsocksr", "global", "adblock", "0") == "1" then
 
- 		log("正在更新【广告屏蔽】数据库")
 
- 		update(uci:get_first("shadowsocksr", "global", "adblock_url"), "/etc/ssrplus/ad.conf", "ad_data", TMP_DNSMASQ_PATH .. "/ad.conf")
 
- 	end
 
- 	if uci:get_first("shadowsocksr", "global", "netflix_enable", "0") == "1" then
 
- 		log("正在更新【Netflix IP段】数据库")
 
- 		update(uci:get_first("shadowsocksr", "global", "nfip_url"), "/etc/ssrplus/netflixip.list", "nfip_data", TMP_DNSMASQ_PATH .. "/netflixip.list")
 
- 	end
 
- 	-- log("正在更新【Netflix IP段】数据库")
 
- 	-- update(uci:get_first("shadowsocksr", "global", "nfip_url"), "/etc/ssrplus/netflixip.list", "nfip_data")
 
- end
 
 
  |