77efe047b489417447645f395d0efe74a521f41a
[project/luci.git] / applications / luci-app-shadowsocks-libev / luasrc / model / shadowsocks-libev.lua
1 -- Copyright 2017 Yousong Zhou <yszhou4tech@gmail.com>
2 -- Licensed to the public under the Apache License 2.0.
3
4 local _up = getfenv(3)
5 local ut = require("luci.util")
6 local sys = require("luci.sys")
7 local ds = require("luci.dispatcher")
8 local nw = require("luci.model.network")
9 nw.init()
10 module("luci.model.shadowsocks-libev", function(m)
11 setmetatable(m, {__index=function (self, k)
12 local tb = _up
13 return rawget(self, k) or _up[k]
14 end})
15 end)
16
17 function values_actions(o)
18 o:value("bypass")
19 o:value("forward")
20 if o.option ~= "dst_default" then
21 o:value("checkdst")
22 end
23 end
24
25 function values_redir(o, xmode)
26 o.map.uci.foreach("shadowsocks-libev", "ss_redir", function(sdata)
27 local disabled = ucival_to_bool(sdata["disabled"])
28 local sname = sdata[".name"]
29 local mode = sdata["mode"] or "tcp_only"
30 if not disabled and mode:find(xmode) then
31 local desc = "%s - %s" % {sname, mode}
32 o:value(sname, desc)
33 end
34 end)
35 o:value("", "<unset>")
36 o.default = ""
37 end
38
39 function values_serverlist(o)
40 o.map.uci.foreach("shadowsocks-libev", "server", function(sdata)
41 local sname = sdata[".name"]
42 local server = sdata["server"]
43 local server_port = sdata["server_port"]
44 if server and server_port then
45 local disabled = ucival_to_bool(sdata[".disabled"]) and " - disabled" or ""
46 local desc = "%s - %s:%s%s" % {sname, server, server_port, disabled}
47 o:value(sname, desc)
48 end
49 end)
50 end
51
52 function values_ipaddr(o)
53 for _, v in ipairs(nw:get_interfaces()) do
54 for _, a in ipairs(v:ipaddrs()) do
55 o:value(a:host():string(), '%s (%s)' %{ a:host(), v:shortname() })
56 end
57 end
58 end
59
60 function values_ifnames(o)
61 for _, v in ipairs(sys.net.devices()) do
62 o:value(v)
63 end
64 end
65
66 function options_client(s, tab)
67 local o
68
69 o = s:taboption(tab, ListValue, "server", translate("Remote server"))
70 values_serverlist(o)
71 o = s:taboption(tab, Value, "local_address", translate("Local address"))
72 o.datatype = "ipaddr"
73 o.placeholder = "0.0.0.0"
74 values_ipaddr(o)
75 o = s:taboption(tab, Value, "local_port", translate("Local port"))
76 o.datatype = "port"
77 end
78
79 function options_server(s, tab)
80 local o
81 local optfunc
82
83 if tab == nil then
84 optfunc = function(...) return s:option(...) end
85 else
86 optfunc = function(...) return s:taboption(tab, ...) end
87 end
88
89 o = optfunc(Value, "server", translate("Server"))
90 o.datatype = "host"
91 o.size = 16
92 o = optfunc(Value, "server_port", translate("Server port"))
93 o.datatype = "port"
94 o.size = 5
95 o = optfunc(ListValue, "method", translate("Method"))
96 for _, m in ipairs(methods) do
97 o:value(m)
98 end
99 o = optfunc(Value, "key", translate("Key (base64 encoding)"))
100 o.datatype = "base64"
101 o.password = true
102 o.size = 12
103 o = optfunc(Value, "password", translate("Password"))
104 o.password = true
105 o.size = 12
106 end
107
108 function options_common(s, tab)
109 local o
110
111 o = s:taboption(tab, ListValue, "mode", translate("Mode of operation"))
112 for _, m in ipairs(modes) do
113 o:value(m)
114 end
115 o.default = "tcp_and_udp"
116 o = s:taboption(tab, Value, "mtu", translate("MTU"))
117 o.datatype = "uinteger"
118 o = s:taboption(tab, Value, "timeout", translate("Timeout (sec)"))
119 o.datatype = "uinteger"
120 s:taboption(tab, Value, "user", translate("Run as"))
121
122 s:taboption(tab, Flag, "verbose", translate("Verbose"))
123 s:taboption(tab, Flag, "ipv6_first", translate("IPv6 First"), translate("Prefer IPv6 addresses when resolving names"))
124 s:taboption(tab, Flag, "fast_open", translate("Enable TCP Fast Open"))
125 s:taboption(tab, Flag, "no_delay", translate("Enable TCP_NODELAY"))
126 s:taboption(tab, Flag, "reuse_port", translate("Enable SO_REUSEPORT"))
127 end
128
129 function ucival_to_bool(val)
130 return val == "true" or val == "1" or val == "yes" or val == "on"
131 end
132
133 function cfgvalue_overview(sdata)
134 local stype = sdata[".type"]
135 local lines = {}
136
137 if stype == "ss_server" then
138 cfgvalue_overview_(sdata, lines, names_options_server)
139 cfgvalue_overview_(sdata, lines, names_options_common)
140 cfgvalue_overview_(sdata, lines, {
141 "bind_address",
142 "manager_address",
143 })
144 elseif stype == "ss_local" or stype == "ss_redir" or stype == "ss_tunnel" then
145 cfgvalue_overview_(sdata, lines, names_options_client)
146 if stype == "ss_tunnel" then
147 cfgvalue_overview_(sdata, lines, {"tunnel_address"})
148 elseif stype == "ss_redir" then
149 cfgvalue_overview_(sdata, lines, {"disable_sni"})
150 end
151 cfgvalue_overview_(sdata, lines, names_options_common)
152 else
153 return nil, nil
154 end
155 local sname = sdata[".name"]
156 local key = "%s.%s" % {stype, sname}
157 local value = {
158 [".name"] = sname,
159 name = '%s.<var>%s</var>' % {stype, sname},
160 overview = table.concat(lines, "</br>"),
161 disabled = ucival_to_bool(sdata["disabled"]),
162 }
163 return key, value
164 end
165
166 function cfgvalue_overview_(sdata, lines, names)
167 local line
168
169 for _, n in ipairs(names) do
170 local v = sdata[n]
171 if v ~= nil then
172 if n == "key" or n == "password" then
173 v = translate("<hidden>")
174 end
175 local fv = "<var>%s</var>" % ut.pcdata(v)
176 if sdata[".type"] ~= "ss_server" and n == "server" then
177 fv = '<a class="label" href="%s">%s</a>' % {
178 ds.build_url("admin/services/shadowsocks-libev/servers", v), fv}
179 end
180 line = n .. ": " .. fv
181 table.insert(lines, line)
182 end
183 end
184 end
185
186 function option_install_package(s, tab)
187 local bin = s.sectiontype:gsub("_", "-", 1)
188 local installed = nixio.fs.access("/usr/bin/" .. bin)
189 if installed then
190 return
191 end
192 local opkg_package = "shadowsocks-libev-" .. bin
193 local p_install
194 if tab then
195 p_install = s:taboption(tab, Button, "_install")
196 else
197 p_install = s:option(Button, "_install")
198 end
199 p_install.title = translate("Package is not installed")
200 p_install.inputtitle = translate("Install package %q" % opkg_package)
201 p_install.inputstyle = "apply"
202
203 function p_install.write()
204 return luci.http.redirect(
205 luci.dispatcher.build_url("admin/system/packages") ..
206 "?submit=1&install=%s" % opkg_package
207 )
208 end
209 end
210
211 names_options_server = {
212 "server",
213 "server_port",
214 "method",
215 "key",
216 "password",
217 }
218
219 names_options_client = {
220 "server",
221 "local_address",
222 "local_port",
223 }
224
225 names_options_common = {
226 "verbose",
227 "ipv6_first",
228 "fast_open",
229 "reuse_port",
230 "mode",
231 "mtu",
232 "timeout",
233 "user",
234 }
235
236 modes = {
237 "tcp_only",
238 "tcp_and_udp",
239 "udp_only",
240 }
241
242 methods = {
243 -- aead
244 "aes-128-gcm",
245 "aes-192-gcm",
246 "aes-256-gcm",
247 "chacha20-ietf-poly1305",
248 "xchacha20-ietf-poly1305",
249 -- stream
250 "table",
251 "rc4",
252 "rc4-md5",
253 "aes-128-cfb",
254 "aes-192-cfb",
255 "aes-256-cfb",
256 "aes-128-ctr",
257 "aes-192-ctr",
258 "aes-256-ctr",
259 "bf-cfb",
260 "camellia-128-cfb",
261 "camellia-192-cfb",
262 "camellia-256-cfb",
263 "salsa20",
264 "chacha20",
265 "chacha20-ietf",
266 }