Merge pull request #371 from aidvu/fstab-swap
[project/luci.git] / modules / luci-mod-admin-full / luasrc / model / cbi / admin_system / fstab.lua
1 -- Copyright 2008 Steven Barth <steven@midlink.org>
2 -- Licensed to the public under the Apache License 2.0.
3
4 require("luci.tools.webadmin")
5
6 local fs   = require "nixio.fs"
7 local util = require "nixio.util"
8 local tp   = require "luci.template.parser"
9
10 local block = io.popen("block info", "r")
11 local ln, dev, devices = nil, nil, {}
12
13 repeat
14         ln = block:read("*l")
15         dev = ln and ln:match("^/dev/(.-):")
16
17         if dev then
18                 local e, s, key, val = { }
19
20                 for key, val in ln:gmatch([[(%w+)="(.-)"]]) do
21                         e[key:lower()] = val
22                         devices[val] = e
23                 end
24
25                 s = tonumber((fs.readfile("/sys/class/block/%s/size" % dev)))
26
27                 e.dev  = "/dev/%s" % dev
28                 e.size = s and math.floor(s / 2048)
29
30                 devices[e.dev] = e
31         end
32 until not ln
33
34 block:close()
35
36
37 m = Map("fstab", translate("Mount Points"))
38
39 local mounts = luci.sys.mounts()
40
41 v = m:section(Table, mounts, translate("Mounted file systems"))
42
43 fs = v:option(DummyValue, "fs", translate("Filesystem"))
44
45 mp = v:option(DummyValue, "mountpoint", translate("Mount Point"))
46
47 avail = v:option(DummyValue, "avail", translate("Available"))
48 function avail.cfgvalue(self, section)
49         return luci.tools.webadmin.byte_format(
50                 ( tonumber(mounts[section].available) or 0 ) * 1024
51         ) .. " / " .. luci.tools.webadmin.byte_format(
52                 ( tonumber(mounts[section].blocks) or 0 ) * 1024
53         )
54 end
55
56 used = v:option(DummyValue, "used", translate("Used"))
57 function used.cfgvalue(self, section)
58         return ( mounts[section].percent or "0%" ) .. " (" ..
59         luci.tools.webadmin.byte_format(
60                 ( tonumber(mounts[section].used) or 0 ) * 1024
61         ) .. ")"
62 end
63
64
65
66 mount = m:section(TypedSection, "mount", translate("Mount Points"), translate("Mount Points define at which point a memory device will be attached to the filesystem"))
67 mount.anonymous = true
68 mount.addremove = true
69 mount.template = "cbi/tblsection"
70 mount.extedit  = luci.dispatcher.build_url("admin/system/fstab/mount/%s")
71
72 mount.create = function(...)
73         local sid = TypedSection.create(...)
74         if sid then
75                 luci.http.redirect(mount.extedit % sid)
76                 return
77         end
78 end
79
80
81 mount:option(Flag, "enabled", translate("Enabled")).rmempty = false
82
83 dev = mount:option(DummyValue, "device", translate("Device"))
84 dev.rawhtml = true
85 dev.cfgvalue = function(self, section)
86         local v, e
87
88         v = m.uci:get("fstab", section, "uuid")
89         e = v and devices[v:lower()]
90         if v and e and e.size then
91                 return "UUID: %s (%s, %d MB)" %{ tp.pcdata(v), e.dev, e.size }
92         elseif v and e then
93                 return "UUID: %s (%s)" %{ tp.pcdata(v), e.dev }
94         elseif v then
95                 return "UUID: %s (<em>%s</em>)" %{ tp.pcdata(v), translate("not present") }
96         end
97
98         v = m.uci:get("fstab", section, "label")
99         e = v and devices[v]
100         if v and e and e.size then
101                 return "Label: %s (%s, %d MB)" %{ tp.pcdata(v), e.dev, e.size }
102         elseif v and e then
103                 return "Label: %s (%s)" %{ tp.pcdata(v), e.dev }
104         elseif v then
105                 return "Label: %s (<em>%s</em>)" %{ tp.pcdata(v), translate("not present") }
106         end
107
108         v = Value.cfgvalue(self, section) or "?"
109         e = v and devices[v]
110         if v and e and e.size then
111                 return "%s (%d MB)" %{ tp.pcdata(v), e.size }
112         elseif v and e then
113                 return tp.pcdata(v)
114         elseif v then
115                 return "%s (<em>%s</em>)" %{ tp.pcdata(v), translate("not present") }
116         end
117 end
118
119 mp = mount:option(DummyValue, "target", translate("Mount Point"))
120 mp.cfgvalue = function(self, section)
121         if m.uci:get("fstab", section, "is_rootfs") == "1" then
122                 return "/overlay"
123         else
124                 return Value.cfgvalue(self, section) or "?"
125         end
126 end
127
128 fs = mount:option(DummyValue, "fstype", translate("Filesystem"))
129 fs.cfgvalue = function(self, section)
130         local v, e
131
132         v = m.uci:get("fstab", section, "uuid")
133         v = v and v:lower() or m.uci:get("fstab", section, "label")
134         v = v or m.uci:get("fstab", section, "device")
135
136         e = v and devices[v]
137
138         return e and e.type or m.uci:get("fstab", section, "fstype") or "?"
139 end
140
141 op = mount:option(DummyValue, "options", translate("Options"))
142 op.cfgvalue = function(self, section)
143         return Value.cfgvalue(self, section) or "defaults"
144 end
145
146 rf = mount:option(DummyValue, "is_rootfs", translate("Root"))
147 rf.cfgvalue = function(self, section)
148         local target = m.uci:get("fstab", section, "target")
149         if target == "/" then
150                 return translate("yes")
151         elseif target == "/overlay" then
152                 return translate("overlay")
153         else
154                 return translate("no")
155         end
156 end
157
158 ck = mount:option(DummyValue, "enabled_fsck", translate("Check"))
159 ck.cfgvalue = function(self, section)
160         return Value.cfgvalue(self, section) == "1"
161                 and translate("yes") or translate("no")
162 end
163
164
165 swap = m:section(TypedSection, "swap", "SWAP", translate("If your physical memory is insufficient unused data can be temporarily swapped to a swap-device resulting in a higher amount of usable <abbr title=\"Random Access Memory\">RAM</abbr>. Be aware that swapping data is a very slow process as the swap-device cannot be accessed with the high datarates of the <abbr title=\"Random Access Memory\">RAM</abbr>."))
166 swap.anonymous = true
167 swap.addremove = true
168 swap.template = "cbi/tblsection"
169 swap.extedit  = luci.dispatcher.build_url("admin/system/fstab/swap/%s")
170
171 swap.create = function(...)
172         local sid = TypedSection.create(...)
173         if sid then
174                 luci.http.redirect(swap.extedit % sid)
175                 return
176         end
177 end
178
179
180 swap:option(Flag, "enabled", translate("Enabled")).rmempty = false
181
182 dev = swap:option(DummyValue, "device", translate("Device"))
183 dev.cfgvalue = function(self, section)
184         local v
185
186         v = m.uci:get("fstab", section, "uuid")
187         if v then return "UUID: %s" % v end
188
189         v = m.uci:get("fstab", section, "label")
190         if v then return "Label: %s" % v end
191
192         v = Value.cfgvalue(self, section) or "?"
193         e = v and devices[v]
194         if v and e and e.size then
195                 return "%s (%s MB)" % {v, e.size}
196         else
197                 return v
198         end
199 end
200
201 return m