Write backend

This commit is contained in:
Michael Soukup 2021-05-09 18:23:32 +02:00
commit f1ee604638
12 changed files with 202 additions and 0 deletions

3
.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
__pycache__
venv
surprise.txt

10
Dockerfile Normal file
View File

@ -0,0 +1,10 @@
FROM xxx
# Install flask and elm
# Copy files
# Compile elm
# Run flask

6
README.md Normal file
View File

@ -0,0 +1,6 @@
# Surprise countdown
This is a web application for revealing a surprise at a given time.

113
backend/app.py Normal file
View File

@ -0,0 +1,113 @@
import os
import datetime
import base64
import random
import string
from dateutil import parser as dt_parser
from toolz import compose
from toolz.curried import map, filter
from atomicwrites import atomic_write
SURPRISE_TITLE = os.environ.get("SURPRISE_TITLE", "Surprise")
print("SURPRISE_TITLE =", SURPRISE_TITLE)
SURPRISE_STORE_FILE = os.environ.get("SURPRISE_STORE_FILE", "surprise.txt")
print("SURPRISE_STORE_FILE =", SURPRISE_STORE_FILE)
SURPRISE_SECRET = os.environ.get("SURPRISE_SECRET")
print("SURPRISE_SECRET =", SURPRISE_SECRET)
assert bool(SURPRISE_SECRET)
SURPRISE_ASSET_DIR = os.environ.get("SURPRISE_ASSET_DIR", "../surprise-assets/gender-reveal")
print("SURPRISE_ASSET_DIR =", SURPRISE_ASSET_DIR)
assert os.path.isdir(SURPRISE_ASSET_DIR)
SURPRISE_REVEAL_DATETIME = dt_parser.parse(
os.environ.get(
"SURPRISE_REVEAL_DATETIME",
(datetime.datetime.now() + datetime.timedelta(minutes=5)).isoformat()
)
)
print("SURPRISE_REVEAL_DATETIME =", SURPRISE_REVEAL_DATETIME)
def _get_surprise_assets():
IMG_TYPES = (
'.png',
'.jpeg',
'.jpg',
'.svg',
'.apng',
'.gif',
'.webp',
'.avif'
)
_, _, files = next(os.walk(SURPRISE_ASSET_DIR))
surprise_assets = compose(
dict,
map(lambda xy: (xy[0], os.path.join('/assets', ''.join(xy)))),
filter(lambda xy: xy[1] in IMG_TYPES),
map(lambda s: os.path.splitext(s)),
map(lambda s: s.lower().strip())
)(files)
if '_null' not in surprise_assets:
surprise_assets['_null'] = os.path.join('/static', '_null.png')
if '_secret' not in surprise_assets:
surprise_assets['_secret'] = os.path.join('/static', '_secret.png')
return surprise_assets
SURPRISE_ASSETS = _get_surprise_assets()
assert len(SURPRISE_ASSETS) >= 4
print("SURPRISE_ASSETS =", SURPRISE_ASSETS)
from flask import (
Flask, jsonify, send_from_directory, render_template, request, url_for,
redirect, flash
)
app = Flask(__name__)
app.config['SECRET_KEY'] = ''.join(random.choice(string.ascii_lowercase) for i in range(20))
@app.route('/')
def index():
return render_template('index.html', title=SURPRISE_TITLE)
@app.route('/assets/<path:path>')
def assets(path):
return send_from_directory(SURPRISE_ASSET_DIR, path)
@app.route('/get-surprise.json')
def get_surprise_json():
diff = (SURPRISE_REVEAL_DATETIME - datetime.datetime.now()).total_seconds()
if os.path.isfile(SURPRISE_STORE_FILE):
if diff < 0:
with open(SURPRISE_STORE_FILE) as f:
surprise = f.read()
else:
surprise = '_secret'
else:
surprise = '_null'
return jsonify({
'diff_seconds': diff,
'surprise': surprise,
'asset': SURPRISE_ASSETS[surprise]
})
@app.route('/set-surprise')
def set_surprise():
enum = [v for v in SURPRISE_ASSETS.keys() if v not in ('_null', '_secret')]
return render_template('set_surprise.html', title=SURPRISE_TITLE, enum=enum)
@app.route('/post-surprise', methods=['POST'])
def post_surprise():
if request.form.get('password') != SURPRISE_SECRET:
flash("Password is wrong")
return redirect(url_for('set_surprise'))
value = request.form.get('surprise')
if value not in SURPRISE_ASSETS.keys():
flash("Invalid value '%s'" % value)
return redirect(url_for('set_surprise'))
with atomic_write(SURPRISE_STORE_FILE, overwrite=True) as f:
f.write(value)
flash("Value was successfully set to '%s'" % value)
return redirect(url_for('index'))

10
backend/requirements.pip Normal file
View File

@ -0,0 +1,10 @@
atomicwrites==1.4.0
click==7.1.2
Flask==1.1.2
itsdangerous==1.1.0
Jinja2==2.11.3
MarkupSafe==1.1.1
python-dateutil==2.8.1
six==1.16.0
toolz==0.11.1
Werkzeug==1.0.1

0
backend/static/_null.png Normal file
View File

View File

View File

@ -0,0 +1,23 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>{{ title }}</title>
</head>
<body>
{% with messages = get_flashed_messages() %}
{% if messages %}
<ul class="messages">
{% for message in messages %}
<li>{{ message }}</li>
{% endfor %}
</ul>
{% endif %}
{% endwith %}
<h1 style="color: blue">Hei</h1>
<p>TODO</p>
</body>
</html>

View File

@ -0,0 +1,37 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>{{ title }}</title>
</head>
<body>
{% with messages = get_flashed_messages() %}
{% if messages %}
<ul class="messages">
{% for message in messages %}
<li>{{ message }}</li>
{% endfor %}
</ul>
{% endif %}
{% endwith %}
<form class="form-signin" method="POST" action="/post-surprise">
<h2 class="form-signin-heading">Set surprise</h2>
{% for v in enum %}
<div>
<input type="radio" name="surprise" id="{{ v }}" value="{{ v }}">
<label for="{{ v }}">{{ v }}</label>
</div>
{% endfor %}
<div>
<input class="form-control" type="password" required name="password" placeholder="Password">
</div>
<div>
<button class="btn btn-lg btn-primary btn-block" type="submit">Set surprise</button>
</div>
</form>
</body>
</html>

View File

View File