Created database class.

Tidied app.py so that it only contains controller logic.
Added error handling for if number of entrants exceeds number of possible identifiers
Updated test coverage to test for all identifiers, as well as error case
This commit is contained in:
2019-07-06 12:43:49 +01:00
parent 9c42cb7890
commit 4da511a7d5
7 changed files with 153 additions and 55 deletions

View File

@@ -1,52 +1,16 @@
from flask import Flask, render_template, request
from flask_bootstrap import Bootstrap
import sqlite3
import os
import string
import db
import config
import pprint
app = Flask(__name__)
app.config.from_object(config.BaseConfig)
Bootstrap(app)
my_db = db.Database(app.config['DB_PATH'])
def get_db_connection():
db_path = os.environ.get('HBC_DB_PATH')
if not db_path:
raise Exception("DB Path not defined")
return sqlite3.connect(db_path + '/hbc.db')
def db_setup():
conn = get_db_connection()
sql_create_table = """ CREATE TABLE IF NOT EXISTS brewers (
id integer PRIMARY KEY,
name text NOT NULL,
identifier text NOT NULL );"""
c = conn.cursor()
c.execute(sql_create_table)
conn.close()
db_setup()
brew_name = "Fruit Beer"
brew_month = "October"
identifiers = list(string.ascii_uppercase) + list(range(1, 100))
def get_identifier(name):
conn = get_db_connection()
c = conn.cursor()
for identifier in identifiers:
identifier = str(identifier)
c.execute('''SELECT identifier FROM brewers WHERE identifier=?''', (identifier,))
data = c.fetchone()
if data is None:
c.execute("INSERT INTO brewers (name,identifier) VALUES(?, ?)", (name, identifier))
conn.commit()
conn.close()
return identifier
@app.route('/')
@@ -56,9 +20,19 @@ def hello_world():
@app.route('/generate', methods=["POST"])
def generate():
identifier = get_identifier(request.form['name'])
return render_template('generate.html', brew_name=brew_name, brew_month=brew_month, identifier=identifier)
try:
error = None
identifier = my_db.get_identifier(request.form['name'])
pass
except StopIteration:
identifier = ''
error = 'Maximum entry limit reached - please contact Sean or Joe'
pass
return render_template('generate.html', brew_name=brew_name, brew_month=brew_month, identifier=identifier, error=error)
if __name__ == '__main__':
app.run(debug=True, host='0.0.0.0')

6
web/config.py Normal file
View File

@@ -0,0 +1,6 @@
import os
class BaseConfig(object):
TESTING = True
DB_PATH = os.environ.get('HBC_DB_PATH', 'tests/')

92
web/db.py Normal file
View File

@@ -0,0 +1,92 @@
import sqlite3
import string
import utils
class Database:
def __init__(self, db_path):
if not db_path:
raise Exception("DB Path not defined")
self.db_path = db_path
self.identifiers = self.get_identifiers_list()
self.setup_database_tables()
@staticmethod
def get_identifiers_list():
"""Returns a list of non ambiguous identifiers"""
numbers = list(range(1, 100))
identifiers = list(string.ascii_uppercase) + [str(item) for item in
numbers] # All identifiers are treated as strings
ambiguous = ['I', 'O', 'V', '1', '5', '9', '99']
utils.remove_common_elements(identifiers, ambiguous)
return identifiers
def get_connection(self):
"""Returns sqlite db connection when provided with base directory"""
return sqlite3.connect(self.db_path + '/hbc.db')
def setup_database_tables(self):
"""Creates sqlite database and set up the sqlite table if it doesnt already exist"""
conn = self.get_connection()
sql_create_table = """ CREATE TABLE IF NOT EXISTS brewers (
id integer PRIMARY KEY,
name text NOT NULL,
identifier text NOT NULL );"""
c = conn.cursor()
c.execute(sql_create_table)
conn.close()
def get_identifier(self, name):
"""Returns the next availible identifier, passing the result through record_entry to make sure it is not reused"""
conn = self.get_connection()
c = conn.cursor()
c.execute('''select identifier from brewers ORDER BY id DESC LIMIT 1;''')
identifier_search_result = c.fetchone()
conn.close()
if identifier_search_result is None:
return self.record_entry(self.identifiers[0], name)
else:
if identifier_search_result[0] == self.identifiers[-1]:
raise StopIteration
else:
i = self.identifiers.index(identifier_search_result[0])
return self.record_entry(self.identifiers[i + 1], name)
def record_entry(self, identifier, name):
"""Returns identifier after recording entry in sqlite database"""
conn = self.get_connection()
c = conn.cursor()
c.execute("INSERT INTO brewers (name,identifier) VALUES(?, ?)", (name, identifier))
conn.commit()
conn.close()
return identifier
# def get_identifier(name, db_path):
# conn = get_connection(db_path)
# c = conn.cursor()
# c.execute('''select identifier from brewers ORDER BY id DESC LIMIT 1;''')
# identifier_search_result = c.fetchone()
# conn.close()
#
# if identifier_search_result is None:
# return record_entry(DataStore.identifiers[0], name, db_path)
# else:
# if identifier_search_result[0] == DataStore.identifiers[-1]:
# raise StopIteration
# else:
# i = DataStore.identifiers.index(identifier_search_result[0])
# return record_entry(DataStore.identifiers[i+1], name, db_path)
# def record_entry(identifier, name, db_path):
# conn = get_connection(db_path)
# c = conn.cursor()
# c.execute("INSERT INTO brewers (name,identifier) VALUES(?, ?)", (name, identifier))
# conn.commit()
# conn.close()
# return identifier

View File

@@ -16,10 +16,15 @@
<div class="row">
<div class="col-md-2"></div>
<div class="col-md-8 content-div">
<p>You have been entered into the competition to brew a <strong>{{ brew_name }}</strong>, which will be judged at the meeting in <strong>{{ brew_month }}</strong>.</p>
<p>To make sure your beer isn't easily identifiable, most people tend to use brown bottles of either 330ml or 500ml size. You probably don't want to put your own label on the bottle!</p>
<br/>
<div class="alert alert-success lead" role="alert"> <span class="glyphicon glyphicon-tags" aria-hidden="true"></span> &nbsp; &nbsp;Please mark all of your bottlecaps with the following identifier: <strong>{{ identifier }}</strong></div>
{% if error == None %}
<p>You have been entered into the competition to brew a <strong>{{ brew_name }}</strong>, which will be judged at the meeting in <strong>{{ brew_month }}</strong>.</p>
<p>To make sure your beer isn't easily identifiable, most people tend to use brown bottles of either 330ml or 500ml size. You probably don't want to put your own label on the bottle!</p>
<br/>
<div class="alert alert-success lead" role="alert"> <span class="glyphicon glyphicon-tags" aria-hidden="true"></span> &nbsp; &nbsp;Please mark all of your bottlecaps with the following identifier: <strong>{{ identifier }}</strong></div>
{% else %}
<p>Unfortunately it has not been possible to enter you into the competition at this time</p>
<div class="alert alert-danger lead" role="alert"> {{ error }} </div>
{% endif %}
</div>
</div>

6
web/utils.py Normal file
View File

@@ -0,0 +1,6 @@
def remove_common_elements(a, b):
"""Removes the common elements from both supplied lists (in place), doesnt not return a new list"""
for e in a[:]:
if e in b:
a.remove(e)
b.remove(e)