aboutsummaryrefslogtreecommitdiffstats
path: root/PolygLua.lua
diff options
context:
space:
mode:
Diffstat (limited to 'PolygLua.lua')
-rwxr-xr-xPolygLua.lua301
1 files changed, 301 insertions, 0 deletions
diff --git a/PolygLua.lua b/PolygLua.lua
new file mode 100755
index 0000000..c2ebb02
--- /dev/null
+++ b/PolygLua.lua
@@ -0,0 +1,301 @@
1#!/usr/bin/env luajit
2
3--[[ PolygLua. Gluing things onto Lua, making it a polyglot language.
4
5TODO - Allow -abc style, expand to -a -b -c.
6
7TODO - Make the parsing recursive. So the command "--fancy" could have it's own options table.
8 --fancy option0 opt1=foo 'Random string!' --somethingElse
9 When to stop and hand back?
10
11TODO - Add some sort of alias mechanism for the #! thing. _.bash'echo "This is bash."':Do()
12 The user can define their own aliases in a table, with defaults for bash, sh, luajit, and maybe some others.
13 Then extend it to changing language on the fly _.bash'echo "This is bash."':luajit("print('This is Lua.')"):Do()
14
15]]
16
17-- Most of this _ stuff was copied from apt-panopticon.
18local _ = {}
19_.version = '0.0 crap'
20
21
22_.verbosity = 2
23local log = function(v, t, s)
24 if v <= _.verbosity then
25 if 3 <= _.verbosity then t = os.date('!%F %T') .. " " .. t end
26 print(t .. ": " .. s)
27 end
28 io.flush()
29end
30
31-- This sets the global values, here and in the caller.
32D = function(s) log(4, 'DEBUG ', s) end
33I = function(s) log(3, 'INFO ', s) end
34T = function(s) log(2, 'TIMEOUT ', s) end
35W = function(s) log(1, 'WARNING ', s) end
36E = function(s) log(0, 'ERROR ', s) end
37C = function(s) log(-1, 'CRITICAL ', s) end
38
39
40local optionsCommon =
41{
42 help = {help = 'Print the help text.',
43 func = function(self, options, a, args, i)
44 print(Help)
45 _.usage(args, options, true)
46 os.exit(0)
47 end
48 },
49 ['--version'] = {help = 'Print the version details.',
50 func = function(self, options, a, args, i)
51 print('This is version ' ..Version .. ' of ' .. args[0])
52 os.exit(0)
53 end
54 },
55 ['-q'] = {help = 'Decrease verbosity level.',
56 func = function(self, options, a, args, i)
57 if _.verbosity > -1 then _.verbosity = _.verbosity - 1 end
58 print('New verbosity level is ' .. _.verbosity)
59 end
60 },
61 ['-v'] = {help = 'Increase verbosity level.',
62 func = function(self, options, a, args, i)
63 if _.verbosity < 4 then _.verbosity = _.verbosity + 1 end
64 print('New verbosity level is ' .. _.verbosity)
65 end
66 },
67}
68optionsCommon['--help'] = optionsCommon['help']
69
70_.usage = function(args, options, all)
71 local h = ''
72 for k, v in pairs(options) do
73 if 'table' == type(v) then h = h .. k .. ' | ' end
74 end
75 for k, v in pairs(optionsCommon) do
76 if 'table' == type(v) then h = h .. k .. ' | ' end
77 end
78 print('Usage: ' .. args[0] .. ' {' .. string.sub(h, 1, -2) .. '}')
79 if true == all then
80 for k, v in pairs(options) do
81 if 'table' == type(v) then
82 if nil ~= v.help then print(k .. '\t\t' .. v.help) end
83 end
84 end
85 for k, v in pairs(optionsCommon) do
86 if 'table' == type(v) then
87 if nil ~= v.help then print(k .. '\t\t' .. v.help) end
88 end
89 end
90 end
91end
92
93_.parse = function(args, options, confFile)
94 local o = nil
95
96 local doIt = function(name, val, a, args, i)
97 local o = options[name]
98 if nil == o then o = optionsCommon[name] end
99 if nil ~= o then
100 if nil ~= val then o.value = val; D(name .. ' = ' .. tostring(val)) end
101 if nil ~= o.func then o:func(options, a, args, i) end
102 end
103 return o
104 end
105
106 if nil ~= confFile then
107 for i,v in ipairs{'/etc/', '~/.', './.'} do
108 local p = v .. confFile .. '.conf.lua'
109 local h, e = io.open(p, "r")
110 if nil ~= h then
111 D('Found configuration file '.. p)
112 h:close()
113 local ar = dofile(p)
114 for k, w in pairs(ar) do
115 if nil == doIt(k, w, k .. '=' .. tostring(w), args, i) then W('config variable not found ' .. k .. ' = ' .. tostring(w)) end
116 end
117 end
118 end
119 end
120
121 if 0 ~= #args then
122 for i,a in ipairs(args) do
123 D('Argument ' .. i .. ' = ' .. a)
124 local s, e = a:find("=")
125 if nil == s then
126 e = 0
127 o = doIt(a:sub(1, e - 1), nil, a, args, i)
128 else
129 o = doIt(a:sub(1, e - 1), a:sub(e + 1, -1), a, args, i)
130 end
131 end
132 end
133
134 if nil == o then
135 _.usage(args, options)
136 os.exit(0)
137 end
138end
139
140_.runnable = function(c)
141 return ( 0 == __('which ' .. c):Do().status )
142end
143
144_.running = function(c)
145 return ( 1 ~= tonumber(__("pgrep -u $USER -cf " .. c):Do().lines[1]) )
146end
147
148_.exists = function(f)
149 local h, e = io.open(f, "r")
150 if nil == h then return false else h:close(); return true end
151end
152
153
154_.killEmAll = function(all)
155 for i,l in ipairs(all) do
156 local c = 0
157 while 0 ~= tonumber(__("pgrep -u $USER -xc " .. l):Do().lines[1]) do
158 local s = 'TERM'
159 if c > 1 then s = 'KILL'; __("sleep " .. c):Do() end
160 __("pkill -" .. s .. " -u $USER -x " .. l):log():Do()
161 c = c + 1
162 end
163 end
164end
165
166
167_.readCmd = function(cmd)
168 local result = {}
169 local output = io.popen(cmd)
170 if nil ~= output then
171 for l in output:lines() do
172 table.insert(result, l)
173 end
174 end
175 -- While this does return the same things as os.execute(), it's just as useless.
176 output:close()
177 return result
178end
179
180
181__ = function(c)
182 local exe = {status = 0, lines = {}, logging = false, showing = false, cmd = '', command = c, isScript = false, script = ''}
183 local n = 0
184
185 exe.cmd = '{ '
186 if 'table' == type(c) then
187 for i, l in ipairs(c) do
188 n = n + 1
189 exe.cmd = exe.cmd .. l .. ' ; '
190 end
191 elseif 'string' == type(c) then
192 exe.isScript = ('#!' == c:sub(1,2)) and (n == 0)
193 for l in string.gmatch(c, "\n*([^\n]+)\n*") do
194 if '' ~= l then
195 if exe.isScript then
196 if '' == exe.script then
197 exe.scriptFile = os.tmpname()
198 D('Creating temporary script file at ' .. exe.scriptFile)
199 exe.cmd = exe.cmd .. l:sub(3) .. ' ' .. exe.scriptFile .. ' ; '
200 end
201 exe.script = exe.script .. l .. '\n'
202 else
203 n = n + 1
204 exe.cmd = exe.cmd .. l .. ' ; '
205 end
206 end
207 end
208 end
209 if exe.isScript then
210 local a, e = io.open(exe.scriptFile, "w")
211 if nil == a then E("Could not open " .. exe.scriptFile) else
212 a:write(exe.script)
213 a:close()
214 end
215 exe.cmd = exe.cmd .. 'rm ' .. exe.scriptFile .. ' ; '
216 end
217 exe.cmd = exe.cmd .. ' } '
218 if 1 == n then exe.cmd = c end
219
220
221 function exe:Nice(c)
222 if nil == c then
223 self.cmd = 'ionice -c3 nice -n 19 ' .. self.cmd
224 else
225 self.cmd = self.cmd .. ' ionice -c3 nice -n 19 ' .. c .. ' '
226 end
227 return self
228 end
229
230 function exe:timeout(c)
231 -- timeout returns a status of - command status if --preserve-status; "128+9" (actually 137) if --kill-after ends up being done; 124 if it had to TERM; command status if all went well.
232 -- --kill-after means "send KILL after TERM fails.
233 if nil == c then
234 self.cmd = 'timeout --kill-after=10.0 --foreground 42.0s ' .. self.cmd
235 else
236 self.cmd = 'timeout --kill-after=10.0 --foreground ' .. c .. ' ' .. self.cmd
237 end
238 return self
239 end
240
241 function exe:log() self.logging = true return self end
242 function exe:show() self.showing = true return self end
243 -- Should be called "then" but that's a reserved word.
244 function exe:Then(c) if nil == c then c = '' else c = ' ' .. c end self.cmd = self.cmd .. '; ' .. c .. ' ' return self end
245 function exe:And(c) if nil == c then c = '' else c = ' ' .. c end self.cmd = self.cmd .. ' && ' .. c .. ' ' return self end
246 function exe:Or(c) if nil == c then c = '' else c = ' ' .. c end self.cmd = self.cmd .. ' || ' .. c .. ' ' return self end
247 function exe:noErr() self.cmd = self.cmd .. ' 2>/dev/null ' return self end
248 function exe:noOut() self.cmd = self.cmd .. ' 1>/dev/null ' return self end
249 function exe:wait(w) self.cmd = self.cmd .. ' && touch ' .. w .. ' ' return self end
250
251 function exe:Do()
252 --[[ "The condition expression of a control structure can return any
253 value. Both false and nil are considered false. All values different
254 from nil and false are considered true (in particular, the number 0
255 and the empty string are also true)."
256 says the docs, I beg to differ.]]
257 if true == self.logging then D(" executing - " .. self.cmd) end
258 --[[ Damn os.execute()
259 Lua 5.1 says it returns "a status code, which is system-dependent"
260 Lua 5.2 says it returns true/nil, "exit"/"signal", the status code.
261 I'm getting 7168 or 0. No idea what the fuck that is.
262 local ok, rslt, status = os.execute(s)
263 ]]
264 self.lines = _.readCmd(self.cmd .. '; echo "$?"', 'r')
265 -- The last line will be the command's returned status, fish that out and collect everything else in lines.
266 self.status = tonumber(self.lines[#self.lines])
267 self.lines[#self.lines] = nil
268 if true == self.showing then for i, l in ipairs(self.lines) do I(l) end end
269
270 if (nil == self.status) then D("STATUS |" .. "NIL" .. '| ' .. self.command)
271 elseif (137 == self.status) or (124 == self.status) then T("timeout killed " .. self.status .. ' ' .. self.command)
272 elseif (0 ~= self.status) then D("STATUS |" .. self.status .. '| ' .. self.command)
273 end
274
275 return self
276 end
277
278 function exe:fork(after, host)
279-- The host part is from apt-panopticon, likely needed there, but makes no sense here.
280-- if nil ~= host then self.cmd = self.cmd .. '; r=$?; if [ $r -ge 124 ]; then echo "$r ' .. host .. ' failed forked command ' .. string.gsub(self.cmd, '"', "'") .. '"; fi' end
281 if nil == after then after = '' end
282 if '' ~= after then after = ' ; ' .. after end
283 self.cmd = '{ ' .. self.cmd .. after .. ' ; } & '
284 if true == self.logging then D(" forking - " .. self.cmd) end
285 os.execute(self.cmd)
286 return self
287 end
288
289 function exe:forkOnce()
290 if _.running(self.command) then
291 D('Already running ' .. self.command)
292 else
293 self:fork()
294 end
295 end
296
297 return exe
298end
299
300
301return _