#!/usr/bin/lua --[[ Part of AREDN -- Used for creating Amateur Radio Emergency Data Networks Copyright (C) 2021 Tim Wilkinson Original Perl Copyright (C) 2015 Conrad Lara 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 . Additional Terms: Additional use restrictions exist on the AREDN(TM) 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("nixio") require("aredn.hardware") require("aredn.http") require("aredn.utils") local html = require("aredn.html") require("uci") local aredn_info = require("aredn.info") -- helpers start function mesh_ip_to_hostnames(ip) if not ip or ip == "" or ip == "none" then return "" end local pattern = "^" .. ip .. "%s+([%w%-%.]+)" local host = "none" for line in io.lines("/etc/hosts") do local host = line:match(pattern) if host then return host.gsub("%s+", " / ") end end if nixio.fs.stat("/var/run/hosts_olsr.stable") then for line in io.lines("/var/run/hosts_olsr.stable") do local host = line:match(pattern) if host then host = host:gsub("^dtdlink%.","") host = host:gsub("^mid[0-9]*%.","") host = host:gsub("%.local.mesh$","") return host end end end return host end function get_default_gw(iface) -- wan will route via table 254 default gw -- wifi will route via OLSR table 31 default gw local rtable = "" if iface == "wan" then rtable = "ip route list table 254" elseif iface == "wifi" then rtable = "ip route list table 31" else return "none" end local p = io.popen(rtable) if p then for line in p:lines() do local gw = line:match("^default%svia%s([%d%.]+)") if gw then p:close() return gw end end p:close() end return "none" end function get_memavail() local f = io.open("/proc/meminfo", "r") if f then for line in f:lines() do local memavail = line:match("^MemAvailable:%s+(%d+)%s+kB") if memavail then f:close() return tonumber(memavail) end end f:close() end return 0 end function get_wifi_signal(wifiif) local signal = -1000 local noise = -1000 for mac, station in pairs(iwinfo.nl80211.assoclist(wifiif)) do if station.signal ~= 0 and station.signal > signal then signal = station.signal end if station.noise ~= 0 and station.noise > noise then noise = station.noise end end if signal == -1000 or noise == -1000 then return "none", "none" else if signal > 0 then signal = (0 - signal) end if noise > 0 then noise = (0 - noise) end return signal, noise end end function css_options() html.print("") for file in nixio.fs.glob("/www/*.css") do if file ~= "/www/style.css" then file = file:match("/www/(.*).css") html.print("") end end end -- helpers end local node = aredn_info.get_nvram("node") if node == "" then node = "NOCALL" end local tactical = aredn_info.get_nvram("tactical") local config = aredn_info.get_nvram("config") if config == "" or nixio.fs.stat("/etc/config.mesh", "type") ~= "dir" then config = "not set" end local wifi_iface = aredn.hardware.get_iface_name("wifi") local wifi_nr = wifi_iface:match("wlan(%d+)") local wifi_disabled = true local radio = "radio0" if wifi_nr then wifi_disabled = false radio = "radio" .. wifi_nr end local cursor = uci.cursor() local wifi_channel local wifi_chanbw local wifi_ssid if not wifi_disabled then wifi_channel = cursor:get("wireless", radio, "channel") wifi_channel = tonumber(wifi_channel) or 0 if wifi_channel >= 76 and wifi_channel <= 99 then wifi_channel = wifi_channel * 5 + 3000 end wifi_chanbw = cursor:get("wireless", radio, "chanbw") or "20" wifi_ssid = "none" cursor:foreach("wireless", "wifi-iface", function (section) if section.network == "wifi" then wifi_ssid = section.ssid return false end end ) end local node_desc = cursor:get("system", "@system[0]", "description") local lat_lon = "Location Not Available" local lat = cursor:get("aredn", "@location[0]", "lat") local lon = cursor:get("aredn", "@location[0]", "lon") if lat and lon then lat_lon = string.format("
Location: %s %s
", lat, lon) end local host_total = 0 local host_nodes = 0 if nixio.fs.stat("/var/run/hosts_olsr.stable") then for line in io.lines("/var/run/hosts_olsr.stable") do if line:match("^10%.") and not line:match("%smid%d+%.") then host_total = host_total + 1 local host = line:match("^10%..+%sdtdlink%.") if host then host_nodes = host_nodes + 1 end end end end -- post data if os.getenv("REQUEST_METHOD") == "POST" then require('luci.http') local request = luci.http.Request(nixio.getenv(), function() local v = io.read(1024) if not v then io.close() end return v end ) local css = request:formvalue("css") if css and css:match("%.css$") and nixio.fs.stat("/www/" .. css) then nixio.fs.unlink("/tmp/web/style.css") nixio.fs.symlink("/www/" .. css, "/tmp/web/style.css") end end -- generate page http_header() html.header(node .. " status", true) html.print("
") html.print("
") html.alert_banner() html.msg_banner() -- page header html.print("

