2013-09-07 12:06:42 +02:00
|
|
|
--[[
|
|
|
|
|
2017-09-04 12:43:00 +02:00
|
|
|
Licensed under GNU General Public License v2
|
2017-09-11 06:51:52 +02:00
|
|
|
* (c) 2013, Luca CPZ
|
2017-09-04 12:43:00 +02:00
|
|
|
|
|
|
|
--]]
|
2013-09-13 00:07:44 +02:00
|
|
|
|
2022-06-11 12:25:17 +02:00
|
|
|
local spawn = require("awful.spawn")
|
|
|
|
local timer = require("gears.timer")
|
2021-09-14 23:34:48 +02:00
|
|
|
local gdebug = require("gears.debug")
|
2022-06-11 12:25:17 +02:00
|
|
|
local debug = require("debug")
|
|
|
|
local io = { lines = io.lines,
|
|
|
|
open = io.open }
|
|
|
|
local pairs = pairs
|
|
|
|
local rawget = rawget
|
|
|
|
local tsort = table.sort
|
|
|
|
local unpack = unpack or table.unpack -- lua 5.1 retro-compatibility
|
2021-09-14 23:34:48 +02:00
|
|
|
local lgi = require("lgi")
|
2013-09-07 12:06:42 +02:00
|
|
|
|
|
|
|
-- Lain helper functions for internal use
|
|
|
|
-- lain.helpers
|
|
|
|
local helpers = {}
|
|
|
|
|
|
|
|
helpers.lain_dir = debug.getinfo(1, 'S').source:match[[^@(.*/).*$]]
|
|
|
|
helpers.icons_dir = helpers.lain_dir .. 'icons/'
|
|
|
|
helpers.scripts_dir = helpers.lain_dir .. 'scripts/'
|
|
|
|
|
|
|
|
-- {{{ Modules loader
|
|
|
|
|
2020-11-30 15:58:27 +01:00
|
|
|
function helpers.wrequire(t, k)
|
|
|
|
return rawget(t, k) or require(t._NAME .. '.' .. k)
|
2013-09-07 12:06:42 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
-- }}}
|
|
|
|
|
2014-01-29 11:56:32 +01:00
|
|
|
-- {{{ File operations
|
2013-09-07 12:06:42 +02:00
|
|
|
|
2018-02-23 13:24:10 +01:00
|
|
|
-- check if the file exists and is readable
|
|
|
|
function helpers.file_exists(path)
|
|
|
|
local file = io.open(path, "rb")
|
|
|
|
if file then file:close() end
|
|
|
|
return file ~= nil
|
2014-01-29 11:56:32 +01:00
|
|
|
end
|
|
|
|
|
2018-02-23 13:24:10 +01:00
|
|
|
-- get a table with all lines from a file
|
|
|
|
function helpers.lines_from(path)
|
|
|
|
local lines = {}
|
|
|
|
for line in io.lines(path) do
|
|
|
|
lines[#lines + 1] = line
|
|
|
|
end
|
|
|
|
return lines
|
2014-01-29 11:56:32 +01:00
|
|
|
end
|
|
|
|
|
2018-02-23 13:24:10 +01:00
|
|
|
-- get a table with all lines from a file matching regexp
|
|
|
|
function helpers.lines_match(regexp, path)
|
|
|
|
local lines = {}
|
|
|
|
for line in io.lines(path) do
|
|
|
|
if string.match(line, regexp) then
|
|
|
|
lines[#lines + 1] = line
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return lines
|
2016-03-30 16:34:41 +02:00
|
|
|
end
|
|
|
|
|
2018-02-23 13:24:10 +01:00
|
|
|
-- get first line of a file
|
|
|
|
function helpers.first_line(path)
|
|
|
|
local file, first = io.open(path, "rb"), nil
|
|
|
|
if file then
|
|
|
|
first = file:read("*l")
|
|
|
|
file:close()
|
|
|
|
end
|
|
|
|
return first
|
2014-01-29 11:56:32 +01:00
|
|
|
end
|
2013-09-07 12:06:42 +02:00
|
|
|
|
2018-02-23 13:24:10 +01:00
|
|
|
-- get first non empty line from a file
|
|
|
|
function helpers.first_nonempty_line(path)
|
|
|
|
for line in io.lines(path) do
|
|
|
|
if #line then return line end
|
|
|
|
end
|
|
|
|
return nil
|
2013-09-07 12:06:42 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
-- }}}
|
|
|
|
|
2013-09-13 18:15:25 +02:00
|
|
|
-- {{{ Timer maker
|
2013-09-10 23:02:11 +02:00
|
|
|
|
|
|
|
helpers.timer_table = {}
|
|
|
|
|
2017-01-24 16:15:40 +01:00
|
|
|
function helpers.newtimer(name, timeout, fun, nostart, stoppable)
|
|
|
|
if not name or #name == 0 then return end
|
|
|
|
name = (stoppable and name) or timeout
|
2016-03-02 13:48:55 +01:00
|
|
|
if not helpers.timer_table[name] then
|
2017-01-20 20:58:22 +01:00
|
|
|
helpers.timer_table[name] = timer({ timeout = timeout })
|
2016-08-11 13:43:35 +02:00
|
|
|
helpers.timer_table[name]:start()
|
2016-03-02 13:48:55 +01:00
|
|
|
end
|
2013-09-10 23:02:11 +02:00
|
|
|
helpers.timer_table[name]:connect_signal("timeout", fun)
|
|
|
|
if not nostart then
|
|
|
|
helpers.timer_table[name]:emit_signal("timeout")
|
|
|
|
end
|
2017-01-24 16:15:40 +01:00
|
|
|
return stoppable and helpers.timer_table[name]
|
2013-09-10 23:02:11 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
-- }}}
|
|
|
|
|
2015-08-13 01:56:37 +02:00
|
|
|
-- {{{ Pipe operations
|
|
|
|
|
2017-01-20 20:58:22 +01:00
|
|
|
-- run a command and execute a function on its output (asynchronous pipe)
|
|
|
|
-- @param cmd the input command
|
|
|
|
-- @param callback function to execute on cmd output
|
|
|
|
-- @return cmd PID
|
|
|
|
function helpers.async(cmd, callback)
|
2017-12-19 17:20:59 +01:00
|
|
|
return spawn.easy_async(cmd,
|
2020-11-29 22:48:32 +01:00
|
|
|
function (stdout, _, _, exit_code)
|
2018-01-22 18:55:37 +01:00
|
|
|
callback(stdout, exit_code)
|
2017-12-19 17:20:59 +01:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- like above, but call spawn.easy_async with a shell
|
|
|
|
function helpers.async_with_shell(cmd, callback)
|
|
|
|
return spawn.easy_async_with_shell(cmd,
|
2020-11-29 22:48:32 +01:00
|
|
|
function (stdout, _, _, exit_code)
|
2018-01-22 18:55:37 +01:00
|
|
|
callback(stdout, exit_code)
|
2017-01-20 20:58:22 +01:00
|
|
|
end)
|
2016-11-06 12:18:34 +01:00
|
|
|
end
|
|
|
|
|
2018-01-22 18:55:37 +01:00
|
|
|
-- run a command and execute a function on its output line by line
|
|
|
|
function helpers.line_callback(cmd, callback)
|
|
|
|
return spawn.with_line_callback(cmd, {
|
|
|
|
stdout = function (line)
|
|
|
|
callback(line)
|
|
|
|
end,
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2015-08-13 01:56:37 +02:00
|
|
|
-- }}}
|
|
|
|
|
2021-09-14 23:34:48 +02:00
|
|
|
-- {{{ Network functions
|
|
|
|
|
|
|
|
-- fetch a http uri using lgi.Soup
|
|
|
|
function helpers.uri(uri, callback, error_handler)
|
|
|
|
error_handler = error_handler or gdebug.print_error
|
|
|
|
local ss = lgi.Soup.Session()
|
|
|
|
local msg = lgi.Soup.Message.new('GET', uri)
|
|
|
|
ss:send_and_read_async(msg, lgi.GLib.PRIORITY_DEFAULT, nil, function(_, resp, _)
|
|
|
|
local buf, err = ss:send_and_read_finish(resp)
|
|
|
|
if err then
|
|
|
|
error_handler(err)
|
|
|
|
elseif buf then
|
|
|
|
callback(buf:get_data())
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- }}}
|
|
|
|
|
2013-09-07 12:06:42 +02:00
|
|
|
-- {{{ A map utility
|
|
|
|
|
|
|
|
helpers.map_table = {}
|
|
|
|
|
|
|
|
function helpers.set_map(element, value)
|
|
|
|
helpers.map_table[element] = value
|
|
|
|
end
|
|
|
|
|
|
|
|
function helpers.get_map(element)
|
|
|
|
return helpers.map_table[element]
|
|
|
|
end
|
|
|
|
|
|
|
|
-- }}}
|
|
|
|
|
2017-01-03 12:21:50 +01:00
|
|
|
-- {{{ Misc
|
|
|
|
|
2017-01-09 20:21:56 +01:00
|
|
|
-- check if an element exist on a table
|
|
|
|
function helpers.element_in_table(element, tbl)
|
|
|
|
for _, i in pairs(tbl) do
|
|
|
|
if i == element then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2017-01-03 12:21:50 +01:00
|
|
|
-- iterate over table of records sorted by keys
|
2015-11-18 09:02:16 +01:00
|
|
|
function helpers.spairs(t)
|
|
|
|
-- collect the keys
|
|
|
|
local keys = {}
|
|
|
|
for k in pairs(t) do keys[#keys+1] = k end
|
|
|
|
|
2022-06-11 12:25:17 +02:00
|
|
|
tsort(keys)
|
2015-11-18 09:02:16 +01:00
|
|
|
|
|
|
|
-- return the iterator function
|
|
|
|
local i = 0
|
|
|
|
return function()
|
|
|
|
i = i + 1
|
|
|
|
if keys[i] then
|
|
|
|
return keys[i], t[keys[i]]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-02-17 14:06:20 +01:00
|
|
|
-- create the partition of singletons of a given set
|
|
|
|
-- example: the trivial partition set of {a, b, c}, is {{a}, {b}, {c}}
|
2018-02-14 23:00:36 +01:00
|
|
|
function helpers.trivial_partition_set(set)
|
|
|
|
local ss = {}
|
|
|
|
for _,e in pairs(set) do
|
|
|
|
ss[#ss+1] = {e}
|
|
|
|
end
|
|
|
|
return ss
|
|
|
|
end
|
|
|
|
|
2019-05-27 14:12:00 +02:00
|
|
|
-- create the powerset of a given set
|
2018-02-14 23:00:36 +01:00
|
|
|
function helpers.powerset(s)
|
|
|
|
if not s then return {} end
|
|
|
|
local t = {{}}
|
|
|
|
for i = 1, #s do
|
|
|
|
for j = 1, #t do
|
|
|
|
t[#t+1] = {s[i],unpack(t[j])}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return t
|
|
|
|
end
|
|
|
|
|
2017-01-03 12:21:50 +01:00
|
|
|
-- }}}
|
2016-11-06 12:18:34 +01:00
|
|
|
|
2013-09-07 12:06:42 +02:00
|
|
|
return helpers
|