wifidog OpenWrt luci页面
OpenWrt luci添加上传下载及网络摄像头功能。
花了几天时间给OpenWrt弄了个上传下载及网络摄像头功能。对lua及luci不熟,时间花的有点多。此软件包是纯luci应用,可以安装在任意平台,网络摄像头要依赖mjpg-streamer。效果图如下:


主要源码如下:
controller/updownload.lua文件:
--[[
Other module
Description: File upload / download, web camera
Author: yuleniwo  xzm2@qq.com  QQ:529698939
]]--
module("luci.controller.other", package.seeall)
function index()
    local page = entry({"admin", "system", "other"}, alias("admin", "system", "other", "updownload"), _("Other"), 89)
    entry({"admin", "system", "other", "updownload"}, form("updownload"), _("Upload / Download"))
    if nixio.fs.access("/etc/config/mjpg-streamer") then
        entry({"admin", "system", "other", "webcam"}, call("Webcam"), _("Web Camera"))
    end
    page.i18n = "other"
    page.dependent = true
end
local translate = luci.i18n.translate
local http = luci.http
function Webcam()
    local iframe = '<iframe src="http://%s:%s@%s:%s" frameborder="no" border="0" width="800" height="600" marginwidth="0" marginheight="0" allowtransparency="yes"></iframe>'
    local html, msg, status
    local act = http.formvalue("act")
    if act then
        if act == "start" then
            luci.sys.call("/etc/init.d/mjpg-streamer start")
        elseif act == "stop" then
            luci.sys.call("/etc/init.d/mjpg-streamer stop")
            luci.sys.call("sleep 1")
        end
    end
    local v = nixio.fs.glob("/dev/video[0-9]")()
    if v then
        if luci.sys.call("pidof mjpg_streamer > /dev/null") == 0 then
            local uci, user, pwd, ip, port
            uci = require "luci.model.uci".cursor()
            user = uci:get("mjpg-streamer", "core", "username")
            pwd = uci:get("mjpg-streamer", "core", "password")
            ip = uci:get("network", "lan", "ipaddr")
            port = uci:get("mjpg-streamer", "core", "port")
            html = string.format(iframe, user, pwd, ip, port)
            status = true
        else
            status = false
            msg = translate("Service 'mjpg_streamer' not started.")
        end
    else
        msg = translate("Video device not found.")
    end
    luci.template.render("webcam", {html = html, msg = msg, status = status})
