mirror of https://github.com/aredn/aredn.git
454 lines
13 KiB
Lua
Executable File
454 lines
13 KiB
Lua
Executable File
#!/usr/bin/lua
|
|
--[[
|
|
|
|
Part of AREDN® -- Used for creating Amateur Radio Emergency Data Networks
|
|
Copyright (C) 2018 Darryl Quinn
|
|
See Contributors file for additional contributors
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation version 3 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
Additional Terms:
|
|
|
|
Additional use restrictions exist on the AREDN® trademark and logo.
|
|
See AREDNLicense.txt for more info.
|
|
|
|
Attributions to the AREDN® Project must be retained in the source code.
|
|
If importing this code into a new or existing project attribution
|
|
to the AREDN® project must be added to the source code.
|
|
|
|
You must not misrepresent the origin of the material contained within.
|
|
|
|
Modified versions must be modified to attribute to the original source
|
|
and be marked in reasonable ways as differentiate it from the original
|
|
version
|
|
|
|
--]]
|
|
|
|
require("uci")
|
|
require("aredn.uci")
|
|
require("aredn.utils")
|
|
require("aredn.http")
|
|
require("aredn.hardware")
|
|
require("aredn.olsr")
|
|
require("aredn.info")
|
|
require("nixio")
|
|
local json = require("luci.jsonc")
|
|
require("iwinfo")
|
|
local nettools = require("aredn.nettools")
|
|
|
|
local API_VERSION="1.6"
|
|
|
|
-- Function extensions
|
|
os.capture = capture
|
|
|
|
|
|
function getSysinfo()
|
|
local info={}
|
|
info['api_version']=API_VERSION
|
|
info['node']=aredn.info.getNodeName()
|
|
info['tactical']=aredn.info.getTacticalName()
|
|
info['description']=aredn.info.getNodeDescription()
|
|
info['firmware_version']=aredn.info.getFirmwareVersion()
|
|
info['model']=aredn.info.getModel()
|
|
--
|
|
info['date']=aredn.info.getDate()
|
|
info['time']=aredn.info.getTime()
|
|
--
|
|
info['uptime']=aredn.info.getUptime()
|
|
info['loads']=aredn.info.getLoads()
|
|
info['first_boot']=aredn.info.getFirstBoot()
|
|
info['target_type']=aredn.info.getTargetType()
|
|
return info
|
|
end
|
|
|
|
function getAlerts()
|
|
local info={}
|
|
info['aredn']=aredn.info.getArednAlert()
|
|
info['local']=aredn.info.getLocalAlert()
|
|
return info
|
|
end
|
|
|
|
function getStatusMeshRF()
|
|
local info={}
|
|
local dev = aredn.info.getMeshRadioDevice()
|
|
if dev and dev ~= "" then
|
|
info['device']= dev
|
|
info['ssid']=aredn.info.getSSID()
|
|
info['channel']=aredn.info.getChannel(info['device'])
|
|
info['chanbw']=aredn.info.getChannelBW(info['device'])
|
|
info['band']=aredn.info.getBand(info['device'])
|
|
info['frequency']=aredn.info.getFreq(info['device'])
|
|
end
|
|
return info
|
|
end
|
|
|
|
function getStatusIp()
|
|
local info={}
|
|
info['gateway']=aredn.info.getDefaultGW()
|
|
info['wifi']=aredn.info.getInterfaceIPAddress('wifi')
|
|
info['lan']=aredn.info.getInterfaceIPAddress('lan')
|
|
info['wan']=aredn.info.getInterfaceIPAddress('wan')
|
|
return info
|
|
end
|
|
|
|
function getLocationInfo()
|
|
local info={}
|
|
local lat, lon= aredn.info.getLatLon()
|
|
local gs= aredn.info.getGridSquare()
|
|
info['lat']=lat
|
|
info['lon']=lon
|
|
info['gridsquare']=gs
|
|
return info
|
|
end
|
|
|
|
function getFreeMemory()
|
|
local info = aredn.info.getFreeMemory()
|
|
return info
|
|
end
|
|
|
|
|
|
function get_key_for_value( t, value )
|
|
for k,v in pairs(t) do
|
|
if v==value then return k end
|
|
end
|
|
return nil
|
|
end
|
|
|
|
function getRemoteNodes()
|
|
local info = {}
|
|
local neighbors = {}
|
|
for _, v in ipairs(aredn.olsr.getOLSRLinks())
|
|
do
|
|
local remoteIP = v.remoteIP
|
|
neighbors[remoteIP] = true
|
|
local name = nixio.getnameinfo(remoteIP)
|
|
if name then
|
|
local cname = name:match("^dtdlink%.(.*)$") or name:match("^mid%d+%.(.*)$")
|
|
if cname then
|
|
local addrs = nixio.getaddrinfo(cname)
|
|
if addrs then
|
|
neighbors[addrs[1].address] = true
|
|
end
|
|
end
|
|
end
|
|
end
|
|
local routeetx = {}
|
|
for _, v in ipairs(aredn.olsr.getOLSRRoutes())
|
|
do
|
|
routeetx[v.destination] = string.format("%.2f", v.etx)
|
|
end
|
|
for line in aredn.olsr.getHostAsLines()
|
|
do
|
|
local ip, hostname = line:match("^(%d+%.%d+%.%d+%.%d+)%s+(%S+)%s+#.*$")
|
|
if ip and not neighbors[ip] and not (hostname:match("^dtdlink%.") or hostname:match("^mid%d+%.")) then
|
|
local etx = routeetx[ip]
|
|
if etx then
|
|
info[#info + 1] = {
|
|
name = hostname,
|
|
ip = ip,
|
|
etx = etx
|
|
}
|
|
end
|
|
end
|
|
end
|
|
table.sort(info, function(a,b) return tonumber(a.etx) < tonumber(b.etx) end)
|
|
return info
|
|
end
|
|
|
|
-- gets signal and noise data from either iwinfo (realtime = true) or the files archived in /tmp/snrlog (realtime = false)
|
|
-- the files in /tmp/snrlog are written once per minute by the script /usr/local/bin/snrlog
|
|
function getSignal(realtime)
|
|
local defnoise=-95
|
|
local dpa={} -- datapoint table/array
|
|
local dirname="/tmp/snrlog"
|
|
|
|
local datepattern="(%d+)/(%d+)/(%d+)%s(%d+):(%d+):(%d+)"
|
|
|
|
local parms={}
|
|
parms=parsecgi(nixio.getenv("QUERY_STRING") or "")
|
|
|
|
-- make sure snrlog dir is created
|
|
if dir_exists(dirname) then
|
|
nixio.fs.mkdir(dirname)
|
|
end
|
|
|
|
-- get wifi ifname
|
|
local wifiiface=get_ifname("wifi")
|
|
|
|
-- get bandwidth
|
|
local radio=aredn.info.getMeshRadioDevice()
|
|
local bandwidth=aredn.info.getChannelBW(radio)
|
|
|
|
local timestamp_s=os.time()
|
|
local signal_dbm, noise_dbm, tx_rate_mbps, rx_rate_mbps, tx_rate_mcs_index, rx_rate_mcs_index
|
|
|
|
--[[
|
|
-- insertResult() transforms data from iwinfo (realtime) and snrlog (archive) so that the responses are always the same.
|
|
-- the front end expects data points that look like this:
|
|
{
|
|
timestamp: string;
|
|
signal_dbm: number;
|
|
noise_dbm: number;
|
|
tx_rate_mbps: number;
|
|
rx_rate_mbps: number;
|
|
tx_rate_mcs_index: number | undefined;
|
|
rx_rate_mcs_index: number | undefined;
|
|
}
|
|
]]--
|
|
|
|
function insertResult(timestamp_s_epoch, signal_dbm, noise_dbm, tx_rate_mbps, rx_rate_mbps, tx_rate_mcs_index, rx_rate_mcs_index)
|
|
local dp={} -- datapoint
|
|
local timestamp
|
|
|
|
-- snrlog sometimes has the string "null" for signal/noise, if so then assume they are zero
|
|
if signal_dbm=="null" then
|
|
signal_dbm = -95
|
|
end
|
|
if noise_dbm=="null" then
|
|
noise_dbm = -95
|
|
end
|
|
|
|
-- signal and noise might also be nil, so convert them to numbers
|
|
signal_dbm = tonumber(signal_dbm or -95)
|
|
noise_dbm = tonumber(noise_dbm or -95)
|
|
|
|
--snrlog sometimes has -1 for mcs indices, if so then make them undefined
|
|
if tx_rate_mcs_index == -1 or tx_rate_mcs_index == "-1" then
|
|
tx_rate_mcs_index = nil
|
|
end
|
|
if rx_rate_mcs_index == -1 or rx_rate_mcs_index == "-1" then
|
|
rx_rate_mcs_index = nil
|
|
end
|
|
|
|
-- convert from seconds since epoch to iso 8601 format
|
|
timestamp=os.date("%Y-%m-%dT%H:%M:%SZ", timestamp_s_epoch)
|
|
|
|
dp.timestamp = timestamp
|
|
dp.signal_dbm=signal_dbm
|
|
dp.noise_dbm=noise_dbm
|
|
dp.tx_rate_mbps=tonumber(tx_rate_mbps or 0)
|
|
dp.rx_rate_mbps=tonumber(rx_rate_mbps or 0)
|
|
dp.tx_rate_mcs_index=tonumber(tx_rate_mcs_index)
|
|
dp.rx_rate_mcs_index=tonumber(rx_rate_mcs_index)
|
|
table.insert(dpa,dp)
|
|
end
|
|
|
|
if realtime then
|
|
-- REALTIME
|
|
if (parms.device=="strongest" or parms.device=="" or (not parms.device)) then
|
|
-- REALTIME/STRONGEST SIGNAL
|
|
signal_dbm=iwinfo["nl80211"].signal(wifiiface)
|
|
noise_dbm=iwinfo["nl80211"].noise(wifiiface)
|
|
else
|
|
-- REALTIME/SPECIFIC SIGNAL
|
|
-- split out device to mac-host
|
|
local mac,host=string.match(parms.device,"([^-]*)-(.*)")
|
|
local macinfo=iwinfo["nl80211"].assoclist(wifiiface)[mac:upper()]
|
|
if macinfo then
|
|
signal_dbm=macinfo.signal
|
|
noise_dbm=macinfo.noise
|
|
tx_rate_mbps=macinfo.tx_rate/1000
|
|
tx_rate_mbps=adjust_rate(tx_rate_mbps,bandwidth)
|
|
rx_rate_mbps=macinfo.rx_rate/1000
|
|
rx_rate_mbps=adjust_rate(rx_rate_mbps,bandwidth)
|
|
tx_rate_mcs_index=macinfo.tx_mcs
|
|
rx_rate_mcs_index=macinfo.rx_mcs
|
|
end
|
|
end
|
|
insertResult(timestamp_s, signal_dbm, noise_dbm, tx_rate_mbps, rx_rate_mbps, tx_rate_mcs_index, rx_rate_mcs_index )
|
|
else
|
|
-- ARCHIVE
|
|
local filename
|
|
if not parms.device or parms.device=="" then
|
|
-- get the first file from dirlist
|
|
for maclist in nixio.fs.glob(dirname.."/".."*") do
|
|
filename=maclist
|
|
parms.device=maclist
|
|
break
|
|
end
|
|
else
|
|
filename=dirname.."/"..parms.device
|
|
end
|
|
|
|
if filename then
|
|
for line in io.lines(filename) do
|
|
local timestamp
|
|
timestamp,signal_dbm,noise_dbm,tx_rate_mcs_index,tx_rate_mbps,rx_rate_mcs_index,rx_rate_mbps=line:match("(.*)%,(.*)%,(.*),(.*),(.*),(.*),(.*)")
|
|
|
|
-- convert the timestamp to seconds since epoch
|
|
local dtm,dtd,dty,dth,dtmin,dts=timestamp:match(datepattern)
|
|
timestamp_s = os.time({month=dtm,day=dtd,year=dty,hour=dth,min=dtmin,sec=dts})
|
|
|
|
insertResult(timestamp_s, signal_dbm, noise_dbm, tx_rate_mbps, rx_rate_mbps, tx_rate_mcs_index, rx_rate_mcs_index)
|
|
end
|
|
end
|
|
end
|
|
return dpa
|
|
end
|
|
|
|
function getScanList()
|
|
local device = aredn.info.getMeshRadioDevice()
|
|
local scanlist = iwinfo["nl80211"].scanlist(device)
|
|
return scanlist
|
|
end
|
|
|
|
function getFreqList()
|
|
local device = aredn.info.getMeshRadioDevice()
|
|
local freqlist = iwinfo["nl80211"].freqlist(device)
|
|
return freqlist
|
|
end
|
|
|
|
-- ==== MAIN =====
|
|
ctx = uci.cursor()
|
|
if not ctx then
|
|
error("Failed to get uci cursor")
|
|
end
|
|
|
|
info={}
|
|
info['api_version'] = API_VERSION
|
|
|
|
-- get/process query string
|
|
local qsset={}
|
|
if (arg[1]==nil and arg[2]==nil) then
|
|
qs=nixio.getenv("QUERY_STRING")
|
|
if qs~="" then
|
|
qsset=parseQueryString(qs)
|
|
else
|
|
-- maybe default to a help page
|
|
qsset["api"]="help"
|
|
end
|
|
else
|
|
qsset[arg[1]]=arg[2]
|
|
end
|
|
|
|
info['pages']={}
|
|
|
|
for page, comps in pairs(qsset) do
|
|
-- ---------------- /mesh page
|
|
if not setContains(info['pages'],page) then
|
|
info['pages'][page]={}
|
|
end
|
|
|
|
if page=="api" then
|
|
info['pages'][page]=nil
|
|
info['api_help']="AREDN API. This API's primary function is to drive the Web UI."
|
|
elseif page=="common" then
|
|
for i,comp in pairs(comps:split(',')) do
|
|
if comp=="sysinfo" then
|
|
info['pages'][page][comp]=getSysinfo()
|
|
elseif comp=="alerts" then
|
|
info['pages'][page][comp]=getAlerts()
|
|
end
|
|
end
|
|
elseif page=="status" then
|
|
for i,comp in pairs(comps:split(',')) do
|
|
if comp=="meshrf" then
|
|
info['pages'][page][comp]=getStatusMeshRF()
|
|
elseif comp=="ip" then
|
|
info['pages'][page][comp]=getStatusIp()
|
|
elseif comp=="sysinfo" then
|
|
info['pages'][page][comp]=getSysinfo()
|
|
elseif comp=="memory" then
|
|
info['pages'][page][comp]=getFreeMemory()
|
|
elseif comp=="storage" then
|
|
info['pages'][page][comp]=aredn.info.getFSFree()
|
|
elseif comp=="olsr" then
|
|
info['pages'][page][comp]=aredn.info.getOLSRInfo()
|
|
elseif comp=="location" then
|
|
info['pages'][page][comp]=getLocationInfo()
|
|
elseif comp=="freqlist" then
|
|
info['pages'][page][comp]=getFreqList()
|
|
elseif comp=="alerts" then
|
|
info['pages'][page][comp]=getAlerts()
|
|
end
|
|
end
|
|
elseif page=="traceroute" then
|
|
for i,tonode in pairs(comps:split(',')) do
|
|
-- Validate that input as ip or hostname inside the mesh
|
|
if tonode:match("^[%d%.]+$") or tonode:match("^[%d%a%-%.%_]+$") then
|
|
info['pages'][page][tonode]=nettools.getTraceroute(tonode)
|
|
else
|
|
info['pages'][page][tonode]="Invalid input!"
|
|
end
|
|
end
|
|
elseif page=="ping" then
|
|
for i,tonode in pairs(comps:split(',')) do
|
|
-- Validate that input as ip or hostname inside the mesh
|
|
if tonode:match("^[%d%.]+$") or tonode:match("^[%d%a%-%.%_]+$") then
|
|
info['pages'][page][tonode]=nettools.getPing(tonode)
|
|
else
|
|
info['pages'][page][tonode]="Invalid input!"
|
|
end
|
|
end
|
|
elseif page=="iperf3" then
|
|
local protocol = "tcp"
|
|
for i,tonode in pairs(comps:split(',')) do
|
|
-- Validate that input as ip or hostname inside the mesh
|
|
if tonode == "tcp" or tonode == "udp" then
|
|
protocol = tonode
|
|
elseif tonode:match("^[%d%.]+$") or tonode:match("^[%d%a%-%.%_]+$") then
|
|
info['pages'][page][tonode]=nettools.getIperf3(tonode, protocol)
|
|
else
|
|
info['pages'][page][tonode]="Invalid input!"
|
|
end
|
|
end
|
|
elseif page=="mesh" then
|
|
for i,comp in pairs(comps:split(',')) do
|
|
if comp=="sysinfo" then
|
|
info['pages'][page][comp]=getSysinfo()
|
|
elseif comp=="allhosts" then
|
|
info['pages'][page][comp]=aredn.info.all_hosts()
|
|
elseif comp=="localhosts" then
|
|
info['pages'][page][comp]=aredn.info.getLocalHosts()
|
|
elseif comp=="remotenodes" then
|
|
info['pages'][page][comp]=getRemoteNodes()
|
|
elseif comp=="currentneighbors" then
|
|
info['pages'][page][comp]=aredn.olsr.getCurrentNeighbors(true)
|
|
elseif comp=="services" then
|
|
info['pages'][page][comp]=aredn.info.all_services()
|
|
elseif comp=="previousneighbors" then
|
|
info['pages'][page][comp]={}
|
|
elseif comp=="topology" then
|
|
info['pages'][page][comp]=fetch_json("http://127.0.0.1:9090/topology")
|
|
end
|
|
end
|
|
elseif page=="services" then
|
|
for i,comp in pairs(comps:split(',')) do
|
|
if comp=="sysinfo" then
|
|
info['pages'][page][comp]=getSysinfo()
|
|
end
|
|
end
|
|
elseif page=="chart" then
|
|
for i,comp in pairs(comps:split(',')) do
|
|
if comp=="realtime" then
|
|
info['pages'][page][comp]=getSignal(true)
|
|
elseif comp=="archive" then
|
|
info['pages'][page][comp]=getSignal(false)
|
|
end
|
|
end
|
|
elseif page=="scan" then
|
|
for i,comp in pairs(comps:split(',')) do
|
|
if comp=="scanlist" then
|
|
info['pages'][page][comp]=getScanList()
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
-- Output the HTTP header for JSON
|
|
-- json_header()
|
|
json_header()
|
|
|
|
-- Output the info table as json
|
|
print(json.stringify(info,true))
|