" .. node) if tactical ~= "" then html.print(" / " .. tactical) end html.print("

") html.print("
" .. lat_lon .. "
") if node_desc then html.print("
" .. node_desc .. "
") end html.print("
") -- nav buttons html.print("") html.print("Help") html.print("  ") html.print("") if config == "mesh" then html.print("  ") html.print("") if cursor:get("aredn", "@lqm[0]", "enable") == "1" then html.print("  ") html.print("") end if not wifi_disabled then html.print("  ") html.print("") end end html.print("  ") html.print("") html.print("  ") html.print("") html.print("") html.print("") if config == "not set" then html.print("

This node is not yet configured.
") local overlay = false for line in io.lines("/proc/mounts") do if line:match("overlay") then overlay = true break end end if overlay then html.print("Go to the setup page and set your node name and password.
") html.print("Click Save Changes, even if you didn't make any changes, then the node will reboot.
") else html.print("Go the the administration page and upload new firmware.
") end html.print("

") html.print("
") html.print("

This device can be configured to either permit or prohibit known encrypted traffic on its RF link. It is up to the user to decide which is appropriate based on how it will be used and the license under which it will be operated. These rules vary by country, frequency, and intended use. You are encouraged to read and understand these rules before going further.

") html.print("

This device is pre-configured with no restrictions as to the type of data being passed.

") html.print("

Follow these steps if you wish to prohibit known encrypted traffic on the RF link. These instructions will disappear, so copy them for your reference:

") html.print("

    ") html.print("
  1. Setup your node name and password as instructed at the top of this page
  2. ") html.print("
  3. After you Save Changes allow your node to reboot
  4. ") html.print("
  5. Return to the Node Status page and navigate to Setup > Administration
  6. ") html.print("
  7. Obtain the blockknownencryption package from the AREDN™ website OR refresh the Package list (node must be connected to the internet)
  8. ") html.print("
  9. Install the blockknownencryption package by uploading it or choosing it from the package drop-down list
  10. ") html.print("
  11. Wait until the package installs and then reboot your node
  12. ") html.print("