end
model/cbi/updownload.lua文件:
local fs = require "luci.fs"
local http = luci.http
ful = SimpleForm("upload", translate("Upload"), nil)
ful.reset = false
ful.submit = false
sul = ful:section(SimpleSection, "", translate("Upload file to '/tmp/upload/'"))
fu = sul:option(FileUpload, "")
fu.template = "cbi/other_upload"
um = sul:option(DummyValue, "", nil)
um.template = "cbi/other_dvalue"
fdl = SimpleForm("download", translate("Download"), nil)
fdl.reset = false
fdl.submit = false
sdl = fdl:section(SimpleSection, "", translate("Download file"))
fd = sdl:option(FileUpload, "")
fd.template = "cbi/other_download"
dm = sdl:option(DummyValue, "", nil)
dm.template = "cbi/other_dvalue"
function Download()
    local sPath, sFile, fd, block
    sPath = http.formvalue("dlfile")
    sFile = nixio.fs.basename(sPath)
    if luci.fs.isdirectory(sPath) then
        fd = io.popen('tar -C "%s" -cz .' % {sPath}, "r")
        sFile = sFile .. ".tar.gz"
    else
        fd = nixio.open(sPath, "r")
    end
    if not fd then
        dm.value = translate("Couldn't open file: ") .. sPath
        return
    end
    dm.value = nil
    http.header('Content-Disposition', 'attachment; filename="%s"' % {sFile})
    http.prepare_content("application/octet-stream")
    while true do
        block = fd:read(nixio.const.buffersize)
        if (not block) or (#block ==0) then
            break
        else
            http.write(block)
        end
    end
    fd:close()
    http.close()
end
local dir, fd
dir = "/tmp/upload/"
nixio.fs.mkdir(dir)
http.setfilehandler(
    function(meta, chunk, eof)
        if not fd then
            if not meta then return end
            fd = nixio.open(dir .. meta.file, "w")
            if not fd then
                um.value = translate("Create upload file error.")
                return
            end
        end
        if chunk and fd then
            fd:write(chunk)
        end
        if eof and fd then
            fd:close()
            fd = nil
            um.value = translate("File saved to") .. ' "/tmp/upload/' .. meta.file .. '"'
        end
    end
)
if luci.http.formvalue("upload") then
    local f = luci.http.formvalue("ulfile")
    if #f <= 0 then
        um.value = translate("No specify upload file.")
    end
elseif luci.http.formvalue("download") then
    Download()
end
local inits, attr = {}
for i, f in ipairs(fs.glob("/tmp/upload/*")) do
    attr = fs.stat(f)
    if attr then
        inits[i] = {}
        inits[i].name = fs.basename(f)
        inits[i].mtime = os.date("%Y-%m-%d %H:%M:%S", attr.mtime)
        inits[i].modestr = attr.modestr
        inits[i].size = tostring(attr.size)
        inits[i].remove = 0
        inits[i].install = false
    end
end
form = SimpleForm("filelist", translate("Upload file list"), nil)
form.reset = false
form.submit = false
tb = form:section(Table, inits)
nm = tb:option(DummyValue, "name", translate("File name"))
mt = tb:option(DummyValue, "mtime", translate("Modify time"))
ms = tb:option(DummyValue, "modestr", translate("Mode string"))
sz = tb:option(DummyValue, "size", translate("Size"))
btnrm = tb:option(Button, "remove", translate("Remove"))
btnrm.render = function(self, section, scope)
    self.inputstyle = "remove"
    Button.render(self, section, scope)
end
btnrm.write = function(self, section)
    local v = luci.fs.unlink("/tmp/upload/" .. luci.fs.basename(inits[section].name))
    if v then table.remove(inits, section) end
    return v
end
function IsIpkFile(name)
    name = name or ""
    local ext = string.lower(string.sub(name, -4, -1))
    return ext == ".ipk"
end
btnis = tb:option(Button, "install", translate("Install"))
btnis.template = "cbi/other_button"
btnis.render = function(self, section, scope)
    if not inits[section] then return false end
    if IsIpkFile(inits[section].name) then
        scope.display = ""
    else
        scope.display = "none"
    end
    self.inputstyle = "apply"
    Button.render(self, section, scope)
end
btnis.write = function(self, section)
    local r = luci.sys.exec(string.format('opkg --force-depends install "/tmp/upload/%s"', inits[section].name))
    form.description = string.format('<span style="color: red">%s</span>', r)
end
return ful, fdl, form
view/cbi/other_button.htm文件:
<%+cbi/valueheader%>
    <% if self:cfgvalue(section) ~= false then %>
        <input class="cbi-button cbi-input-<%=self.inputstyle or "button" %>" style="display: <%= display %>" type="submit"<%= attr("name", cbid) .. attr("id", cbid) .. attr("value", self.inputtitle or self.title)%> />
    <% else %>
        -
    <% end %>
<%+cbi/valuefooter%>
view/cbi/other_dvalue.htm文件:
<%+cbi/valueheader%>
<span style="color: red">
<%
    local val = self:cfgvalue(section) or self.default or ""
    write(pcdata(val))
%>
</span>
<%+cbi/valuefooter%>
view/cbi/other_upload.htm文件:
<%+cbi/valueheader%>
    <label class="cbi-value" style="display:inline-block; width: 80px" for="ulfile"><%:Upload file:%></label>
    <input class="cbi-input-file" style="width: 400px" type="file" id="ulfile" name="ulfile" />
    <input type="submit" class="cbi-button cbi-input-apply" name="upload" value="<%:Upload%>" />
<%+cbi/valuefooter%>
view/cbi/other_download.htm文件:
<%+cbi/valueheader%>
    <label class="cbi-value" style="display:inline-block; width: 80px" for="dlfile"><%:Download file:%></label>
    <input class="cbi-input-file" style="width: 400px" type="text" id="dlfile" name="dlfile" />
    <input type="submit" class="cbi-button cbi-input-apply" name="download" value="<%:Download%>" />
<%+cbi/valuefooter%>
view/webcam.htm文件:
<%+header%>
<div class="cbi-section-error"<% if not msg then %> style="display:none"<% end %>><%=msg%></div>
<form method="post" action="<%=REQUEST_URI%>"<%if status == nil then %> style="display:none"<% end %>>
    <div class="cbi-section-node">
        <div class="cbi-value cbi-value-last">
            <input type="hidden" name="act" value="<% if status then write('stop') else write('start') end %>" />
            <div class="cbi-value-field">
                <input class="cbi-button cbi-input-<% if status then write('remove') else write('apply') end %>" type="submit" value="<% if status then write(translate('Stop')) else write(translate('Start')) end %>" />
            </div>
        </div>
    </div>
</form>
<div style="text-align: center">
    <% if html then write(html) end %>
</div>
<%+footer%>
为了使添加的软件包能在openwrt源码make menuconfig时识别出来,需要在./feeds/luci/contrib/package/luci/Makefile增加如下语句:
$(eval $(call application,other,luci my other application))
软件包下载地址:luci-app-other_0.12.ipk
完整源码下载地址:luci-other_src.tar.gz
本文章由 http://www.wifidog.pro/2015/01/04/wifidog-luci.html 整理编辑,转载请注明出处