app.py 2.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. #!/usr/bin/env python3
  2. import pickle
  3. import atexit
  4. import datetime
  5. from flask import Flask, jsonify, request, abort
  6. from flask_cors import CORS
  7. DB_FILE = "vacation.db"
  8. last_update = datetime.datetime.now()
  9. app = Flask(__name__)
  10. CORS(app)
  11. app.config["APPLICATION_ROOT"] = "/vacation/api"
  12. app.url_map.strict_slashes = False
  13. try:
  14. with open(DB_FILE, "rb") as infile:
  15. db = pickle.load(infile)
  16. except FileNotFoundError:
  17. db = {}
  18. @atexit.register
  19. def save_db(prefix=""):
  20. with open(prefix + DB_FILE, "wb") as outfile:
  21. pickle.dump(db, outfile)
  22. def backup_db():
  23. save_db(f"backup-{datetime.datetime.now().isoformat()}-")
  24. @app.route("/")
  25. def health():
  26. return jsonify({"status": "healthy"})
  27. @app.route("/availability", methods=["GET", "POST"])
  28. def status():
  29. global last_update
  30. if request.method == "POST":
  31. body = request.get_json()
  32. if body is None:
  33. abort(400)
  34. name = body.get("name", None)
  35. availability = body.get("availability", None)
  36. if not isinstance(name, str) or not isinstance(availability, list):
  37. abort(400)
  38. transaction = []
  39. for a in availability:
  40. month = a.get("month", None)
  41. day = a.get("day", None)
  42. status = a.get("status", None)
  43. if (month, day) not in db or status not in ("yes", "no", "maybe", "unknown"):
  44. abort(400)
  45. transaction.append((month, day, status))
  46. last_update = datetime.datetime.now()
  47. for (month, day, status) in transaction:
  48. db[(month, day)][name] = status
  49. return jsonify({
  50. "lastUpdated": last_update.isoformat(),
  51. "availability": [
  52. {
  53. "month": month,
  54. "day": day,
  55. "availability": [{
  56. "name": name,
  57. "status": status,
  58. } for (name, status) in avail.items()],
  59. } for (month, day), avail in db.items()
  60. ],
  61. })
  62. if __name__ == "__main__":
  63. app.run("0.0.0.0", 5000, debug=True)