modules/freifunk: Make status page update dynamically with javascript and small cosmetic changes.
This commit is contained in:
parent
d0c8c5b532
commit
6951da02ce
3 changed files with 258 additions and 32 deletions
|
@ -166,6 +166,7 @@ end
|
||||||
-- @return String containing the memory used for caching in kB
|
-- @return String containing the memory used for caching in kB
|
||||||
-- @return String containing the memory used for buffering in kB
|
-- @return String containing the memory used for buffering in kB
|
||||||
-- @return String containing the free memory amount in kB
|
-- @return String containing the free memory amount in kB
|
||||||
|
-- @return String containing the cpu bogomips (number)
|
||||||
function sysinfo()
|
function sysinfo()
|
||||||
local cpuinfo = fs.readfile("/proc/cpuinfo")
|
local cpuinfo = fs.readfile("/proc/cpuinfo")
|
||||||
local meminfo = fs.readfile("/proc/meminfo")
|
local meminfo = fs.readfile("/proc/meminfo")
|
||||||
|
@ -176,6 +177,7 @@ function sysinfo()
|
||||||
local memcached = tonumber(meminfo:match("\nCached:%s*(%d+)"))
|
local memcached = tonumber(meminfo:match("\nCached:%s*(%d+)"))
|
||||||
local memfree = tonumber(meminfo:match("MemFree:%s*(%d+)"))
|
local memfree = tonumber(meminfo:match("MemFree:%s*(%d+)"))
|
||||||
local membuffers = tonumber(meminfo:match("Buffers:%s*(%d+)"))
|
local membuffers = tonumber(meminfo:match("Buffers:%s*(%d+)"))
|
||||||
|
local bogomips = tonumber(cpuinfo:match("BogoMIPS.-:%s*([^\n]+)"))
|
||||||
|
|
||||||
if not system then
|
if not system then
|
||||||
system = nixio.uname().machine
|
system = nixio.uname().machine
|
||||||
|
@ -187,7 +189,7 @@ function sysinfo()
|
||||||
model = cpuinfo:match("cpu model.-:%s*([^\n]+)")
|
model = cpuinfo:match("cpu model.-:%s*([^\n]+)")
|
||||||
end
|
end
|
||||||
|
|
||||||
return system, model, memtotal, memcached, membuffers, memfree
|
return system, model, memtotal, memcached, membuffers, memfree, bogomips
|
||||||
end
|
end
|
||||||
|
|
||||||
--- Retrieves the output of the "logread" command.
|
--- Retrieves the output of the "logread" command.
|
||||||
|
|
|
@ -51,6 +51,7 @@ function index()
|
||||||
|
|
||||||
entry({"freifunk", "status.json"}, call("jsonstatus"))
|
entry({"freifunk", "status.json"}, call("jsonstatus"))
|
||||||
entry({"freifunk", "status", "zeroes"}, call("zeroes"), "Testdownload")
|
entry({"freifunk", "status", "zeroes"}, call("zeroes"), "Testdownload")
|
||||||
|
entry({"freifunk", "status", "public_status_json"}, call("public_status_json")).leaf = true
|
||||||
|
|
||||||
assign({"freifunk", "olsr"}, {"admin", "status", "olsr"}, "OLSR", 30)
|
assign({"freifunk", "olsr"}, {"admin", "status", "olsr"}, "OLSR", 30)
|
||||||
|
|
||||||
|
@ -217,3 +218,65 @@ function jsonstatus()
|
||||||
http.prepare_content("application/json")
|
http.prepare_content("application/json")
|
||||||
ltn12.pump.all(json.Encoder(root):source(), http.write)
|
ltn12.pump.all(json.Encoder(root):source(), http.write)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
function public_status_json()
|
||||||
|
local twa = require "luci.tools.webadmin"
|
||||||
|
local sys = require "luci.sys"
|
||||||
|
local i18n = require "luci.i18n"
|
||||||
|
local path = luci.dispatcher.context.requestpath
|
||||||
|
local rv = { }
|
||||||
|
|
||||||
|
local dev
|
||||||
|
for dev in path[#path]:gmatch("[%w%.%-]+") do
|
||||||
|
local j = { id = dev }
|
||||||
|
local iw = luci.sys.wifi.getiwinfo(dev)
|
||||||
|
if iw then
|
||||||
|
local f
|
||||||
|
for _, f in ipairs({
|
||||||
|
"channel", "txpower", "bitrate", "signal", "noise",
|
||||||
|
"quality", "quality_max", "mode", "ssid", "bssid", "encryption", "ifname"
|
||||||
|
}) do
|
||||||
|
j[f] = iw[f]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
rv[#rv+1] = j
|
||||||
|
end
|
||||||
|
|
||||||
|
local load1, load5, load15 = sys.loadavg()
|
||||||
|
|
||||||
|
local _, _, memtotal, memcached, membuffers, memfree = sys.sysinfo()
|
||||||
|
local mem = string.format("%.2f MB (%.2f %s, %.2f %s, %.2f %s, %.2f %s)",
|
||||||
|
tonumber(memtotal) / 1024,
|
||||||
|
tonumber(memtotal - memfree) / 1024,
|
||||||
|
tostring(i18n.translate("used")),
|
||||||
|
memfree / 1024,
|
||||||
|
tostring(i18n.translate("free")),
|
||||||
|
memcached / 1024,
|
||||||
|
tostring(i18n.translate("cached")),
|
||||||
|
membuffers / 1024,
|
||||||
|
tostring(i18n.translate("buffered"))
|
||||||
|
)
|
||||||
|
|
||||||
|
local dr4 = sys.net.defaultroute()
|
||||||
|
local dr6 = sys.net.defaultroute6()
|
||||||
|
|
||||||
|
rv[#rv+1] = {
|
||||||
|
time = os.date("%c"),
|
||||||
|
uptime = twa.date_format(tonumber(sys.uptime())),
|
||||||
|
load = string.format("%.2f, %.2f, %.2f", load1, load5, load15),
|
||||||
|
mem = mem,
|
||||||
|
defroutev4 = { gateway = dr4.gateway:string(),
|
||||||
|
dest = dr4.dest:string(),
|
||||||
|
dev = dr4.device,
|
||||||
|
metr = dr4.metric },
|
||||||
|
defroutev6 = { gateway = dr6.nexthop:string(),
|
||||||
|
dest = dr6.dest:string(),
|
||||||
|
dev = dr6.device,
|
||||||
|
metr = dr6.metric }
|
||||||
|
}
|
||||||
|
|
||||||
|
luci.http.prepare_content("application/json")
|
||||||
|
luci.http.write_json(rv)
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
|
|
@ -2,45 +2,207 @@
|
||||||
local sys = require "luci.sys"
|
local sys = require "luci.sys"
|
||||||
local twa = require "luci.tools.webadmin"
|
local twa = require "luci.tools.webadmin"
|
||||||
-- System
|
-- System
|
||||||
local system, model, memtotal, memcached, membuffers, memfree = sys.sysinfo()
|
local system, model, memtotal, memcached, membuffers, memfree, bogomips = sys.sysinfo()
|
||||||
local uptime = twa.date_format(tonumber(sys.uptime()))
|
local uptime = twa.date_format(tonumber(sys.uptime()))
|
||||||
local_time = os.date("%c")
|
local_time = os.date("%c")
|
||||||
local load1, load5, load15 = sys.loadavg()
|
local load1, load5, load15 = sys.loadavg()
|
||||||
local load = string.format("%.2f, %.2f, %.2f", load1, load5, load15)
|
local load = string.format("%.2f, %.2f, %.2f", load1, load5, load15)
|
||||||
local memory = string.format("%.2f MB (%.0f%% %s, %.0f%% %s, %.0f%% %s)",
|
|
||||||
tonumber(memtotal) / 1024,
|
local mem = string.format("%.2f MB (%.2f %s, %.2f %s, %.2f %s, %.2f %s)",
|
||||||
100 * memcached / memtotal,
|
tonumber(memtotal) / 1024,
|
||||||
tostring(translate("cached")),
|
tonumber(memtotal - memfree) / 1024,
|
||||||
100 * membuffers / memtotal,
|
tostring(i18n.translate("used")),
|
||||||
tostring(translate("buffered")),
|
memfree / 1024,
|
||||||
100 * memfree / memtotal,
|
tostring(i18n.translate("free")),
|
||||||
tostring(translate("free"))
|
memcached / 1024,
|
||||||
|
tostring(i18n.translate("cached")),
|
||||||
|
membuffers / 1024,
|
||||||
|
tostring(i18n.translate("buffered"))
|
||||||
)
|
)
|
||||||
|
|
||||||
|
-- update interval
|
||||||
|
local bogomips = bogomips or 100
|
||||||
|
local interval
|
||||||
|
if bogomips > 350 then
|
||||||
|
interval = "5000"
|
||||||
|
else
|
||||||
|
interval = "10000"
|
||||||
|
end
|
||||||
|
|
||||||
-- wireless
|
-- wireless
|
||||||
local ntm = require "luci.model.network".init()
|
local ntm = require "luci.model.network".init()
|
||||||
local devices = ntm:get_wifidevs()
|
local devices = ntm:get_wifidevs()
|
||||||
local netlist = { }
|
local netlist = { }
|
||||||
local netdevs = { }
|
local netdevs = { }
|
||||||
local dev
|
local dev
|
||||||
|
for _, dev in ipairs(devices) do
|
||||||
|
local net
|
||||||
|
for _, net in ipairs(dev:get_wifinets()) do
|
||||||
|
netlist[#netlist+1] = net:ifname()
|
||||||
|
netdevs[net:ifname()] = dev:name()
|
||||||
|
end
|
||||||
|
end
|
||||||
local has_iwinfo = pcall(require, "iwinfo")
|
local has_iwinfo = pcall(require, "iwinfo")
|
||||||
|
|
||||||
-- Routes
|
-- Routes
|
||||||
local defroutev4 = sys.net.defaultroute()
|
local defroutev4 = sys.net.defaultroute()
|
||||||
local defroutev6 = sys.net.defaultroute6 ()
|
local defroutev6 = sys.net.defaultroute6 ()
|
||||||
|
|
||||||
%>
|
%>
|
||||||
|
|
||||||
<%+header%>
|
<%+header%>
|
||||||
|
|
||||||
|
|
||||||
|
<script type="text/javascript" src="<%=resource%>/cbi.js"></script>
|
||||||
|
|
||||||
|
<script type="text/javascript">//<![CDATA[
|
||||||
|
var iwxhr = new XHR();
|
||||||
|
|
||||||
|
var update_wifi_status = function() {
|
||||||
|
iwxhr.get('<%=luci.dispatcher.build_url("freifunk", "status", "public_status_json", table.concat(netlist, ","))%>', null,
|
||||||
|
function(x, st)
|
||||||
|
{
|
||||||
|
if (st)
|
||||||
|
{
|
||||||
|
for( var i = 0; i < st.length; i++ )
|
||||||
|
{
|
||||||
|
var iw = st[i];
|
||||||
|
var is_assoc = (iw.bssid && iw.channel);
|
||||||
|
var p = (100 / iw.quality_max * iw.quality);
|
||||||
|
var q = is_assoc ? p : -1;
|
||||||
|
|
||||||
|
var icon;
|
||||||
|
if (q < 0)
|
||||||
|
icon = "<%=resource%>/icons/signal-none.png";
|
||||||
|
else if (q == 0)
|
||||||
|
icon = "<%=resource%>/icons/signal-0.png";
|
||||||
|
else if (q < 25)
|
||||||
|
icon = "<%=resource%>/icons/signal-0-25.png";
|
||||||
|
else if (q < 50)
|
||||||
|
icon = "<%=resource%>/icons/signal-25-50.png";
|
||||||
|
else if (q < 75)
|
||||||
|
icon = "<%=resource%>/icons/signal-50-75.png";
|
||||||
|
else
|
||||||
|
icon = "<%=resource%>/icons/signal-75-100.png";
|
||||||
|
|
||||||
|
var power = document.getElementById(iw.id + '-txpower');
|
||||||
|
if (power)
|
||||||
|
power.innerHTML = String.format('%s dbm', iw.txpower);
|
||||||
|
|
||||||
|
var signal = document.getElementById(iw.id + '-signal');
|
||||||
|
if (signal)
|
||||||
|
signal.innerHTML = String.format(
|
||||||
|
'<img src="%s" title="Signal: %s db / Noise: %s db" alt="Signal Quality"</img>', icon, iw.signal, iw.noise);
|
||||||
|
|
||||||
|
var bitrate = document.getElementById(iw.id + '-bitrate');
|
||||||
|
if (bitrate)
|
||||||
|
bitrate.innerHTML = String.format('%s Mb/s', iw.bitrate ? iw.bitrate / 1000 : '?');
|
||||||
|
|
||||||
|
var ssid = document.getElementById(iw.id + '-ssid');
|
||||||
|
if (ssid)
|
||||||
|
ssid.innerHTML = iw.ssid;
|
||||||
|
|
||||||
|
var bssid = document.getElementById(iw.id + '-bssid');
|
||||||
|
if (bssid)
|
||||||
|
bssid.innerHTML = iw.bssid;
|
||||||
|
|
||||||
|
var channel = document.getElementById(iw.id + '-channel');
|
||||||
|
if (channel)
|
||||||
|
channel.innerHTML = iw.channel;
|
||||||
|
|
||||||
|
var mode = document.getElementById(iw.id + '-mode');
|
||||||
|
if (mode)
|
||||||
|
mode.innerHTML = iw.mode;
|
||||||
|
}
|
||||||
|
|
||||||
|
i = st.length - 1
|
||||||
|
var u = document.getElementById('dynuptime');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].uptime;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('dynload');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].load;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('dynmem');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].mem;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v4dst');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev4.dest;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v4gw');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev4.gateway;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v4dev');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev4.dev;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v4metr');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev4.metr;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v6dst');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev6.dest;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v6gw');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev6.gateway;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v6dev');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev6.dev;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('v6metr');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].defroutev6.metr;
|
||||||
|
}
|
||||||
|
|
||||||
|
var u = document.getElementById('dyntime');
|
||||||
|
if (u)
|
||||||
|
{
|
||||||
|
u.innerHTML = st[i].time;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
window.setTimeout(update_wifi_status, <%=interval%>);
|
||||||
|
}
|
||||||
|
)
|
||||||
|
};
|
||||||
|
update_wifi_status();
|
||||||
|
//]]></script>
|
||||||
|
|
||||||
<div class="cbi-map">
|
<div class="cbi-map">
|
||||||
<h2><%:System%></h2>
|
<h2><%:System%></h2>
|
||||||
<div class="cbi-section-node">
|
<div class="cbi-section-node">
|
||||||
<div class="cbi-value"><label class="cbi-value-title"><%:System%></label><div class="cbi-value-field"><%=system%></div></div>
|
<div class="cbi-value"><label class="cbi-value-title"><%:System%></label><div class="cbi-value-field"><%=system%></div></div>
|
||||||
<div class="cbi-value"><label class="cbi-value-title"><%:Processor%></label><div class="cbi-value-field"><%=model%></div></div>
|
<div class="cbi-value"><label class="cbi-value-title"><%:Processor%></label><div class="cbi-value-field"><%=model%></div></div>
|
||||||
<div class="cbi-value"><label class="cbi-value-title"><%:Load%></label><div class="cbi-value-field"><%=load%></div></div>
|
<div class="cbi-value"><label class="cbi-value-title"><%:Load%></label><div class="cbi-value-field" id="dynload"><%=load%></div></div>
|
||||||
<div class="cbi-value"><label class="cbi-value-title"><%:Memory%></label><div class="cbi-value-field"><%=memory%></div></div>
|
<div class="cbi-value"><label class="cbi-value-title"><%:Memory%></label><div class="cbi-value-field" id="dynmem"><%=mem%></div></div>
|
||||||
<div class="cbi-value"><label class="cbi-value-title"><%:Local Time%></label><div class="cbi-value-field"><%=local_time%></div></div>
|
<div class="cbi-value"><label class="cbi-value-title"><%:Local Time%></label><div class="cbi-value-field" id="dyntime"><%=local_time%></div></div>
|
||||||
<div class="cbi-value"><label class="cbi-value-title"><%:Uptime%></label><div class="cbi-value-field"><%=uptime%></div></div>
|
<div class="cbi-value"><label class="cbi-value-title"><%:Uptime%></label><div class="cbi-value-field" id="dynuptime"><%=uptime%></div></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -114,13 +276,13 @@ local defroutev6 = sys.net.defaultroute6 ()
|
||||||
local interface = net.iwdata.ifname or "N/A"
|
local interface = net.iwdata.ifname or "N/A"
|
||||||
%>
|
%>
|
||||||
<tr class="cbi-section-table-row cbi-rowstyle-1">
|
<tr class="cbi-section-table-row cbi-rowstyle-1">
|
||||||
<td class="cbi-value-field"><%=signal_string%></td>
|
<td class="cbi-value-field" id="<%=net:ifname()%>-signal"><%=signal_string%></td>
|
||||||
<td class="cbi-value-field"><%=bitrate%></td>
|
<td class="cbi-value-field" id="<%=net:ifname()%>-bitrate"><%=bitrate%></td>
|
||||||
<td class="cbi-value-field"><%=ssid%></td>
|
<td class="cbi-value-field" id="<%=net:ifname()%>-ssid"><%=ssid%></td>
|
||||||
<td class="cbi-value-field"><%=bssid%></td>
|
<td class="cbi-value-field" id="<%=net:ifname()%>-bssid"><%=bssid%></td>
|
||||||
<td class="cbi-value-field"><%=chan%></td>
|
<td class="cbi-value-field" id="<%=net:ifname()%>-channel"><%=chan%></td>
|
||||||
<td class="cbi-value-field"><%=mode%></td>
|
<td class="cbi-value-field" id="<%=net:ifname()%>-mode"><%=mode%></td>
|
||||||
<td class="cbi-value-field"><%=txpwr%></td>
|
<td class="cbi-value-field" id="<%=net:ifname()%>-txpower"><%=txpwr%></td>
|
||||||
<td class="cbi-value-field"><%=interface%></td>
|
<td class="cbi-value-field"><%=interface%></td>
|
||||||
</tr>
|
</tr>
|
||||||
<% end
|
<% end
|
||||||
|
@ -151,20 +313,20 @@ local defroutev6 = sys.net.defaultroute6 ()
|
||||||
<% if defroutev4 then %>
|
<% if defroutev4 then %>
|
||||||
|
|
||||||
<tr class="cbi-section-table-row cbi-rowstyle-1">
|
<tr class="cbi-section-table-row cbi-rowstyle-1">
|
||||||
<td class="cbi-value-field"><%=defroutev4.dest:string()%></td>
|
<td class="cbi-value-field" id="v4dst"><%=defroutev4.dest:string()%></td>
|
||||||
<td class="cbi-value-field"><%=defroutev4.device%></td>
|
<td class="cbi-value-field" id="v4dev"><%=defroutev4.device%></td>
|
||||||
<td class="cbi-value-field"><%=defroutev4.gateway:string()%></td>
|
<td class="cbi-value-field" id="v4gw"><%=defroutev4.gateway:string()%></td>
|
||||||
<td class="cbi-value-field"><%=defroutev4.metric%></td>
|
<td class="cbi-value-field" id="v4metr"><%=defroutev4.metric%></td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<% end
|
<% end
|
||||||
if defroutev6 then %>
|
if defroutev6 then %>
|
||||||
|
|
||||||
<tr class="cbi-section-table-row cbi-rowstyle-2">
|
<tr class="cbi-section-table-row cbi-rowstyle-2">
|
||||||
<td class="cbi-value-field"><%=defroutev6.dest:string()%></td>
|
<td class="cbi-value-field" id="v6dst"><%=defroutev6.dest:string()%></td>
|
||||||
<td class="cbi-value-field"><%=defroutev6.device%></td>
|
<td class="cbi-value-field" id="v6dev"><%=defroutev6.device%></td>
|
||||||
<td class="cbi-value-field"><%=defroutev6.nexthop:string()%></td>
|
<td class="cbi-value-field" id="v6gw"><%=defroutev6.nexthop:string()%></td>
|
||||||
<td class="cbi-value-field"><%=defroutev6.metric%></td>
|
<td class="cbi-value-field" id="v6metr"><%=defroutev6.metric%></td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<% end %>
|
<% end %>
|
||||||
|
@ -174,5 +336,4 @@ local defroutev6 = sys.net.defaultroute6 ()
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|
||||||
<%+footer%>
|
<%+footer%>
|
||||||
|
|
Loading…
Reference in a new issue