applications/siitwizard: add sw_merge and txpower to wifi-iface config
[project/luci.git] / applications / luci-siitwizard / luasrc / model / cbi / siitwizard.lua
1 --[[
2 LuCI - Lua Configuration Interface
3
4 Copyright 2008 Steven Barth <steven@midlink.org>
5 Copyright 2008 Jo-Philipp Wich <xm@leipzig.freifunk.net>
6
7 Licensed under the Apache License, Version 2.0 (the "License");
8 you may not use this file except in compliance with the License.
9 You may obtain a copy of the License at
10
11 http://www.apache.org/licenses/LICENSE-2.0
12
13 $Id$
14
15 ]]--
16
17 local uci = require "luci.model.uci".cursor()
18
19 -------------------- Init --------------------
20
21 --
22 -- Find link-local address
23 --
24 LL_PREFIX = luci.ip.IPv6("fe80::/64")
25 function find_ll()
26         for _, r in ipairs(luci.sys.net.routes6()) do
27                 if LL_PREFIX:contains(r.dest) and r.dest:higher(LL_PREFIX) then
28                         return r.dest:sub(LL_PREFIX)
29                 end
30         end
31         return luci.ip.IPv6("::")
32 end
33
34 --
35 -- Determine defaults
36 --
37 local ula_prefix  = uci:get("siit", "ipv6", "ula_prefix")  or "fd00::"
38 local ula_global  = uci:get("siit", "ipv6", "ula_global")  or "00ca:ffee:babe::"                -- = Freifunk
39 local ula_subnet  = uci:get("siit", "ipv6", "ula_subnet")  or "0000:0000:0000:4223::"   -- = Berlin
40 local siit_prefix = uci:get("siit", "ipv6", "siit_prefix") or "::ffff:0000:0000"
41 local ipv4_pool   = uci:get("siit", "ipv4", "pool")        or "172.16.0.0/12"
42 local ipv4_netsz  = uci:get("siit", "ipv4", "netsize")     or "24"
43
44 --
45 -- Find IPv4 allocation pool
46 --
47 local gv4_net = luci.ip.IPv4(ipv4_pool)
48
49 --
50 -- Generate ULA
51 --
52 local ula = luci.ip.IPv6("::/64")
53
54 for _, prefix in ipairs({ ula_prefix, ula_global, ula_subnet }) do
55         ula = ula:add(luci.ip.IPv6(prefix))
56 end
57
58 ula = ula:add(find_ll())
59
60
61 -------------------- View --------------------
62 f = SimpleForm("siitwizward", "SIIT-Wizzard",
63  "This wizzard helps to setup SIIT (IPv4-over-IPv6) translation according to RFC2765.")
64
65 f:field(DummyValue, "info_ula", "Mesh ULA address").value = ula:string()
66
67 f:field(DummyValue, "ipv4_pool", "IPv4 allocation pool").value =
68         "%s (%i hosts)" %{ gv4_net:string(), 2 ^ gv4_net:prefix() - 2 }
69
70 f:field(DummyValue, "ipv4_size", "IPv4 LAN network prefix").value =
71         "%i bit (%i hosts)" %{ ipv4_netsz, 2 ^ ( 32 - ipv4_netsz ) - 2 }
72
73 mode = f:field(ListValue, "mode", "Operation mode")
74 mode:value("client", "Client")
75 mode:value("gateway", "Gateway")
76
77 dev = f:field(ListValue, "device", "Wireless device")
78 uci:foreach("wireless", "wifi-device",
79         function(section)
80                 dev:value(section[".name"])
81         end)
82
83 lanip = f:field(Value, "ipaddr", "LAN IPv4 subnet")
84 function lanip.formvalue(self, section)
85         local val = self.map:formvalue(self:cbid(section))
86         local net = luci.ip.IPv4("%s/%i" %{ val, ipv4_netsz })
87
88         if net then
89                 if gv4_net:contains(net) then
90                         if not net:minhost():equal(net:host()) then
91                                 self.error = { [section] = true }
92                                 f.errmessage = "IPv4 address is not the first host of " ..
93                                         "subnet, expected " .. net:minhost():string()
94                         end
95                 else
96                         self.error = { [section] = true }
97                         f.errmessage = "IPv4 address is not within the allocation pool"
98                 end
99         else
100                 self.error = { [section] = true }
101                 f.errmessage = "Invalid IPv4 address given"
102         end
103
104         return val
105 end
106
107 dns = f:field(Value, "dns", "DNS server for LAN clients")
108 dns.value = "141.1.1.1"
109
110 -------------------- Control --------------------
111 function f.handle(self, state, data)
112         if state == FORM_VALID then
113                 luci.http.redirect(luci.dispatcher.build_url("admin", "uci", "changes"))
114                 return false
115         end
116         return true
117 end
118
119 function mode.write(self, section, value)
120
121         --
122         -- Find LAN IPv4 range
123         --
124         local lan_net = luci.ip.IPv4(
125                 ( lanip:formvalue(section) or "172.16.0.1" ) .. "/" .. ipv4_netsz
126         )
127
128         if not lan_net then return end
129
130         --
131         -- Find wifi interface, dns server and hostname
132         --
133         local device      = dev:formvalue(section)
134         local dns_server  = dns:formvalue(section) or "141.1.1.1"
135         local hostname    = "siit-" .. lan_net:host():string():gsub("%.","-")
136
137         --
138         -- Configure wifi device
139         --
140         local wifi_device  = dev:formvalue(section)
141         local wifi_essid   = uci:get("siit", "wifi", "essid")   or "6mesh.freifunk.net"
142         local wifi_bssid   = uci:get("siit", "wifi", "bssid")   or "02:ca:ff:ee:ba:be"
143         local wifi_channel = uci:get("siit", "wifi", "channel") or "1"
144
145         -- nuke old device definition
146         uci:delete_all("wireless", "wifi-iface",
147                 function(s) return s.device == wifi_device end )
148
149         uci:delete_all("network", "interface",
150                 function(s) return s['.name'] == wifi_device end )
151
152         -- create wifi device definition
153         uci:tset("wireless", wifi_device, {
154                 disabled  = 0,
155                 channel   = wifi_channel,
156 --              txantenna = 1,
157 --              rxantenna = 1,
158 --              diversity = 0
159         })
160
161         uci:section("wireless", "wifi-iface", nil, {
162                 encryption = "none",
163                 mode       = "adhoc",
164                 txpower    = 10,
165                 sw_merge   = 1,
166                 network    = wifi_device,
167                 device     = wifi_device,
168                 ssid       = wifi_essid,
169                 bssid      = wifi_bssid,
170         })
171
172         --
173         -- Gateway mode
174         --
175         --      * wan port is dhcp, lan port is 172.23.1.1/24
176         --      * siit0 gets a dummy address: 169.254.42.42
177         --      * wl0 gets an ipv6 address, in this case the fdca:ffee:babe::1:1/64
178         --      * we do a ::ffff:ffff:0/96 route into siit0, so everything from 6mesh goes into translation.
179         --      * an HNA6 of ::ffff:ffff:0:0/96 announces the mapped 0.0.0.0/0 ipv4 space.
180         --      * MTU on WAN, LAN down to 1400, ipv6 headers are slighly larger.
181
182         if value == "gateway" then
183
184                 -- wan mtu
185                 uci:set("network", "wan", "mtu", 1400)
186
187                 -- lan settings
188                 uci:tset("network", "lan", {
189                         mtu     = 1400,
190                         ipaddr  = lan_net:host():string(),
191                         netmask = lan_net:mask():string()
192                 })
193
194                 -- use full siit subnet
195                 siit_route = luci.ip.IPv6(siit_prefix .. "/96")
196
197                 -- v4 <-> siit route
198                 uci:delete_all("network", "route",
199                         function(s) return s.interface == "siit0" end)
200
201                 uci:section("network", "route", nil, {
202                         interface = "siit0",
203                         target    = gv4_net:network():string(),
204                         netmask   = gv4_net:mask():string()
205                 })
206
207         --
208         -- Client mode
209         --
210         --      * 172.23.2.1/24 on its lan, fdca:ffee:babe::1:2 on wl0 and the usual dummy address on siit0.
211         --      * we do a ::ffff:ffff:172.13.2.0/120 to siit0, because in this case, only traffic directed to clients needs to go into translation.
212         --      * same route as HNA6 announcement to catch the traffic out of the mesh.
213         --      * Also, MTU on LAN reduced to 1400.
214
215         else
216
217                 -- lan settings
218                 uci:tset("network", "lan", {
219                         mtu     = 1400,
220                         ipaddr  = lan_net:host():string(),
221                         netmask = lan_net:mask():string()
222                 })
223
224                 -- derive siit subnet from lan config
225                 siit_route = luci.ip.IPv6(
226                         siit_prefix .. "/" .. (96 + lan_net:prefix())
227                 ):add(lan_net[2])
228
229                 -- ipv4 <-> siit route
230                 uci:delete_all("network", "route",
231                         function(s) return s.interface == "siit0" end)
232
233                 -- XXX: kind of a catch all, gv4_net would be better
234                 --      but does not cover non-local v4 space
235                 uci:section("network", "route", nil, {
236                         interface = "siit0",
237                         target    = "0.0.0.0",
238                         netmask   = "0.0.0.0"
239                 })
240         end
241
242         -- setup the firewall
243         uci:delete_all("firewall", "zone",
244                 function(s) return (
245                         s['.name'] == "siit0" or s.name == "siit0" or
246                         s.network == "siit0" or s['.name'] == wifi_device or
247                         s.name == wifi_device or s.network == wifi_device
248                 ) end)
249
250         uci:delete_all("firewall", "forwarding",
251                 function(s) return (
252                         s.src == wifi_device and s.dest == "siit0" or
253                         s.dest == wifi_device and s.src == "siit0" or
254                         s.src == "lan" and s.dest == "siit0" or
255                         s.dest == "lan" and s.src == "siit0"
256                 ) end)
257
258         uci:section("firewall", "zone", "siit0", {
259                 name    = "siit0",
260                 network = "siit0",
261                 input   = "ACCEPT",
262                 output  = "ACCEPT",
263                 forward = "ACCEPT"
264         })
265
266         uci:section("firewall", "zone", wifi_device, {
267                 name    = wifi_device,
268                 network = wifi_device,
269                 input   = "ACCEPT",
270                 output  = "ACCEPT",
271                 forward = "ACCEPT"
272         })
273
274         uci:section("firewall", "forwarding", nil, {
275                 src  = wifi_device,
276                 dest = "siit0"
277         })
278
279         uci:section("firewall", "forwarding", nil, {
280                 src  = "siit0",
281                 dest = wifi_device
282         })
283
284         uci:section("firewall", "forwarding", nil, {
285                 src  = "lan",
286                 dest = "siit0"
287         })
288
289         uci:section("firewall", "forwarding", nil, {
290                 src  = "siit0",
291                 dest = "lan"
292         })
293
294         -- firewall include
295         uci:delete_all("firewall", "include",
296                 function(s) return s.path == "/etc/firewall.user" end)
297
298         uci:section("firewall", "include", nil, {
299                 path = "/etc/firewall.user"
300         })
301
302
303         -- siit0 interface
304         uci:delete_all("network", "interface",
305                 function(s) return ( s.ifname == "siit0" ) end)
306
307         uci:section("network", "interface", "siit0", {
308                 ifname  = "siit0",
309                 proto   = "none"
310         })
311
312         -- siit0 route
313         uci:delete_all("network", "route6",
314                 function(s) return siit_route:contains(luci.ip.IPv6(s.target)) end)
315
316         uci:section("network", "route6", nil, {
317                 interface = "siit0",
318                 target    = siit_route:string()
319         })
320
321         -- create wifi network interface
322         uci:section("network", "interface", wifi_device, {
323                 proto   = "static",
324                 mtu     = 1400,
325                 ip6addr = ula:string()
326         })
327
328         -- nuke old olsrd interfaces
329         uci:delete_all("olsrd", "Interface",
330                 function(s) return s.interface == wifi_device end)
331
332         -- configure olsrd interface
333         uci:foreach("olsrd", "olsrd",
334                 function(s) uci:set("olsrd", s['.name'], "IpVersion", 6) end)
335
336         uci:section("olsrd", "Interface", nil, {
337                 ignore      = 0,
338                 interface   = wifi_device,
339                 Ip6AddrType = "global"
340         })
341
342         -- hna6
343         uci:delete_all("olsrd", "Hna6",
344                 function(s)
345                         if s.netaddr and s.prefix then
346                                 return siit_route:contains(luci.ip.IPv6(s.netaddr.."/"..s.prefix))
347                         end
348                 end)
349
350         uci:section("olsrd", "Hna6", nil, {
351                 netaddr = siit_route:host():string(),
352                 prefix  = siit_route:prefix()
353         })
354
355         -- txtinfo v6 & olsrd nameservice
356         uci:foreach("olsrd", "LoadPlugin",
357                 function(s)
358                         if s.library == "olsrd_txtinfo.so.0.1" then
359                                 uci:set("olsrd", s['.name'], "accept", "::1")
360                         elseif s.library == "olsrd_nameservice.so.0.3" then
361                                 uci:set("olsrd", s['.name'], "name", hostname)
362                         end
363                 end)
364
365         -- lan dns
366         uci:tset("dhcp", "lan", {
367                 dhcp_option = "6," .. dns_server
368         })
369
370         -- hostname
371         uci:foreach("system", "system",
372                 function(s)
373                         uci:set("system", s['.name'], "hostname", hostname)
374                 end)
375
376         uci:save("wireless")
377         uci:save("firewall")
378         uci:save("network")
379         uci:save("system")
380         uci:save("olsrd")
381         uci:save("dhcp")
382 end
383
384 return f