luci-mod-admin-full: protect clock, flash and opkg ops with submit token

* Use post_on() target to require csrf token verification for modifying actions
* Ensure that package and flash operation handlers guard modifying operations
  with parameter check

Signed-off-by: Jo-Philipp Wich <jow@openwrt.org>
This commit is contained in:
Jo-Philipp Wich 2015-10-20 21:01:41 +02:00
parent 562c47e5fd
commit b5826f1ffb
4 changed files with 258 additions and 219 deletions

View file

@ -9,12 +9,12 @@ function index()
entry({"admin", "system"}, alias("admin", "system", "system"), _("System"), 30).index = true entry({"admin", "system"}, alias("admin", "system", "system"), _("System"), 30).index = true
entry({"admin", "system", "system"}, cbi("admin_system/system"), _("System"), 1) entry({"admin", "system", "system"}, cbi("admin_system/system"), _("System"), 1)
entry({"admin", "system", "clock_status"}, call("action_clock_status")) entry({"admin", "system", "clock_status"}, post_on({ set = true }, "action_clock_status"))
entry({"admin", "system", "admin"}, cbi("admin_system/admin"), _("Administration"), 2) entry({"admin", "system", "admin"}, cbi("admin_system/admin"), _("Administration"), 2)
if fs.access("/bin/opkg") then if fs.access("/bin/opkg") then
entry({"admin", "system", "packages"}, call("action_packages"), _("Software"), 10) entry({"admin", "system", "packages"}, post_on({ exec = "1" }, "action_packages"), _("Software"), 10)
entry({"admin", "system", "packages", "ipkg"}, form("admin_system/ipkg")) entry({"admin", "system", "packages", "ipkg"}, form("admin_system/ipkg"))
end end
@ -31,7 +31,7 @@ function index()
entry({"admin", "system", "leds"}, cbi("admin_system/leds"), _("<abbr title=\"Light Emitting Diode\">LED</abbr> Configuration"), 60) entry({"admin", "system", "leds"}, cbi("admin_system/leds"), _("<abbr title=\"Light Emitting Diode\">LED</abbr> Configuration"), 60)
end end
entry({"admin", "system", "flashops"}, call("action_flashops"), _("Backup / Flash Firmware"), 70) entry({"admin", "system", "flashops"}, post_on({ exec = "1" }, "action_flashops"), _("Backup / Flash Firmware"), 70)
entry({"admin", "system", "flashops", "backupfiles"}, form("admin_system/backupfiles")) entry({"admin", "system", "flashops", "backupfiles"}, form("admin_system/backupfiles"))
entry({"admin", "system", "reboot"}, template("admin_system/reboot"), _("Reboot"), 90) entry({"admin", "system", "reboot"}, template("admin_system/reboot"), _("Reboot"), 90)
@ -56,7 +56,8 @@ end
function action_packages() function action_packages()
local fs = require "nixio.fs" local fs = require "nixio.fs"
local ipkg = require "luci.model.ipkg" local ipkg = require "luci.model.ipkg"
local submit = luci.http.formvalue("submit") local submit = (luci.http.formvalue("exec") == "1")
local update, upgrade
local changes = false local changes = false
local install = { } local install = { }
local remove = { } local remove = { }
@ -76,59 +77,62 @@ function action_packages()
query = (query ~= '') and query or nil query = (query ~= '') and query or nil
-- Packets to be installed -- Modifying actions
local ninst = submit and luci.http.formvalue("install") if submit then
local uinst = nil -- Packets to be installed
local ninst = luci.http.formvalue("install")
local uinst = nil
-- Install from URL -- Install from URL
local url = luci.http.formvalue("url") local url = luci.http.formvalue("url")
if url and url ~= '' and submit then if url and url ~= '' then
uinst = url uinst = url
end end
-- Do install -- Do install
if ninst then if ninst then
install[ninst], out, err = ipkg.install(ninst) install[ninst], out, err = ipkg.install(ninst)
stdout[#stdout+1] = out
stderr[#stderr+1] = err
changes = true
end
if uinst then
local pkg
for pkg in luci.util.imatch(uinst) do
install[uinst], out, err = ipkg.install(pkg)
stdout[#stdout+1] = out stdout[#stdout+1] = out
stderr[#stderr+1] = err stderr[#stderr+1] = err
changes = true changes = true
end end
end
-- Remove packets if uinst then
local rem = submit and luci.http.formvalue("remove") local pkg
if rem then for pkg in luci.util.imatch(uinst) do
remove[rem], out, err = ipkg.remove(rem) install[uinst], out, err = ipkg.install(pkg)
stdout[#stdout+1] = out stdout[#stdout+1] = out
stderr[#stderr+1] = err stderr[#stderr+1] = err
changes = true changes = true
end end
end
-- Remove packets
local rem = luci.http.formvalue("remove")
if rem then
remove[rem], out, err = ipkg.remove(rem)
stdout[#stdout+1] = out
stderr[#stderr+1] = err
changes = true
end
-- Update all packets -- Update all packets
local update = luci.http.formvalue("update") update = luci.http.formvalue("update")
if update then if update then
update, out, err = ipkg.update() update, out, err = ipkg.update()
stdout[#stdout+1] = out stdout[#stdout+1] = out
stderr[#stderr+1] = err stderr[#stderr+1] = err
end end
-- Upgrade all packets -- Upgrade all packets
local upgrade = luci.http.formvalue("upgrade") upgrade = luci.http.formvalue("upgrade")
if upgrade then if upgrade then
upgrade, out, err = ipkg.upgrade() upgrade, out, err = ipkg.upgrade()
stdout[#stdout+1] = out stdout[#stdout+1] = out
stderr[#stderr+1] = err stderr[#stderr+1] = err
end
end end
@ -168,8 +172,11 @@ function action_packages()
end end
function action_flashops() function action_flashops()
local sys = require "luci.sys" local http = require "luci.http"
local fs = require "nixio.fs" local sys = require "luci.sys"
local fs = require "nixio.fs"
local submit = (http.formvalue("exec") == "1")
local upgrade_avail = fs.access("/lib/upgrade/platform.sh") local upgrade_avail = fs.access("/lib/upgrade/platform.sh")
local reset_avail = os.execute([[grep '"rootfs_data"' /proc/mtd >/dev/null 2>&1]]) == 0 local reset_avail = os.execute([[grep '"rootfs_data"' /proc/mtd >/dev/null 2>&1]]) == 0
@ -208,96 +215,108 @@ function action_flashops()
return size return size
end end
--
-- Handle modifying actions
--
if submit then
local fp local fp
luci.http.setfilehandler( http.setfilehandler(
function(meta, chunk, eof) function(meta, chunk, eof)
if not fp then if not fp then
if meta and meta.name == "image" then if meta and meta.name == "image" then
fp = io.open(image_tmp, "w") fp = io.open(image_tmp, "w")
else else
fp = io.popen(restore_cmd, "w") fp = io.popen(restore_cmd, "w")
end
end
if chunk then
fp:write(chunk)
end
if eof then
fp:close()
end end
end end
if chunk then )
fp:write(chunk)
end
if eof then
fp:close()
end
end
)
if luci.http.formvalue("backup") then if http.formvalue("backup") then
-- --
-- Assemble file list, generate backup -- Assemble file list, generate backup
-- --
local reader = ltn12_popen(backup_cmd) local reader = ltn12_popen(backup_cmd)
luci.http.header('Content-Disposition', 'attachment; filename="backup-%s-%s.tar.gz"' % { http.header('Content-Disposition', 'attachment; filename="backup-%s-%s.tar.gz"' % {
luci.sys.hostname(), os.date("%Y-%m-%d")}) luci.sys.hostname(), os.date("%Y-%m-%d")})
luci.http.prepare_content("application/x-targz") http.prepare_content("application/x-targz")
luci.ltn12.pump.all(reader, luci.http.write) luci.ltn12.pump.all(reader, http.write)
elseif luci.http.formvalue("restore") then return
--
-- Unpack received .tar.gz elseif http.formvalue("restore") then
-- --
local upload = luci.http.formvalue("archive") -- Unpack received .tar.gz
if upload and #upload > 0 then --
luci.template.render("admin_system/applyreboot") local upload = http.formvalue("archive")
luci.sys.reboot() if upload and #upload > 0 then
end luci.template.render("admin_system/applyreboot")
elseif luci.http.formvalue("image") or luci.http.formvalue("step") then luci.sys.reboot()
-- return
-- Initiate firmware flash
--
local step = tonumber(luci.http.formvalue("step") or 1)
if step == 1 then
if image_supported() then
luci.template.render("admin_system/upgrade", {
checksum = image_checksum(),
storage = storage_size(),
size = (fs.stat(image_tmp, "size") or 0),
keep = (not not luci.http.formvalue("keep"))
})
else
fs.unlink(image_tmp)
luci.template.render("admin_system/flashops", {
reset_avail = reset_avail,
upgrade_avail = upgrade_avail,
image_invalid = true
})
end end
--
-- Start sysupgrade flash elseif http.formvalue("image") or http.formvalue("step") then
-- --
elseif step == 2 then -- Initiate firmware flash
local keep = (luci.http.formvalue("keep") == "1") and "" or "-n" --
local step = tonumber(http.formvalue("step") or 1)
if step == 1 then
if image_supported() then
luci.template.render("admin_system/upgrade", {
checksum = image_checksum(),
storage = storage_size(),
size = (fs.stat(image_tmp, "size") or 0),
keep = (not not http.formvalue("keep"))
})
else
fs.unlink(image_tmp)
luci.template.render("admin_system/flashops", {
reset_avail = reset_avail,
upgrade_avail = upgrade_avail,
image_invalid = true
})
end
return
--
-- Start sysupgrade flash
--
elseif step == 2 then
local keep = (http.formvalue("keep") == "1") and "" or "-n"
luci.template.render("admin_system/applyreboot", {
title = luci.i18n.translate("Flashing..."),
msg = luci.i18n.translate("The system is flashing now.<br /> DO NOT POWER OFF THE DEVICE!<br /> Wait a few minutes before you try to reconnect. It might be necessary to renew the address of your computer to reach the device again, depending on your settings."),
addr = (#keep > 0) and "192.168.1.1" or nil
})
fork_exec("killall dropbear uhttpd; sleep 1; /sbin/sysupgrade %s %q" %{ keep, image_tmp })
return
end
elseif reset_avail and http.formvalue("reset") then
--
-- Reset system
--
luci.template.render("admin_system/applyreboot", { luci.template.render("admin_system/applyreboot", {
title = luci.i18n.translate("Flashing..."), title = luci.i18n.translate("Erasing..."),
msg = luci.i18n.translate("The system is flashing now.<br /> DO NOT POWER OFF THE DEVICE!<br /> Wait a few minutes before you try to reconnect. It might be necessary to renew the address of your computer to reach the device again, depending on your settings."), msg = luci.i18n.translate("The system is erasing the configuration partition now and will reboot itself when finished."),
addr = (#keep > 0) and "192.168.1.1" or nil addr = "192.168.1.1"
}) })
fork_exec("killall dropbear uhttpd; sleep 1; /sbin/sysupgrade %s %q" %{ keep, image_tmp }) fork_exec("killall dropbear uhttpd; sleep 1; mtd -r erase rootfs_data")
return
end end
elseif reset_avail and luci.http.formvalue("reset") then
--
-- Reset system
--
luci.template.render("admin_system/applyreboot", {
title = luci.i18n.translate("Erasing..."),
msg = luci.i18n.translate("The system is erasing the configuration partition now and will reboot itself when finished."),
addr = "192.168.1.1"
})
fork_exec("killall dropbear uhttpd; sleep 1; mtd -r erase rootfs_data")
else
--
-- Overview
--
luci.template.render("admin_system/flashops", {
reset_avail = reset_avail,
upgrade_avail = upgrade_avail
})
end end
--
-- Overview
--
luci.template.render("admin_system/flashops", {
reset_avail = reset_avail,
upgrade_avail = upgrade_avail
})
end end
function action_passwd() function action_passwd()

View file

@ -17,8 +17,8 @@
btn.disabled = true; btn.disabled = true;
btn.value = '<%:Synchronizing...%>'; btn.value = '<%:Synchronizing...%>';
XHR.get('<%=url('admin/system/clock_status')%>', (new XHR()).post('<%=url('admin/system/clock_status')%>',
{ set: Math.floor((new Date()).getTime() / 1000) }, { token: '<%=token%>', set: Math.floor((new Date()).getTime() / 1000) },
function() function()
{ {
btn.disabled = false; btn.disabled = false;

View file

@ -1,6 +1,6 @@
<%# <%#
Copyright 2008 Steven Barth <steven@midlink.org> Copyright 2008 Steven Barth <steven@midlink.org>
Copyright 2008 Jo-Philipp Wich <jow@openwrt.org> Copyright 2008-2015 Jo-Philipp Wich <jow@openwrt.org>
Licensed to the public under the Apache License 2.0. Licensed to the public under the Apache License 2.0.
-%> -%>
@ -17,7 +17,9 @@
<fieldset class="cbi-section"> <fieldset class="cbi-section">
<legend><%:Backup / Restore%></legend> <legend><%:Backup / Restore%></legend>
<form method="post" action="<%=REQUEST_URI%>" enctype="multipart/form-data"> <form method="post" action="<%=url('admin/system/flashops')%>" enctype="multipart/form-data">
<input type="hidden" name="exec" value="1" />
<input type="hidden" name="token" value="<%=token%>" />
<div class="cbi-section-descr"><%:Click "Generate archive" to download a tar archive of the current configuration files. To reset the firmware to its initial state, click "Perform reset" (only possible with squashfs images).%></div> <div class="cbi-section-descr"><%:Click "Generate archive" to download a tar archive of the current configuration files. To reset the firmware to its initial state, click "Perform reset" (only possible with squashfs images).%></div>
<div class="cbi-section-node"> <div class="cbi-section-node">
<div class="cbi-value<% if not reset_avail then %> cbi-value-last<% end %>"> <div class="cbi-value<% if not reset_avail then %> cbi-value-last<% end %>">
@ -54,7 +56,9 @@
<fieldset class="cbi-section"> <fieldset class="cbi-section">
<legend><%:Flash new firmware image%></legend> <legend><%:Flash new firmware image%></legend>
<% if upgrade_avail then %> <% if upgrade_avail then %>
<form method="post" action="<%=REQUEST_URI%>" enctype="multipart/form-data"> <form method="post" action="<%=url('admin/system/flashops')%>" enctype="multipart/form-data">
<input type="hidden" name="exec" value="1" />
<input type="hidden" name="token" value="<%=token%>" />
<div class="cbi-section-descr"><%:Upload a sysupgrade-compatible image here to replace the running firmware. Check "Keep settings" to retain the current configuration (requires an OpenWrt compatible firmware image).%></div> <div class="cbi-section-descr"><%:Upload a sysupgrade-compatible image here to replace the running firmware. Check "Keep settings" to retain the current configuration (requires an OpenWrt compatible firmware image).%></div>
<div class="cbi-section-node"> <div class="cbi-section-node">
<div class="cbi-value"> <div class="cbi-value">

View file

@ -46,17 +46,18 @@ end
<h2 name="content"><%:Software%></h2> <h2 name="content"><%:Software%></h2>
<form method="post" action="<%=REQUEST_URI%>"> <div class="cbi-map">
<div class="cbi-map">
<ul class="cbi-tabmenu"> <ul class="cbi-tabmenu">
<li class="cbi-tab"><a href="#"><%:Actions%></a></li> <li class="cbi-tab"><a href="#"><%:Actions%></a></li>
<li class="cbi-tab-disabled"><a href="<%=REQUEST_URI%>/ipkg"><%:Configuration%></a></li> <li class="cbi-tab-disabled"><a href="<%=REQUEST_URI%>/ipkg"><%:Configuration%></a></li>
</ul> </ul>
<form method="post" action="<%=REQUEST_URI%>">
<input type="hidden" name="exec" value="1" />
<input type="hidden" name="token" value="<%=token%>" />
<fieldset class="cbi-section"> <fieldset class="cbi-section">
<fieldset class="cbi-section-node"> <fieldset class="cbi-section-node">
<% if (install and next(install)) or (remove and next(remove)) or update or upgrade then %> <% if (install and next(install)) or (remove and next(remove)) or update or upgrade then %>
<div class="cbi-value"> <div class="cbi-value">
@ -80,7 +81,7 @@ end
<% else %> <% else %>
<%:No package lists available%> <%:No package lists available%>
<% end %> <% end %>
<input type="button" onclick="location.href='?update=1'" href="#" class="cbi-button cbi-button-apply" style="margin-left:3em" value="<%:Update lists%>" /> <input type="submit" name="update" href="#" class="cbi-button cbi-button-apply" style="margin-left:3em" value="<%:Update lists%>" />
</div> </div>
<% end %> <% end %>
@ -101,7 +102,7 @@ end
<label class="cbi-value-title"><%:Download and install package%>:</label> <label class="cbi-value-title"><%:Download and install package%>:</label>
<div class="cbi-value-field"> <div class="cbi-value-field">
<input type="text" name="url" size="30" value="" /> <input type="text" name="url" size="30" value="" />
<input class="cbi-button cbi-input-save" type="submit" name="submit" value="<%:OK%>" /> <input class="cbi-button cbi-input-save" type="submit" name="exec" value="<%:OK%>" />
</div> </div>
</div> </div>
@ -114,83 +115,98 @@ end
</div> </div>
</fieldset> </fieldset>
</fieldset> </fieldset>
<br /> </form>
<h3><%:Status%></h3>
<ul class="cbi-tabmenu"> <h3><%:Status%></h3>
<li class="cbi-tab<% if display ~= "installed" then %>-disabled<% end %>"><a href="?display=installed&amp;query=<%=pcdata(query)%>"><%:Installed packages%><% if query then %> (<%=pcdata(query)%>)<% end %></a></li>
<li class="cbi-tab<% if display ~= "available" then %>-disabled<% end %>"><a href="?display=available&amp;query=<%=pcdata(query)%>"><%:Available packages%><% if query then %> (<%=pcdata(query)%>)<% end %></a></li>
</ul>
<% if display ~= "available" then %>
<fieldset class="cbi-section"> <ul class="cbi-tabmenu">
<table class="cbi-section-table" style="width:100%"> <li class="cbi-tab<% if display ~= "installed" then %>-disabled<% end %>"><a href="?display=installed&amp;query=<%=pcdata(query)%>"><%:Installed packages%><% if query then %> (<%=pcdata(query)%>)<% end %></a></li>
<tr class="cbi-section-table-titles"> <li class="cbi-tab<% if display ~= "available" then %>-disabled<% end %>"><a href="?display=available&amp;query=<%=pcdata(query)%>"><%:Available packages%><% if query then %> (<%=pcdata(query)%>)<% end %></a></li>
<th class="cbi-section-table-cell" style="text-align:left">&#160;</th> </ul>
<th class="cbi-section-table-cell" style="text-align:left"><%:Package name%></th>
<th class="cbi-section-table-cell" style="text-align:left"><%:Version%></th> <% if display ~= "available" then %>
</tr> <fieldset class="cbi-section">
<% local empty = true; luci.model.ipkg.list_installed(querypat, function(n, v, s, d) empty = false; filter[n] = true %> <table class="cbi-section-table" style="width:100%">
<tr class="cbi-section-table-row cbi-rowstyle-<%=rowstyle()%>"> <tr class="cbi-section-table-titles">
<td style="text-align:left; width:10%"><a onclick="return window.confirm('<%:Remove%> &quot;<%=luci.util.pcdata(n)%>&quot; ?')" href="<%=REQUEST_URI%>?submit=1&amp;remove=<%=luci.util.pcdata(n)%>"><%:Remove%></a></td> <th class="cbi-section-table-cell" style="text-align:left">&#160;</th>
<td style="text-align:left"><%=luci.util.pcdata(n)%></td> <th class="cbi-section-table-cell" style="text-align:left"><%:Package name%></th>
<td style="text-align:left"><%=luci.util.pcdata(v)%></td> <th class="cbi-section-table-cell" style="text-align:left"><%:Version%></th>
</tr> </tr>
<% end) %> <% local empty = true; luci.model.ipkg.list_installed(querypat, function(n, v, s, d) empty = false; filter[n] = true %>
<% if empty then %> <tr class="cbi-section-table-row cbi-rowstyle-<%=rowstyle()%>">
<tr class="cbi-section-table-row"> <td style="text-align:left; width:10%">
<td style="text-align:left">&#160;</td> <form method="post" class="inline" action="<%=REQUEST_URI%>">
<td style="text-align:left"><em><%:none%></em></td> <input type="hidden" name="exec" value="1" />
<td style="text-align:left"><em><%:none%></em></td> <input type="hidden" name="token" value="<%=token%>" />
</tr> <input type="hidden" name="remove" value="<%=pcdata(n)%>" />
<% end %> <a onclick="window.confirm('<%:Remove%> &quot;<%=luci.util.pcdata(n)%>&quot; ?') && this.parentNode.submit(); return false" href="#"><%:Remove%></a>
</table> </form>
</fieldset> </td>
<% else %> <td style="text-align:left"><%=luci.util.pcdata(n)%></td>
<fieldset class="cbi-section"> <td style="text-align:left"><%=luci.util.pcdata(v)%></td>
<% if not querypat then %> </tr>
<ul class="cbi-tabmenu"> <% end) %>
<% local i; for i = 65, 90 do %> <% if empty then %>
<li class="cbi-tab<% if letter ~= i then %>-disabled<% end %>"><a href="?display=available&amp;letter=<%=string.char(i)%>"><%=string.char(i)%></a></li> <tr class="cbi-section-table-row">
<% end %> <td style="text-align:left">&#160;</td>
<li class="cbi-tab<% if letter ~= 35 then %>-disabled<% end %>"><a href="?display=available&amp;letter=%23">#</a></li> <td style="text-align:left"><em><%:none%></em></td>
</ul> <td style="text-align:left"><em><%:none%></em></td>
<div class="cbi-section-node"> </tr>
<% end %> <% end %>
<table class="cbi-section-table" style="width:100%"> </table>
<tr class="cbi-section-table-titles"> </fieldset>
<th class="cbi-section-table-cell" style="text-align:left">&#160;</th> <% else %>
<th class="cbi-section-table-cell" style="text-align:left"><%:Package name%></th> <fieldset class="cbi-section">
<th class="cbi-section-table-cell" style="text-align:left"><%:Version%></th> <% if not querypat then %>
<th class="cbi-section-table-cell" style="text-align:right"><%:Size (.ipk)%></th> <ul class="cbi-tabmenu">
<th class="cbi-section-table-cell" style="text-align:left"><%:Description%></th> <% local i; for i = 65, 90 do %>
</tr> <li class="cbi-tab<% if letter ~= i then %>-disabled<% end %>"><a href="?display=available&amp;letter=<%=string.char(i)%>"><%=string.char(i)%></a></li>
<% local empty = true; opkg_list(querypat or letterpat, function(n, v, s, d) if filter[n] then return end; empty = false %> <% end %>
<tr class="cbi-section-table-row cbi-rowstyle-<%=rowstyle()%>"> <li class="cbi-tab<% if letter ~= 35 then %>-disabled<% end %>"><a href="?display=available&amp;letter=%23">#</a></li>
<td style="text-align:left; width:10%"><a onclick="return window.confirm('<%:Install%> &quot;<%=luci.util.pcdata(n)%>&quot; ?')" href="<%=REQUEST_URI%>?submit=1&amp;install=<%=luci.util.pcdata(n)%>"><%:Install%></a></td> </ul>
<td style="text-align:left"><%=luci.util.pcdata(n)%></td> <div class="cbi-section-node">
<td style="text-align:left"><%=luci.util.pcdata(v)%></td>
<td style="text-align:right"><%=luci.util.pcdata(s)%></td>
<td style="text-align:left"><%=luci.util.pcdata(d)%></td>
</tr>
<% end) %>
<% if empty then %>
<tr class="cbi-section-table-row">
<td style="text-align:left">&#160;</td>
<td style="text-align:left"><em><%:none%></em></td>
<td style="text-align:left"><em><%:none%></em></td>
<td style="text-align:right"><em><%:none%></em></td>
<td style="text-align:left"><em><%:none%></em></td>
</tr>
<% end %>
</table>
<% if not querypat then %>
</div>
<% end %>
</fieldset>
<% end %> <% end %>
</div> <table class="cbi-section-table" style="width:100%">
</form> <tr class="cbi-section-table-titles">
<th class="cbi-section-table-cell" style="text-align:left">&#160;</th>
<th class="cbi-section-table-cell" style="text-align:left"><%:Package name%></th>
<th class="cbi-section-table-cell" style="text-align:left"><%:Version%></th>
<th class="cbi-section-table-cell" style="text-align:right"><%:Size (.ipk)%></th>
<th class="cbi-section-table-cell" style="text-align:left"><%:Description%></th>
</tr>
<% local empty = true; opkg_list(querypat or letterpat, function(n, v, s, d) if filter[n] then return end; empty = false %>
<tr class="cbi-section-table-row cbi-rowstyle-<%=rowstyle()%>">
<td style="text-align:left; width:10%">
<form method="post" class="inline" action="<%=REQUEST_URI%>">
<input type="hidden" name="exec" value="1" />
<input type="hidden" name="token" value="<%=token%>" />
<input type="hidden" name="install" value="<%=pcdata(n)%>" />
<a onclick="window.confirm('<%:Install%> &quot;<%=luci.util.pcdata(n)%>&quot; ?') && this.parentNode.submit(); return false" href="#"><%:Install%></a>
</form>
</td>
<td style="text-align:left"><%=luci.util.pcdata(n)%></td>
<td style="text-align:left"><%=luci.util.pcdata(v)%></td>
<td style="text-align:right"><%=luci.util.pcdata(s)%></td>
<td style="text-align:left"><%=luci.util.pcdata(d)%></td>
</tr>
<% end) %>
<% if empty then %>
<tr class="cbi-section-table-row">
<td style="text-align:left">&#160;</td>
<td style="text-align:left"><em><%:none%></em></td>
<td style="text-align:left"><em><%:none%></em></td>
<td style="text-align:right"><em><%:none%></em></td>
<td style="text-align:left"><em><%:none%></em></td>
</tr>
<% end %>
</table>
<% if not querypat then %>
</div>
<% end %>
</fieldset>
<% end %>
</div>
<%+footer%> <%+footer%>