1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
|
#!/usr/bin/env lua
local json = require "luci.jsonc"
local nixio = require "nixio"
local fs = require "nixio.fs"
local UCI = require "luci.model.uci"
local sys = require "luci.sys"
local util = require "luci.util"
local luci_helper = "/usr/lib/ddns/dynamic_dns_lucihelper.sh"
local srv_name = "ddns-scripts"
-- convert epoch date to given format
local function epoch2date(epoch, format)
if not format or #format < 2 then
local uci = UCI.cursor()
format = uci:get("ddns", "global", "ddns_dateformat") or "%F %R"
uci:unload("ddns")
end
format = format:gsub("%%n", "<br />") -- replace newline
format = format:gsub("%%t", " ") -- replace tab
return os.date(format, epoch)
end
-- function to calculate seconds from given interval and unit
local function calc_seconds(interval, unit)
if not tonumber(interval) then
return nil
elseif unit == "days" then
return (tonumber(interval) * 86400) -- 60 sec * 60 min * 24 h
elseif unit == "hours" then
return (tonumber(interval) * 3600) -- 60 sec * 60 min
elseif unit == "minutes" then
return (tonumber(interval) * 60) -- 60 sec
elseif unit == "seconds" then
return tonumber(interval)
else
return nil
end
end
local methods = {
get_services_log = {
args = { service_name = "service_name" },
call = function(args)
local result = "File not found or empty"
local uci = UCI.cursor()
local dirlog = uci:get('ddns', 'global', 'ddns_logdir') or "/var/log/ddns"
-- Fallback to default logdir with unsecure path
if dirlog:match('%.%.%/') then dirlog = "/var/log/ddns" end
if args and args.service_name and fs.access("%s/%s.log" % { dirlog, args.service_name }) then
result = fs.readfile("%s/%s.log" % { dirlog, args.service_name })
end
uci.unload()
return { result = result }
end
},
get_services_status = {
call = function()
local uci = UCI.cursor()
local rundir = uci:get("ddns", "global", "ddns_rundir") or "/var/run/ddns"
local date_format = uci:get("ddns", "global", "ddns_dateformat")
local res = {}
uci:foreach("ddns", "service", function (s)
local ip, last_update, next_update
local section = s[".name"]
if fs.access("%s/%s.ip" % { rundir, section }) then
ip = fs.readfile("%s/%s.ip" % { rundir, section })
else
local dnsserver = s["dns_server"] or ""
local force_ipversion = tonumber(s["force_ipversion"] or 0)
local force_dnstcp = tonumber(s["force_dnstcp"] or 0)
local is_glue = tonumber(s["is_glue"] or 0)
local command = { luci_helper , [[ -]] }
local lookup_host = s["lookup_host"] or "_nolookup_"
if (use_ipv6 == 1) then command[#command+1] = [[6]] end
if (force_ipversion == 1) then command[#command+1] = [[f]] end
if (force_dnstcp == 1) then command[#command+1] = [[t]] end
if (is_glue == 1) then command[#command+1] = [[g]] end
command[#command+1] = [[l ]]
command[#command+1] = lookup_host
command[#command+1] = [[ -S ]]
command[#command+1] = section
if (#dnsserver > 0) then command[#command+1] = [[ -d ]] .. dnsserver end
command[#command+1] = [[ -- get_registered_ip]]
line = util.exec(table.concat(command))
end
local last_update = tonumber(fs.readfile("%s/%s.update" % { rundir, section } ) or 0)
local next_update, converted_last_update
local pid = tonumber(fs.readfile("%s/%s.pid" % { rundir, section } ) or 0)
if pid > 0 and not nixio.kill(pid, 0) then
pid = 0
end
local uptime = sys.uptime()
local force_seconds = calc_seconds(
tonumber(s["force_interval"]) or 72,
s["force_unit"] or "hours" )
-- process running but update needs to happen
-- problems if force_seconds > uptime
force_seconds = (force_seconds > uptime) and uptime or force_seconds
if last_update > 0 then
local epoch = os.time() - uptime + last_update + force_seconds
-- use linux date to convert epoch
converted_last_update = epoch2date(epoch,date_format)
next_update = epoch2date(epoch + force_seconds)
end
if pid > 0 and ( last_update + force_seconds - uptime ) <= 0 then
next_update = "Verify"
-- run once
elseif force_seconds == 0 then
next_update = "Run once"
-- no process running and NOT enabled
elseif pid == 0 and s['enabled'] == '0' then
next_update = "Disabled"
-- no process running and enabled
elseif pid == 0 and s['enabled'] ~= '0' then
next_update = "Stopped"
end
res[section] = {
ip = ip and ip:gsub("\n","") or nil,
last_update = last_update ~= 0 and converted_last_update or nil,
next_update = next_update or nil,
pid = pid or nil,
}
end
)
uci:unload("ddns")
return res
end
},
get_ddns_state = {
call = function()
local ipkg = require "luci.model.ipkg"
local uci = UCI.cursor()
local dateformat = uci:get("ddns", "global", "ddns_dateformat") or "%F %R"
uci:unload("ddns")
local ver, srv_ver_cmd
local res = {}
if ipkg then
ver = ipkg.info(srv_name)[srv_name].Version
else
srv_ver_cmd = luci_helper .. " -V | awk {'print $2'} "
ver = util.exec(srv_ver_cmd)
end
res['_version'] = ver and #ver > 0 and ver or nil
res['_enabled'] = sys.init.enabled("ddns")
res['_curr_dateformat'] = os.date(dateformat)
return res
end
},
get_env = {
call = function()
local res = {}
local cache = {}
local function has_wget()
return (sys.call( [[which wget >/dev/null 2>&1]] ) == 0)
end
local function has_wgetssl()
if cache['has_wgetssl'] then return cache['has_wgetssl'] end
local res = (sys.call( [[which wget-ssl >/dev/null 2>&1]] ) == 0)
cache['has_wgetssl'] = res
return res
end
local function has_curlssl()
return (sys.call( [[$(which curl) -V 2>&1 | grep -qF "https"]] ) == 0)
end
local function has_fetch()
if cache['has_fetch'] then return cache['has_fetch'] end
local res = (sys.call( [[which uclient-fetch >/dev/null 2>&1]] ) == 0)
cache['has_fetch'] = res
return res
end
local function has_fetchssl()
return fs.access("/lib/libustream-ssl.so")
end
local function has_curl()
if cache['has_curl'] then return cache['has_curl'] end
local res = (sys.call( [[which curl >/dev/null 2>&1]] ) == 0)
cache['has_curl'] = res
return res
end
local function has_curlpxy()
return (sys.call( [[grep -i "all_proxy" /usr/lib/libcurl.so* >/dev/null 2>&1]] ) == 0)
end
local function has_bbwget()
return (sys.call( [[$(which wget) -V 2>&1 | grep -iqF "busybox"]] ) == 0)
end
res['has_wget'] = has_wget() or false
res['has_curl'] = has_curl() or false
res['has_ssl'] = has_wgetssl() or has_curlssl() or (has_fetch() and has_fetchssl()) or false
res['has_proxy'] = has_wgetssl() or has_curlpxy() or has_fetch() or has_bbwget or false
res['has_forceip'] = has_wgetssl() or has_curl() or has_fetch() or false
res['has_bindnet'] = has_curl() or has_wgetssl() or false
local function has_bindhost()
if cache['has_bindhost'] then return cache['has_bindhost'] end
local res = (sys.call( [[which host >/dev/null 2>&1]] ) == 0)
if res then
cache['has_bindhost'] = res
return true
end
res = (sys.call( [[which khost >/dev/null 2>&1]] ) == 0)
if res then
cache['has_bindhost'] = res
return true
end
res = (sys.call( [[which drill >/dev/null 2>&1]] ) == 0)
if res then
cache['has_bindhost'] = res
return true
end
cache['has_bindhost'] = false
return false
end
res['has_bindhost'] = cache['has_bindhost'] or has_bindhost() or false
local function has_hostip()
return (sys.call( [[which hostip >/dev/null 2>&1]] ) == 0)
end
local function has_nslookup()
return (sys.call( [[which nslookup >/dev/null 2>&1]] ) == 0)
end
res['has_dnsserver'] = cache['has_bindhost'] or has_nslookup() or has_hostip() or has_bindhost() or false
local function check_certs()
local _, v = fs.glob("/etc/ssl/certs/*.crt")
if ( v == 0 ) then _, v = fs.glob("/etc/ssl/certs/*.pem") end
return (v > 0)
end
res['has_cacerts'] = check_certs() or false
res['has_ipv6'] = (fs.access("/proc/net/ipv6_route") and fs.access("/usr/sbin/ip6tables"))
return res
end
}
}
local function parseInput()
local parse = json.new()
local done, err
while true do
local chunk = io.read(4096)
if not chunk then
break
elseif not done and not err then
done, err = parse:parse(chunk)
end
end
if not done then
print(json.stringify({ error = err or "Incomplete input" }))
os.exit(1)
end
return parse:get()
end
local function validateArgs(func, uargs)
local method = methods[func]
if not method then
print(json.stringify({ error = "Method not found" }))
os.exit(1)
end
if type(uargs) ~= "table" then
print(json.stringify({ error = "Invalid arguments" }))
os.exit(1)
end
uargs.ubus_rpc_session = nil
local k, v
local margs = method.args or {}
for k, v in pairs(uargs) do
if margs[k] == nil or
(v ~= nil and type(v) ~= type(margs[k]))
then
print(json.stringify({ error = "Invalid arguments" }))
os.exit(1)
end
end
return method
end
if arg[1] == "list" then
local _, method, rv = nil, nil, {}
for _, method in pairs(methods) do rv[_] = method.args or {} end
print((json.stringify(rv):gsub(":%[%]", ":{}")))
elseif arg[1] == "call" then
local args = parseInput()
local method = validateArgs(arg[2], args)
local result, code = method.call(args)
print((json.stringify(result):gsub("^%[%]$", "{}")))
os.exit(code or 0)
end
|