import matplotlib
matplotlib.use('Agg')
import matplotlib.pyplot as plt
+from matplotlib.ticker import MaxNLocator
from io import BytesIO
import coffee_db as db
import time
+from datetime import date, timedelta
+
+from json import loads
+from requests import post
db.init_db()
app = Flask(__name__)
CORS(app, supports_credentials=True)
app.secret_key = b'_5#y2L"F4Q8z\n\xec]/'
+
@app.route('/')
def hello():
if "uid" in session:
- uid = session["uid"];
+ uid = session["uid"]
return render_template('hello.html', name=db.get_name(uid))
return render_template('hello.html')
+
@app.route('/login', methods=["POST"])
@app.route('/login/<uid>')
def login(uid=None):
uid = request.data.decode("utf-8")
if uid is not None:
db.add_user(uid)
- session["uid"] = uid;
+ session["uid"] = uid
return redirect(url_for('user'))
+
@app.route('/logout')
def logout():
session.pop('uid', None)
return redirect(url_for('user'))
+
@app.route('/user')
def user():
if "uid" in session:
- uid = session["uid"];
+ uid = session["uid"]
return render_template('user.html',
- name=db.get_name(uid),
- flavors=db.flavors(),
- count=db.coffee_count(uid, 0),
- stamp=time.time()
- )
- return render_template('user.html')
+ name=db.get_name(uid),
+ flavors=db.flavors(),
+ count=db.coffee_count(uid, 0),
+ stamp=time.time()
+ )
+ # TODO: Replace stamp parameter with proper cache control HTTP
+ # headers in response
+ return render_template('user.html', stamp=time.time())
+
@app.route('/user/rename')
def user_rename():
name = request.args.get("name")
if name and "uid" in session:
- uid = session["uid"];
+ uid = session["uid"]
db.name_user(uid, name)
return redirect(url_for('user'))
+
@app.route("/coffee/graph_flavors")
def coffee_graph_flavors():
+ days = request.args.get('days', default = 0, type = int)
+ start = request.args.get('start', default = 0, type = int)
+
b = BytesIO()
if "uid" in session:
uid = session["uid"]
- flavors, counts = zip(*db.coffee_flavors(uid))
+ flavors, counts = zip(*db.coffee_flavors(uid, days, start))
else:
- flavors, counts = zip(*db.coffee_flavors())
- fig = plt.figure()
+ flavors, counts = zip(*db.coffee_flavors(None, days, start))
+ fig = plt.figure(figsize=(3, 3))
ax = fig.add_subplot(111)
ax.set_aspect(1)
- ax.pie(counts)
- ax.legend(flavors)
- ax.set_title("Your taste")
+ ax.pie(counts, autopct=lambda p: '{:.0f}'.format(p * sum(counts)/100) if p != 0 else '')
+ ax.legend(flavors, bbox_to_anchor=(1.0, 1.0))
+
+ if "uid" in session:
+ ax.set_title("Your taste")
+ else:
+ ax.set_title("This week taste")
+
fig.savefig(b, format="svg", bbox_inches="tight")
b.seek(0)
return send_file(b, mimetype="image/svg+xml")
+
@app.route("/coffee/graph_history")
def coffee_graph_history():
b = BytesIO()
else:
hist = db.coffee_history()
if hist == []:
- days = tuple()
+ unix_days = tuple()
counts = tuple()
+ flavors = tuple()
else:
- days, counts = zip(*hist)
- fig = plt.figure()
+ unix_days, counts, flavors = zip(*hist)
+ fig = plt.figure(figsize=(4, 3))
ax = fig.add_subplot(111)
- ax.bar(days, counts)
- ax.set_title("Your week")
+
+ list_flavor = sorted(db.flavors())
+ l = [{} for i in range(len(list_flavor))]
+ for ll in l:
+ for d in unix_days:
+ ll[d] = 0
+
+ for(d, c, f) in zip(unix_days, counts, flavors):
+ if f is None:
+ continue
+ what_f = 0
+ for i in range(len(list_flavor)):
+ if f == list_flavor[i]:
+ what_f = i
+ break
+ l[what_f][d] += c
+
+ z = list(0 for i in range(len(l[0])))
+ for flavor in range(len(list_flavor)):
+ sortedlist = [(k, l[flavor][k]) for k in sorted(l[flavor])]
+ x = [i[0] for i in sortedlist]
+ y = [i[1] for i in sortedlist]
+ ax.bar(range(len(x)), y, bottom=z)
+ z = [sum(i) for i in zip(y, z)]
+
+ unix_days = set(unix_days)
+ xdays = [i.strftime("%a") for i in [
+ date.today() - timedelta(j - 1) for j in
+ range(len(unix_days), 0, -1)]]
+ xdays[-1] = "TDY"
+ xdays[-2] = "YDA"
+ ax.set_xticks(range(len(unix_days)))
+ ax.set_xticklabels(xdays)
+
+ if "uid" in session:
+ ax.set_title("Your week")
+ else:
+ ax.set_title("This week total")
+
+ ax.yaxis.set_major_locator(MaxNLocator(integer=True))
fig.savefig(b, format="svg", bbox_inches="tight")
b.seek(0)
+ plt.close(fig)
return send_file(b, mimetype="image/svg+xml")
+
@app.route("/coffee/add", methods=["POST"])
def coffee_add():
if request.method == "POST":
json = request.json
- if json and "uid" in session:
- db.add_coffee(session["uid"], json["flavor"], json["time"])
+ print("User '%(uid)s' had '%(flavor)s' at %(time)s" % json)
+ db.add_coffee(json["uid"], json["flavor"], json["time"])
return redirect(url_for('user'))
+
@app.route("/coffee/count")
def coffee_count():
start = request.args.get("start")
stop = request.args.get("stop")
return str(db.coffee_count(session.get("uid"), start, stop))
+
@app.route('/js')
def js():
response = make_response(render_template('main.js'))
response.headers['Content-Type'] = "text/javascript"
return response
+
@app.route("/log", methods=["POST"])
def log():
if request.method == "POST":
data = request.data.decode("utf-8")
- print(data)
+ print("Log:", data)
return data
return "nope"
+
+@app.route("/tellCoffeebot", methods=["POST"])
+def tell_coffeebot():
+ err = "Don't worry now! There is a NEW HOPE Tonda is buying NEW PACK!"
+ if request.method == "POST":
+ what = loads(request.data.decode("utf-8"))
+ try:
+ with open(".config", "r") as f:
+ conf = loads(f.read())
+ except:
+ return "Config needed! Please find in git history how it should look."
+ try:
+ res = post(conf["coffeebot"]["url"], json=what)
+ print("res is {}".format(res))
+ except:
+ err = "No connection! No covfefe! We all die here!"
+ if not res.ok:
+ err = "Slack don't like the request! It's discrimination!"
+ return err