diff options
author | Sebastian Reichel <sre@ring0.de> | 2013-05-17 00:06:05 +0200 |
---|---|---|
committer | Sebastian Reichel <sre@ring0.de> | 2013-05-17 00:06:05 +0200 |
commit | ed125d14ad89e37361cb2fffcf26de5a447e020b (patch) | |
tree | 9f4bda852ba86b28680adf95703aac1b69bdc1bb /src/invoice | |
parent | 6eea30b050c1f083090d514e7b3a2dfcffbf2c49 (diff) | |
download | serial-barcode-scanner-ed125d14ad89e37361cb2fffcf26de5a447e020b.tar.bz2 |
new invoice tool
Diffstat (limited to 'src/invoice')
-rw-r--r-- | src/invoice/.gitignore | 1 | ||||
-rw-r--r-- | src/invoice/Makefile | 9 | ||||
-rw-r--r-- | src/invoice/invoice.vala | 288 | ||||
-rw-r--r-- | src/invoice/main.vala | 60 |
4 files changed, 358 insertions, 0 deletions
diff --git a/src/invoice/.gitignore b/src/invoice/.gitignore new file mode 100644 index 0000000..7079d00 --- /dev/null +++ b/src/invoice/.gitignore @@ -0,0 +1 @@ +invoice diff --git a/src/invoice/Makefile b/src/invoice/Makefile new file mode 100644 index 0000000..8b005ec --- /dev/null +++ b/src/invoice/Makefile @@ -0,0 +1,9 @@ +all: invoice + +invoice: main.vala invoice.vala ../mail/mailer-interface.vala ../pdf-invoice/pdf-invoice-interface.vala ../database/db-interface.vala ../config/config-interface.vala ../price.vapi + valac -o $@ --pkg gio-2.0 $^ + +clean: + rm -f invoice + +.PHONY: all clean diff --git a/src/invoice/invoice.vala b/src/invoice/invoice.vala new file mode 100644 index 0000000..63b3d07 --- /dev/null +++ b/src/invoice/invoice.vala @@ -0,0 +1,288 @@ +/* Copyright 2013, Sebastian Reichel <sre@ring0.de> + * + * Permission to use, copy, modify, and/or distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +public const int day_in_seconds = 60*60*24; + +public struct Timespan { + int64 from; + int64 to; +} + +public class InvoiceImplementation { + Mailer mailer; + Database db; + PDFInvoice pdf; + string datapath; + + public InvoiceImplementation() throws IOError, KeyFileError { + mailer = Bus.get_proxy_sync(BusType.SESSION, "io.mainframe.shopsystem.Mail", "/io/mainframe/shopsystem/mailer"); + db = Bus.get_proxy_sync(BusType.SESSION, "io.mainframe.shopsystem.Database", "/io/mainframe/shopsystem/database"); + pdf = Bus.get_proxy_sync(BusType.SESSION, "io.mainframe.shopsystem.InvoicePDF", "/io/mainframe/shopsystem/invoicepdf"); + Config cfg = Bus.get_proxy_sync(BusType.SESSION, "io.mainframe.shopsystem.Config", "/io/mainframe/shopsystem/config"); + datapath = cfg.get_string("INVOICE", "datapath"); + } + + public void send_invoices(bool temporary, int64 timestamp) throws IOError, InvoicePDFError, DatabaseError { + int64 prevtimestamp = timestamp - day_in_seconds; + + if(!temporary) { + prevtimestamp = new DateTime.from_unix_local(timestamp).add_months(-1).to_unix(); + } + + Timespan ts = get_timespan(temporary, prevtimestamp); + Timespan tst = get_timespan(false, prevtimestamp); + int number = 0; + + + var start = new DateTime.from_unix_local(ts.from); + var stop = new DateTime.from_unix_local(ts.to); + var startstring = start.format("%d.%m.%Y %H:%M:%S"); + var stopstring = stop.format("%d.%m.%Y %H:%M:%S"); + + /* title */ + string mailtitle = temporary ? "Getränkezwischenstand" : "Getränkerechnung"; + mailtitle += @" $startstring - $stopstring"; + + stdout.printf(mailtitle + "\n\n"); + + var users = db.get_users_with_sales(ts.from, ts.to); + + string treasurer_path = mailer.create_mail(); + Mail treasurer_mail = Bus.get_proxy_sync(BusType.SESSION, "io.mainframe.shopsystem.Mail", treasurer_path); + treasurer_mail.from = {"KtT Shopsystem", "shop@kreativitaet-trifft-technik.de"}; + treasurer_mail.subject = mailtitle; + treasurer_mail.add_recipient({"Schatzmeister", "schatzmeister@kreativitaet-trifft-technik.de"}, RecipientType.TO); + var csvinvoicedata = ""; + + foreach(var userid in users) { + uint8[] pdfdata = null; + var userdata = db.get_user_info(userid); + + stdout.printf("%d (%s %s)...\n", userdata.id, userdata.firstname, userdata.lastname); + + var invoiceentries = db.get_invoice(userid, ts.from, ts.to); + var total_sum = db.get_user_invoice_sum(userid, tst.from, tst.to); + + /* invoice id */ + number++; + string invoiceid = start.format("%Y%m") + "5" + "%03d".printf(number); + string pdffilename = invoiceid + @"_$(userdata.firstname)_$(userdata.lastname).pdf"; + + /* pdf generation */ + if(!temporary) { + try { + pdf.invoice_id = invoiceid; + pdf.invoice_date = timestamp; + pdf.invoice_recipient = { + userdata.firstname, + userdata.lastname, + userdata.street, + userdata.postcode, + userdata.city, + userdata.gender + }; + pdf.invoice_entries = invoiceentries; + pdfdata = pdf.generate(); + pdf.clear(); + } catch(DBusError e) { + throw new IOError.FAILED("PDF Generation failed"); + } + } + + string mail_path = mailer.create_mail(); + Mail mail = Bus.get_proxy_sync(BusType.SESSION, "io.mainframe.shopsystem.Mail", mail_path); + mail.from = {"KtT Shopsystem", "shop@kreativitaet-trifft-technik.de"}; + mail.subject = mailtitle; + mail.add_recipient({@"$(userdata.firstname) $(userdata.lastname)", userdata.email}, RecipientType.TO); + + if(!temporary) { + mail.add_attachment(pdffilename, "application/pdf", pdfdata); + } + + var plain = generate_invoice_message(MessageType.PLAIN, temporary, get_address(userdata.gender), userdata.lastname, invoiceentries, total_sum); + mail.set_main_part(plain, MessageType.PLAIN); + + var html = generate_invoice_message(MessageType.HTML, temporary, get_address(userdata.gender), userdata.lastname, invoiceentries, total_sum); + mail.set_main_part(html, MessageType.HTML); + + mailer.send_mail(mail_path); + + if(!temporary) { + treasurer_mail.add_attachment(pdffilename, "application/pdf", pdfdata); + csvinvoicedata += @"$(userdata.id),$(userdata.lastname),$(userdata.firstname),$invoiceid,$total_sum\n"; + } + } + + if(!temporary) { + treasurer_mail.set_main_part(get_treasurer_text(), MessageType.PLAIN); + treasurer_mail.add_attachment("invoice.csv", "text/csv; charset=utf-8", csvinvoicedata.data); + mailer.send_mail(treasurer_path); + } + } + + private string get_treasurer_text() throws IOError { + string text; + + try { + FileUtils.get_contents(datapath + "/treasurer.mail.txt", out text); + } catch(GLib.FileError e) { + throw new IOError.FAILED("Could not open invoice template: %s", e.message); + } + + return text; + } + + private Timespan get_timespan(bool temporary, int64 timestamp) throws IOError { + var time = new DateTime.from_unix_local(timestamp); + Timespan ts = {}; + + if(temporary) { + var start = new DateTime.local(time.get_year(), time.get_month(), time.get_day_of_month(), 8, 0, 0); + if(start.compare(time) > 0) + start.add_days(-1); + var stop = start.add_days(1).add_seconds(-1); + + ts.from = start.to_unix(); + ts.to = stop.to_unix(); + } else { + var start = new DateTime.local(time.get_year(), time.get_month(), 1, 0, 0, 0); + var stop = start.add_months(1).add_seconds(-1); + + ts.from = start.to_unix(); + ts.to = stop.to_unix(); + } + + return ts; + } + + private string get_address(string gender) { + switch(gender) { + case "masculinum": + return "Sehr geehrter Herr"; + case "femininum": + return "Sehr geehrte Frau"; + default: + return "Moin"; + } + } + + private string generate_invoice_message(MessageType type, bool temporary, string address, string name, InvoiceEntry[] entries, Price total_sum) throws IOError { + string filename = ""; + string table = ""; + string text; + + if(type == MessageType.HTML && temporary) + filename = "invoice.temporary.html"; + else if(type == MessageType.HTML) + filename = "invoice.final.html"; + else if(type == MessageType.PLAIN && temporary) + filename = "invoice.temporary.txt"; + else if(type == MessageType.PLAIN) + filename = "invoice.final.txt"; + + if(type == MessageType.PLAIN) + table = generate_invoice_table_text(entries); + else if(type == MessageType.HTML) + table = generate_invoice_table_html(entries); + + if(filename == "") + throw new IOError.FAILED("Unknown MessageType"); + + try { + FileUtils.get_contents(datapath + "/" + filename, out text); + } catch(GLib.FileError e) { + throw new IOError.FAILED("Could not open invoice template: %s", e.message); + } + + text = text.replace("{{{ADDRESS}}}", address); + text = text.replace("{{{LASTNAME}}}", name); + text = text.replace("{{{INVOICE_TABLE}}}", table); + text = text.replace("{{{SUM_MONTH}}}", "%d,%02d".printf(total_sum / 100, total_sum % 100)); + + return text; + } + + private string generate_invoice_table_text(InvoiceEntry[] entries) { + string result = ""; + + /* 7 == "Artikel".char_count() */ + const int article_minsize = 7; + + /* no articles bought */ + if(entries.length == 0) + return result; + + /* get length of longest name + invoice sum */ + int namelength = 0; + int total = 0; + foreach(var entry in entries) { + if(namelength < entry.product.name.char_count()) + namelength = entry.product.name.char_count(); + total += entry.price; + } + + /* better safe than sorry */ + if(namelength < article_minsize) + namelength = article_minsize; + + /* generate table header */ + result += " +------------+----------+-" + string.nfill(namelength, '-') + "-+----------+\n"; + result += " | Datum | Uhrzeit | Artikel" + string.nfill(namelength - article_minsize, ' ') + " | Preis |\n"; + result += " +------------+----------+-" + string.nfill(namelength, '-') + "-+----------+\n"; + + /* generate table data */ + string lastdate = ""; + foreach(var entry in entries) { + var dt = new DateTime.from_unix_local(entry.timestamp); + string newdate = dt.format("%Y-%m-%d"); + string date = (lastdate == newdate) ? " " : newdate; + result += " | %s | %s | %s%s | %3d,%02d € |\n".printf(date, dt.format("%H:%M:%S"), entry.product.name, string.nfill(namelength-entry.product.name.char_count(), ' '), entry.price / 100, entry.price % 100); + lastdate = newdate; + } + + /* generate table footer */ + result += " +------------+----------+-" + string.nfill(namelength, '-') + "-+----------+\n"; + result += " | Summe: " + string.nfill(namelength, ' ') + " | %3d,%02d € |\n".printf(total / 100, total % 100); + result += " +-------------------------" + string.nfill(namelength, '-') + "-+----------+\n"; + + return result; + } + + private string generate_invoice_table_html(InvoiceEntry[] entries) { + string result = ""; + + result += "<table cellpadding=\"5\" style=\"border-collapse:collapse;\">\n"; + result += "\t<tr><th style=\"border: 1px solid black;\">Datum</th><th style=\"border: 1px solid black;\">Zeit</th><th style=\"border: 1px solid black;\">Artikel</th><th style=\"border: 1px solid black;\">Preis</th></tr>\n"; + + string lastdate = ""; + int total = 0; + foreach(var entry in entries) { + var dt = new DateTime.from_unix_local(entry.timestamp); + string newdate = dt.format("%Y-%m-%d"); + string time = dt.format("%H:%M:%S"); + string date = (lastdate == newdate) ? "" : newdate; + total += entry.price; + + result += "\t<tr><td style=\"border: 1px solid black;\">%s</td><td style=\"border: 1px solid black;\">%s</td><td style=\"border: 1px solid black;\">%s</td><td style=\"border: 1px solid black;\" align=\"right\"><tt>%d,%02d €</tt></td></tr>\n".printf(date, time, entry.product.name, entry.price / 100, entry.price % 100); + lastdate = newdate; + } + + result += "\t<tr><th style=\"border: 1px solid black;\" colspan=\"3\" align=\"left\">Summe:</th><td style=\"border: 1px solid black;\" align=\"right\"><tt>%d,%02d €</tt></td></tr>\n".printf(total / 100, total % 100); + + result += "</table>\n"; + + return result; + } +} diff --git a/src/invoice/main.vala b/src/invoice/main.vala new file mode 100644 index 0000000..b341adb --- /dev/null +++ b/src/invoice/main.vala @@ -0,0 +1,60 @@ +/* Copyright 2013, Sebastian Reichel <sre@ring0.de> + * + * Permission to use, copy, modify, and/or distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +InvoiceImplementation invoice; + +public static void help(string name) { + stderr.printf("Usage: %s <temporary> [timestamp]\n", name); + stderr.printf("Possible values for <temporary>: temporary, final\n"); +} + +public static int main(string[] args) { + bool temporary = false; + int64 timestamp = new DateTime.now_local().to_unix(); + + if(args.length < 2) { + help(args[0]); + return 1; + } + + if(args[1] == "temporary") { + temporary = true; + } else if(args[1] == "final") { + temporary = false; + } else { + help(args[0]); + return 1; + } + + if(args.length >= 3) { + timestamp = int64.parse(args[2]); + } + + try { + invoice = new InvoiceImplementation(); + } catch(Error e) { + stderr.printf("Error: %s\n", e.message); + return 1; + } + + try { + invoice.send_invoices(temporary, timestamp); + } catch(Error e) { + stderr.printf("Error: %s\n", e.message); + return 1; + } + + return 0; +} |