2022-02-15 20:07:46 -07:00
#!/usr/bin/lua
--[[
Part of AREDN -- Used for creating Amateur Radio Emergency Data Networks
Copyright (C) 2021 Tim Wilkinson
Original Perl Copyright (C) 2020 - 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(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")
aredn.html = require("aredn.html")
require("uci")
aredn.info = require("aredn.info")
require("luci.sys")
local html = aredn.html
local urlprefix
local target = "unknown"
function defaultPackageRepos(repo)
if not urlprefix then
urlprefix = "http://downloads.arednmesh.org"
local release = "unknown"
for line in io.lines("/etc/openwrt_release")
do
local m = line:match("DISTRIB_RELEASE='(.*)'")
if m then
release = m
end
m = line:match("DISTRIB_TARGET='(.*)'")
if m then
target = m
end
end
if release:match("%.") then
local a, b = release:match("^(%d+)%.(%d+)%.")
urlprefix = urlprefix .. "/releases/" .. a .. "/" .. b .. "/" .. release
else
-- nightly
2022-03-05 10:46:57 -07:00
urlprefix = urlprefix .. "/snapshots"
2022-02-15 20:07:46 -07:00
end
end
if repo:match("aredn_core") then
return urlprefix .. "/targets/" .. target .. "/packages"
else
return urlprefix .. "/packages/mips_24kc/" .. repo
end
end
local settings = {
{
2022-05-06 20:03:03 -06:00
category = "Map Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@map[0].maptiles",
type = "string",
desc = "Specifies the URL of the location to access map tiles",
default = "http://stamen-tiles-{s}.a.ssl.fastly.net/terrain/{z}/{x}/{y}.jpg"
},
{
2022-05-06 20:03:03 -06:00
category = "Map Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@map[0].leafletcss",
type = "string",
desc = "Specifies the URL of the leaflet.css file",
2022-03-16 20:42:26 -06:00
default = "http://unpkg.com/leaflet@0.7.7/dist/leaflet.css"
2022-02-15 20:07:46 -07:00
},
{
2022-05-06 20:03:03 -06:00
category = "Map Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@map[0].leafletjs",
type = "string",
desc = "Specifies the URL of the leaflet.js file",
2022-03-16 20:42:26 -06:00
default = "http://unpkg.com/leaflet@0.7.7/dist/leaflet.js"
2022-02-15 20:07:46 -07:00
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].firmwarepath",
type = "string",
desc = "Specifies the URL of the location from which firmware files will be downloaded.",
default = "http://downloads.arednmesh.org/firmware"
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_core",
type = "string",
desc = "Specifies the URL for the 'core' packages: kernel modules and the like",
default = defaultPackageRepos('aredn_core'),
postcallback = "writePackageRepo('core')"
},
2018-12-11 05:47:36 -07:00
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_base",
type = "string",
desc = "Specifies the URL for the 'base' packages: libraries, shells, etc.",
default = defaultPackageRepos('base'),
postcallback = "writePackageRepo('base')"
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_arednpackages",
type = "string",
desc = "Specifies the URL for the 'arednpackages' packages: vtun, etc.",
default = defaultPackageRepos('arednpackages'),
postcallback = "writePackageRepo('arednpackages')"
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_luci",
type = "string",
desc = "Specifies the URL for the 'luci' packages: luci and things needed for luci.",
default = defaultPackageRepos('luci'),
postcallback = "writePackageRepo('luci')"
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_packages",
type = "string",
desc = "Specifies the URL for the 'packages' packages: everything not included in the other dirs.",
default = defaultPackageRepos('packages'),
postcallback = "writePackageRepo('packages')"
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_routing",
type = "string",
desc = "Specifies the URL for the 'routing' packages: olsr, etc.",
default = defaultPackageRepos('routing'),
postcallback = "writePackageRepo('routing')"
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_telephony",
type = "string",
desc = "Specifies the URL for the 'telephony' packages.",
default = defaultPackageRepos('telephony'),
postcallback = "writePackageRepo('telephony')"
},
{
2022-05-06 20:03:03 -06:00
category = "Firmware Paths",
2022-02-15 20:07:46 -07:00
key = "aredn.@downloads[0].pkgs_freifunk",
type = "string",
desc = "Specifies the URL for the 'freifunk' packages.",
default = defaultPackageRepos('freifunk'),
postcallback = "writePackageRepo('freifunk')"
},
2022-03-05 10:46:57 -07:00
2022-02-15 20:07:46 -07:00
{
2022-05-06 20:03:03 -06:00
category = "Power Options",
2022-02-15 20:07:46 -07:00
key = "aredn.@poe[0].passthrough",
type = "boolean",
2022-03-05 16:31:44 -07:00
desc = "Specifies whether a PoE passthrough port should be on or off. (Not all devices have PoE passthrough ports).",
2022-02-15 20:07:46 -07:00
default = "0",
condition = "hasPOE()",
postcallback = "setPOEOutput()"
},
{
2022-05-06 20:03:03 -06:00
category = "Power Options",
2022-02-15 20:07:46 -07:00
key = "aredn.@usb[0].passthrough",
type = "boolean",
2022-03-05 16:31:44 -07:00
desc = "Specifies whether the USB port should be on or off. (Not all devices have USB powered ports).",
2022-02-15 20:07:46 -07:00
default = "1",
postcallback = "setUSBOutput()",
condition = "hasUSB()"
},
{
2022-05-06 20:03:03 -06:00
category = "Tunnel Options",
2022-03-05 10:46:57 -07:00
key = "aredn.@tunnel[0].maxclients",
type = "string",
2022-03-20 19:55:11 -06:00
desc = "Specifies the maximum number of tunnel clients this node can serve; must be an integer in the range [0,100].",
2022-03-05 10:46:57 -07:00
default = "10",
2022-02-15 20:07:46 -07:00
precallback = "restrictTunnelLimitToValidRange()",
postcallback = "adjustTunnelInterfaceCount()"
},
{
2022-05-06 20:03:03 -06:00
category = "Tunnel Options",
2022-03-05 10:46:57 -07:00
key = "aredn.@tunnel[0].maxservers",
type = "string",
2022-03-20 19:55:11 -06:00
desc = "Specifies the maximum number of tunnel servers to which this node can connect; must be an integer in the range [0,100].",
2022-03-05 10:46:57 -07:00
default = "10",
2022-02-15 20:07:46 -07:00
precallback = "restrictTunnelLimitToValidRange()",
postcallback = "adjustTunnelInterfaceCount()"
},
2022-03-20 19:55:11 -06:00
{
2022-05-06 20:03:03 -06:00
category = "Tunnel Options",
2022-03-20 19:55:11 -06:00
key = "aredn.@tunnel[0].wanonly",
type = "boolean",
desc = "Prevents tunnel traffic from being routed over the mesh network itself.",
2022-04-16 17:32:23 -06:00
default = "1",
needreboot= true
2022-03-20 19:55:11 -06:00
},
2022-02-15 20:07:46 -07:00
{
2022-05-06 20:03:03 -06:00
category = "Memory",
2022-02-15 20:07:46 -07:00
key = "aredn.@meshstatus[0].lowmem",
type = "string",
desc = "Specifies the low memory threshold (in KB) when we will truncate the mesh status page",
default = "10000"
},
{
2022-05-06 20:03:03 -06:00
category = "Memory",
2022-02-15 20:07:46 -07:00
key = "aredn.@meshstatus[0].lowroutes",
type = "string",
desc = "When low memory is detected, limit the number of routes shown on the mesh status page",
default = "1000"
},
2022-04-16 17:32:23 -06:00
{
2022-05-06 20:03:03 -06:00
category = "WAN",
2022-04-17 14:41:34 -06:00
key = "aredn.wan.vlanid",
2022-04-16 17:32:23 -06:00
type = "string",
2022-04-17 14:41:34 -06:00
desc = "Specify WAN VLAN #",
2022-04-16 17:32:23 -06:00
default = "",
2022-04-16 22:34:57 -06:00
condition = "supportsVLANChange()",
2022-04-17 14:41:34 -06:00
current = "currentWANVLAN()",
postcallback = "changeWANVLAN()",
2022-04-16 17:32:23 -06:00
needreboot = true
},
2022-02-15 20:07:46 -07:00
{
2022-07-25 15:38:18 -06:00
category = "Network Tools",
2022-02-15 20:07:46 -07:00
key = "aredn.olsr.restart",
type = "none",
desc = "Will restart OLSR when saving setting -- wait up to 2 or 3 minutes to receive response.",
default = "0",
postcallback = "olsr_restart()"
},
2022-07-25 15:38:18 -06:00
{
category = "Network Tools",
key = "aredn.@iperf[0].enable",
type = "boolean",
desc = "Enable the included iperf3 client/server support",
default = "1"
},
2022-02-15 20:07:46 -07:00
{
2022-05-06 20:03:03 -06:00
category = "AREDN Alerts",
2022-02-15 20:07:46 -07:00
key = "aredn.aam.refresh",
type = "none",
desc = "Attempt to pull any AREDN Alert messages.",
default = "0",
postcallback = "aam_refresh()"
},
{
2022-05-06 20:03:03 -06:00
category = "AREDN Alerts",
2022-02-15 20:07:46 -07:00
key = "aredn.@alerts[0].localpath",
type = "string",
desc = "Specifies the URL of the location from which local AREDN Alerts can be downloaded.",
default = ""
},
2022-03-09 13:21:37 -07:00
{
2022-05-06 20:03:03 -06:00
category = "AREDN Alerts",
2022-03-09 13:21:37 -07:00
key = "aredn.@alerts[0].pollrate",
type = "string",
desc = "Specifies how many hours to wait between polling for new AREDN Alerts.",
2022-04-16 17:32:23 -06:00
default = "12",
needreboot = true
2022-03-09 13:21:37 -07:00
},
2022-02-15 20:07:46 -07:00
{
2022-05-06 20:03:03 -06:00
category = "AREDN Alerts",
2022-02-15 20:07:46 -07:00
key = "aredn.aam.purge",
type = "none",
desc = "Immediately purge/delete all AREDN (and local) Alerts from this node.",
default = "",
postcallback = "alert_purge()"
2022-04-26 20:08:46 -06:00
},
2022-05-18 11:49:00 -06:00
{
category = "Link Quality",
key = "aredn.@lqm[0].enable",
type = "boolean",
desc = "Enable experimental link quality management",
2022-06-09 16:57:06 -06:00
default = "1",
2022-05-18 11:49:00 -06:00
postcallback = "lqm_defaults()",
needreboot = true
},
{
category = "Link Quality",
key = "aredn.@lqm[0].margin_snr",
type = "string",
desc = "Margin above minimim SNR a signal must reach to become acceptable",
default = "1",
condition = "lqm_enabled()"
},
{
category = "Link Quality",
key = "aredn.@lqm[0].min_distance",
type = "string",
desc = "Distance neightbor must be over to be acceptable",
default = "0",
condition = "lqm_enabled()"
},
2022-06-27 15:31:43 -06:00
{
category = "Link Quality",
key = "aredn.@lqm[0].auto_distance",
type = "string",
desc = "Distance to use when actual distance cannot be calculated. 0 for auto",
default = "0",
condition = "lqm_enabled()"
},
2022-05-18 11:49:00 -06:00
{
category = "Link Quality",
key = "aredn.@lqm[0].margin_quality",
type = "string",
desc = "Quality increase before neighbor can be re-accepted",
default = "1",
condition = "lqm_enabled()"
},
{
category = "Link Quality",
key = "aredn.@lqm[0].ping_penalty",
type = "string",
desc = "Quality penalty when neighbor cannot be pinged",
2022-05-20 07:10:01 -06:00
default = "5",
2022-05-18 11:49:00 -06:00
condition = "lqm_enabled()"
},
{
category = "Link Quality",
key = "aredn.@lqm[0].user_blocks",
type = "string",
desc = "Comma separated list of blocked MACs",
default = "",
condition = "lqm_enabled()"
2022-06-12 22:01:12 -06:00
},
{
category = "Link Quality",
key = "aredn.@lqm[0].user_allows",
type = "string",
desc = "Comma separated list of always allowed MACs",
default = "",
condition = "lqm_enabled()"
}
2018-10-25 20:06:05 -06:00
}
2022-02-15 20:07:46 -07:00
local msgs = {}
--
-- helpers
--
function msg(m)
msgs[#msgs + 1] = m
end
2022-02-28 13:38:05 -07:00
-- uci cursor
2022-03-04 19:57:16 -07:00
local cursora = uci.cursor()
local cursorb = uci.cursor("/etc/config.mesh")
2022-02-28 13:38:05 -07:00
function cursor_set(a, b, c, d)
2022-03-03 16:03:08 -07:00
if not cursora:get(a, b) and b:match("@(.+)%[0%]") then
cursora:add(a, b:match("@(.+)%[0%]"))
end
2022-02-28 13:38:05 -07:00
cursora:set(a, b, c, d)
2022-03-03 16:03:08 -07:00
if not cursorb:get(a, b) and b:match("@(.+)%[0%]") then
cursorb:add(a, b:match("@(.+)%[0%]"))
end
2022-02-28 13:38:05 -07:00
cursorb:set(a, b, c, d)
cursora:commit(a)
cursorb:commit(a)
end
function cursor_add(a, b, c)
cursora:set(a, b, c)
cursorb:set(a, b, c)
cursora:commit(a)
cursorb:commit(a)
end
function cursor_delete(a, b)
cursora:delete(a, b)
cursorb:delete(a, b)
cursora:commit(a)
cursorb:commit(a)
end
function cursor_get(a, b, c)
return cursora:get(a, b, c)
end
2022-02-15 20:07:46 -07:00
function reboot()
local node = aredn.info.get_nvram("node")
if node == "" then
node = "Node"
end
local lanip, _, lanmask = aredn.hardware.get_interface_ip4(aredn.hardware.get_iface_name("lan"))
local browser = os.getenv("REMOTE_ADDR")
local browser6 = browser:match("::ffff:([%d%.]+)")
if browser6 then
browser = browser6
end
local fromlan = false
local subnet_change = false
if lanip then
fromlan = validate_same_subnet(browser, lanip, lanmask)
if fromlan then
lanmask = ip_to_decimal(lanmask)
local cfgip = cursor_get("network", "lan", "ipaddr")
local cfgmask = ip_to_decimal(cursor_get("network", "lan", "netmask"))
2022-03-06 17:15:07 -07:00
if lanmask ~= cfgmask or nixio.bit.band(ip_to_decimal(lanip), lanmask) ~= nixio.bit.band(ip_to_decimal(cfgip), cfgmask) then
2022-02-15 20:07:46 -07:00
subnet_change = true
end
end
end
http_header()
if fromlan and subnet_change then
html.header(node .. " rebooting", true)
html.print("<body><center>")
html.print("<h1>" .. node .. " is rebooting</h1><br>")
html.print("<h3>The LAN subnet has changed. You will need to acquire a new DHCP lease<br>")
html.print("and reset any name service caches you may be using.</h3><br>")
html.print("<h3>When the node reboots you get your new DHCP lease and reconnect with<br>")
html.print("<a href='http://localnode.local.mesh:8080/'>http://localnode.local.mesh:8080/</a><br>or<br>")
html.print("<a href='http://" .. node .. ".local.mesh:8080/'>http://" .. node .. ".local.mesh:8080/</a></h3>")
else
html.header(node .. " rebooting", false)
html.print("<meta http-equiv='refresh' content='60;url=/cgi-bin/status'>")
html.print("</head><body><center>")
html.print("<h1>" .. node .. " is rebooting</h1><br>")
html.print("<h3>Your browser should return to this node in 60 seconds.</br><br>")
html.print("If something goes astray you can try to connect with<br><br>")
html.print("<a href='http://localnode.local.mesh:8080/'>http://localnode.local.mesh:8080/</a><br>")
if node ~= "Node" then
html.print("or<br><a href='http://" .. node .. ".local.mesh:8080/'>http://" .. node .. ".local.mesh:8080/</a></h3>")
end
end
html.print("</center></body></html>")
http_footer()
luci.sys.reboot()
os.exit()
end
-- conditionals
function hasPOE()
return aredn.hardware.has_poe()
end
function hasUSB()
return aredn.hardware.has_usb()
end
2022-04-16 22:34:57 -06:00
function supportsVLANChange()
local stat = nixio.fs.stat("/etc/aredn_include/swconfig")
2022-07-25 11:02:42 -06:00
-- We always support VLAN changing on devices without switches
if not (stat and stat.size > 0) then
return true
end
-- We also support VLAN changing on hAPs as WAN is on it's own ethernet port
if aredn.hardware.get_type() == "rb-952ui-5ac2nd" then
return true
end
-- Otherwise
return false
2022-04-16 22:34:57 -06:00
end
2022-02-15 20:07:46 -07:00
-- callbacks
local newval
local key
function setPOEOutput()
if not newval then
newval = 0
end
os.execute("/usr/local/bin/poe_passthrough " .. newval)
end
function setUSBOutput()
if not newval then
newval = 0
end
os.execute("/usr/local/bin/usb_passthrough " .. newval)
end
function olsr_restart()
os.execute("/etc/init.d/olsrd restart")
end
function aam_refresh()
os.execute("/usr/local/bin/aredn_message.sh")
end
function alert_purge()
os.remove("/tmp/aredn_message")
os.remove("/tmp/local_message")
end
2022-05-18 11:49:00 -06:00
function lqm_enabled()
return cursor_get("aredn", "@lqm[0]", "enable") == "1"
end
function lqm_defaults()
cursor_set("aredn", "@lqm[0]", "min_snr", "15")
cursor_set("aredn", "@lqm[0]", "margin_snr", "1")
cursor_set("aredn", "@lqm[0]", "min_distance", "0")
2022-06-27 15:31:43 -06:00
cursor_set("aredn", "@lqm[0]", "auto_distance", "0")
2022-05-18 11:49:00 -06:00
cursor_set("aredn", "@lqm[0]", "max_distance", "80467")
cursor_set("aredn", "@lqm[0]", "min_quality", "50")
2022-05-20 07:10:01 -06:00
cursor_set("aredn", "@lqm[0]", "ping_penalty", "5")
2022-05-18 11:49:00 -06:00
cursor_set("aredn", "@lqm[0]", "margin_quality", "1")
end
2022-02-15 20:07:46 -07:00
function writePackageRepo(repo)
2022-03-03 16:03:08 -07:00
local uciurl = cursor_get("aredn", "@downloads[0]", "pkgs_" .. repo)
local disturl = capture("grep aredn_" .. repo .. " /etc/opkg/distfeeds.conf | cut -d' ' -f3")
if uciurl and disturl ~= "" then
os.execute("sed -i 's|" .. disturl:chomp() .. "|" .. uciurl:chomp() .. "|g' /etc/opkg/distfeeds.conf")
end
2022-02-15 20:07:46 -07:00
end
function restrictTunnelLimitToValidRange()
newval = tonumber(newval)
if not newval then
msg(key .. " must be an interger in the range [0,100]")
newval = 0
elseif newval < 0 then
msg("Lower limit of " .. key .. " is 0")
newval = 0
elseif newval > 100 then
msg("Upper limit of " .. key .. " is 100")
newval = 100
end
end
function addTunnelInterface(file, tunnum)
local section = "tun" .. tunnum
cursor_add(file, section, "interface")
cursor_set(file, section, "ifname", section)
cursor_set(file, section, "proto", "none")
end
function deleteTunnelInterface(file, tunnum)
local section = "tun" .. tunnum
cursor_delete(file, section)
end
function adjustTunnelInterfaceCount()
local tunnel_if_count = 0
cursora:foreach('network_tun', 'interface', function(s) tunnel_if_count = tunnel_if_count + 1 end)
2022-03-03 16:03:08 -07:00
local maxclients = cursor_get("aredn", "@tunnel[0]", "maxclients")
2022-02-15 20:07:46 -07:00
if not maxclients then
maxclients = 10
end
2022-03-03 16:03:08 -07:00
local maxservers = cursor_get("aredn", "@tunnel[0]", "maxservers")
2022-02-15 20:07:46 -07:00
if not maxservers then
maxservers = 10
end
local needed_if_count = maxclients + maxservers
if tunnel_if_count ~= needed_if_count then
for i = tunnel_if_count,needed_if_count-1
do
local tunnum = 50 + i
addTunnelInterface("network_tun", tunnum)
addTunnelInterface("network", tunnum)
end
for i = tunnel_if_count-1,needed_if_count,-1
do
local tunnum = 50 + i
deleteTunnelInterface("network_tun", tunnum)
deleteTunnelInterface("network", tunnum)
end
-- can't clone network because it contains macros; re-edit it instead
2022-03-05 10:46:57 -07:00
os.execute("sed -i -e '$r /etc/config.mesh/network_tun' -e '/interface.*tun',$d' /etc/config.mesh/network")
2022-02-15 20:07:46 -07:00
end
end
2022-04-17 14:41:34 -06:00
function currentWANVLAN()
for line in io.lines("/etc/config.mesh/_setup")
do
local vlan = line:match("^wan_intf = %w+%d+%.(%d+)")
if vlan then
return vlan
end
end
local vlan = aredn.hardware.get_board().network.wan.ifname:match("^%w+%.(%d+)")
if vlan then
return vlan
end
return ""
end
function changeWANVLAN()
2022-04-16 17:32:23 -06:00
local lines = {}
for line in io.lines("/etc/config.mesh/_setup")
do
if not line:match("^wan_intf = ") then
lines[#lines + 1] = line
end
end
if newval ~= "" then
2022-04-17 14:41:34 -06:00
local wan_intf = ""
for dev in aredn.hardware.get_board().network.wan.ifname:gmatch("%S+")
do
wan_intf = wan_intf .. " " .. dev:match("^([^%.]+)") .. "." .. newval
end
if wan_intf ~= "" then
lines[#lines + 1] = "wan_intf =" .. wan_intf
end
2022-04-16 17:32:23 -06:00
end
local f = io.open("/etc/config.mesh/_setup", "w")
if f then
for _, line in ipairs(lines)
do
f:write(line .. "\n")
end
f:close()
end
os.execute("/usr/local/bin/node-setup -a mesh")
end
2022-02-15 20:07:46 -07:00
-- read_postdata
local parms = {}
if os.getenv("REQUEST_METHOD") == "POST" then
require('luci.http')
local request = luci.http.Request(luci.sys.getenv(),
function()
local v = io.read(1024)
if not v then
io.close()
end
return v
end
)
parms = request:formvalue()
end
if parms.button_firstboot then
os.execute("firstboot -y")
end
if parms.button_firstboot or parms.button_reboot then
reboot()
end
local node = aredn.info.get_nvram("node")
for i, setting in ipairs(settings)
do
if parms["button_save_" .. i] then
newval = parms["newval_" .. i]
2022-03-05 16:31:44 -07:00
if not newval then
newval = ""
else
newval = newval:gsub("^%s+", ""):gsub("%s+$", "")
end
2022-02-15 20:07:46 -07:00
if setting.type == "boolean" then
if newval == "1" or newval == "true" then
newval = "1"
else
newval = "0"
end
end
key = setting.key
if setting.precallback then
loadstring(setting.precallback)()
end
local a, b, c = setting.key:match("(.+)%.(.+)%.(.*)")
cursor_set(a, b, c, newval)
msg("Changed " .. key)
if setting.postcallback then
loadstring(setting.postcallback)()
end
2022-04-16 17:32:23 -06:00
if setting.needreboot then
io.open("/tmp/reboot-required", "w"):close()
end
2022-02-15 20:07:46 -07:00
break
end
end
-- generate the page
http_header()
html.header(node .. " Advanced Configuration", false)
html.print([[
2020-08-26 17:34:30 -06:00
<style>
2022-02-15 20:07:46 -07:00
/* The switch - the box around the slider */
2020-08-26 17:34:30 -06:00
.switch {
2022-02-15 20:07:46 -07:00
position: relative;
display: inline-block;
2022-03-22 05:33:00 -06:00
vertical-align:middle;
width: 48px;
height: 20px;
2020-08-26 17:34:30 -06:00
}
/* Hide default HTML checkbox */
.switch input {
2022-02-15 20:07:46 -07:00
opacity: 0;
width: 0;
height: 0;
2020-08-26 17:34:30 -06:00
}
/* The slider */
.slider {
2022-02-15 20:07:46 -07:00
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #ccc;
-webkit-transition: .4s;
transition: .4s;
2020-08-26 17:34:30 -06:00
}
.slider:before {
2022-02-15 20:07:46 -07:00
position: absolute;
content: "";
2022-03-22 05:33:00 -06:00
height: 13px;
width: 13px;
2022-02-15 20:07:46 -07:00
left: 4px;
bottom: 4px;
background-color: white;
-webkit-transition: .4s;
transition: .4s;
2020-08-26 17:34:30 -06:00
}
input:checked + .slider {
2022-02-15 20:07:46 -07:00
background-color: #2196F3;
2020-08-26 17:34:30 -06:00
}
input:focus + .slider {
2022-02-15 20:07:46 -07:00
box-shadow: 0 0 1px #2196F3;
2020-08-26 17:34:30 -06:00
}
input:checked + .slider:before {
2022-02-15 20:07:46 -07:00
-webkit-transform: translateX(26px);
-ms-transform: translateX(26px);
transform: translateX(26px);
2020-08-26 17:34:30 -06:00
}
/* Rounded sliders */
.slider.round {
2022-02-15 20:07:46 -07:00
border-radius: 34px;
2020-08-26 17:34:30 -06:00
}
.slider.round:before {
2022-02-15 20:07:46 -07:00
border-radius: 50%;
2021-03-27 10:28:17 -06:00
}
2020-08-26 17:34:30 -06:00
</style>
<script>
function toggleDefault(fname, defval) {
2022-02-15 20:07:46 -07:00
if(document.getElementById(fname).checked) {
2020-08-26 17:34:30 -06:00
cval = '1'
2022-02-15 20:07:46 -07:00
} else {
2020-08-26 17:34:30 -06:00
cval = '0'
2022-02-15 20:07:46 -07:00
}
if(cval != defval) {
2020-08-26 17:34:30 -06:00
document.getElementById(fname).click();
2022-02-15 20:07:46 -07:00
}
return true;
2020-08-26 17:34:30 -06:00
}
</script>
</head>
2022-02-15 20:07:46 -07:00
]])
html.print("<body><center>")
html.alert_banner();
html.print("<div style=\"padding:5px;background-color:#FF0000;color:#FFFFFF;width:650px;\"><strong>WARNING:</strong> Changing advanced settings can be harmful to the stability, security, and performance of this node and potentially the entire mesh network.<br><strong>You should only continue if you are sure of what you are doing.</strong></div><form method=post action=advancedconfig enctype='multipart/form-data'><table width=790><tr><td>")
-- navbar
html.print("<hr><table cellpadding=5 border=0 width=100%><tr>")
html.print("<td align=center width=15%><a href='status'>Node Status</a></td>")
html.print("<td align=center width=15%><a href='setup'>Basic Setup</a></td>")
html.print("<td align=center width=15%><a href='ports'>Port Forwarding,<br>DHCP, and Services</a></td>")
html.print("<td align=center width=15%><a href='vpn'>Tunnel<br>Server</a></td>")
html.print("<td align=center width=15%><a href='vpnc'>Tunnel<br>Client</a></td>")
html.print("<td align=center width=15%><a href='admin'>Administration</a></td>")
html.print("<td align=center width=15% class=navbar_select><a href='advancedconfig'>Advanced<br>Configuration</a></td>")
html.print("</tr></table><hr>")
html.print("</td></tr>")
html.print("<tr><td align=center><a href='/help.html#advancedconfig' target='_blank'>Help</a> <input type=submit name=button_reboot value=Reboot style='font-weight:bold' title='Immediately reboot this node'> <input type=submit name=button_firstboot value='Reset to Firstboot' onclick=\"return confirm('All config settings and add-on packages will be lost back to first boot state. Continue?')\" title='Reset this node to the initial/firstboot status and reboot.'></td></tr>")
2022-04-16 17:32:23 -06:00
if nixio.fs.stat("/tmp/reboot-required") then
html.print("<tr><td align=center><h3>Reboot is required for changes to take effect</h3></td></tr>")
end
2022-02-15 20:07:46 -07:00
for _, m in ipairs(msgs)
do
html.print("<tr><td align='center'><strong>" .. m .. "</strong></td></tr>")
end
html.print([[
<tr><td align=center>
<table border=1>
<thead>
<tr>
<th>Help<br><small>(hover)</small></th>
<th>Config Setting</th>
<th>Value</th>
<th>Actions</th>
</tr>
</thead>
]])
-- settings
2022-05-06 20:03:03 -06:00
local prior_category = ""
2022-02-15 20:07:46 -07:00
for i, setting in ipairs(settings)
do
if not setting.condition or loadstring("return " .. setting.condition)() then
2022-04-17 14:41:34 -06:00
local sval
if setting.current then
sval = loadstring("return " .. setting.current)()
else
local a, b, c = setting.key:match("(.+)%.(.+)%.(.*)")
sval = cursor_get(a, b, c)
end
2022-03-03 16:03:08 -07:00
sval = sval and tostring(sval) or ""
2022-05-06 20:03:03 -06:00
if setting.category ~= prior_category then
2022-07-25 15:38:18 -06:00
html.print([[<tr style=background-color:lightseagreen><td align="center" colspan="4"><b>]] .. setting.category .. [[</b></td></tr>]])
2022-05-06 20:03:03 -06:00
prior_category = setting.category
end
2022-02-15 20:07:46 -07:00
html.print([[<tr><td align="center"><span title="]] .. setting.desc .. [["><img src="/qmark.png" /></span></td><td>]] .. setting.key .. [[</td><td>]])
if setting.type == "string" then
html.print("<input type='text' id='field_" .. i .. "' name='newval_" .. i .. "' size='65' value='" .. sval .. "'>")
2022-03-20 19:55:11 -06:00
elseif setting.type == "boolean" then
if sval == "" then
sval = setting.default
end
if sval == "1" then
html.print("OFF<label class='switch'><input type='checkbox' id='field_" .. i .. "' name='newval_" .. i .."' value='1' checked><span class='slider round'></span></label>ON")
else
html.print("OFF<label class='switch'><input type='checkbox' id='field_" .. i .. "' name='newval_" .. i .. "' value='1'><span class='slider round'></span></label>ON")
end
2022-02-15 20:07:46 -07:00
elseif setting.type == "none" then
html.print("Click EXECUTE button to trigger this action<input type='hidden' id='field_" .. i .. "' name='newval_" .. i .."' value='" .. sval .."'>")
end
html.print("</td>")
if setting.type ~= "none" then
html.print("<td align='center'><input type='submit' name='button_save_" .. i .. "' value='Save Setting' /><br><br>")
else
html.print("<td align='center'><input type='submit' name='button_save_" .. i .. "' value='Execute' /><br><br>")
end
if setting.type == "string" then
html.print("<input value='Set to Default' type='button' onclick=\"document.getElementById('field_" .. i .. "').value='" .. setting.default .. "';\">")
elseif setting.type == "boolean" then
html.print("<input value='Set to Default' type='button' onclick=\"return toggleDefault('field_" .. i .. "', '" .. setting.default .. "' );\">")
end
html.print("</td></tr>")
end
end
html.print("</table></td></tr></table></form></center>")
html.footer()
html.print("</body></html>")
http_footer()