310 lines
8.5 KiB
Lua
310 lines
8.5 KiB
Lua
|
-- helper functions
|
||
|
-- Copyright (C) 2009 Benedikt Sauer <filmor@gmail.com>
|
||
|
-- Copyright (C) 2009 Henning Glawe <glaweh@debian.org>
|
||
|
-- Copyright (C) 2009 Lucas de Vries <lucas@glacicle.com>
|
||
|
-- Copyright (C) 2009 Rémy C. <shikamaru@mandriva.org>
|
||
|
-- Copyright (C) 2009-2012 Adrian C. (anrxc) <anrxc@sysphere.org>
|
||
|
-- Copyright (C) 2011 Joerg T. (Mic92) <jthalheim@gmail.com>
|
||
|
-- Copyright (C) 2012 Arvydas Sidorenko <asido4@gmail.com>
|
||
|
-- Copyright (C) 2012 Jörg Thalheim <jthalheim@gmail.com>
|
||
|
-- Copyright (C) 2014-2015 Jörg Thalheim <joerg@higgsboson.tk>
|
||
|
-- Copyright (C) 2017 Joerg Thalheim <joerg@thalheim.io>
|
||
|
-- Copyright (C) 2017,2019 mutlusun <mutlusun@github.com>
|
||
|
-- Copyright (C) 2017-2018 Jörg Thalheim <joerg@thalheim.io>
|
||
|
-- Copyright (C) 2018-2019 Nguyễn Gia Phong <vn.mcsinyx@gmail.com>
|
||
|
-- Copyright (C) 2019 Alexander Koch <lynix47@gmail.com>
|
||
|
-- Copyright (C) 2019 Enric Morales <me@enric.me>
|
||
|
--
|
||
|
-- This file is part of Vicious.
|
||
|
--
|
||
|
-- Vicious is free software: you can redistribute it and/or modify
|
||
|
-- it under the terms of the GNU General Public License as
|
||
|
-- published by the Free Software Foundation, either version 2 of the
|
||
|
-- License, or (at your option) any later version.
|
||
|
--
|
||
|
-- Vicious is distributed in the hope that it will be useful,
|
||
|
-- but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
-- GNU General Public License for more details.
|
||
|
--
|
||
|
-- You should have received a copy of the GNU General Public License
|
||
|
-- along with Vicious. If not, see <https://www.gnu.org/licenses/>.
|
||
|
|
||
|
-- {{{ Grab environment
|
||
|
local ipairs = ipairs
|
||
|
local pairs = pairs
|
||
|
local rawget = rawget
|
||
|
local require = require
|
||
|
local tonumber = tonumber
|
||
|
local tostring = tostring
|
||
|
local io = { open = io.open, popen = io.popen }
|
||
|
local setmetatable = setmetatable
|
||
|
local getmetatable = getmetatable
|
||
|
local string = {
|
||
|
upper = string.upper,
|
||
|
lower = string.lower,
|
||
|
format = string.format,
|
||
|
match = string.match,
|
||
|
find = string.find,
|
||
|
}
|
||
|
local table = { concat = table.concat }
|
||
|
local pcall = pcall
|
||
|
local assert = assert
|
||
|
local spawn = require("vicious.spawn")
|
||
|
-- }}}
|
||
|
|
||
|
|
||
|
-- Helpers: provides helper functions for vicious widgets
|
||
|
-- vicious.helpers
|
||
|
local helpers = {}
|
||
|
|
||
|
-- {{{ Constants definitions
|
||
|
local OS_UNSUPPORTED_ERR = "Vicious: platform not supported: %s"
|
||
|
local NOT_FOUND_MSG = "module '%s' not found"
|
||
|
local NOT_FOUND_ERR = [[
|
||
|
Vicious: %s is not available for the current platform or does not exist]]
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Variable definitions
|
||
|
local scroller = {}
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Helper functions
|
||
|
-- {{{ Determine operating system
|
||
|
local kernel_name
|
||
|
function helpers.getos()
|
||
|
if kernel_name ~= nil then
|
||
|
return kernel_name
|
||
|
end
|
||
|
|
||
|
local f = io.popen("uname -s")
|
||
|
kernel_name = string.lower(f:read("*line"))
|
||
|
f:close()
|
||
|
|
||
|
return kernel_name
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Loader of vicious modules
|
||
|
function helpers.wrequire(collection, key)
|
||
|
local ret = rawget(collection, key)
|
||
|
|
||
|
if ret then
|
||
|
return ret
|
||
|
end
|
||
|
|
||
|
local ostable = {
|
||
|
linux = { "linux", "all" },
|
||
|
freebsd = { "freebsd", "bsd", "all" },
|
||
|
openbsd = { "openbsd", "bsd", "all" }
|
||
|
}
|
||
|
|
||
|
local platform = ostable[helpers.getos()]
|
||
|
assert(platform, OS_UNSUPPORTED_ERR:format(helpers.getos()))
|
||
|
|
||
|
local basename = collection._NAME .. '.' .. key
|
||
|
for i = 1, #platform do
|
||
|
local name = basename .. '_' .. platform[i]
|
||
|
local status, value = pcall(require, name)
|
||
|
if status then
|
||
|
ret = value
|
||
|
break
|
||
|
end
|
||
|
|
||
|
-- This is ugly but AFAWK there is no other way to check for
|
||
|
-- the type of error. If other error get caught, raise it.
|
||
|
if value:find(NOT_FOUND_MSG:format(name), 1, true) == nil then
|
||
|
require(name)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
assert(ret, NOT_FOUND_ERR:format(basename))
|
||
|
return ret
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Set widget type's __call metamethod to given worker function
|
||
|
function helpers.setcall(worker)
|
||
|
return setmetatable(
|
||
|
{}, { __call = function(_, ...) return worker(...) end })
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Set __call metamethod to widget type table having async key
|
||
|
function helpers.setasyncall(wtype)
|
||
|
local function worker(format, warg)
|
||
|
local ret
|
||
|
wtype.async(format, warg, function (data) ret = data end)
|
||
|
while ret == nil do end
|
||
|
return ret
|
||
|
end
|
||
|
local metatable = { __call = function (_, ...) return worker(...) end }
|
||
|
return setmetatable(wtype, metatable)
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Expose path as a Lua table
|
||
|
function helpers.pathtotable(dir)
|
||
|
return setmetatable({ _path = dir },
|
||
|
{ __index = function(self, index)
|
||
|
local path = self._path .. '/' .. index
|
||
|
local f = io.open(path)
|
||
|
if f then
|
||
|
local s = f:read("*all")
|
||
|
f:close()
|
||
|
if s then
|
||
|
return s
|
||
|
else
|
||
|
local o = { _path = path }
|
||
|
setmetatable(o, getmetatable(self))
|
||
|
return o
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
})
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Format a string with args
|
||
|
function helpers.format(format, args)
|
||
|
for var, val in pairs(args) do
|
||
|
format = format:gsub("$" .. (tonumber(var) and var or
|
||
|
var:gsub("[-+?*]", function(i) return "%"..i end)),
|
||
|
val)
|
||
|
end
|
||
|
|
||
|
return format
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Format units to one decimal point
|
||
|
function helpers.uformat(array, key, value, unit)
|
||
|
for u, v in pairs(unit) do
|
||
|
array["{"..key.."_"..u.."}"] = string.format("%.1f", value/v)
|
||
|
end
|
||
|
|
||
|
return array
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Escape a string
|
||
|
function helpers.escape(text)
|
||
|
local xml_entities = {
|
||
|
["\""] = """,
|
||
|
["&"] = "&",
|
||
|
["'"] = "'",
|
||
|
["<"] = "<",
|
||
|
[">"] = ">"
|
||
|
}
|
||
|
|
||
|
return text and text:gsub("[\"&'<>]", xml_entities)
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Escape a string for safe usage on the command line
|
||
|
function helpers.shellquote(arg)
|
||
|
local s = tostring(arg)
|
||
|
if s == nil then return "" end
|
||
|
-- use single quotes, and put single quotes into double quotes
|
||
|
-- the string $'b is then quoted as '$'"'"'b'"'"'
|
||
|
return "'" .. s:gsub("'", "'\"'\"'") .. "'"
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Capitalize a string
|
||
|
function helpers.capitalize(text)
|
||
|
return text and text:gsub("([%w])([%w]*)", function(c, s)
|
||
|
return string.upper(c) .. s
|
||
|
end)
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Truncate a string
|
||
|
function helpers.truncate(text, maxlen)
|
||
|
local txtlen = text:len()
|
||
|
|
||
|
if txtlen > maxlen then
|
||
|
text = text:sub(1, maxlen - 3) .. "..."
|
||
|
end
|
||
|
|
||
|
return text
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Scroll through a string
|
||
|
function helpers.scroll(text, maxlen, widget)
|
||
|
if not scroller[widget] then
|
||
|
scroller[widget] = { i = 1, d = true }
|
||
|
end
|
||
|
|
||
|
local txtlen = text:len()
|
||
|
local state = scroller[widget]
|
||
|
|
||
|
if txtlen > maxlen then
|
||
|
if state.d then
|
||
|
text = text:sub(state.i, state.i + maxlen) .. "..."
|
||
|
state.i = state.i + 3
|
||
|
|
||
|
if maxlen + state.i >= txtlen then
|
||
|
state.d = false
|
||
|
end
|
||
|
else
|
||
|
text = "..." .. text:sub(state.i, state.i + maxlen)
|
||
|
state.i = state.i - 3
|
||
|
|
||
|
if state.i <= 1 then
|
||
|
state.d = true
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
return text
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
-- {{{ Parse output of sysctl command calling the `parse` function
|
||
|
function helpers.sysctl_async(path_table, parse)
|
||
|
local ret = {}
|
||
|
local path = {}
|
||
|
|
||
|
for i=1,#path_table do
|
||
|
path[i] = helpers.shellquote(path_table[i])
|
||
|
end
|
||
|
|
||
|
path = table.concat(path, " ")
|
||
|
|
||
|
spawn.with_line_callback("sysctl " .. path, {
|
||
|
stdout = function (line)
|
||
|
local separators = {
|
||
|
freebsd = ": ",
|
||
|
linux = " = ",
|
||
|
openbsd = "="
|
||
|
}
|
||
|
local pattern = ("(.+)%s(.+)"):format(separators[helpers.getos()])
|
||
|
local key, value = string.match(line, pattern)
|
||
|
ret[key] = value
|
||
|
end,
|
||
|
stderr = function (line)
|
||
|
local messages = {
|
||
|
openbsd = { "level name .+ in (.+) is invalid" },
|
||
|
linux = { "cannot stat /proc/sys/(.+):",
|
||
|
"permission denied on key '(.+)'" },
|
||
|
freebsd = { "unknown oid '(.+)'" }
|
||
|
}
|
||
|
|
||
|
for _, error_message in ipairs(messages[helpers.getos()]) do
|
||
|
local key = line:match(error_message)
|
||
|
if key then
|
||
|
key = key:gsub("/", ".")
|
||
|
ret[key] = "N/A"
|
||
|
end
|
||
|
end
|
||
|
end,
|
||
|
output_done = function () parse(ret) end
|
||
|
})
|
||
|
end
|
||
|
-- }}}
|
||
|
|
||
|
return helpers
|
||
|
-- }}}
|