split application into multiple modules
This commit is contained in:
parent
b3ba10fae2
commit
e434f8277a
255
anonkun.py
255
anonkun.py
|
@ -4,17 +4,13 @@ Simple file host using Flask.
|
||||||
"""
|
"""
|
||||||
import os
|
import os
|
||||||
import time
|
import time
|
||||||
import string
|
|
||||||
import urllib
|
|
||||||
import functools
|
|
||||||
|
|
||||||
from flask import Flask, session, request, abort, redirect, url_for, g, \
|
from flask import Flask
|
||||||
render_template
|
|
||||||
from flask_socketio import SocketIO, emit, join_room
|
|
||||||
from flask_paranoid import Paranoid
|
from flask_paranoid import Paranoid
|
||||||
import MySQLdb
|
from flask_socketio import SocketIO
|
||||||
import bleach
|
|
||||||
from passlib.hash import argon2
|
from events import socketio
|
||||||
|
from views import views
|
||||||
|
|
||||||
class ReverseProxied(object):
|
class ReverseProxied(object):
|
||||||
"""
|
"""
|
||||||
|
@ -53,21 +49,11 @@ class ReverseProxied(object):
|
||||||
|
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
app.wsgi_app = ReverseProxied(app.wsgi_app)
|
app.wsgi_app = ReverseProxied(app.wsgi_app)
|
||||||
|
app.register_blueprint(views)
|
||||||
app.config['MAX_CONTENT_LENGTH'] = 16 * 1024 * 1024
|
app.config['MAX_CONTENT_LENGTH'] = 16 * 1024 * 1024
|
||||||
socketio = SocketIO(app)
|
socketio.init_app(app)
|
||||||
paranoid = Paranoid(app)
|
paranoid = Paranoid(app)
|
||||||
paranoid.redirect_view = 'index'
|
paranoid.redirect_view = 'views.index'
|
||||||
|
|
||||||
|
|
||||||
def db_execute(*args, **kwargs):
|
|
||||||
"""
|
|
||||||
Opens a connection to the app's database and executes the SQL statements
|
|
||||||
passed to this function.
|
|
||||||
"""
|
|
||||||
passwd = app.config.get("DB_KEY")
|
|
||||||
with MySQLdb.connect(user="anonkun", passwd=passwd, db="anonkun") as cur:
|
|
||||||
cur.execute(*args, **kwargs)
|
|
||||||
return cur
|
|
||||||
|
|
||||||
|
|
||||||
def init():
|
def init():
|
||||||
|
@ -84,116 +70,6 @@ def init():
|
||||||
file.write(secret_key)
|
file.write(secret_key)
|
||||||
app.secret_key = secret_key
|
app.secret_key = secret_key
|
||||||
|
|
||||||
# init db
|
|
||||||
with open("db_key", "r") as file:
|
|
||||||
passwd = file.read().strip() # TODO: encrypt this
|
|
||||||
app.config["DB_KEY"] = passwd
|
|
||||||
try:
|
|
||||||
db_execute("SELECT * FROM `users`").fetchone()
|
|
||||||
except MySQLdb.ProgrammingError: # database not initialized
|
|
||||||
with open("anonkun.sql", "r") as file:
|
|
||||||
commands = file.read().split(";")
|
|
||||||
for cmd in commands:
|
|
||||||
cmd = cmd.strip()
|
|
||||||
if not cmd:
|
|
||||||
continue
|
|
||||||
db_execute(cmd)
|
|
||||||
|
|
||||||
|
|
||||||
def login_required(url=None):
|
|
||||||
"""
|
|
||||||
A decorator function to protect certain endpoints by requiring the user
|
|
||||||
to either pass a valid session cookie, or pass thier username and
|
|
||||||
password along with the request to login.
|
|
||||||
"""
|
|
||||||
def actual_decorator(func):
|
|
||||||
@functools.wraps(func)
|
|
||||||
def _nop(*args, **kwargs):
|
|
||||||
username = session.get("username")
|
|
||||||
if verify_username(username):
|
|
||||||
return func(*args, **kwargs)
|
|
||||||
|
|
||||||
username = request.form.get("user")
|
|
||||||
password = request.form.get("pass")
|
|
||||||
if verify_password(username, password):
|
|
||||||
return func(*args, **kwargs)
|
|
||||||
|
|
||||||
if url:
|
|
||||||
return redirect(url_for(url))
|
|
||||||
else:
|
|
||||||
abort(401)
|
|
||||||
return _nop
|
|
||||||
return actual_decorator
|
|
||||||
|
|
||||||
|
|
||||||
def add_user(username, password):
|
|
||||||
"""
|
|
||||||
Adds a user to the database.
|
|
||||||
"""
|
|
||||||
if verify_username(username): # username taken
|
|
||||||
return "username_taken"
|
|
||||||
elif len(username) > 20:
|
|
||||||
return "username_too_long"
|
|
||||||
|
|
||||||
pw_hash = argon2.hash(password)
|
|
||||||
db_execute(
|
|
||||||
"INSERT INTO `users` (`username`, `password_hash`) VALUES (%s, %s)",
|
|
||||||
(username, pw_hash))
|
|
||||||
return "success"
|
|
||||||
|
|
||||||
|
|
||||||
def verify_password(username, password):
|
|
||||||
"""
|
|
||||||
Verifies a user's password.
|
|
||||||
"""
|
|
||||||
user = verify_username(username)
|
|
||||||
if not user:
|
|
||||||
return False
|
|
||||||
|
|
||||||
user_id, _, pw_hash = user
|
|
||||||
|
|
||||||
if argon2.verify(password, pw_hash):
|
|
||||||
session["user_id"] = user_id
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
def verify_username(username):
|
|
||||||
"""
|
|
||||||
Checks to see if the given username is in the database.
|
|
||||||
"""
|
|
||||||
user = db_execute("SELECT * FROM `users` WHERE `username` = %s",
|
|
||||||
(username,)).fetchone()
|
|
||||||
if user:
|
|
||||||
return user
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
@socketio.on('joined', namespace="/chat")
|
|
||||||
def joined(data):
|
|
||||||
"""
|
|
||||||
Sent by clients when they enter a room.
|
|
||||||
"""
|
|
||||||
room = data["room"]
|
|
||||||
join_room(room)
|
|
||||||
|
|
||||||
|
|
||||||
@socketio.on('message', namespace="/chat")
|
|
||||||
def text(data):
|
|
||||||
"""
|
|
||||||
Sent by a client when the user entered a new message.
|
|
||||||
"""
|
|
||||||
room = data["room"]
|
|
||||||
|
|
||||||
message = data["message"]
|
|
||||||
name = data["name"]
|
|
||||||
date = int(time.time())
|
|
||||||
data["date"] = date
|
|
||||||
|
|
||||||
emit('message', data, room=room)
|
|
||||||
|
|
||||||
|
|
||||||
@app.template_filter("strftime")
|
@app.template_filter("strftime")
|
||||||
def unix2string(unix):
|
def unix2string(unix):
|
||||||
|
@ -205,121 +81,6 @@ def unix2string(unix):
|
||||||
return time.strftime(form, t)
|
return time.strftime(form, t)
|
||||||
|
|
||||||
|
|
||||||
messages = [{"name":"Anonymous", "date":1528998539, "message":"lol"}, {"name":"Namefag", "date":1528998521, "message":"kek"}]
|
|
||||||
@app.route("/quest/<path:quest_title>")
|
|
||||||
def quest(quest_title):
|
|
||||||
"""
|
|
||||||
An arbituary quest page.
|
|
||||||
"""
|
|
||||||
quest_title, _, extra = quest_title.partition("/")
|
|
||||||
res = db_execute("SELECT * FROM `quests` WHERE `ident_title` = %s",
|
|
||||||
(quest_title,))
|
|
||||||
data = res.fetchone()
|
|
||||||
if not data:
|
|
||||||
abort(404)
|
|
||||||
quest_id, quest_title, _, owner_id, quest_data = data
|
|
||||||
|
|
||||||
res = db_execute("SELECT * FROM `chat_messages` WHERE `room_id` = '%s'",
|
|
||||||
(quest_id,))
|
|
||||||
messages = res.fetchall()
|
|
||||||
return render_template('quest.html',
|
|
||||||
quest_title=quest_title,
|
|
||||||
quest_body=quest_data,
|
|
||||||
room_id=quest_id,
|
|
||||||
messages=messages)
|
|
||||||
|
|
||||||
|
|
||||||
@app.route("/create_quest", methods=["GET", "POST"])
|
|
||||||
@login_required("login")
|
|
||||||
def create_quest():
|
|
||||||
"""
|
|
||||||
Starts a new quest.
|
|
||||||
"""
|
|
||||||
if request.method == "GET":
|
|
||||||
return render_template("create_quest.html")
|
|
||||||
canon_title = request.form.get("quest_title")
|
|
||||||
quest_body = request.form.get("quest_body")
|
|
||||||
|
|
||||||
ident_title = sanitize_title(canon_title)
|
|
||||||
quest_body = bleach.clean(quest_body.strip())
|
|
||||||
quest_body = quest_body.replace("\n", "<br />")
|
|
||||||
|
|
||||||
db_execute(
|
|
||||||
"INSERT INTO `quests` (`canon_title`, `ident_title`, `quest_data`) " \
|
|
||||||
"VALUES (%s, %s, %s)", (canon_title, ident_title, quest_body))
|
|
||||||
return redirect(url_for('quest', quest_title=ident_title))
|
|
||||||
|
|
||||||
|
|
||||||
@app.route("/login", methods=["GET", "POST"])
|
|
||||||
def login():
|
|
||||||
"""
|
|
||||||
Logs the user in.
|
|
||||||
"""
|
|
||||||
if request.method == "GET":
|
|
||||||
return render_template("login.html")
|
|
||||||
|
|
||||||
username = request.form.get("user")
|
|
||||||
password = request.form.get("pass")
|
|
||||||
|
|
||||||
if verify_password(username, password):
|
|
||||||
session["username"] = username
|
|
||||||
return redirect(url_for("index"))
|
|
||||||
else:
|
|
||||||
abort(401)
|
|
||||||
|
|
||||||
|
|
||||||
@app.route("/signup", methods=["GET", "POST"])
|
|
||||||
def signup():
|
|
||||||
"""
|
|
||||||
Create a new account.
|
|
||||||
"""
|
|
||||||
if request.method == "GET":
|
|
||||||
return render_template("signup.html")
|
|
||||||
|
|
||||||
username = request.form.get("user")
|
|
||||||
password = request.form.get("pass")
|
|
||||||
verify_password = request.form.get("verify_pass")
|
|
||||||
|
|
||||||
if len(username) > 20:
|
|
||||||
"username_too_long"
|
|
||||||
elif len(username) < 3:
|
|
||||||
"username_too_short"
|
|
||||||
chrs = [c not in string.ascii_letters + string.digits for c in username]
|
|
||||||
if any(chrs):
|
|
||||||
"username_bad_chars"
|
|
||||||
if verify_username(username):
|
|
||||||
"username_taken"
|
|
||||||
|
|
||||||
if len(password) > 1024:
|
|
||||||
"password_too_long"
|
|
||||||
elif len(password) < 8:
|
|
||||||
"password_too_short"
|
|
||||||
|
|
||||||
if password != verify_password:
|
|
||||||
"passwords_dont_match"
|
|
||||||
|
|
||||||
res = add_user(username, password)
|
|
||||||
return redirect(url_for("index"))
|
|
||||||
|
|
||||||
|
|
||||||
@app.route("/")
|
|
||||||
def index():
|
|
||||||
"""
|
|
||||||
The index page.
|
|
||||||
"""
|
|
||||||
return render_template("index.html")
|
|
||||||
|
|
||||||
|
|
||||||
def sanitize_title(canon_title):
|
|
||||||
"""
|
|
||||||
Sanitizes the given canonical title for a quest and returns a
|
|
||||||
url-friendly version.
|
|
||||||
"""
|
|
||||||
ident_title = canon_title.lower().replace(" ", "-")
|
|
||||||
ident_title = urllib.parse.quote(ident_title)
|
|
||||||
return ident_title
|
|
||||||
|
|
||||||
|
|
||||||
init()
|
init()
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
app.run(host='0.0.0.0', port=5050)
|
app.run(host='0.0.0.0', port=5050)
|
||||||
|
|
86
database.py
Normal file
86
database.py
Normal file
|
@ -0,0 +1,86 @@
|
||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
Database tools and functions.
|
||||||
|
"""
|
||||||
|
import MySQLdb
|
||||||
|
from flask import session
|
||||||
|
from passlib.hash import argon2
|
||||||
|
|
||||||
|
class Database():
|
||||||
|
"""
|
||||||
|
An interface to interact with the database.
|
||||||
|
"""
|
||||||
|
def __init__(self):
|
||||||
|
"""
|
||||||
|
Initalizes the database.
|
||||||
|
"""
|
||||||
|
with open("db_key", "r") as file:
|
||||||
|
# TODO: encrypt this
|
||||||
|
self.user, self.db, self.key = file.read().strip().split()
|
||||||
|
try:
|
||||||
|
self.execute("SELECT * FROM `users`").fetchone()
|
||||||
|
except MySQLdb.ProgrammingError: # database not initialized
|
||||||
|
with open("anonkun.sql", "r") as file:
|
||||||
|
commands = file.read().split(";")
|
||||||
|
for cmd in commands:
|
||||||
|
cmd = cmd.strip()
|
||||||
|
if not cmd:
|
||||||
|
continue
|
||||||
|
self.execute(cmd)
|
||||||
|
|
||||||
|
|
||||||
|
def execute(self, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
Opens a connection to the app's database and executes the SQL
|
||||||
|
statements passed to this function.
|
||||||
|
"""
|
||||||
|
with MySQLdb.connect(user=self.user,passwd=self.key,db=self.db) as cur:
|
||||||
|
cur.execute(*args, **kwargs)
|
||||||
|
return cur
|
||||||
|
|
||||||
|
_DB = Database()
|
||||||
|
|
||||||
|
|
||||||
|
def add_user(username, password):
|
||||||
|
"""
|
||||||
|
Adds a user to the database.
|
||||||
|
"""
|
||||||
|
if verify_username(username): # username taken
|
||||||
|
return "username_taken"
|
||||||
|
elif len(username) > 20:
|
||||||
|
return "username_too_long"
|
||||||
|
|
||||||
|
pw_hash = argon2.hash(password)
|
||||||
|
_DB.execute(
|
||||||
|
"INSERT INTO `users` (`username`, `password_hash`) VALUES (%s, %s)",
|
||||||
|
(username, pw_hash))
|
||||||
|
return "success"
|
||||||
|
|
||||||
|
|
||||||
|
def verify_password(username, password):
|
||||||
|
"""
|
||||||
|
Verifies a user's password.
|
||||||
|
"""
|
||||||
|
user = verify_username(username)
|
||||||
|
if not user:
|
||||||
|
return False
|
||||||
|
|
||||||
|
user_id, _, pw_hash = user
|
||||||
|
|
||||||
|
if argon2.verify(password, pw_hash):
|
||||||
|
session["user_id"] = user_id
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def verify_username(username):
|
||||||
|
"""
|
||||||
|
Checks to see if the given username is in the database.
|
||||||
|
"""
|
||||||
|
user = _DB.execute("SELECT * FROM `users` WHERE `username` = %s",
|
||||||
|
(username,)).fetchone()
|
||||||
|
if user:
|
||||||
|
return user
|
||||||
|
else:
|
||||||
|
return False
|
32
events.py
Normal file
32
events.py
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
SocketIO events.
|
||||||
|
"""
|
||||||
|
import time
|
||||||
|
|
||||||
|
from flask_socketio import SocketIO, emit, join_room
|
||||||
|
|
||||||
|
socketio = SocketIO()
|
||||||
|
|
||||||
|
@socketio.on('joined', namespace="/chat")
|
||||||
|
def joined(data):
|
||||||
|
"""
|
||||||
|
Sent by clients when they enter a room.
|
||||||
|
"""
|
||||||
|
room = data["room"]
|
||||||
|
join_room(room)
|
||||||
|
|
||||||
|
|
||||||
|
@socketio.on('message', namespace="/chat")
|
||||||
|
def text(data):
|
||||||
|
"""
|
||||||
|
Sent by a client when the user entered a new message.
|
||||||
|
"""
|
||||||
|
room = data["room"]
|
||||||
|
|
||||||
|
message = data["message"]
|
||||||
|
name = data["name"]
|
||||||
|
date = int(time.time())
|
||||||
|
data["date"] = date
|
||||||
|
|
||||||
|
emit('message', data, room=room)
|
|
@ -4,7 +4,7 @@
|
||||||
<title>Make a new quest</title>
|
<title>Make a new quest</title>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<form method="post" action="{{ url_for('create_quest') }}">
|
<form method="post" action="{{ url_for('views.create_quest') }}">
|
||||||
<input type="text" placeholder="Quest Title" name="quest_title" maxlength="300" required/><br/>
|
<input type="text" placeholder="Quest Title" name="quest_title" maxlength="300" required/><br/>
|
||||||
<textarea id="create_textarea" name="quest_body"></textarea>
|
<textarea id="create_textarea" name="quest_body"></textarea>
|
||||||
<input type="submit" name="submit" value="Submit"/>
|
<input type="submit" name="submit" value="Submit"/>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<title>Login</title>
|
<title>Login</title>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<form method="post" action="{{ url_for('login') }}">
|
<form method="post" action="{{ url_for('views.login') }}">
|
||||||
<input type="text" placeholder="Username" name="user" maxlength="20" required/><br />
|
<input type="text" placeholder="Username" name="user" maxlength="20" required/><br />
|
||||||
<input type="password" placeholder="Password" name="pass" maxlength="1024" required/><br />
|
<input type="password" placeholder="Password" name="pass" maxlength="1024" required/><br />
|
||||||
<input type="submit" value="Log in" name="submit"/>
|
<input type="submit" value="Log in" name="submit"/>
|
||||||
|
|
|
@ -14,10 +14,10 @@
|
||||||
<ul>
|
<ul>
|
||||||
<li>Must be between 8 and 1024 characters</li>
|
<li>Must be between 8 and 1024 characters</li>
|
||||||
</ul>
|
</ul>
|
||||||
<form method="post" action="{{ url_for('signup') }}">
|
<form method="post" action="{{ url_for('views.signup') }}">
|
||||||
<input type="text" placeholder="Username" name="user" maxlength="20" required/><br />
|
<input type="text" placeholder="Username" name="user" maxlength="20" required/><br />
|
||||||
<input type="password" placeholder="Password" name="pass" maxlength="1024" required/><br />
|
<input type="password" placeholder="Password" name="pass" maxlength="1024" required/><br />
|
||||||
<input type="password" placeholder="Verify password" name="verify_pass" maxlength="1024" required/><br />
|
<input type="password" placeholder="Verify password" name="pass_verify" maxlength="1024" required/><br />
|
||||||
<input type="submit" value="Sign up" name="submit"/>
|
<input type="submit" value="Sign up" name="submit"/>
|
||||||
</form>
|
</form>
|
||||||
</body>
|
</body>
|
||||||
|
|
154
views.py
Normal file
154
views.py
Normal file
|
@ -0,0 +1,154 @@
|
||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
Database tools and functions.
|
||||||
|
"""
|
||||||
|
import string
|
||||||
|
import urllib
|
||||||
|
import functools
|
||||||
|
|
||||||
|
import bleach
|
||||||
|
from flask import session, request, abort, redirect, url_for, render_template
|
||||||
|
from flask import Blueprint
|
||||||
|
|
||||||
|
import database as db
|
||||||
|
|
||||||
|
views = Blueprint("views", __name__)
|
||||||
|
|
||||||
|
def login_required(url=None):
|
||||||
|
"""
|
||||||
|
A decorator function to protect certain endpoints by requiring the user
|
||||||
|
to either pass a valid session cookie, or pass thier username and
|
||||||
|
password along with the request to login.
|
||||||
|
"""
|
||||||
|
def actual_decorator(func):
|
||||||
|
@functools.wraps(func)
|
||||||
|
def _nop(*args, **kwargs):
|
||||||
|
username = session.get("username")
|
||||||
|
if db.verify_username(username):
|
||||||
|
return func(*args, **kwargs)
|
||||||
|
|
||||||
|
username = request.form.get("user")
|
||||||
|
password = request.form.get("pass")
|
||||||
|
if db.verify_password(username, password):
|
||||||
|
return func(*args, **kwargs)
|
||||||
|
|
||||||
|
if url:
|
||||||
|
return redirect(url_for(url))
|
||||||
|
else:
|
||||||
|
abort(401)
|
||||||
|
return _nop
|
||||||
|
return actual_decorator
|
||||||
|
|
||||||
|
|
||||||
|
def sanitize_title(canon_title):
|
||||||
|
"""
|
||||||
|
Sanitizes the given canonical title for a quest and returns a
|
||||||
|
url-friendly version.
|
||||||
|
"""
|
||||||
|
ident_title = canon_title.lower().replace(" ", "-")
|
||||||
|
ident_title = urllib.parse.quote(ident_title)
|
||||||
|
return ident_title
|
||||||
|
|
||||||
|
|
||||||
|
@views.route("/quest/<path:quest_title>")
|
||||||
|
def quest(quest_title):
|
||||||
|
"""
|
||||||
|
An arbituary quest page.
|
||||||
|
"""
|
||||||
|
quest_title, _, extra = quest_title.partition("/")
|
||||||
|
res = db._DB.execute("SELECT * FROM `quests` WHERE `ident_title` = %s",
|
||||||
|
(quest_title,))
|
||||||
|
data = res.fetchone()
|
||||||
|
if not data:
|
||||||
|
abort(404)
|
||||||
|
quest_id, quest_title, _, owner_id, quest_data = data
|
||||||
|
|
||||||
|
res = db._DB.execute("SELECT * FROM `chat_messages` WHERE `room_id` = '%s'",
|
||||||
|
(quest_id,))
|
||||||
|
messages = res.fetchall()
|
||||||
|
return render_template('quest.html',
|
||||||
|
quest_title=quest_title,
|
||||||
|
quest_body=quest_data,
|
||||||
|
room_id=quest_id,
|
||||||
|
messages=messages)
|
||||||
|
|
||||||
|
|
||||||
|
@views.route("/create_quest", methods=["GET", "POST"])
|
||||||
|
@login_required("login")
|
||||||
|
def create_quest():
|
||||||
|
"""
|
||||||
|
Starts a new quest.
|
||||||
|
"""
|
||||||
|
if request.method == "GET":
|
||||||
|
return render_template("create_quest.html")
|
||||||
|
canon_title = request.form.get("quest_title")
|
||||||
|
quest_body = request.form.get("quest_body")
|
||||||
|
|
||||||
|
ident_title = sanitize_title(canon_title)
|
||||||
|
quest_body = bleach.clean(quest_body.strip())
|
||||||
|
quest_body = quest_body.replace("\n", "<br />")
|
||||||
|
|
||||||
|
db._DB.execute(
|
||||||
|
"INSERT INTO `quests` (`canon_title`, `ident_title`, `quest_data`) " \
|
||||||
|
"VALUES (%s, %s, %s)", (canon_title, ident_title, quest_body))
|
||||||
|
return redirect(url_for('views.quest', quest_title=ident_title))
|
||||||
|
|
||||||
|
|
||||||
|
@views.route("/login", methods=["GET", "POST"])
|
||||||
|
def login():
|
||||||
|
"""
|
||||||
|
Logs the user in.
|
||||||
|
"""
|
||||||
|
if request.method == "GET":
|
||||||
|
return render_template("login.html")
|
||||||
|
|
||||||
|
username = request.form.get("user")
|
||||||
|
password = request.form.get("pass")
|
||||||
|
|
||||||
|
if db.verify_password(username, password):
|
||||||
|
session["username"] = username
|
||||||
|
return redirect(url_for("views.index"))
|
||||||
|
else:
|
||||||
|
abort(401)
|
||||||
|
|
||||||
|
|
||||||
|
@views.route("/signup", methods=["GET", "POST"])
|
||||||
|
def signup():
|
||||||
|
"""
|
||||||
|
Create a new account.
|
||||||
|
"""
|
||||||
|
if request.method == "GET":
|
||||||
|
return render_template("signup.html")
|
||||||
|
|
||||||
|
username = request.form.get("user")
|
||||||
|
password = request.form.get("pass")
|
||||||
|
password_verify = request.form.get("pass_verify")
|
||||||
|
|
||||||
|
if len(username) > 20:
|
||||||
|
"username_too_long"
|
||||||
|
elif len(username) < 3:
|
||||||
|
"username_too_short"
|
||||||
|
chrs = [c not in string.ascii_letters + string.digits for c in username]
|
||||||
|
if any(chrs):
|
||||||
|
"username_bad_chars"
|
||||||
|
if db.verify_username(username):
|
||||||
|
"username_taken"
|
||||||
|
|
||||||
|
if len(password) > 1024:
|
||||||
|
"password_too_long"
|
||||||
|
elif len(password) < 8:
|
||||||
|
"password_too_short"
|
||||||
|
|
||||||
|
if password != password_verify:
|
||||||
|
"passwords_dont_match"
|
||||||
|
|
||||||
|
res = db.add_user(username, password)
|
||||||
|
return redirect(url_for("views.index"))
|
||||||
|
|
||||||
|
|
||||||
|
@views.route("/")
|
||||||
|
def index():
|
||||||
|
"""
|
||||||
|
The index page.
|
||||||
|
"""
|
||||||
|
return render_template("index.html")
|
Loading…
Reference in New Issue
Block a user