#!/bin/sh _EXEC="$(realpath "${0%/*}")" . "$_EXEC/cgilite/cgilite.sh" . "$_EXEC/cgilite/storage.sh" debug(){ if [ $# = 0 ]; then tee /dev/stderr else printf %s\\n "$*" >/dev/stderr fi } timeid(){ # generate time based ID # Fixme: Unix time stamps assumed to be 32bit always d=$(date +%s) { printf $( while [ "$d" -gt 0 ]; do printf \\%o\\n $((d % 256)) d=$((d / 256)) done |tac |tr -d \\n ) head -c5 /dev/urandom } \ | uuencode -m - \ | sed -n '2{y;+/;:_;;p}' } checkid(){ grep -m 1 -xE '[0-9a-zA-Z:_]{12}'; } yield_page(){ printf 'Content-Type: text/html; charset=utf-8\r\n\r\n' ./cgilite/html-sh.sed < %s [label for=hourly Hourly Rate:] [input #hourly type=number name=hourly value="%s"] [submit "update" "update" Update] ]' \ "$(HTML $id)" \ "$(UNSTRING "${address#address=}" |HTML)" \ "$(UNSTRING "${hourly#hourly=}" |grep -xE '[0-9]+')" } edit_sender(){ id="$1" if [ -f "senders/$id" ]; then address="$(cat "senders/$id")" fi [ "$address" ] || address="Name Street City Phone: 000 000000 Tax no. xxx / 000 / ### " printf ' [form method="POST" action="/update_sender" [hidden "id" "%s"] [submit "update" "update" Update] ]' \ "$(HTML $id)" \ "$(HTML "${address}")" } list_clients(){ [ -d clients/ ] && for c in clients/*; do read -r address hourly <"$c" address="$(UNSTRING "${address#address=}")" [ "$address" ] || address="(no address)" printf '[div .client .address %s [label Hourly Rate:] %s€ [a href="/clients/%s" Edit]] ' "$(HTML "$address")" \ "$(HTML "${hourly#hourly=}")" \ "$(HTML "${c#clients/}")" done } list_senders(){ [ -d senders/ ] && for s in senders/*; do address=$(cat "$s") [ "$address" ] || address="(no address)" printf '[div .sender .address %s[a href="/senders/%s" Edit]] ' "$(HTML "$address")" "$(HTML "${s#senders/}")" done } sender_list(){ local select="$1" n name [ -d senders/ ] && for n in '' senders/*; do [ "$n" ] && name="$(sed q "$n" |HTML)" [ "${n#senders/}" = "$select" ] \ && printf '' "${n#senders/}" "$name" \ || printf '' "${n#senders/}" "$name" done } client_list(){ local select="$1" n address hourly name [ -d clients/ ] && for n in '' clients/*; do [ "$n" ] && read -r address hourly <"$n" name="$(UNSTRING "${address#address=}" |sed q |HTML)" [ "${n#clients/}" = "$select" ] \ && printf '' "${n#clients/}" "$name" \ || printf '' "${n#clients/}" "$name" done } list_invoices(){ [ -d invoices/ ] && for i in invoices/*; do read -r sender client date number vat vatrate<<-EOF $(sed q "$i") EOF [ -f "senders/${sender#sender=}" ] \ && sender="$(sed q "senders/${sender#sender=}")" \ || sender="(unset)" [ ! -f "clients/${client#client=}" ] \ && client="(unset)" \ || read -r client hourly <"clients/${client#client=}" [ "${date#date=}" -ge 0 ] 2>&- \ && date="$(date -d "@${date#date=}" +%x)" \ || date="(unset)" printf '[div .invoice [h2 %s] [label From:] %s [label To:] %s [label on] %s [a href="/invoice/%s" Edit] ]' "$(UNSTRING "${number#number=}" |HTML)" \ "$(HTML "$sender")" \ "$(UNSTRING "${client#address=}" |sed q |HTML)" "$(HTML "$date")" \ "$(HTML ${i#invoices/})" done } edit_invoice(){ id="$1" if [ -f "invoices/$id" ]; then read -r sender client date number vat vatrate<<-EOF $(sed q "invoices/$id") EOF fi [ "${date#date=}" -ge 0 ] 2>&- \ && date="$(date -d "@${date#date=}" +%F)" \ || date="$(date +%F)" [ "${number#number=}" ] || number="number=$(date +%s)" [ "${vatrate#vatrate=}" -ge 0 ] 2>&- || vatrate="vatrate=19" cat <<-EOF [form method="POST" action="/update_invoice" [hidden "id" "$(HTML "$id")"] [label Sender:] [select name=sender $(sender_list "${sender#sender=}") ] [label Client:] [select name=client $(client_list "${client#client=}") ] [label for=number Invoice Number:] [input #number name=number value="$(UNSTRING "${number#number=}" |HTML)"] [label for=date Date:] [input #date name=date value="${date}" placeholder="YYYY-MM-TT"] [radio "vat" "smallbusiness" #vatsb $([ "${vat#vat=}" = smallbusiness ] && printf checked) ] [label for=vatsb Small business exemption from VAT] [radio "vat" "nett" #vatnett $([ "${vat#vat=}" = nett ] && printf checked)] [label for=vatnett Nett][radio "vat" "gross" #vatgross $([ "${vat#vat=}" = gross ] && printf checked)][label for=vatgross Gross] [label for=vatrate VAT Rate: [input type=number name="vatrate" value="${vatrate#vatrate=}"]% ] [submit "update" "update" Update] ] EOF } info="$(PATH "${PATH_INFO}")" case $info in /invoices.css) . "$_EXEC/cgilite/file.sh" FILE "$_EXEC/invoices.css" ;; /clients) { list_clients printf '[a .new href="/clients/%s" New]' "$(timeid)" } | yield_page clients ;; /clients/*) edit_client "${info#/clients/}" |yield_page client ;; /update_client) id="$(POST id |checkid)" if [ "$(POST update)" = update -a "$id" ]; then mkdir -p clients printf 'address=%s hourly=%s' \ "$(POST address |STRING)" "$(POST hourly |STRING)" \ >"clients/$id" else echo Invalid Data "$(POST id)" "$(POST update)" >&2 fi REDIRECT /clients/ ;; /senders) { list_senders printf '[a .new href="/senders/%s" New]' "$(timeid)" } | yield_page senders ;; /senders/*) edit_sender "${info#/senders/}" |yield_page sender ;; /update_sender) id="$(POST id |checkid)" if [ "$(POST update)" = update -a "$id" ]; then mkdir -p senders POST address >"senders/$id" fi REDIRECT /senders/ ;; /invoices) { list_invoices printf '[a .new href="/invoice/%s" New]' "$(timeid)" } | yield_page invoices ;; /invoice/*) edit_invoice "${info#/invoice/}" |yield_page invoice ;; /update_invoice) id="$(POST id |checkid)" if [ "$(POST update)" = update -a "$id" ]; then mkdir -p invoices printf 'sender=%s client=%s date=%s number=%s vat=%s vatrate=%s\n' \ "$(POST sender)" "$(POST client)" \ "$(date -d "$(POST date)" +%s)" \ "$(POST number |STRING)" \ "$(POST vat |grep -m1 -xE 'smallbusiness|gross|nett')" \ "$(POST vatrate |grep -m1 -xE '[0-9]+')" \ >"invoices/$id" fi REDIRECT "/invoice/$id" ;; *) REDIRECT /invoices ;; esac