libs/core: implement special treatment of wireless in network model

This commit is contained in:
Jo-Philipp Wich 2009-10-15 16:30:17 +00:00
parent d8facca1b3
commit 09c5e7241a
2 changed files with 247 additions and 11 deletions

View file

@ -17,8 +17,8 @@ limitations under the License.
]]-- ]]--
local type, pairs, ipairs, table, i18n local type, pairs, ipairs, loadfile, table, i18n
= type, pairs, ipairs, table, luci.i18n = type, pairs, ipairs, loadfile, table, luci.i18n
local lmo = require "lmo" local lmo = require "lmo"
local nxo = require "nixio" local nxo = require "nixio"
@ -30,6 +30,28 @@ local uct = require "luci.model.uci.bind"
module "luci.model.network" module "luci.model.network"
-- load extensions
local ext
local handler = { }
for ext in nfs.glob(utl.libpath() .. "/model/network/*.lua") do
if nfs.access(ext) then
local m = loadfile(ext)
if m then
handler[#handler+1] = m()
end
end
end
function foreach_handler(code, ...)
local h
for _, h in ipairs(handler) do
if code(h, ...) then
return true
end
end
return false
end
local ub = uct.bind("network") local ub = uct.bind("network")
local ifs, brs, sws local ifs, brs, sws
@ -44,6 +66,12 @@ function init(cursor)
brs = { } brs = { }
sws = { } sws = { }
-- init handler
foreach_handler(function(h)
h:init(cursor)
h:find_interfaces(ifs, brs)
end)
-- read interface information -- read interface information
local n, i local n, i
for n, i in ipairs(nxo.getifaddrs()) do for n, i in ipairs(nxo.getifaddrs()) do
@ -150,6 +178,8 @@ function del_network(self, n)
ub.uci:delete("network", s['.name']) ub.uci:delete("network", s['.name'])
end end
end) end)
foreach_handler(function(h) h:del_network(n) end)
end end
return r return r
end end
@ -179,6 +209,8 @@ function rename_network(self, old, new)
ub.uci:set("network", s['.name'], "interface", new) ub.uci:set("network", s['.name'], "interface", new)
end end
end) end)
foreach_handler(function(h) h:rename_network(old, new) end)
end end
end end
return r or false return r or false
@ -198,8 +230,12 @@ function get_interfaces(self)
end end
function ignore_interface(self, x) function ignore_interface(self, x)
return (x:match("^wmaster%d") or x:match("^wifi%d") if foreach_handler(function(h) return h:ignore_interface(x) end) then
or x:match("^hwsim%d") or x:match("^imq%d") or x == "lo") return true
else
return (x:match("^wmaster%d") or x:match("^wifi%d")
or x:match("^hwsim%d") or x:match("^imq%d") or x == "lo")
end
end end
@ -218,11 +254,27 @@ function network.is_bridge(self)
end end
function network.add_interface(self, ifname) function network.add_interface(self, ifname)
local ifaces, iface
if type(ifname) ~= "string" then if type(ifname) ~= "string" then
ifname = ifname:name() ifaces = { ifname:name() }
else
ifaces = ub:list(ifname)
end end
if ifs[ifname] then
self:ifname(ub:list((self:ifname() or ''), ifname)) for _, iface in ipairs(ifaces) do
if ifs[iface] then
-- make sure the interface is removed from all networks
local i = interface(iface)
local n = i:get_network()
if n then n:del_interface(iface) end
if ifs[iface].handler then
ifs[iface].handler:add_interface(self, iface, ifs[iface])
else
self:ifname(ub:list((self:ifname() or ''), iface))
end
end
end end
end end
@ -230,7 +282,12 @@ function network.del_interface(self, ifname)
if type(ifname) ~= "string" then if type(ifname) ~= "string" then
ifname = ifname:name() ifname = ifname:name()
end end
self:ifname(ub:list((self:ifname() or ''), nil, ifname))
if ifs[ifname] and ifs[ifname].handler then
ifs[ifname].handler:del_interface(self, ifname, ifs[ifname])
else
self:ifname(ub:list((self:ifname() or ''), nil, ifname))
end
end end
function network.get_interfaces(self) function network.get_interfaces(self)
@ -242,6 +299,11 @@ function network.get_interfaces(self)
ifaces[#ifaces+1] = interface(iface) ifaces[#ifaces+1] = interface(iface)
end end
end end
for iface, _ in pairs(ifs) do
if ifs[iface].network == self:name() then
ifaces[#ifaces+1] = interface(iface)
end
end
return ifaces return ifaces
end end
@ -259,6 +321,12 @@ function network.contains_interface(self, iface)
end end
end end
for i, _ in pairs(ifs) do
if ifs[i].dev and ifs[i].dev.network == self:name() then
return true
end
end
return false return false
end end
@ -289,8 +357,8 @@ function interface.ip6addrs(self)
end end
function interface.type(self) function interface.type(self)
if iwi.type(self.ifname) and iwi.type(self.ifname) ~= "dummy" then if self.dev and self.dev.type then
return "wifi" return self.dev.type
elseif brs[self.ifname] then elseif brs[self.ifname] then
return "bridge" return "bridge"
elseif sws[self.ifname] or self.ifname:match("%.") then elseif sws[self.ifname] or self.ifname:match("%.") then
@ -300,6 +368,22 @@ function interface.type(self)
end end
end end
function interface.shortname(self)
if self.dev and self.dev.handler then
return self.dev.handler:shortname(self)
else
return self.ifname
end
end
function interface.get_i18n(self)
if self.dev and self.dev.handler then
return self.dev.handler:get_i18n(self)
else
return "%s: %q" %{ self:get_type_i18n(), self:name() }
end
end
function interface.get_type_i18n(self) function interface.get_type_i18n(self)
local x = self:type() local x = self:type()
if x == "wifi" then if x == "wifi" then
@ -373,6 +457,10 @@ function interface.rx_packets(self)
end end
function interface.get_network(self) function interface.get_network(self)
if self.dev and self.dev.network then
self.network = _M:get_network(self.dev.network)
end
if not self.network then if not self.network then
local net local net
for _, net in ipairs(_M:get_networks()) do for _, net in ipairs(_M:get_networks()) do

View file

@ -0,0 +1,148 @@
--[[
LuCI - Network model - Wireless extension
Copyright 2009 Jo-Philipp Wich <xm@subsignal.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.
]]--
local pairs, i18n, uci = pairs, luci.i18n, luci.model.uci
local iwi = require "iwinfo"
local utl = require "luci.util"
local uct = require "luci.model.uci.bind"
module "luci.model.network.wireless"
local ub = uct.bind("wireless")
local st, ifs
function init(self, cursor)
cursor:unload("wireless")
cursor:load("wireless")
ub:init(cursor)
st = uci.cursor_state()
ifs = { }
local count = 0
ub.uci:foreach("wireless", "wifi-iface",
function(s)
count = count + 1
local device = s.device or "wlan0"
local state = st:get_all("wireless", s['.name'])
local name = state.ifname or device .. ".network" .. count
ifs[state and state.ifname or name] = {
idx = count,
name = state and state.ifname or name,
rawname = state and state.ifname or name,
flags = { },
ipaddrs = { },
ip6addrs = { },
type = "wifi",
network = s.network,
handler = self,
wifi = state or s,
sid = s['.name']
}
end)
end
function shortname(self, iface)
if iface.dev and iface.dev.wifi then
return "%s %q" %{
i18n.translate("a_s_if_iwmode_" .. (iface.dev.wifi.mode or "ap")),
iface.dev.wifi.ssid or iface.dev.wifi.bssid or "(hidden)"
}
else
return iface:name()
end
end
function get_i18n(self, iface)
if iface.dev and iface.dev.wifi then
return "%s: %s %q" %{
i18n.translate("a_s_if_wifinet", "Wireless Network"),
i18n.translate("a_s_if_iwmode_" .. (iface.dev.wifi.mode or "ap"), iface.dev.wifi.mode or "AP"),
iface.dev.wifi.ssid or iface.dev.wifi.bssid or "(hidden)"
}
else
return "%s: %q" %{ i18n.translate("a_s_if_wifinet", "Wireless Network"), iface:name() }
end
end
function rename_network(self, old, new)
local i
for i, _ in pairs(ifs) do
if ifs[i].network == old then
ifs[i].network = new
end
end
ub.uci:foreach("wireless", "wifi-iface",
function(s)
if s.network == old then
if new then
ub.uci:set("wireless", s['.name'], "network", new)
else
ub.uci:delete("wireless", s['.name'], "network")
end
end
end)
end
function del_network(self, old)
return self:rename_network(old, nil)
end
function find_interfaces(self, iflist, brlist)
local iface
for iface, _ in pairs(ifs) do
iflist[iface] = ifs[iface]
end
end
function ignore_interface(self, iface)
if ifs and ifs[iface] then
return false
else
return iwi.type(iface) and true or false
end
end
function add_interface(self, net, iface)
if ifs and ifs[iface] and ifs[iface].sid then
ub.uci:set("wireless", ifs[iface].sid, "network", net:name())
ifs[iface].network = net:name()
return true
end
return false
end
function del_interface(self, net, iface)
if ifs and ifs[iface] and ifs[iface].sid then
ub.uci:delete("wireless", ifs[iface].sid, "network")
--return true
end
return false
end
return _M