#! /usr/bin/env python # -*- coding: utf-8 -*- # # Copyright (C) 2011 Sebastian Pipping # Licensed under GPL v3 or later from __future__ import print_function import freitagslib.network as net from freitagslib.commands import BuyCommand, DepositCommand import colorama from colorama import Fore, Style import sys from decimal import Decimal import os import time import urllib2 COLOR_WARN = Fore.YELLOW COLOR_ERROR = Fore.RED COLOR_DEPOSIT = Fore.GREEN + Style.BRIGHT COLOR_WITHDRAW = Fore.RED + Style.BRIGHT COLOR_SOME = Fore.WHITE + Style.BRIGHT COLOR_MUCH = Fore.YELLOW + Style.BRIGHT COLOR_RESET = Style.RESET_ALL def clear(): os.system('clear') CODES = { 'UNDO':('undo',), 'COMMIT':('commit',), 'DEPOSIT 0.01':('deposit', Decimal('0.01')), 'DEPOSIT 0.05':('deposit', Decimal('0.05')), 'DEPOSIT 0.10':('deposit', Decimal('0.10')), 'DEPOSIT 0.50':('deposit', Decimal('0.50')), 'DEPOSIT 1.00':('deposit', Decimal('1.00')), 'DEPOSIT 5.00':('deposit', Decimal('5.00')), 'DEPOSIT 10.00':('deposit', Decimal('10.00')), 'DEPOSIT 50.00':('deposit', Decimal('50.00')), } def delay(what, seconds): for i in xrange(seconds, 0, -1): if i < seconds: sys.stdout.write('\r') sys.stdout.write('%s in %d Sekunden... ' % (what, i)) sys.stdout.flush() time.sleep(1) def warn_balance(): print('Kontostrand im Minus, bitte Geld aufladen.') def error_page(message): clear() print(COLOR_ERROR + message + COLOR_RESET) print() delay('Weiter', 3) class Status: def __init__(self): self._reset() self.item_cache = dict() def _reset(self): self.auth_blob = None self.login_name = None self.balance = None self.transfers = None def dump(self): def sign(amount, plus='+'): return '-' if amount < 0 else plus def color(amount): return COLOR_WITHDRAW if amount < 0 else COLOR_DEPOSIT def show_total(balance, plus=' '): print('%3s %-40s %s%c %6.2f Euro%s' \ % ('', '', color(balance), sign(balance, plus), abs(balance), COLOR_RESET)) def show_item(position, command): diff = command.difference() label = command.label() print('%2d) %-40s %s%c %6.2f Euro%s' \ % (position, label, color(diff), sign(diff), abs(diff), COLOR_RESET)) def show_bar(): print('%3s %-40s %13s' % ('', '', '=============')) if self.logged_in(): print('Eingeloggt als: %s%s%s' % (COLOR_SOME, self.login_name, COLOR_RESET)) print() if self.transfers: initial_command, initial_balance = self.transfers[0] print('Geplante Änderungen:') show_total(initial_balance) for i, (command, balance_backup) in enumerate(self.transfers): show_item(i + 1, command) show_bar() show_total(self.balance) if self.balance < 0: warn_balance() print() print(COLOR_SOME + 'Committen nicht vergessen.' + COLOR_RESET) else: print('Kontostand beträgt: %s%.2f Euro%s' % (COLOR_MUCH, self.balance, COLOR_RESET)) if self.balance < 0: print() warn_balance() else: print(COLOR_MUCH + 'Bitte einloggen.' + COLOR_RESET) print() print('Scanne dazu deine ID-Karte mit dem Barcode-Leser.') print() def logged_in(self): return self.auth_blob is not None def login(self, auth_blob): assert(not self.logged_in()) user_name = net.get_user_name_from_auth_blob(auth_blob) balance = net.get_balance(user_name) self.auth_blob = auth_blob self.login_name = user_name self.balance = balance self.transfers = list() def commit(self): assert(self.logged_in()) # Process command queue for (command, balance_backup) in list(self.transfers): try: command.run(self.login_name) except urllib2.HTTPError as e: error_page('FEHLER bei Kommunikation mit Server "%s"' % str(e)) break else: self.transfers.pop(0) if not self.transfers: # Show final balance for some time clear() self.dump() delay('Logout', 3) # Logout self._reset() def find(self, barcode): try: return self.item_cache[barcode] except KeyError: item = net.get_item(barcode) self.item_cache[barcode] = item return item def buy(self, item): assert(self.logged_in()) log_entry = (BuyCommand(item), self.balance) self.transfers.append(log_entry) self.balance = self.balance - item.price def deposit(self, amount): assert(self.logged_in()) log_entry = (DepositCommand(amount), self.balance) self.transfers.append(log_entry) self.balance = self.balance + amount def undo(self): assert(self.logged_in()) if self.transfers: (command, balance_backup) = self.transfers[-1] self.transfers.pop() self.balance = balance_backup else: error_page('FEHLER: Nichts da, was ich rückgängig machen könnte.') def print_prompt(): sys.stdout.write(">>> ") def handle(line, status): if line == 'exit': clear() sys.exit(0) if status.logged_in(): if line in CODES: call = CODES[line] method = call[0] params = call[1:] getattr(status, method)(*params) else: try: item = status.find(line) except urllib2.HTTPError as e: if e.code == 404: # URL not found == item not found with REST error_page('FEHLER: Aktion oder Ware "%s" nicht bekannt' % line) else: error_page('FEHLER bei Kommunikation mit Server "%s"' % str(e)) else: status.buy(item) else: try: status.login(line) except urllib2.URLError as e: error_page('FEHLER bei Kommunikation mit Server "%s"' % e.reason) def main(): colorama.init() status = Status() while True: clear() status.dump() print_prompt() l = sys.stdin.readline() if not l: break line = l.rstrip() handle(line, status) if __name__ == '__main__': try: main() except KeyboardInterrupt: pass