luci/libs/web/luasrc/dispatcher.lua

787 lines
18 KiB
Lua
Raw Normal View History

--[[
LuCI - Dispatcher
Description:
The request dispatcher and module dispatcher generators
FileId:
$Id$
License:
Copyright 2008 Steven Barth <steven@midlink.org>
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
]]--
--- LuCI web dispatcher.
2009-07-19 00:24:58 +00:00
local fs = require "nixio.fs"
local sys = require "luci.sys"
local init = require "luci.init"
local util = require "luci.util"
local http = require "luci.http"
local nixio = require "nixio", require "nixio.util"
module("luci.dispatcher", package.seeall)
2008-11-29 21:58:39 +00:00
context = util.threadlocal()
2009-11-10 16:02:48 +00:00
uci = require "luci.model.uci"
i18n = require "luci.i18n"
2009-11-14 18:41:16 +00:00
_M.fs = fs
2008-08-10 12:58:05 +00:00
authenticator = {}
-- Index table
local index = nil
2008-06-02 15:36:13 +00:00
-- Fastindex
local fi
--- Build the URL relative to the server webroot from given virtual path.
-- @param ... Virtual path
-- @return Relative URL
function build_url(...)
2008-12-15 10:40:45 +00:00
local path = {...}
local url = { http.getenv("SCRIPT_NAME") or "" }
local k, v
2008-12-15 10:40:45 +00:00
for k, v in pairs(context.urltoken) do
url[#url+1] = "/;"
url[#url+1] = http.urlencode(k)
url[#url+1] = "="
url[#url+1] = http.urlencode(v)
2008-12-15 10:40:45 +00:00
end
local p
for _, p in ipairs(path) do
if p:match("^[a-zA-Z0-9_%-%./,;]+$") then
url[#url+1] = "/"
url[#url+1] = p
end
end
return table.concat(url, "")
end
--- Send a 404 error code and render the "error404" template if available.
-- @param message Custom error message (optional)
-- @return false
function error404(message)
luci.http.status(404, "Not Found")
message = message or "Not Found"
require("luci.template")
if not luci.util.copcall(luci.template.render, "error404") then
luci.http.prepare_content("text/plain")
luci.http.write(message)
end
return false
end
--- Send a 500 error code and render the "error500" template if available.
-- @param message Custom error message (optional)#
-- @return false
function error500(message)
2009-03-07 13:21:27 +00:00
luci.util.perror(message)
if not context.template_header_sent then
luci.http.status(500, "Internal Server Error")
luci.http.prepare_content("text/plain")
luci.http.write(message)
else
require("luci.template")
if not luci.util.copcall(luci.template.render, "error500", {message=message}) then
luci.http.prepare_content("text/plain")
luci.http.write(message)
end
end
return false
end
function authenticator.htmlauth(validator, accs, default)
local user = luci.http.formvalue("username")
local pass = luci.http.formvalue("password")
2008-08-10 12:58:05 +00:00
if user and validator(user, pass) then
return user
end
2008-08-10 12:58:05 +00:00
require("luci.i18n")
require("luci.template")
context.path = {}
luci.template.render("sysauth", {duser=default, fuser=user})
return false
end
--- Dispatch an HTTP request.
-- @param request LuCI HTTP Request object
function httpdispatch(request, prefix)
luci.http.context.request = request
local r = {}
context.request = r
2009-01-20 19:40:14 +00:00
local pathinfo = http.urldecode(request:getenv("PATH_INFO") or "", true)
if prefix then
for _, node in ipairs(prefix) do
r[#r+1] = node
end
end
for node in pathinfo:gmatch("[^/]+") do
r[#r+1] = node
end
2009-02-26 16:45:01 +00:00
local stat, err = util.coxpcall(function()
2009-07-24 15:45:29 +00:00
dispatch(context.request)
2009-02-26 16:45:01 +00:00
end, error500)
luci.http.close()
2008-09-05 20:32:20 +00:00
--context._disable_memtrace()
2008-05-04 20:53:31 +00:00
end
--- Dispatches a LuCI virtual path.
-- @param request Virtual path
2009-07-24 15:45:29 +00:00
function dispatch(request)
--context._disable_memtrace = require "luci.debug".trap_memtrace("l")
local ctx = context
ctx.path = request
ctx.urltoken = ctx.urltoken or {}
2009-01-14 23:47:56 +00:00
local conf = require "luci.config"
assert(conf.main,
"/etc/config/luci seems to be corrupt, unable to find section 'main'")
local lang = conf.main.lang or "auto"
2009-01-14 23:47:56 +00:00
if lang == "auto" then
local aclang = http.getenv("HTTP_ACCEPT_LANGUAGE") or ""
for lpat in aclang:gmatch("[%w-]+") do
lpat = lpat and lpat:gsub("-", "_")
2009-01-14 23:47:56 +00:00
if conf.languages[lpat] then
lang = lpat
break
end
end
end
2009-01-14 23:47:56 +00:00
require "luci.i18n".setlanguage(lang)
local c = ctx.tree
local stat
2009-07-24 15:45:29 +00:00
if not c then
c = createtree()
end
2008-05-22 14:04:03 +00:00
local track = {}
2008-08-07 19:03:25 +00:00
local args = {}
2008-11-01 18:49:41 +00:00
ctx.args = args
ctx.requestargs = ctx.requestargs or args
2008-08-07 19:03:25 +00:00
local n
local t = true
local token = ctx.urltoken
local preq = {}
local freq = {}
2008-05-22 14:04:03 +00:00
for i, s in ipairs(request) do
local tkey, tval
if t then
tkey, tval = s:match(";(%w+)=([a-fA-F0-9]*)")
end
if tkey then
token[tkey] = tval
else
t = false
preq[#preq+1] = s
freq[#freq+1] = s
c = c.nodes[s]
n = i
if not c then
break
end
util.update(track, c)
if c.leaf then
break
end
2008-08-22 21:52:36 +00:00
end
2008-05-04 20:53:31 +00:00
end
2008-08-07 19:03:25 +00:00
if c and c.leaf then
for j=n+1, #request do
args[#args+1] = request[j]
freq[#freq+1] = request[j]
2008-08-07 19:03:25 +00:00
end
end
2009-09-11 10:46:06 +00:00
ctx.requestpath = ctx.requestpath or freq
ctx.path = preq
2008-05-22 14:04:03 +00:00
if track.i18n then
require("luci.i18n").loadc(track.i18n)
2008-05-04 20:53:31 +00:00
end
-- Init template engine
if (c and c.index) or not track.notemplate then
local tpl = require("luci.template")
local media = track.mediaurlbase or luci.config.main.mediaurlbase
if not pcall(tpl.Template, "themes/%s/header" % fs.basename(media)) then
media = nil
for name, theme in pairs(luci.config.themes) do
if name:sub(1,1) ~= "." and pcall(tpl.Template,
"themes/%s/header" % fs.basename(theme)) then
media = theme
end
end
assert(media, "No valid theme found")
end
tpl.context.viewns = setmetatable({
2009-07-24 15:45:29 +00:00
write = luci.http.write;
include = function(name) tpl.Template(name):render(getfenv(2)) end;
translate = function(...) return require("luci.i18n").translate(...) end;
export = function(k, v) if tpl.context.viewns[k] == nil then tpl.context.viewns[k] = v end end;
2009-07-24 15:45:29 +00:00
striptags = util.striptags;
pcdata = util.pcdata;
2009-07-24 15:45:29 +00:00
media = media;
theme = fs.basename(media);
resource = luci.config.main.resourcebase
}, {__index=function(table, key)
if key == "controller" then
2008-12-15 10:40:45 +00:00
return build_url()
elseif key == "REQUEST_URI" then
return build_url(unpack(ctx.requestpath))
else
return rawget(table, key) or _G[key]
end
end})
end
track.dependent = (track.dependent ~= false)
assert(not track.dependent or not track.auto, "Access Violation")
if track.sysauth then
local sauth = require "luci.sauth"
2008-08-22 20:04:04 +00:00
local authen = type(track.sysauth_authenticator) == "function"
and track.sysauth_authenticator
or authenticator[track.sysauth_authenticator]
local def = (type(track.sysauth) == "string") and track.sysauth
local accs = def and {track.sysauth} or track.sysauth
local sess = ctx.authsession
2008-12-15 10:40:45 +00:00
local verifytoken = false
if not sess then
sess = luci.http.getcookie("sysauth")
2009-07-25 10:47:26 +00:00
sess = sess and sess:match("^[a-f0-9]*$")
2008-12-15 10:40:45 +00:00
verifytoken = true
end
local sdat = sauth.read(sess)
local user
if sdat then
2009-06-20 07:14:36 +00:00
sdat = loadstring(sdat)
setfenv(sdat, {})
sdat = sdat()
if not verifytoken or ctx.urltoken.stok == sdat.token then
user = sdat.user
end
2009-07-25 10:47:26 +00:00
else
local eu = http.getenv("HTTP_AUTH_USER")
local ep = http.getenv("HTTP_AUTH_PASS")
if eu and ep and luci.sys.user.checkpasswd(eu, ep) then
authen = function() return eu end
end
end
if not util.contains(accs, user) then
2008-08-10 12:58:05 +00:00
if authen then
2008-12-15 10:40:45 +00:00
ctx.urltoken.stok = nil
local user, sess = authen(luci.sys.user.checkpasswd, accs, def)
if not user or not util.contains(accs, user) then
2008-08-10 12:58:05 +00:00
return
else
local sid = sess or luci.sys.uniqueid(16)
if not sess then
local token = luci.sys.uniqueid(16)
sauth.write(sid, util.get_bytecode({
user=user,
token=token,
secret=luci.sys.uniqueid(16)
}))
2008-12-15 10:40:45 +00:00
ctx.urltoken.stok = token
end
2008-12-15 10:40:45 +00:00
luci.http.header("Set-Cookie", "sysauth=" .. sid.."; path="..build_url())
ctx.authsession = sid
2009-09-11 10:46:06 +00:00
ctx.authuser = user
2008-08-10 12:58:05 +00:00
end
else
luci.http.status(403, "Forbidden")
return
end
else
ctx.authsession = sess
2009-09-11 10:46:06 +00:00
ctx.authuser = user
end
end
if track.setgroup then
luci.sys.process.setgroup(track.setgroup)
end
if track.setuser then
luci.sys.process.setuser(track.setuser)
end
local target = nil
if c then
if type(c.target) == "function" then
target = c.target
elseif type(c.target) == "table" then
target = c.target.target
end
end
if c and (c.index or type(target) == "function") then
ctx.dispatched = c
ctx.requested = ctx.requested or ctx.dispatched
end
if c and c.index then
local tpl = require "luci.template"
if util.copcall(tpl.render, "indexer", {}) then
return true
end
end
if type(target) == "function" then
util.copcall(function()
local oldenv = getfenv(target)
local module = require(c.module)
local env = setmetatable({}, {__index=
function(tbl, key)
return rawget(tbl, key) or module[key] or oldenv[key]
end})
setfenv(target, env)
end)
if type(c.target) == "table" then
target(c.target, unpack(args))
else
target(unpack(args))
end
else
2008-05-22 14:04:03 +00:00
error404()
end
end
--- Generate the dispatching index using the best possible strategy.
2008-05-22 14:04:03 +00:00
function createindex()
2008-08-06 20:20:40 +00:00
local path = luci.util.libpath() .. "/controller/"
local suff = { ".lua", ".lua.gz" }
if luci.util.copcall(require, "luci.fastindex") then
2009-07-24 15:45:29 +00:00
createindex_fastindex(path, suff)
else
2009-07-24 15:45:29 +00:00
createindex_plain(path, suff)
2008-06-02 15:36:13 +00:00
end
end
--- Generate the dispatching index using the fastindex C-indexer.
-- @param path Controller base directory
-- @param suffixes Controller file suffixes
function createindex_fastindex(path, suffixes)
2009-07-24 15:45:29 +00:00
index = {}
2008-06-02 15:36:13 +00:00
if not fi then
fi = luci.fastindex.new("index")
for _, suffix in ipairs(suffixes) do
fi.add(path .. "*" .. suffix)
fi.add(path .. "*/*" .. suffix)
end
2008-06-02 15:36:13 +00:00
end
fi.scan()
for k, v in pairs(fi.indexes) do
index[v[2]] = v[1]
end
end
--- Generate the dispatching index using the native file-cache based strategy.
-- @param path Controller base directory
-- @param suffixes Controller file suffixes
function createindex_plain(path, suffixes)
local controllers = { }
for _, suffix in ipairs(suffixes) do
2009-07-19 00:24:58 +00:00
nixio.util.consume((fs.glob(path .. "*" .. suffix)), controllers)
nixio.util.consume((fs.glob(path .. "*/*" .. suffix)), controllers)
end
2008-11-05 14:10:02 +00:00
if indexcache then
2009-07-19 00:24:58 +00:00
local cachedate = fs.stat(indexcache, "mtime")
2008-11-05 14:10:02 +00:00
if cachedate then
local realdate = 0
for _, obj in ipairs(controllers) do
2009-07-19 00:24:58 +00:00
local omtime = fs.stat(path .. "/" .. obj, "mtime")
2008-11-05 14:10:02 +00:00
realdate = (omtime and omtime > realdate) and omtime or realdate
end
2008-11-05 14:10:02 +00:00
if cachedate > realdate then
assert(
sys.process.info("uid") == fs.stat(indexcache, "uid")
and fs.stat(indexcache, "modestr") == "rw-------",
2008-11-05 14:10:02 +00:00
"Fatal: Indexcache is not sane!"
)
2008-11-05 14:10:02 +00:00
index = loadfile(indexcache)()
return index
end
end
end
2009-07-24 15:45:29 +00:00
index = {}
2008-06-02 15:36:13 +00:00
for i,c in ipairs(controllers) do
local module = "luci.controller." .. c:sub(#path+1, #c):gsub("/", ".")
for _, suffix in ipairs(suffixes) do
module = module:gsub(suffix.."$", "")
end
local mod = require(module)
local idx = mod.index
if type(idx) == "function" then
index[module] = idx
end
end
if indexcache then
local f = nixio.open(indexcache, "w", 600)
f:writeall(util.get_bytecode(index))
f:close()
end
end
--- Create the dispatching tree from the index.
-- Build the index before if it does not exist yet.
function createtree()
2009-07-24 15:45:29 +00:00
if not index then
createindex()
end
2009-07-24 15:45:29 +00:00
local ctx = context
local tree = {nodes={}}
local modi = {}
ctx.treecache = setmetatable({}, {__mode="v"})
ctx.tree = tree
2008-11-11 18:55:07 +00:00
ctx.modifiers = modi
-- Load default translation
require "luci.i18n".loadc("base")
2008-09-05 19:17:48 +00:00
local scope = setmetatable({}, {__index = luci.dispatcher})
2009-07-24 15:45:29 +00:00
for k, v in pairs(index) do
scope._NAME = k
setfenv(v, scope)
2009-07-24 15:45:29 +00:00
v()
end
2008-11-11 18:55:07 +00:00
local function modisort(a,b)
return modi[a].order < modi[b].order
end
for _, v in util.spairs(modi, modisort) do
scope._NAME = v.module
setfenv(v.func, scope)
2009-07-24 15:45:29 +00:00
v.func()
2008-11-11 18:55:07 +00:00
end
2009-07-24 15:45:29 +00:00
return tree
end
2008-11-11 18:55:07 +00:00
--- Register a tree modifier.
-- @param func Modifier function
-- @param order Modifier order value (optional)
function modifier(func, order)
context.modifiers[#context.modifiers+1] = {
func = func,
order = order or 0,
module
= getfenv(2)._NAME
2008-11-11 18:55:07 +00:00
}
end
--- Clone a node of the dispatching tree to another position.
-- @param path Virtual path destination
-- @param clone Virtual path source
-- @param title Destination node title (optional)
-- @param order Destination node order value (optional)
-- @return Dispatching tree node
function assign(path, clone, title, order)
local obj = node(unpack(path))
obj.nodes = nil
obj.module = nil
obj.title = title
obj.order = order
2008-09-15 16:50:55 +00:00
setmetatable(obj, {__index = _create_node(clone)})
return obj
end
--- Create a new dispatching node and define common parameters.
-- @param path Virtual path
-- @param target Target function to call when dispatched.
-- @param title Destination node title
-- @param order Destination node order value (optional)
-- @return Dispatching tree node
function entry(path, target, title, order)
local c = node(unpack(path))
c.target = target
c.title = title
c.order = order
2008-05-27 20:39:48 +00:00
c.module = getfenv(2)._NAME
return c
end
2008-05-04 20:53:31 +00:00
--- Fetch or create a dispatching node without setting the target module or
-- enabling the node.
-- @param ... Virtual path
-- @return Dispatching tree node
function get(...)
return _create_node({...})
end
--- Fetch or create a new dispatching node.
-- @param ... Virtual path
-- @return Dispatching tree node
2008-05-22 14:04:03 +00:00
function node(...)
2008-09-15 16:50:55 +00:00
local c = _create_node({...})
c.module = getfenv(2)._NAME
c.auto = nil
2008-05-22 14:04:03 +00:00
return c
end
function _create_node(path, cache)
if #path == 0 then
return context.tree
end
cache = cache or context.treecache
local name = table.concat(path, ".")
local c = cache[name]
if not c then
2008-11-16 13:52:50 +00:00
local new = {nodes={}, auto=true, path=util.clone(path)}
local last = table.remove(path)
2008-11-16 13:52:50 +00:00
c = _create_node(path, cache)
c.nodes[last] = new
cache[name] = new
return new
else
return c
end
end
2008-05-22 14:04:03 +00:00
-- Subdispatchers --
--- Create a redirect to another dispatching node.
-- @param ... Virtual path destination
2008-05-22 14:04:03 +00:00
function alias(...)
2008-10-30 19:09:52 +00:00
local req = {...}
return function(...)
for _, r in ipairs({...}) do
req[#req+1] = r
end
dispatch(req)
2008-05-04 20:53:31 +00:00
end
end
--- Rewrite the first x path values of the request.
-- @param n Number of path values to replace
-- @param ... Virtual path to replace removed path values with
function rewrite(n, ...)
2008-10-30 19:09:52 +00:00
local req = {...}
return function(...)
local dispatched = util.clone(context.dispatched)
for i=1,n do
2008-10-30 19:09:52 +00:00
table.remove(dispatched, 1)
end
for i, r in ipairs(req) do
table.insert(dispatched, i, r)
end
2008-10-30 19:09:52 +00:00
for _, r in ipairs({...}) do
dispatched[#dispatched+1] = r
end
2008-10-30 19:09:52 +00:00
dispatch(dispatched)
end
end
local function _call(self, ...)
if #self.argv > 0 then
return getfenv()[self.name](unpack(self.argv), ...)
else
return getfenv()[self.name](...)
end
end
--- Create a function-call dispatching target.
-- @param name Target function of local controller
-- @param ... Additional parameters passed to the function
function call(name, ...)
return {type = "call", argv = {...}, name = name, target = _call}
end
local _template = function(self, ...)
require "luci.template".render(self.view)
2008-05-27 20:39:48 +00:00
end
--- Create a template render dispatching target.
-- @param name Template to be rendered
2008-05-22 14:04:03 +00:00
function template(name)
return {type = "template", view = name, target = _template}
2008-05-22 14:04:03 +00:00
end
2008-05-04 20:53:31 +00:00
local function _cbi(self, ...)
local cbi = require "luci.cbi"
local tpl = require "luci.template"
local http = require "luci.http"
local config = self.config or {}
local maps = cbi.load(self.model, ...)
2008-10-20 22:35:11 +00:00
local state = nil
for i, res in ipairs(maps) do
2009-04-03 18:08:25 +00:00
res.flow = config
local cstate = res:parse()
2009-03-07 16:28:27 +00:00
if cstate and (not state or cstate < state) then
state = cstate
2008-11-05 21:12:19 +00:00
end
end
2008-11-05 21:12:19 +00:00
2009-06-20 07:14:36 +00:00
local function _resolve_path(path)
return type(path) == "table" and build_url(unpack(path)) or path
end
if config.on_valid_to and state and state > 0 and state < 2 then
2009-06-20 07:14:36 +00:00
http.redirect(_resolve_path(config.on_valid_to))
return
end
2008-11-02 13:26:41 +00:00
if config.on_changed_to and state and state > 1 then
2009-06-20 07:14:36 +00:00
http.redirect(_resolve_path(config.on_changed_to))
return
end
if config.on_success_to and state and state > 0 then
2009-06-20 07:14:36 +00:00
http.redirect(_resolve_path(config.on_success_to))
return
end
if config.state_handler then
if not config.state_handler(state, maps) then
return
2008-11-01 18:32:02 +00:00
end
end
2008-11-01 18:32:02 +00:00
local pageaction = true
http.header("X-CBI-State", state or 0)
if not config.noheader then
tpl.render("cbi/header", {state = state})
end
for i, res in ipairs(maps) do
res:render()
if res.pageaction == false then
pageaction = false
end
end
if not config.nofooter then
2009-04-03 18:08:25 +00:00
tpl.render("cbi/footer", {flow = config, pageaction=pageaction, state = state, autoapply = config.autoapply})
end
end
--- Create a CBI model dispatching target.
-- @param model CBI model to be rendered
function cbi(model, config)
return {type = "cbi", config = config, model = model, target = _cbi}
end
local function _arcombine(self, ...)
local argv = {...}
local target = #argv > 0 and self.targets[2] or self.targets[1]
setfenv(target.target, self.env)
target:target(unpack(argv))
end
2008-10-20 22:35:11 +00:00
--- Create a combined dispatching target for non argv and argv requests.
-- @param trg1 Overview Target
-- @param trg2 Detail Target
function arcombine(trg1, trg2)
return {type = "arcombine", env = getfenv(), target = _arcombine, targets = {trg1, trg2}}
end
local function _form(self, ...)
local cbi = require "luci.cbi"
local tpl = require "luci.template"
local http = require "luci.http"
local maps = luci.cbi.load(self.model, ...)
local state = nil
for i, res in ipairs(maps) do
local cstate = res:parse()
2009-03-07 16:28:27 +00:00
if cstate and (not state or cstate < state) then
state = cstate
end
end
http.header("X-CBI-State", state or 0)
tpl.render("header")
for i, res in ipairs(maps) do
res:render()
end
tpl.render("footer")
end
--- Create a CBI form model dispatching target.
-- @param model CBI form model tpo be rendered
function form(model)
return {type = "cbi", model = model, target = _form}
end