") html.print("
") end -- status display local col1 = {} local col2 = {} -- left column - network interface info local ip = cursor:get("network", "wifi", "ipaddr") local cidr = netmask_to_cidr(cursor:get("network", "wifi", "netmask")) if wifi_disabled then col1[#col1 + 1] = "primary address:" .. ip .. " / " .. cidr .. "
" else wifi_gw = get_default_gw("wifi") col1[#col1 + 1] = "mesh RF address:
mesh gateway:
gateway node:
SSID:
channel:
channel width:" .. ip .. " / " .. cidr .. "
" .. wifi_gw .. "
" .. mesh_ip_to_hostnames(wifi_gw) .. "
" .. wifi_ssid .. "
" .. wifi_channel .. "
" .. wifi_chanbw .. " MHz" end ip = cursor:get("network", "lan", "ipaddr") mask = cursor:get("network", "lan", "netmask") local browser_ip local remote_ip = os.getenv("REMOTE_ADDRESS") if remote_ip then remote_ip = remote_ip.match("::ffff:([%d%.]+)") end local hide_local = false if remote_ip then browser_ip = remote_ip if not validate_same_subnet(remote_ip, ip, mask) then hide_local = true end end if ip:match("^10%.") or not hide_local then cidr = netmask_to_cidr(mask) local lan_wifi_ssid = "none" cursor:foreach("wireless", "wifi-iface", function (section) if section.network == "lan" then lan_wifi_ssid = section.ssid return false end end ) if lan_wifi_ssid ~= "none" then col1[#col1 + 1] = "LAN address:
LAN AP SSID:" .. ip .. " / " .. cidr .. "
" .. lan_wifi_ssid .. "" else col1[#col1 + 1] = "LAN address:" .. ip .. " / " .. cidr .. "" end end local wan_iface = aredn.hardware.get_iface_name("wan") if wan_iface and not hide_local then local ip, bcast, mask = aredn.hardware.get_interface_ip4(wan_iface) if not ip then if not wifi_gw or wifi_gw == "none" then col1[#col1 + 1] = "WAN address:
default gateway:none
none" else col1[#col1 + 1] = "WAN address:
default gateway:
gateway node:none
" .. wifi_gw .. "
" .. mesh_ip_to_hostnames(wifi_gw) .. "" end else local wprefix = "" local wan_wifi_snr = "none" local wan_wifi_ssid if wan_iface:match("^wlan%d+$") then wprefix = "wifi " local s, n = get_wifi_signal(wan_iface) if s ~= "none" and n ~= "none" then wan_wifi_snr = math.abs(s - n) end cursor:foreach("wireless", "wifi-iface", function (section) if section.network == "wan" then wan_wifi_ssid = section.ssid return false end end ) if not wan_wifi_ssid then -- if still nil then set default wan_wifi_ssid = "none" end end cidr = netmask_to_cidr(mask) wan_gw = get_default_gw("wan") if wprefix == "" then -- no wifi wan if wan_gw:match("^10%.") or not hide_local then if wan_gw:match("^10%.") then col1[#col1 + 1] = "WAN address:
gateway:
gateway node:" .. ip .. " / " .. cidr .. "
" .. wan_gw .. "
" .. mesh_ip_to_hostnames(wan_gw) .. "" else col1[#col1 + 1] = "WAN address:
default gateway:" .. ip .. " / " .. cidr .. "
" .. wan_gw .. "" end end else -- with wifi wan if wan_wifi_ssid ~= "none" and wan_wifi_snr ~= "none" then if wan_gw:match("^10%.") or not hide_local then if wan_gw:match("^10%.") then col1[#col1 + 1] = "" .. wprefix .. "WAN address:
SSID | SNR:
gateway:
gateway node:" .. ip .. " / " .. cidr .. "
" .. wan_wifi_ssid .. " | " .. wan_wifi_snr .. " dB
" .. wan_gw .. "
" .. mesh_ip_to_hostnames(wan_gw) .. "" else col1[#col1 + 1] = "" .. wprefix .. "WAN address:
SSID | SNR:
default gateway:" .. ip .. " / " .. cidr .. "
" .. wan_wifi_ssid .. " | " .. wan_wifi_snr .. " dB
" .. wan_gw .. "" end end else if wan_gw:match("^10%.") or not hide_local then if wan_gw:match("^10%.") then col1[#col1 + 1] = "
" .. wprefix .. "WAN address:
gateway:
gateway node:" .. ip .. " / " .. cidr .. "
" .. wan_gw .. "
" .. mesh_ip_to_hostnames(wan_gw) .. "" else col1[#col1 + 1] = "" .. wprefix .. "WAN address:
default gateway:" .. ip .. " / " .. cidr .. "
" .. wan_gw .. "" end end end end end end if browser_ip then col1[#col1 + 1] = "your address:" .. browser_ip .. "
" .. mesh_ip_to_hostnames(browser_ip) .. "" end -- right column - system info if config == "mesh" and not wifi_disabled then col2[#col2 + 1] = "signal|noise|SNR:" local s, n = get_wifi_signal(wifi_iface) if s == "none" then col2[#col2] = col2[#col2] .. "no RF links" col2[#col2] = col2[#col2] .. "   " else col2[#col2] = col2[#col2] .. "" .. s .. " | " .. n .. " | " .. math.abs(s - n) .. " dB" col2[#col2] = col2[#col2] .. "   " end end col2[#col2 + 1] = "firmware version:
model:" .. read_all("/etc/mesh-release") .. "
" .. (aredn.hardware.get_radio() or { name = "unknown" }).name .. "" local sysinfo = nixio.sysinfo() local uptime = string.format("%d:%02d", math.floor(sysinfo.uptime / 3600) % 24, math.floor(sysinfo.uptime / 60) % 60) if sysinfo.uptime >= 172800 then uptime = math.floor(sysinfo.uptime / 86400) .. " days, " .. uptime elseif sysinfo.uptime >= 86400 then uptime = "1 day, " .. uptime end col2[#col2 + 1] = "system time:
uptime:" .. os.date("%a %b %e %Y") .. " " .. os.date("%T %Z") .. "
" .. uptime .. "" local vfs = nixio.fs.statvfs("/overlay") local fspace = vfs.bfree * vfs.bsize / 1024 if fspace < 100 then fspace = "" .. fspace .. " KB" else fspace = fspace .. " KB" end local rspace = sysinfo.freeram / 1024 local mavail = get_memavail() if rspace < mavail then rspace = mavail end if rspace < 500 then rspace = "" .. rspace .. " KB" else rspace = rspace .. " KB" end col2[#col2 + 1] = "load average:
available space:" .. string.format("%.2f, %.2f, %.2f", sysinfo.loads[1], sysinfo.loads[2], sysinfo.loads[3]) .. "
flash = " .. fspace .. "
memory = " .. rspace .. "" col2[#col2 + 1] = "host entries:" .. host_nodes .. " nodes / " .. host_total .. " total devices" -- now print the tables html.print("

") html.print("
") for i, line in ipairs(col1) do html.print("" .. line .. "") end html.print("
") for i, line in ipairs(col2) do html.print("" .. line .. "") end html.print("
") -- end html.print("
") html.footer() html.print("")