Lua
Tables
Sorting
Create an iterator
function utils.spairs(t, order)
-- collect the keys
local keys = {}
for k in pairs(t) do keys[#keys + 1] = k end
-- if order function given, sort by it by passing the table and keys a, b,
-- otherwise just sort the keys
if order then
table.sort(keys, function(a, b) return order(t, a, b) end)
else
table.sort(keys)
end
-- return the iterator function
local i = 0
return function()
i = i + 1
if keys![](i) then
return keys![t[keys[i](i])]
end
end
end
Example usage (from {mmx}):
local function make_sorted_entry_iterator(entries)
return utils.spairs(entries, function(es, name_a, name_b)
local entry_a = es![](name_a)
local entry_b = es![](name_b)
if entry_a.date or entry_b.date then
return (entry_a.date or MIN_DATE) > (entry_b.date or MIN_DATE)
else
return name_a < name_b
end
end)
end
utils.lua
local utils = {}
-- file io
function utils.file_exists(path)
local file = io.open(path, "r")
if file ~= nil then
file:close()
return true
else
return false
end
end
function utils.read_file(path)
local file = io.open(path, "rb") -- r read mode and b binary mode
if not file then
error("unable to open file for reading at " .. path)
end
local content = file:read "*a" -- *a or *all reads the whole file
file:close()
return content
end
function utils.write_file(path, content)
local file = io.open(path, "w")
if not file then
error("unable to open file for writing at " .. path)
end
file:write(content)
file:close()
end
function utils.delete_file(path)
assert(os.remove(path))
end
function utils.list_folders(directory)
local pfile = io.popen('cd ' .. directory .. ' && find ' .. directory .. ' -type d')
if not pfile then
error("unable to list folders in directory " .. directory)
end
local folder_names = {}
for folder_name in pfile:lines() do
folder_name = folder_name:sub(#directory + 1)
if not utils.starts_with(folder_name, '.') and #folder_name > 0 then
table.insert(folder_names, folder_name)
end
end
pfile:close()
return folder_names
end
function utils.list_files(directory, extension)
-- resulting command is:
-- find [dir] -type f -name "*.[ext]"
local pfile = io.popen('find -L ' .. directory .. ' -type f -name "*' .. (extension or "") .. '"')
if not pfile then
error("unable to list files in directory " .. directory .. " with extension " .. extension)
end
local file_names = {}
for file_name in pfile:lines() do
file_name = file_name:sub(#directory + 1)
table.insert(file_names, file_name)
end
pfile:close()
return file_names
end
-- strings
function utils.starts_with(str, start)
return str:sub(1, #start) == start
end
function utils.ends_with(str, ending)
return ending == "" or str:sub(-#ending) == ending
end
function utils.lines(str)
if str:sub(-1) ~= "\n" then
str = str .. "\n"
end
return str:gmatch("(.-)\n")
end
function utils.split(input, sep)
if sep == nil then
sep = "%s"
end
local t = {}
for str in string.gmatch(input, "([^" .. sep .. "]+)") do
table.insert(t, str)
end
return t
end
function utils.capitalize(str)
return (str:gsub("^%l", string.upper))
end
function utils.title_case(str)
return utils.capitalize(str:gsub(" %l", string.upper))
end
function utils.slugify(str)
return (str:gsub("[%s%p]", "_")):lower()
end
-- tables
function utils.has_keys(table, keys)
for _, key in pairs(keys) do
if table[key] == nil then
return false
end
end
return true
end
function utils.get_key_case_insensitive(table, key)
return table[key] or table[utils.capitalize(key)] or table[key:lower()] or table[key:upper()] or
table[utils.title_case(key)]
end
function utils.spairs(t, order)
-- collect the keys
local keys = {}
for k in pairs(t) do
keys[#keys + 1] = k
end
-- if order function given, sort by it by passing the table and keys a, b,
-- otherwise just sort the keys
if order then
table.sort(keys, function(a, b)
return order(t, a, b)
end)
else
table.sort(keys)
end
-- 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
function utils.dump(o)
if type(o) == 'table' then
local s = '{ '
for k, v in pairs(o) do
if type(k) ~= 'number' then
k = '"' .. k .. '"'
end
s = s .. '[' .. k .. '] = ' .. utils.dump(v) .. ','
end
return s .. '} '
else
return tostring(o)
end
end
-- dates
function utils.today()
local date_table = os.date("*t")
local year, month, day = date_table.year, date_table.month, date_table.day -- date_table.wday to date_table.day
return string.format("%d-%d-%d", year, month, day)
end
function utils.rss_date(date_str)
local handle = io.popen("date -R -d " .. date_str)
if not handle then
error("unable to run command in rss_date")
end
local result = handle:read("*a")
handle:close()
return result
end
return utils
Compiled 2025-01-19