-- Copyright 2008 Steven Barth -- Copyright 2008-2011 Jo-Philipp Wich -- Licensed to the public under the Apache License 2.0. local fs = require "nixio.fs" local ut = require "luci.util" local pt = require "luci.tools.proto" local nw = require "luci.model.network" local fw = require "luci.model.firewall" arg[1] = arg[1] or "" local has_dnsmasq = fs.access("/etc/config/dhcp") local has_firewall = fs.access("/etc/config/firewall") m = Map("network", translate("Interfaces") .. " - " .. arg[1]:upper(), translate("On this page you can configure the network interfaces. You can bridge several interfaces by ticking the \"bridge interfaces\" field and enter the names of several network interfaces separated by spaces. You can also use VLAN notation INTERFACE.VLANNR (e.g.: eth0.1).")) m.redirect = luci.dispatcher.build_url("admin", "network", "network") m:chain("wireless") m:chain("luci") if has_firewall then m:chain("firewall") end nw.init(m.uci) fw.init(m.uci) local net = nw:get_network(arg[1]) local function set_ifstate(name, option, value) local found = false m.uci:foreach("luci", "ifstate", function (s) if s.interface == name then m.uci:set("luci", s[".name"], option, value) found = true return false end end) if not found then local sid = m.uci:add("luci", "ifstate") m.uci:set("luci", sid, "interface", name) m.uci:set("luci", sid, option, value) end m.uci:save("luci") end local function get_ifstate(name, option) local val m.uci:foreach("luci", "ifstate", function (s) if s.interface == name then val = m.uci:get("luci", s[".name"], option) return false end end) return val end local function backup_ifnames(is_bridge) if not net:is_floating() and not get_ifstate(net:name(), "ifname") then local ifcs = net:get_interfaces() or { net:get_interface() } if ifcs then local _, ifn local ifns = { } for _, ifn in ipairs(ifcs) do local wif = ifn:get_wifinet() ifns[#ifns+1] = wif and wif:id() or ifn:name() end if #ifns > 0 then set_ifstate(net:name(), "ifname", table.concat(ifns, " ")) set_ifstate(net:name(), "bridge", tostring(net:is_bridge())) end end end end -- redirect to overview page if network does not exist anymore (e.g. after a revert) if not net then luci.http.redirect(luci.dispatcher.build_url("admin/network/network")) return end -- protocol switch was requested, rebuild interface config and reload page if m:formvalue("cbid.network.%s._switch" % net:name()) then -- get new protocol local ptype = m:formvalue("cbid.network.%s.proto" % net:name()) or "-" local proto = nw:get_protocol(ptype, net:name()) if proto then -- backup default backup_ifnames() -- if current proto is not floating and target proto is not floating, -- then attempt to retain the ifnames --error(net:proto() .. " > " .. proto:proto()) if not net:is_floating() and not proto:is_floating() then -- if old proto is a bridge and new proto not, then clip the -- interface list to the first ifname only if net:is_bridge() and proto:is_virtual() then local _, ifn local first = true for _, ifn in ipairs(net:get_interfaces() or { net:get_interface() }) do if first then first = false else net:del_interface(ifn) end end m:del(net:name(), "type") end -- if the current proto is floating, the target proto not floating, -- then attempt to restore ifnames from backup elseif net:is_floating() and not proto:is_floating() then -- if we have backup data, then re-add all orphaned interfaces -- from it and restore the bridge choice local br = (get_ifstate(net:name(), "bridge") == "true") local ifn local ifns = { } for ifn in ut.imatch(get_ifstate(net:name(), "ifname")) do ifn = nw:get_interface(ifn) if ifn and not ifn:get_network() then proto:add_interface(ifn) if not br then break end end end if br then m:set(net:name(), "type", "bridge") end -- in all other cases clear the ifnames else local _, ifc for _, ifc in ipairs(net:get_interfaces() or { net:get_interface() }) do net:del_interface(ifc) end m:del(net:name(), "type") end -- clear options local k, v for k, v in pairs(m:get(net:name())) do if k:sub(1,1) ~= "." and k ~= "type" and k ~= "ifname" then m:del(net:name(), k) end end -- set proto m:set(net:name(), "proto", proto:proto()) m.uci:save("network") m.uci:save("wireless") -- reload page luci.http.redirect(luci.dispatcher.build_url("admin/network/network", arg[1])) return end end -- dhcp setup was requested, create section and reload page if m:formvalue("cbid.dhcp._enable._enable") then m.uci:section("dhcp", "dhcp", arg[1], { interface = arg[1], start = "100", limit = "150", leasetime = "12h" }) m.uci:save("dhcp") luci.http.redirect(luci.dispatcher.build_url("admin/network/network", arg[1])) return end local ifc = net:get_interface() s = m:section(NamedSection, arg[1], "interface", translate("Common Configuration")) s.addremove = false s:tab("general", translate("General Setup")) s:tab("advanced", translate("Advanced Settings")) s:tab("physical", translate("Physical Settings")) if has_firewall then s:tab("firewall", translate("Firewall Settings")) end st = s:taboption("general", DummyValue, "__status", translate("Status")) local function set_status() -- if current network is empty, print a warning if not net:is_floating() and net:is_empty() then st.template = "cbi/dvalue" st.network = nil st.value = translate("There is no device assigned yet, please attach a network device in the \"Physical Settings\" tab") else st.template = "admin_network/iface_status" st.network = arg[1] st.value = nil end end m.on_init = set_status m.on_after_save = set_status p = s:taboption("general", ListValue, "proto", translate("Protocol")) p.default = net:proto() if not net:is_installed() then p_install = s:taboption("general", Button, "_install") p_install.title = translate("Protocol support is not installed") p_install.inputtitle = translate("Install package %q" % net:opkg_package()) p_install.inputstyle = "apply" p_install:depends("proto", net:proto()) function p_install.write() return luci.http.redirect( luci.dispatcher.build_url("admin/system/packages") .. "?submit=1&install=%s" % net:opkg_package() ) end end p_switch = s:taboption("general", Button, "_switch") p_switch.title = translate("Really switch protocol?") p_switch.inputtitle = translate("Switch protocol") p_switch.inputstyle = "apply" local _, pr for _, pr in ipairs(nw:get_protocols()) do p:value(pr:proto(), pr:get_i18n()) if pr:proto() ~= net:proto() then p_switch:depends("proto", pr:proto()) end end auto = s:taboption("advanced", Flag, "auto", translate("Bring up on boot")) auto.default = (net:proto() == "none") and auto.disabled or auto.enabled delegate = s:taboption("advanced", Flag, "delegate", translate("Use builtin IPv6-management")) delegate.default = delegate.enabled force_link = s:taboption("advanced", Flag, "force_link", translate("Force link"), translate("Set interface properties regardless of the link carrier (If set, carrier sense events do not invoke hotplug handlers).")) force_link.default = (net:proto() == "static") and force_link.enabled or force_link.disabled if not net:is_virtual() then br = s:taboption("physical", Flag, "type", translate("Bridge interfaces"), translate("creates a bridge over specified interface(s)")) br.enabled = "bridge" br.rmempty = true br:depends("proto", "static") br:depends("proto", "dhcp") br:depends("proto", "none") stp = s:taboption("physical", Flag, "stp", translate("Enable STP"), translate("Enables the Spanning Tree Protocol on this bridge")) stp:depends("type", "bridge") stp.rmempty = true igmp = s:taboption("physical", Flag, "igmp_snooping", translate("Enable IGMP snooping"), translate("Enables IGMP snooping on this bridge")) igmp:depends("type", "bridge") igmp.rmempty = true end if not net:is_floating() then ifname_single = s:taboption("physical", Value, "ifname_single", translate("Interface")) ifname_single.template = "cbi/network_ifacelist" ifname_single.widget = "radio" ifname_single.nobridges = true ifname_single.rmempty = false ifname_single.network = arg[1] ifname_single:depends("type", "") function ifname_single.cfgvalue(self, s) -- let the template figure out the related ifaces through the network model return nil end function ifname_single.write(self, s, val) local i local new_ifs = { } local old_ifs = { } for _, i in ipairs(net:get_interfaces() or { net:get_interface() }) do old_ifs[#old_ifs+1] = i:name() end for i in ut.imatch(val) do new_ifs[#new_ifs+1] = i -- if this is not a bridge, only assign first interface if self.option == "ifname_single" then break end end table.sort(old_ifs) table.sort(new_ifs) for i = 1, math.max(#old_ifs, #new_ifs) do if old_ifs[i] ~= new_ifs[i] then backup_ifnames() for i = 1, #old_ifs do net:del_interface(old_ifs[i]) end for i = 1, #new_ifs do net:add_interface(new_ifs[i]) end break end end end end if not net:is_virtual() then ifname_multi = s:taboption("physical", Value, "ifname_multi", translate("Interface")) ifname_multi.template = "cbi/network_ifacelist" ifname_multi.nobridges = true ifname_multi.rmempty = false ifname_multi.network = arg[1] ifname_multi.widget = "checkbox" ifname_multi:depends("type", "bridge") ifname_multi.cfgvalue = ifname_single.cfgvalue ifname_multi.write = ifname_single.write end if has_firewall then fwzone = s:taboption("firewall", Value, "_fwzone", translate("Create / Assign firewall-zone"), translate("Choose the firewall zone you want to assign to this interface. Select unspecified to remove the interface from the associated zone or fill out the create field to define a new zone and attach the interface to it.")) fwzone.template = "cbi/firewall_zonelist" fwzone.network = arg[1] fwzone.rmempty = false function fwzone.cfgvalue(self, section) self.iface = section local z = fw:get_zone_by_network(section) return z and z:name() end function fwzone.write(self, section, value) local zone = fw:get_zone(value) if not zone and value == '-' then value = m:formvalue(self:cbid(section) .. ".newzone") if value and #value > 0 then zone = fw:add_zone(value) else fw:del_network(section) end end if zone then fw:del_network(section) zone:add_network(section) end end end function p.write() end function p.remove() end function p.validate(self, value, section) if value == net:proto() then if not net:is_floating() and net:is_empty() then local ifn = ((br and (br:formvalue(section) == "bridge")) and ifname_multi:formvalue(section) or ifname_single:formvalue(section)) for ifn in ut.imatch(ifn) do return value end return nil, translate("The selected protocol needs a device assigned") end end return value end local form, ferr = loadfile( ut.libpath() .. "/model/cbi/admin_network/proto_%s.lua" % net:proto() ) if not form then s:taboption("general", DummyValue, "_error", translate("Missing protocol extension for proto %q" % net:proto()) ).value = ferr else setfenv(form, getfenv(1))(m, s, net) end local _, field for _, field in ipairs(s.children) do if field ~= st and field ~= p and field ~= p_install and field ~= p_switch then if next(field.deps) then local _, dep for _, dep in ipairs(field.deps) do dep.proto = net:proto() end else field:depends("proto", net:proto()) end end end -- -- Display DNS settings if dnsmasq is available -- if has_dnsmasq and net:proto() == "static" then m2 = Map("dhcp", "", "") local has_section = false m2.uci:foreach("dhcp", "dhcp", function(s) if s.interface == arg[1] then has_section = true return false end end) if not has_section and has_dnsmasq then s = m2:section(TypedSection, "dhcp", translate("DHCP Server")) s.anonymous = true s.cfgsections = function() return { "_enable" } end x = s:option(Button, "_enable") x.title = translate("No DHCP Server configured for this interface") x.inputtitle = translate("Setup DHCP Server") x.inputstyle = "apply" elseif has_section then s = m2:section(TypedSection, "dhcp", translate("DHCP Server")) s.addremove = false s.anonymous = true s:tab("general", translate("General Setup")) s:tab("advanced", translate("Advanced Settings")) s:tab("ipv6", translate("IPv6 Settings")) function s.filter(self, section) return m2.uci:get("dhcp", section, "interface") == arg[1] end local ignore = s:taboption("general", Flag, "ignore", translate("Ignore interface"), translate("Disable DHCP for " .. "this interface.")) local start = s:taboption("general", Value, "start", translate("Start"), translate("Lowest leased address as offset from the network address.")) start.optional = true start.datatype = "or(uinteger,ip4addr)" start.default = "100" local limit = s:taboption("general", Value, "limit", translate("Limit"), translate("Maximum number of leased addresses.")) limit.optional = true limit.datatype = "uinteger" limit.default = "150" local ltime = s:taboption("general", Value, "leasetime", translate("Lease time"), translate("Expiry time of leased addresses, minimum is 2 minutes (2m).")) ltime.rmempty = true ltime.default = "12h" local dd = s:taboption("advanced", Flag, "dynamicdhcp", translate("Dynamic DHCP"), translate("Dynamically allocate DHCP addresses for clients. If disabled, only " .. "clients having static leases will be served.")) dd.default = dd.enabled s:taboption("advanced", Flag, "force", translate("Force"), translate("Force DHCP on this network even if another server is detected.")) -- XXX: is this actually useful? --s:taboption("advanced", Value, "name", translate("Name"), -- translate("Define a name for this network.")) mask = s:taboption("advanced", Value, "netmask", translate("IPv4-Netmask"), translate("Override the netmask sent to clients. Normally it is calculated " .. "from the subnet that is served.")) mask.optional = true mask.datatype = "ip4addr" s:taboption("advanced", DynamicList, "dhcp_option", translate("DHCP-Options"), translate("Define additional DHCP options, for example \"6,192.168.2.1," .. "192.168.2.2\" which advertises different DNS servers to clients.")) for i, n in ipairs(s.children) do if n ~= ignore then n:depends("ignore", "") end end o = s:taboption("ipv6", ListValue, "ra", translate("Router Advertisement-Service")) o:value("", translate("disabled")) o:value("server", translate("server mode")) o:value("relay", translate("relay mode")) o:value("hybrid", translate("hybrid mode")) o = s:taboption("ipv6", ListValue, "dhcpv6", translate("DHCPv6-Service")) o:value("", translate("disabled")) o:value("server", translate("server mode")) o:value("relay", translate("relay mode")) o:value("hybrid", translate("hybrid mode")) o = s:taboption("ipv6", ListValue, "ndp", translate("NDP-Proxy")) o:value("", translate("disabled")) o:value("relay", translate("relay mode")) o:value("hybrid", translate("hybrid mode")) o = s:taboption("ipv6", ListValue, "ra_management", translate("DHCPv6-Mode"), translate("Default is stateless + stateful")) o:value("0", translate("stateless")) o:value("1", translate("stateless + stateful")) o:value("2", translate("stateful-only")) o:depends("dhcpv6", "server") o:depends("dhcpv6", "hybrid") o.default = "1" o = s:taboption("ipv6", Flag, "ra_default", translate("Always announce default router"), translate("Announce as default router even if no public prefix is available.")) o:depends("ra", "server") o:depends("ra", "hybrid") s:taboption("ipv6", DynamicList, "dns", translate("Announced DNS servers")) s:taboption("ipv6", DynamicList, "domain", translate("Announced DNS domains")) else m2 = nil end end return m, m2