2024-10-30 01:00:27 +00:00
|
|
|
#!/usr/bin/env luajit
|
2024-11-04 17:12:10 +00:00
|
|
|
--[[
|
|
|
|
Abrechenbarkeit - A simple trust-based ledger
|
|
|
|
Copyright 2024 metamuffin
|
|
|
|
Copyright 2024 dasriley
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Affero General Public License as published by
|
|
|
|
the Free Software Foundation, version 3 of the License only.
|
|
|
|
|
|
|
|
This program 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 Affero General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2024-11-04 19:05:04 +00:00
|
|
|
]] --
|
2024-10-30 01:00:27 +00:00
|
|
|
|
|
|
|
local function escape(s)
|
|
|
|
return s:gsub("<", "<"):gsub("<", "<")
|
|
|
|
end
|
2024-10-31 01:00:42 +00:00
|
|
|
|
2024-10-30 12:50:36 +00:00
|
|
|
local function urldecode(s)
|
|
|
|
if s == nil then return nil end
|
|
|
|
return s:gsub("+", " "):gsub("%%20", " ")
|
|
|
|
end
|
2024-10-31 01:00:42 +00:00
|
|
|
|
2024-10-30 12:50:36 +00:00
|
|
|
local function urlencode(s)
|
|
|
|
if s == nil then return nil end
|
|
|
|
return s:gsub(" ", "%%20")
|
|
|
|
end
|
|
|
|
|
|
|
|
local function parse_query(q)
|
|
|
|
if q == nil then return {} end
|
|
|
|
local data = {}
|
|
|
|
for pair in string.gmatch(q, "([^&]+)") do
|
|
|
|
local flag = string.match(pair, "^([^=]+)$")
|
|
|
|
if flag ~= nil then
|
|
|
|
data[flag] = "1"
|
|
|
|
else
|
|
|
|
local key, value = string.match(pair, "^([^=]+)=([^=]*)$")
|
|
|
|
if key ~= nil and value ~= nil then
|
|
|
|
data[key] = urldecode(value)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return data
|
|
|
|
end
|
|
|
|
|
2024-10-30 22:09:58 +00:00
|
|
|
local function load_config()
|
|
|
|
local log = io.open("config", "r")
|
|
|
|
if log == nil then return {} end
|
|
|
|
local config = {}
|
|
|
|
for l in log:lines("l") do
|
|
|
|
if l ~= "" and l[0] ~= "#" then
|
|
|
|
local key, value = string.match(l, "^([^=]+)=([^=]*)")
|
|
|
|
if key ~= nil and value ~= nil then
|
|
|
|
config[key] = value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return config
|
|
|
|
end
|
|
|
|
|
2024-11-04 19:05:04 +00:00
|
|
|
local function load_translations(langs)
|
|
|
|
local t = {}
|
|
|
|
for _, lcode in ipairs(langs) do
|
|
|
|
local file = io.open(string.format("locale/%s.ini", lcode), "r")
|
|
|
|
if file ~= nil then
|
|
|
|
for l in file:lines("l") do
|
|
|
|
if l ~= "" then
|
|
|
|
local key, value = string.match(l, "^([^=]+)=([^=]*)")
|
|
|
|
if key ~= nil and value ~= nil then
|
|
|
|
t[key] = value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return t
|
|
|
|
end
|
|
|
|
|
2024-10-30 22:09:58 +00:00
|
|
|
local config = load_config()
|
2024-10-30 12:50:36 +00:00
|
|
|
local path = os.getenv("PATH_INFO")
|
|
|
|
local method = os.getenv("REQUEST_METHOD")
|
|
|
|
local query = parse_query(os.getenv("QUERY_STRING"))
|
2024-11-04 19:05:04 +00:00
|
|
|
local translations = load_translations({ "en", config.language })
|
2024-10-30 12:50:36 +00:00
|
|
|
|
2024-11-03 17:38:21 +00:00
|
|
|
local stylesheet = io.open("style.css"):read("a")
|
|
|
|
local script = io.open("script.js"):read("a")
|
2024-10-30 12:50:36 +00:00
|
|
|
|
2024-11-04 19:05:04 +00:00
|
|
|
local function format(template, params)
|
|
|
|
return string.gsub(template, "{([%w\\.!]+)}", function(n)
|
|
|
|
local raw = n:sub(1,1) ~= "!"
|
|
|
|
if raw then n = n:sub(2) end
|
|
|
|
local s = format(params[n] or translations[n] or "NIL TEMPLATE", params)
|
|
|
|
return raw and s or escape(s)
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
local function format_amount(amount, tag, classes)
|
|
|
|
local s = string.format("%.02f%s", amount / 100, config.unit or "€")
|
|
|
|
if tag == nil then return s end
|
|
|
|
return string.format([[<%s class="amount-%s %s">%s</%s>"]], tag, amount >= 0 and "pos" or "neg", s, tag, classes or "")
|
|
|
|
end
|
|
|
|
|
2024-11-03 22:33:09 +00:00
|
|
|
local function get_user_theme(username)
|
|
|
|
local c = ""
|
|
|
|
if username == "_jeb" then
|
|
|
|
c = "html { animation: 2s jeb infinite; }"
|
|
|
|
c = c .. "@keyframes jeb {\n"
|
2024-11-04 15:16:28 +00:00
|
|
|
for i = 0, 100 do
|
2024-11-03 22:33:09 +00:00
|
|
|
c = c .. string.format("%.02f%% { --hue: %.02f; } \n", i, i / 100 * 360)
|
|
|
|
end
|
|
|
|
c = c .. "\n}"
|
|
|
|
elseif username == "Dinnerbone" then
|
2024-11-04 10:59:53 +00:00
|
|
|
c = "html { transform: scale(-1); } "
|
2024-11-03 22:33:09 +00:00
|
|
|
end
|
2024-11-04 19:05:04 +00:00
|
|
|
return c
|
2024-11-03 22:33:09 +00:00
|
|
|
end
|
|
|
|
|
2024-11-04 19:05:04 +00:00
|
|
|
local function format_duration(t)
|
|
|
|
if t > 86400 then return string.format("%d day%s", t / 86400, math.floor(t / 86400) ~= 1 and "s" or "") end
|
|
|
|
if t > 3600 then return string.format("%d hour%s", t / 3600, math.floor(t / 3600) ~= 1 and "s" or "") end
|
|
|
|
if t > 60 then return string.format("%d minute%s", t / 60, math.floor(t / 60) ~= 1 and "s" or "") end
|
|
|
|
return string.format("%d seconds", t)
|
|
|
|
end
|
2024-11-04 10:59:53 +00:00
|
|
|
|
2024-11-04 19:05:04 +00:00
|
|
|
local function respond(status, title, body)
|
2024-10-30 01:00:27 +00:00
|
|
|
print(string.format("Status: %d", status))
|
|
|
|
print("Content-Type: text/html")
|
|
|
|
print("")
|
2024-11-04 19:05:04 +00:00
|
|
|
print(format([[
|
2024-10-30 21:00:15 +00:00
|
|
|
<!DOCTYPE html>
|
2024-10-30 01:00:27 +00:00
|
|
|
<html><head>
|
2024-11-04 19:05:04 +00:00
|
|
|
<title>{title}</title>
|
2024-10-30 01:00:27 +00:00
|
|
|
<meta charset="utf-8" />
|
2024-11-04 19:05:04 +00:00
|
|
|
<style>{style}</style>
|
|
|
|
<style>{user_style}</style>
|
|
|
|
<script>{script}</script>
|
|
|
|
{head_extra}
|
2024-10-30 12:50:36 +00:00
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<nav>
|
2024-11-03 17:31:13 +00:00
|
|
|
<a class="logo" href="/">
|
|
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-receipt-cutoff" viewBox="0 0 16 16">
|
|
|
|
<path d="M3 4.5a.5.5 0 0 1 .5-.5h6a.5.5 0 1 1 0 1h-6a.5.5 0 0 1-.5-.5m0 2a.5.5 0 0 1 .5-.5h6a.5.5 0 1 1 0 1h-6a.5.5 0 0 1-.5-.5m0 2a.5.5 0 0 1 .5-.5h6a.5.5 0 1 1 0 1h-6a.5.5 0 0 1-.5-.5m0 2a.5.5 0 0 1 .5-.5h6a.5.5 0 0 1 0 1h-6a.5.5 0 0 1-.5-.5m0 2a.5.5 0 0 1 .5-.5h6a.5.5 0 0 1 0 1h-6a.5.5 0 0 1-.5-.5M11.5 4a.5.5 0 0 0 0 1h1a.5.5 0 0 0 0-1zm0 2a.5.5 0 0 0 0 1h1a.5.5 0 0 0 0-1zm0 2a.5.5 0 0 0 0 1h1a.5.5 0 0 0 0-1zm0 2a.5.5 0 0 0 0 1h1a.5.5 0 0 0 0-1zm0 2a.5.5 0 0 0 0 1h1a.5.5 0 0 0 0-1z"/>
|
|
|
|
<path d="M2.354.646a.5.5 0 0 0-.801.13l-.5 1A.5.5 0 0 0 1 2v13H.5a.5.5 0 0 0 0 1h15a.5.5 0 0 0 0-1H15V2a.5.5 0 0 0-.053-.224l-.5-1a.5.5 0 0 0-.8-.13L13 1.293l-.646-.647a.5.5 0 0 0-.708 0L11 1.293l-.646-.647a.5.5 0 0 0-.708 0L9 1.293 8.354.646a.5.5 0 0 0-.708 0L7 1.293 6.354.646a.5.5 0 0 0-.708 0L5 1.293 4.354.646a.5.5 0 0 0-.708 0L3 1.293zm-.217 1.198.51.51a.5.5 0 0 0 .707 0L4 1.707l.646.647a.5.5 0 0 0 .708 0L6 1.707l.646.647a.5.5 0 0 0 .708 0L8 1.707l.646.647a.5.5 0 0 0 .708 0L10 1.707l.646.647a.5.5 0 0 0 .708 0L12 1.707l.646.647a.5.5 0 0 0 .708 0l.509-.51.137.274V15H2V2.118z"/>
|
|
|
|
</svg>
|
2024-11-04 19:05:04 +00:00
|
|
|
{appname}
|
|
|
|
</a>
|
|
|
|
<a href="/?log">{log}</a>
|
|
|
|
<a href="/?products">{products}</a>
|
|
|
|
<a href="/?about">{about}</a>
|
2024-10-30 12:50:36 +00:00
|
|
|
</nav>
|
2024-11-04 19:05:04 +00:00
|
|
|
]], {
|
|
|
|
title = escape(title),
|
|
|
|
style = stylesheet,
|
|
|
|
user_style = get_user_theme(path and path:sub(2)),
|
|
|
|
script = script,
|
|
|
|
head_extra = config.head_extra or ""
|
|
|
|
}))
|
2024-10-30 01:00:27 +00:00
|
|
|
body()
|
2024-11-04 19:05:04 +00:00
|
|
|
print("</body></html>")
|
2024-10-30 01:00:27 +00:00
|
|
|
end
|
|
|
|
|
2024-11-04 13:44:56 +00:00
|
|
|
local function error_box(message)
|
|
|
|
return string.format([[<div class="notif error"><p>Error: %s</p></div>]], message)
|
|
|
|
end
|
|
|
|
|
2024-10-30 01:00:27 +00:00
|
|
|
local function respond_error(message)
|
|
|
|
respond(400, "Error", function()
|
2024-11-04 13:44:56 +00:00
|
|
|
print(error_box(message))
|
2024-10-30 01:00:27 +00:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
2024-10-30 11:20:03 +00:00
|
|
|
local function redirect(path)
|
|
|
|
print("Status: 307")
|
|
|
|
print(string.format("Location: %s", path))
|
|
|
|
print()
|
|
|
|
end
|
|
|
|
|
2024-10-30 01:00:27 +00:00
|
|
|
local function form_data()
|
2024-10-30 12:50:36 +00:00
|
|
|
return parse_query(io.read())
|
2024-10-30 01:00:27 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
local function read_log()
|
|
|
|
local log = io.open("log", "r")
|
|
|
|
if log == nil then
|
|
|
|
return function() return nil end
|
|
|
|
end
|
|
|
|
local lines = log:lines("l")
|
|
|
|
return function()
|
|
|
|
local l = lines()
|
2024-10-30 12:50:36 +00:00
|
|
|
if l == "" or l == nil then
|
|
|
|
return nil
|
|
|
|
end
|
2024-11-03 18:38:24 +00:00
|
|
|
local time, username, amount, pcode, pcount, comment = string.match(l,
|
|
|
|
"(%d+),([%w_ -]+),(-?%d+),([%w_-]*),(-?%d*),([%w_ -]*)")
|
|
|
|
return tonumber(time), username, tonumber(amount), pcode, tonumber(pcount), comment
|
2024-10-30 01:00:27 +00:00
|
|
|
end
|
|
|
|
end
|
2024-10-31 01:00:42 +00:00
|
|
|
|
2024-10-30 12:50:36 +00:00
|
|
|
local function read_products()
|
|
|
|
local log = io.open("products", "r")
|
|
|
|
if log == nil then
|
|
|
|
return function() return nil end
|
|
|
|
end
|
|
|
|
local lines = log:lines("l")
|
|
|
|
return function()
|
|
|
|
local l = lines()
|
|
|
|
if l == "" or l == nil then
|
|
|
|
return nil
|
|
|
|
end
|
2024-11-04 11:24:17 +00:00
|
|
|
local barcode, price, name, owner = string.match(l, "([%w_-]+),(-?%d+),([%w_ -]*),([%w_ -]*)")
|
|
|
|
return barcode, tonumber(price), name, owner
|
2024-10-30 12:50:36 +00:00
|
|
|
end
|
|
|
|
end
|
2024-10-30 01:00:27 +00:00
|
|
|
|
2024-10-30 01:11:09 +00:00
|
|
|
local function balances()
|
|
|
|
local users = {}
|
2024-11-03 18:38:24 +00:00
|
|
|
for _, username, amount, _, _, _ in read_log() do
|
2024-10-30 01:11:09 +00:00
|
|
|
users[username] = (users[username] or 0) + amount
|
|
|
|
end
|
|
|
|
return users
|
|
|
|
end
|
2024-10-31 01:00:42 +00:00
|
|
|
|
2024-11-03 18:38:24 +00:00
|
|
|
local function product_balances()
|
|
|
|
local products = {}
|
|
|
|
for _, _, _, pcode, pcount, _ in read_log() do
|
|
|
|
if pcode ~= nil and pcount ~= nil then
|
|
|
|
products[pcode] = (products[pcode] or 0) + pcount
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return products
|
|
|
|
end
|
|
|
|
|
2024-10-30 21:00:15 +00:00
|
|
|
local function last_txns()
|
|
|
|
local users = {}
|
2024-11-03 18:38:24 +00:00
|
|
|
for time, username, _, _, _, _ in read_log() do
|
2024-10-30 21:00:15 +00:00
|
|
|
users[username] = time
|
2024-10-30 01:00:27 +00:00
|
|
|
end
|
2024-10-30 21:00:15 +00:00
|
|
|
return users
|
|
|
|
end
|
|
|
|
|
2024-11-03 21:23:21 +00:00
|
|
|
local function get_active_users()
|
2024-11-03 21:34:04 +00:00
|
|
|
local user_balances = {}
|
2024-11-03 21:23:21 +00:00
|
|
|
for time, username, amount, _, _, _ in read_log() do
|
|
|
|
user_balances[username] = {
|
2024-11-03 21:34:04 +00:00
|
|
|
time = time,
|
|
|
|
username = username,
|
|
|
|
balance = (user_balances[username] or { balance = 0 }).balance + amount
|
2024-11-03 21:23:21 +00:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2024-11-03 21:34:04 +00:00
|
|
|
local users = {}
|
|
|
|
for _, user in pairs(user_balances) do
|
|
|
|
table.insert(users, user)
|
|
|
|
end
|
|
|
|
|
|
|
|
table.sort(users, function(a, b) return a.time > b.time end)
|
2024-11-03 21:23:21 +00:00
|
|
|
return users
|
|
|
|
end
|
|
|
|
|
2024-10-30 21:00:15 +00:00
|
|
|
local function r_user_post(username)
|
|
|
|
local data = form_data()
|
2024-11-03 20:21:35 +00:00
|
|
|
local amount = tonumber(data.amount)
|
|
|
|
local comment = data.comment
|
2024-11-03 18:38:24 +00:00
|
|
|
local pcode = nil
|
|
|
|
local pcount = nil
|
2024-11-04 11:24:17 +00:00
|
|
|
local powner = nil
|
2024-11-04 15:24:41 +00:00
|
|
|
local powner_comment = nil
|
2024-11-03 20:21:35 +00:00
|
|
|
if data.pcode then
|
2024-11-04 11:24:17 +00:00
|
|
|
for p_barcode, p_amount, p_name, p_owner in read_products() do
|
2024-11-03 20:21:35 +00:00
|
|
|
if p_barcode == data.pcode then
|
2024-11-04 15:16:28 +00:00
|
|
|
powner = p_owner
|
2024-11-03 20:21:35 +00:00
|
|
|
pcount = (tonumber(data.pcount) or 1) * (data.negate_pcount ~= nil and -1 or 1)
|
2024-11-03 18:38:24 +00:00
|
|
|
pcode = p_barcode
|
2024-11-03 20:21:35 +00:00
|
|
|
if amount == nil then amount = pcount * p_amount end
|
2024-11-04 11:24:17 +00:00
|
|
|
if comment == nil then
|
2024-11-04 15:16:28 +00:00
|
|
|
comment = string.format("%s %d %s", pcount < 0 and "Buy" or "Restock",
|
2024-11-04 11:24:17 +00:00
|
|
|
math.abs(pcount), p_name)
|
2024-11-04 15:16:28 +00:00
|
|
|
|
|
|
|
powner_comment = string.format("%s %d %s %s %s",
|
|
|
|
pcount < 0 and "Sell" or "Restock",
|
|
|
|
math.abs(pcount), p_name,
|
|
|
|
pcount < 0 and "to" or "by",
|
|
|
|
username)
|
|
|
|
end
|
2024-10-30 12:50:36 +00:00
|
|
|
end
|
|
|
|
end
|
2024-10-30 01:00:27 +00:00
|
|
|
if amount == nil then
|
2024-10-30 21:00:15 +00:00
|
|
|
return error_box("unknown product")
|
2024-10-30 01:00:27 +00:00
|
|
|
end
|
|
|
|
end
|
2024-10-30 21:00:15 +00:00
|
|
|
if amount == nil then
|
|
|
|
return error_box("amount invalid")
|
|
|
|
end
|
2024-11-03 20:21:35 +00:00
|
|
|
if comment == nil or comment:match("^[%w_ -]*$") == nil then
|
2024-10-30 21:00:15 +00:00
|
|
|
return error_box("comment invalid")
|
|
|
|
end
|
|
|
|
local log = io.open("log", "a+")
|
|
|
|
if log == nil then
|
|
|
|
return error_box("failed to open log")
|
|
|
|
end
|
|
|
|
local time = os.time()
|
2024-11-04 15:16:28 +00:00
|
|
|
-- subtract from buyer
|
2024-11-03 18:38:24 +00:00
|
|
|
log:write(string.format("%d,%s,%d,%s,%s,%s\n", time, username, amount, pcode or "", pcount or "", comment))
|
2024-11-04 11:24:17 +00:00
|
|
|
-- add to owner
|
|
|
|
if powner then
|
2024-11-04 15:16:28 +00:00
|
|
|
-- count is always zero as doesn't affect stock
|
|
|
|
log:write(string.format("%d,%s,%d,%s,%s,%s\n", time, powner, -amount, pcode or "", "", powner_comment))
|
2024-11-04 11:24:17 +00:00
|
|
|
end
|
2024-10-30 21:00:15 +00:00
|
|
|
log:flush()
|
|
|
|
log:close()
|
2024-11-04 19:05:04 +00:00
|
|
|
return format([[
|
|
|
|
<div class="notif"><p>Transaction successful: {amount} ({!comment})</p></div>
|
|
|
|
<audio src="{sound}" autoplay></audio>
|
|
|
|
]], {
|
|
|
|
sign = amount >= 0 and "pos" or "neg",
|
|
|
|
amount = format_amount(amount, "strong"),
|
|
|
|
comment = escape(comment),
|
|
|
|
sound = config.transaction_sound or ""
|
|
|
|
})
|
2024-10-30 21:00:15 +00:00
|
|
|
end
|
2024-10-30 01:00:27 +00:00
|
|
|
|
2024-10-30 21:00:15 +00:00
|
|
|
local function r_user(username)
|
|
|
|
local notif = nil
|
|
|
|
if method == "POST" then
|
|
|
|
notif = r_user_post(username)
|
|
|
|
end
|
2024-11-03 17:31:13 +00:00
|
|
|
return respond(200, string.format("Abrechenbarheit: %s", username), function()
|
2024-11-04 19:05:04 +00:00
|
|
|
print(format("<h1>{username}</h1>", { username = username }))
|
2024-10-30 12:50:36 +00:00
|
|
|
local balance = balances()[username]
|
2024-10-30 21:00:15 +00:00
|
|
|
local last_txn = last_txns()[username]
|
2024-10-30 12:50:36 +00:00
|
|
|
local new_user = balance == nil
|
|
|
|
balance = balance or 0
|
2024-10-30 22:09:58 +00:00
|
|
|
if notif then print(notif) end
|
2024-10-30 12:50:36 +00:00
|
|
|
if new_user then
|
2024-11-04 19:05:04 +00:00
|
|
|
print([[<div class="notif"><p><i>{user.lazy_creation}</i></p></div>]])
|
2024-10-30 22:09:58 +00:00
|
|
|
else
|
2024-11-04 15:16:28 +00:00
|
|
|
print([[<div class="backgroundbox userinfo">]])
|
2024-11-04 19:05:04 +00:00
|
|
|
print(format([[{user.balance}: <br>{amount}<br>]],
|
|
|
|
{ sign = balance >= 0 and "pos" or "neg", amount = format_amount(balance, "span", "balance-value") }))
|
|
|
|
print(format([[{user.last_txn} <a href="/{username}?log">{user.view_log}</a>]],
|
|
|
|
{ time = format_duration(os.time() - last_txn), username = urlencode(username) }))
|
2024-11-04 15:16:28 +00:00
|
|
|
print([[</div>]])
|
2024-10-30 12:50:36 +00:00
|
|
|
end
|
2024-11-03 18:38:22 +00:00
|
|
|
print([[<div class="transactions container firstchildlarge">]])
|
2024-11-03 20:21:35 +00:00
|
|
|
print([[<div class="amount-presets backgroundbox">]])
|
2024-11-03 17:31:13 +00:00
|
|
|
for _, type in ipairs({ 1, -1 }) do
|
|
|
|
for _, amount in ipairs({ 50, 100, 150, 200, 500, 1000 }) do
|
|
|
|
print(string.format([[
|
|
|
|
<form action="" method="POST">
|
|
|
|
<input type="number" name="amount" id="amount" value="%d" hidden />
|
|
|
|
<input type="text" name="comment" id="comment" value="" hidden />
|
2024-11-03 19:58:57 +00:00
|
|
|
<input type="submit" value="%s%.02f€" class="amount-%s button" />
|
2024-11-03 17:31:13 +00:00
|
|
|
</form>
|
|
|
|
]], amount * type, ({ [-1] = "-", [1] = "+" })[type], amount / 100,
|
|
|
|
({ [-1] = "neg", [1] = "pos" })[type]))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
print("</div>")
|
2024-11-04 19:05:04 +00:00
|
|
|
print(format([[
|
2024-11-03 18:38:22 +00:00
|
|
|
<form class="transaction box backgroundbox" action="" method="POST">
|
2024-11-04 19:05:04 +00:00
|
|
|
<h3>{user.form.transaction}</h3>
|
2024-11-03 19:58:57 +00:00
|
|
|
<label for="amount">Amount (ct): </label>
|
2024-11-03 17:31:13 +00:00
|
|
|
<input type="number" name="amount" id="amount" />
|
2024-10-30 01:00:27 +00:00
|
|
|
<label for="comment">Comment: </label>
|
2024-11-03 17:31:13 +00:00
|
|
|
<input type="text" name="comment" id="comment" />
|
2024-11-04 19:05:04 +00:00
|
|
|
<input type="submit" value="{user.form.transaction.submit}" class="amount-ntr button" />
|
2024-10-30 01:00:27 +00:00
|
|
|
</form>
|
2024-11-03 18:38:22 +00:00
|
|
|
<form class="transaction box backgroundbox" action="" method="POST" id="buy_product">
|
2024-11-04 19:05:04 +00:00
|
|
|
<h3>{user.form.buy}</h3>
|
2024-11-03 20:21:35 +00:00
|
|
|
<input type="text" name="negate_pcount" value="1" hidden />
|
|
|
|
<label for="pcount">Count: </label>
|
|
|
|
<input type="number" name="pcount" id="pcount" value="1" />
|
2024-11-04 19:05:04 +00:00
|
|
|
<label for="pcode">{field.barcode}: </label>
|
2024-11-03 20:21:35 +00:00
|
|
|
<input type="text" name="pcode" id="pcode" />
|
2024-11-04 19:05:04 +00:00
|
|
|
<input class="amount-neg button" type="submit" value="{user.form.buy.submit}" />
|
2024-10-30 21:00:15 +00:00
|
|
|
</form>
|
2024-11-03 20:21:35 +00:00
|
|
|
<form class="transaction box backgroundbox" action="" method="POST" id="buy_product">
|
2024-11-04 19:05:04 +00:00
|
|
|
<h3>{user.form.restock}</h3>
|
|
|
|
<label for="pcount">{field.count}: </label>
|
2024-11-03 20:21:35 +00:00
|
|
|
<input type="number" name="pcount" id="pcount" value="1" />
|
2024-11-04 19:05:04 +00:00
|
|
|
<label for="amount">{field.upstream_price}: </label>
|
2024-11-03 20:21:35 +00:00
|
|
|
<input type="number" name="amount" id="amount" />
|
2024-11-04 19:05:04 +00:00
|
|
|
<label for="pcode">{field.barcode}: </label>
|
2024-11-03 20:21:35 +00:00
|
|
|
<input type="text" name="pcode" id="pcode" />
|
2024-11-04 19:05:04 +00:00
|
|
|
<input type="submit" value="{user.form.restock.submit}" class="button amount-pos" />
|
2024-11-03 20:21:35 +00:00
|
|
|
</form>
|
2024-11-04 19:05:04 +00:00
|
|
|
]]))
|
2024-10-30 12:50:36 +00:00
|
|
|
print("</div>")
|
2024-10-30 01:00:27 +00:00
|
|
|
end)
|
|
|
|
end
|
2024-10-30 11:20:03 +00:00
|
|
|
|
2024-10-30 21:00:15 +00:00
|
|
|
local function r_log(filter)
|
2024-11-03 17:31:13 +00:00
|
|
|
return respond(200, "Abrechnungen", function()
|
2024-11-03 21:55:29 +00:00
|
|
|
print([[<table class="log"]])
|
2024-11-04 12:47:45 +00:00
|
|
|
print([[<thead><tr>
|
2024-11-03 20:21:35 +00:00
|
|
|
<th>Time</th>
|
|
|
|
<th>Username</th>
|
|
|
|
<th>Amount</th>
|
|
|
|
<th>P.-Barcode</th>
|
|
|
|
<th>P.-Count</th>
|
|
|
|
<th>Comment</th>
|
|
|
|
<th>Actions</th>
|
2024-11-04 12:47:45 +00:00
|
|
|
</tr></thead>]])
|
|
|
|
print("<tbody>")
|
2024-11-03 18:38:24 +00:00
|
|
|
for time, username, amount, pcode, pcount, comment in read_log() do
|
2024-10-30 21:00:15 +00:00
|
|
|
if filter == nil or filter == username then
|
2024-10-30 23:19:10 +00:00
|
|
|
print(string.format([[
|
|
|
|
<tr>
|
2024-11-03 21:34:04 +00:00
|
|
|
<td>%s (%s ago)</td>
|
2024-10-30 23:19:10 +00:00
|
|
|
<td>%s</td>
|
2024-10-31 00:44:12 +00:00
|
|
|
<td class="amount-%s">%.02f€</td>
|
2024-10-30 23:19:10 +00:00
|
|
|
<td>%s</td>
|
2024-11-03 20:21:35 +00:00
|
|
|
<td>%s %s</td>
|
2024-11-03 18:38:24 +00:00
|
|
|
<td>%s</td>
|
2024-10-30 23:19:10 +00:00
|
|
|
<td>
|
|
|
|
<form action="/%s" method="POST">
|
|
|
|
<input type="number" name="amount" id="amount" value="%d" hidden />
|
|
|
|
<input type="text" name="comment" id="comment" value="Revert %s" hidden />
|
2024-11-03 19:58:57 +00:00
|
|
|
<input type="submit" class="amount-ntr button" value="Revert" />
|
2024-10-30 23:19:10 +00:00
|
|
|
</form>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
]],
|
2024-11-03 21:34:04 +00:00
|
|
|
os.date("!%Y-%m-%dT%H:%M:%SZ", time), format_duration(os.time() - time),
|
2024-10-30 21:00:15 +00:00
|
|
|
escape(username),
|
|
|
|
amount >= 0 and "pos" or "neg", amount / 100,
|
2024-11-03 18:38:24 +00:00
|
|
|
escape(pcode) or "",
|
2024-11-03 20:21:35 +00:00
|
|
|
pcount and (pcount < 0 and "buy" or "stock") or "", pcount and tostring(math.abs(pcount)) or "",
|
2024-10-30 23:19:10 +00:00
|
|
|
escape(comment),
|
|
|
|
escape(username),
|
|
|
|
-amount,
|
2024-10-30 21:00:15 +00:00
|
|
|
escape(comment)
|
|
|
|
))
|
|
|
|
end
|
2024-10-30 11:20:03 +00:00
|
|
|
end
|
2024-11-04 12:47:45 +00:00
|
|
|
print("</tbody>")
|
2024-10-30 11:20:03 +00:00
|
|
|
print("</table>")
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
local function r_index()
|
2024-11-03 17:31:13 +00:00
|
|
|
return respond(200, "Abrechenbarkeit", function()
|
2024-10-30 11:20:03 +00:00
|
|
|
print([[
|
2024-11-04 15:16:28 +00:00
|
|
|
<form action="/" method="GET" id="user_creation">
|
2024-10-30 13:15:18 +00:00
|
|
|
<h3>User Creation</h3>
|
2024-10-30 11:20:03 +00:00
|
|
|
<label for="username">Username: </label>
|
2024-11-03 17:31:13 +00:00
|
|
|
<input type="text" name="create_user" id="username" />
|
2024-11-03 21:23:21 +00:00
|
|
|
<input type="submit" value="Continue" class="button amount-ntr" />
|
2024-10-30 11:20:03 +00:00
|
|
|
</form>
|
|
|
|
]])
|
2024-11-03 21:55:29 +00:00
|
|
|
print([[<div class="userlist"></div>]]) -- for printing
|
|
|
|
print([[<ul class="userlist">]])
|
2024-11-03 21:23:21 +00:00
|
|
|
for _, user in ipairs(get_active_users()) do
|
2024-10-31 00:44:12 +00:00
|
|
|
print(string.format([[
|
2024-11-03 17:31:13 +00:00
|
|
|
<li><a href="/%s"><span class="name">%s</span> <span class="amount amount-%s">%.02f€</span></a></li>
|
2024-10-31 00:44:12 +00:00
|
|
|
]],
|
2024-11-03 21:23:21 +00:00
|
|
|
urlencode(user.username),
|
|
|
|
escape(user.username),
|
|
|
|
user.balance >= 0 and "pos" or "neg", user.balance / 100
|
2024-10-31 00:44:12 +00:00
|
|
|
))
|
2024-10-30 11:20:03 +00:00
|
|
|
end
|
|
|
|
print("</ul>")
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
2024-11-04 13:44:56 +00:00
|
|
|
local function validate_username(username)
|
2024-11-04 15:16:28 +00:00
|
|
|
-- disallow leading or traling whitespace
|
|
|
|
return username ~= nil
|
|
|
|
and username:match("^([%w_ -]+)$") ~= nil
|
2024-11-04 13:44:56 +00:00
|
|
|
and username:match("^%s") == nil
|
|
|
|
and username:match("%s$") == nil
|
|
|
|
end
|
|
|
|
|
2024-10-30 11:20:03 +00:00
|
|
|
local function r_create_user()
|
2024-10-30 12:50:36 +00:00
|
|
|
local username = query.create_user
|
2024-11-04 13:44:56 +00:00
|
|
|
if not validate_username(username) then
|
2024-10-30 12:50:36 +00:00
|
|
|
return respond_error("invalid username " .. username)
|
2024-10-30 11:20:03 +00:00
|
|
|
end
|
2024-10-30 12:50:36 +00:00
|
|
|
return redirect(string.format("/%s", urlencode(username)))
|
2024-10-30 11:20:03 +00:00
|
|
|
end
|
|
|
|
|
2024-11-03 01:20:52 +00:00
|
|
|
local function r_products_post()
|
|
|
|
local data = form_data()
|
|
|
|
local barcode = data.barcode
|
|
|
|
if barcode == nil then
|
|
|
|
return error_box("barcode unset")
|
|
|
|
end
|
|
|
|
if barcode:match("^[%w_-]*$") == nil then
|
|
|
|
return error_box("barcode invalid")
|
|
|
|
end
|
|
|
|
if data.delete then
|
|
|
|
local new_products = io.open("products.new", "w+")
|
|
|
|
if new_products == nil then
|
|
|
|
return error_box("failed to open new products")
|
|
|
|
end
|
2024-11-04 11:24:17 +00:00
|
|
|
for a_barcode, price, name, owner in read_products() do
|
2024-11-03 01:20:52 +00:00
|
|
|
if barcode ~= a_barcode then
|
2024-11-04 11:24:17 +00:00
|
|
|
new_products:write(string.format("%s,%d,%s,%s\n", a_barcode, price, name, owner))
|
2024-11-03 01:20:52 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
new_products:flush()
|
|
|
|
new_products:close()
|
|
|
|
os.rename("products.new", "products")
|
|
|
|
else
|
|
|
|
local price = tonumber(data.price)
|
|
|
|
local name = data.name
|
|
|
|
if price == nil then
|
|
|
|
return error_box("price invalid")
|
|
|
|
end
|
|
|
|
if name:match("^[%w_ -]*$") == nil then
|
|
|
|
return error_box("name invalid")
|
|
|
|
end
|
|
|
|
local products = io.open("products", "a+")
|
|
|
|
if products == nil then
|
|
|
|
return error_box("failed to open products")
|
|
|
|
end
|
2024-11-04 11:24:17 +00:00
|
|
|
local owner = data.owner or ""
|
|
|
|
if name:match("^[%w_ -]*$") == nil then
|
|
|
|
return error_box("owner invalid")
|
|
|
|
end
|
|
|
|
products:write(string.format("%s,%d,%s,%s\n", barcode, price, name, owner))
|
2024-11-03 01:20:52 +00:00
|
|
|
products:flush()
|
|
|
|
products:close()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-10-31 01:00:42 +00:00
|
|
|
local function r_products()
|
2024-11-03 01:20:52 +00:00
|
|
|
local notif = nil
|
|
|
|
if method == "POST" then
|
|
|
|
notif = r_products_post()
|
|
|
|
end
|
|
|
|
respond(200, "Abrechenbare Product List", function()
|
2024-10-31 01:00:42 +00:00
|
|
|
print("<h1>Product List</h1>")
|
2024-11-03 01:20:52 +00:00
|
|
|
if notif then print(notif) end
|
|
|
|
print([[
|
2024-11-03 18:38:22 +00:00
|
|
|
<div class="container">
|
|
|
|
<form action="/?products" method="POST" class="box backgroundbox">
|
2024-11-03 01:20:52 +00:00
|
|
|
<h3>Add Product</h3>
|
|
|
|
<label for="name">Name: </label>
|
2024-11-03 18:38:22 +00:00
|
|
|
<input type="text" name="name" id="name" />
|
2024-11-03 19:58:57 +00:00
|
|
|
<label for="price">Price (ct): </label>
|
2024-11-03 20:05:03 +00:00
|
|
|
<input type="number" name="price" id="price" />
|
2024-11-04 11:24:17 +00:00
|
|
|
<label for="owner">Owner: </label>
|
|
|
|
<input type="text" name="owner" id="owner" />
|
2024-11-03 20:21:35 +00:00
|
|
|
<label for="barcode">Barcode: </label>
|
|
|
|
<input type="text" name="barcode" id="barcode" />
|
2024-11-03 20:05:03 +00:00
|
|
|
<input type="submit" value="Add" class="amount-ntr button" />
|
2024-11-03 01:20:52 +00:00
|
|
|
</form>
|
2024-11-03 18:38:22 +00:00
|
|
|
<form action="/?products" method="POST" class="box backgroundbox">
|
2024-11-03 01:20:52 +00:00
|
|
|
<h3>Remove Product</h3>
|
|
|
|
<input type="text" name="delete" value="1" hidden />
|
|
|
|
<label for="barcode">Barcode: </label>
|
2024-11-03 18:38:22 +00:00
|
|
|
<input type="text" name="barcode" id="barcode" />
|
2024-11-03 20:05:03 +00:00
|
|
|
<input type="submit" value="Remove"class="amount-ntr button" />
|
2024-11-03 01:20:52 +00:00
|
|
|
</form>
|
2024-11-03 18:38:22 +00:00
|
|
|
</div>
|
2024-11-03 01:20:52 +00:00
|
|
|
]])
|
2024-11-04 15:16:28 +00:00
|
|
|
print([[<table class="productlist"><tr>
|
|
|
|
<th>Name</th>
|
|
|
|
<th>Price</th>
|
|
|
|
<th>Barcode</th>
|
|
|
|
<th>Count</th>
|
|
|
|
<th>Owner</th>
|
|
|
|
</tr>]])
|
2024-11-03 18:38:24 +00:00
|
|
|
local pbals = product_balances()
|
2024-11-04 11:24:17 +00:00
|
|
|
for barcode, price, name, owner in read_products() do
|
2024-10-31 01:00:42 +00:00
|
|
|
print(string.format([[
|
2024-11-04 11:24:17 +00:00
|
|
|
<tr><td>%s</td><td class="amount-%s">%.02f€</td><td>%s</td><td>%s</td><td>%s</td></tr>
|
2024-10-31 01:00:42 +00:00
|
|
|
]],
|
|
|
|
name,
|
2024-11-03 20:21:35 +00:00
|
|
|
-price >= 0 and "pos" or "neg", -price / 100,
|
2024-11-03 18:38:24 +00:00
|
|
|
barcode,
|
2024-11-04 11:24:17 +00:00
|
|
|
pbals[barcode] or "0",
|
|
|
|
owner
|
2024-10-31 01:00:42 +00:00
|
|
|
))
|
|
|
|
end
|
|
|
|
print("</table>")
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
2024-11-04 17:12:10 +00:00
|
|
|
local function r_about()
|
|
|
|
respond(200, "About Abrechenbarkeit", function()
|
|
|
|
print([[
|
|
|
|
<h1>About Abrechenbarkeit</h1>
|
|
|
|
<p>Abrechenbarkeit is a simple trust-based ledger for keeping track of money spent on product.</p>
|
|
|
|
<p>Abrechenbarkeit is free software. It is licensed exclusively GNU Affero General Public License Version 3 only.<p>
|
|
|
|
<p>The source code is published on <a href="https://codeberg.org/metamuffin/strichliste">Codeberg</a>. This is also where <a href="https://codeberg.org/metamuffin/strichliste/issues">issues with this software</a> should be reported.</p>
|
|
|
|
<p>Thanks for choosing Abrechenbarkeit.</p>
|
|
|
|
]])
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
2024-10-30 21:00:15 +00:00
|
|
|
local function extract_username()
|
|
|
|
if path == nil then
|
|
|
|
return respond_error("no path")
|
|
|
|
end
|
|
|
|
local username = urldecode(path:sub(2))
|
|
|
|
if username == nil or username:match("^([%w_ -]+)$") == nil then
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
return username
|
|
|
|
end
|
|
|
|
|
2024-10-30 11:20:03 +00:00
|
|
|
if path == "/" then
|
2024-11-04 17:12:10 +00:00
|
|
|
if query.about then
|
|
|
|
return r_about()
|
|
|
|
elseif query.products then
|
2024-10-31 01:00:42 +00:00
|
|
|
return r_products()
|
|
|
|
elseif query.log then
|
2024-10-30 11:20:03 +00:00
|
|
|
return r_log()
|
2024-10-30 12:50:36 +00:00
|
|
|
elseif query.create_user then
|
2024-10-30 11:20:03 +00:00
|
|
|
return r_create_user()
|
|
|
|
else
|
|
|
|
return r_index()
|
|
|
|
end
|
|
|
|
else
|
2024-10-30 21:00:15 +00:00
|
|
|
local username = extract_username()
|
2024-11-04 13:44:56 +00:00
|
|
|
if username == nil or not validate_username(username) then
|
2024-10-30 21:00:15 +00:00
|
|
|
return respond_error("username invalid")
|
|
|
|
elseif query.log then
|
|
|
|
return r_log(username)
|
|
|
|
else
|
|
|
|
return r_user(username)
|
|
|
|
end
|
2024-10-30 11:20:03 +00:00
|
|
|
end
|