Unit 2.4a Using Programs with Data, SQLAlchemy
Using Programs with Data is focused on SQL and database actions. Part A focuses on SQLAlchemy and an OOP programming style,
Database and SQLAlchemy
In this blog we will explore using programs with data, focused on Databases. We will use SQLite Database to learn more about using Programs with Data. Use Debugging through these examples to examine Objects created in Code.
-
College Board talks about ideas like
- Program Usage. "iterative and interactive way when processing information"
- Managing Data. "classifying data are part of the process in using programs", "data files in a Table"
- Insight "insight and knowledge can be obtained from ... digitally represented information"
- Filter systems. 'tools for finding information and recognizing patterns"
- Application. "the preserve has two databases", "an employee wants to count the number of book"
-
PBL, Databases, Iterative/OOP
- Iterative. Refers to a sequence of instructions or code being repeated until a specific end result is achieved
- OOP. A computer programming model that organizes software design around data, or objects, rather than functions and logic
- SQL. Structured Query Language, abbreviated as SQL, is a language used in programming, managing, and structuring data
Imports and Flask Objects
Defines and key object creations
- Comment on where you have observed these working? Provide a defintion of purpose.
- Flask app object
- SQLAlchemy db object
I have seen and used SQLAlchemy in the old flask projects from last trimester. We used Object Relational Mapping (ORM) when learning about how to use models in code to represent data. The flask application object was what we used when creating the api file. It was a blueprint/ framework that allowed us to build applications. Within it, we defined create and read functions.
"""
These imports define the key objects
"""
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
"""
These object and definitions are used throughout the Jupyter Notebook.
"""
# Setup of key Flask object (app)
app = Flask(__name__)
# Setup SQLAlchemy object and properties for the database (db)
database = 'sqlite:///sqlite.db' # path and filename of database
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
app.config['SQLALCHEMY_DATABASE_URI'] = database
app.config['SECRET_KEY'] = 'SECRET_KEY'
db = SQLAlchemy()
# This belongs in place where it runs once per project
db.init_app(app)
Model Definition
Define columns, initialization, and CRUD methods for users table in sqlite.db
- Comment on these items in the class, purpose and defintion.
- class User:the class user contains the user's unique id, a separate uid, name, password, and date of birth. - db.Model inheritance: by inheriting from an existing class, you are using a model blueprint that you may modify to match your prgramming needs, or resuse as is. In this way, you can add more attributes or
- init method: the init method serves constructor of a User object, and initializes the instance variables within object. Basically, anything with "self" in front of it updates to the database. it's chewsday innit
-
@property
,@<column>.setter
- create, read, update, delete methods: create, update, and delete methods do exactly as their name implies to objects in the model.
""" database dependencies to support sqlite examples """
import datetime
from datetime import datetime
import json
from sqlalchemy.exc import IntegrityError
from werkzeug.security import generate_password_hash, check_password_hash
''' Tutorial: https://www.sqlalchemy.org/library.html#tutorials, try to get into a Python shell and follow along '''
# Define the User class to manage actions in the 'users' table
# -- Object Relational Mapping (ORM) is the key concept of SQLAlchemy
# -- a.) db.Model is like an inner layer of the onion in ORM
# -- b.) User represents data we want to store, something that is built on db.Model
# -- c.) SQLAlchemy ORM is layer on top of SQLAlchemy Core, then SQLAlchemy engine, SQL
class User(db.Model):
__tablename__ = 'users' # table name is plural, class name is singular
# Define the User schema with "vars" from object
id = db.Column(db.Integer, primary_key=True)
_name = db.Column(db.String(255), unique=False, nullable=False)
_uid = db.Column(db.String(255), unique=True, nullable=False)
_password = db.Column(db.String(255), unique=False, nullable=False)
_dob = db.Column(db.Date)
# constructor of a User object, initializes the instance variables within object (self)
def __init__(self, name, uid, password="123qwerty", dob=datetime.today()):
self._name = name # variables with self prefix become part of the object,
self._uid = uid
self.set_password(password)
if isinstance(dob, str): # not a date type
dob = date=datetime.today()
self._dob = dob
# a name getter method, extracts name from object
@property
def name(self):
return self._name
# a setter function, allows name to be updated after initial object creation
@name.setter
def name(self, name):
self._name = name
# a getter method, extracts uid from object
@property
def uid(self):
return self._uid
# a setter function, allows uid to be updated after initial object creation
@uid.setter
def uid(self, uid):
self._uid = uid
# check if uid parameter matches user id in object, return boolean
def is_uid(self, uid):
return self._uid == uid
@property
def password(self):
return self._password[0:10] + "..." # because of security only show 1st characters
# update password, this is conventional method used for setter
def set_password(self, password):
"""Create a hashed password."""
self._password = generate_password_hash(password, method='sha256')
# check password parameter against stored/encrypted password
def is_password(self, password):
"""Check against hashed password."""
result = check_password_hash(self._password, password)
return result
# dob property is returned as string, a string represents date outside object
@property
def dob(self):
dob_string = self._dob.strftime('%m-%d-%Y')
return dob_string
# dob setter, verifies date type before it is set or default to today
@dob.setter
def dob(self, dob):
if isinstance(dob, str): # not a date type
dob = date=datetime.today()
self._dob = dob
# age is calculated field, age is returned according to date of birth
@property
def age(self):
today = datetime.today()
return today.year - self._dob.year - ((today.month, today.day) < (self._dob.month, self._dob.day))
# output content using str(object) is in human readable form
# output content using json dumps, this is ready for API response
def __str__(self):
return json.dumps(self.read())
# CRUD create/add a new record to the table
# returns self or None on error
def create(self):
try:
# creates a person object from User(db.Model) class, passes initializers
db.session.add(self) # add prepares to persist person object to Users table
db.session.commit() # SqlAlchemy "unit of work pattern" requires a manual commit
return self
except IntegrityError:
db.session.remove()
return None
# CRUD read converts self to dictionary
# returns dictionary
def read(self):
return {
"id": self.id,
"name": self.name,
"uid": self.uid,
"dob": self.dob,
"age": self.age,
}
# CRUD update: updates user name, password, phone
# returns self
def update(self, name="", uid="", password=""):
"""only updates values with length"""
if len(name) > 0:
self.name = name
if len(uid) > 0:
self.uid = uid
if len(password) > 0:
self.set_password(password)
db.session.commit()
return self
# CRUD delete: remove self
# None
def delete(self):
db.session.delete(self)
db.session.commit()
return None
Initial Data
Uses SQLALchemy db.create_all() to initialize rows into sqlite.db
- Comment on how these work?
- Create All Tables from db Object:You can input a few user data, then the db createall function creates the tables in which the data will be stored from schema. 2. User Object Constructors: this is the way that tester table data is created. each row fro the data consists of a user and its attributes. The user object instantiated (instance is created) using attributes like name, uid, password, and dob that are directly specified at the time of construction by passing these as arguments in the constructor.
- Try / Except: this is how test data is passed to see if there are errors during code execution. This is a debugging method that includes a try block surrounding code that must be monitored. If there is an exception or runtime error, then the program execution goes to the exception block. This also catches bad or duplicated data. This mechanism gracefully exit out of a runtime error rather than crashing an entire application.
"""Database Creation and Testing """
# Builds working data for testing
def initUsers():
with app.app_context():
"""Create database and tables"""
db.create_all()
"""Tester data for table"""
u1 = User(name='Thomas Edison', uid='toby', password='123toby', dob=datetime(1847, 2, 11))
u2 = User(name='Nikola Tesla', uid='niko', password='123niko')
u3 = User(name='Alexander Graham Bell', uid='lex', password='123lex')
u4 = User(name='Eli Whitney', uid='whit', password='123whit')
u5 = User(name='Indiana Jones', uid='indi', dob=datetime(1920, 10, 21))
u6 = User(name='Marion Ravenwood', uid='raven', dob=datetime(1921, 10, 21))
users = [u1, u2, u3, u4, u5, u6]
"""Builds sample user/note(s) data"""
for user in users:
try:
'''add user to table'''
object = user.create()
print(f"Created new uid {object.uid}")
except: # error raised if object nit created
'''fails with bad or duplicate data'''
print(f"Records exist uid {user.uid}, or error.")
initUsers()
Check for given Credentials in users table in sqlite.db
Use of ORM Query object and custom methods to identify user to credentials uid and password
- Comment on purpose of following
- User.query.filter_by:this function's purpose is to organize the user table by the uid numerical order 2. user.password: the purpose of this code segment is to check an if statement condition- if the password entered matches the one corresponding to the user specified in the database
def find_by_uid(uid):
with app.app_context():
user = User.query.filter_by(_uid=uid).first()
return user # returns user object
# Check credentials by finding user and verify password
def check_credentials(uid, password):
# query email and return user record
user = find_by_uid(uid)
if user == None:
return False
if (user.is_password(password)):
return True
return False
#check_credentials("indi", "123qwerty")
Create a new User in table in Sqlite.db
Uses SQLALchemy and custom user.create() method to add row.
- Comment on purpose of following
- user.find_by_uid() and try/except:this is how test data is passed to see if there are errors during code execution. This is a debugging method that includes a try block surrounding code that must be monitored. If there is an exception or runtime error, then the program execution goes to the exception block. This also catches bad or duplicated data. This mechanism gracefully exit out of a runtime error rather than crashing an entire application. 2. user = User(...): this is the way that tester table data is created. each row fro the data consists of a user and its attributes. The user object instantiated (instance is created) using attributes like name, uid, password, and dob that are directly specified at the time of construction by passing these as arguments in the constructor.
- user.dob and try/except: (see above for try/ except)
- user.create() and try/except: (see above for try/ except) the create function's purpose is to create and verify a valid user object
def create():
# optimize user time to see if uid exists
uid = input("Enter your user id:")
user = find_by_uid(uid)
try:
print("Found\n", user.read())
return
except:
pass # keep going
# request value that ensure creating valid object
name = input("Enter your name:")
password = input("Enter your password")
# Initialize User object before date
user = User(name=name,
uid=uid,
password=password
)
# create user.dob, fail with today as dob
dob = input("Enter your date of birth 'YYYY-MM-DD'")
try:
user.dob = datetime.strptime(dob, '%Y-%m-%d').date()
except ValueError:
user.dob = datetime.today()
print(f"Invalid date {dob} require YYYY-mm-dd, date defaulted to {user.dob}")
# write object to database
with app.app_context():
try:
object = user.create()
print("Created\n", object.read())
except: # error raised if object not created
print("Unknown error uid {uid}")
create()
Reading users table in sqlite.db
Uses SQLALchemy query.all method to read data
- Comment on purpose of following
- User.query.all:this is a method that extracts data for each user table row. 2. json_ready assignment, google List Comprehension: According to ChatGPT, list comprehension is "a concise way to create lists in Python. It allows you to create a new list by applying a transformation to each element of an existing list or other iterable." In this context, the read function iterates through each user and its attributes in the database, and adds it to the list.
# SQLAlchemy extracts all users from database, turns each user into JSON
def read():
with app.app_context():
table = User.query.all()
json_ready = [user.read() for user in table] # "List Comprehensions", for each user add user.read() to list
return json_ready
read()
""" database dependencies to support sqlite examples """
import json
from sqlalchemy.exc import IntegrityError
#''' Tutorial: https://www.sqlalchemy.org/library.html#tutorials, try to get into a Python shell and follow along '''
# Define the User class to manage actions in the 'users' table
# -- Object Relational Mapping (ORM) is the key concept of SQLAlchemy
# -- a.) db.Model is like an inner layer of the onion in ORM
# -- b.) User represents data we want to store, something that is built on db.Model
# -- c.) SQLAlchemy ORM is layer on top of SQLAlchemy Core, then SQLAlchemy engine, SQL
class Recipe(db.Model):
__tablename__ = 'recipes' # table name is plural, class name is singular
# Define the User schema with "vars" from object
id = db.Column(db.Integer, unique=True, primary_key=True)
_recipename = db.Column(db.String(255), unique=False, nullable=False)
_recipelink = db.Column(db.String(255), unique=False, nullable=False)
_recipetype = db.Column(db.String(255), unique=False, nullable=False)
_recipecuisine = db.Column(db.String(255), unique=False, nullable=False)
# constructor of a User object, initializes the instance variables within object (self)
def __init__(self, recipename, recipelink, recipetype, recipecuisine):
self._recipename = recipename # variables with self prefix become part of the object,
self._recipelink = recipelink
self._recipetype = recipetype
self._recipecuisine = recipecuisine
# a name getter method, extracts name from object
@property
def recipename(self):
return self._recipename
# a setter function, allows name to be updated after initial object creation
@recipename.setter
def recipename(self, recipename):
self._recipename = recipename
# a getter method, extracts link from object
@property
def recipelink(self):
return self._recipelink
# a setter function, allows link to be updated after initial object creation
@recipelink.setter
def recipelink(self, recipelink):
self._recipelink = recipelink
# a getter method, extracts link from object
@property
def recipetype(self):
return self._recipetype
# a setter function, allows link to be updated after initial object creation
@recipetype.setter
def recipetype(self, recipetype):
self._recipetype = recipetype
# a getter method, extracts link from object
@property
def recipecuisine(self):
return self._recipecuisine
# a setter function, allows link to be updated after initial object creation
@recipecuisine.setter
def recipecuisine(self, recipecuisine):
self._recipecuisine = recipecuisine
@property
# output content using str(object) in human readable form, uses getter
# output content using json dumps, this is ready for API response
def __str__(self):
return json.dumps(self.read())
# CRUD create/add a new record to the table
# returns self or None on error
def create(self):
try:
# creates a person object from User(db.Model) class, passes initializers
db.session.add(self) # add prepares to persist person object to Users table
db.session.commit() # SqlAlchemy "unit of work pattern" requires a manual commit
return self
except IntegrityError:
db.session.remove()
return None
# CRUD read converts self to dictionary
# returns dictionary
def read(self):
return {
"id": self.id,
"recipename" : self.recipename,
"recipelink" : self.recipelink,
"recipetype" : self.recipetype,
"recipecuisine" : self.recipecuisine,
}
# CRUD update: updates user name, password, phone
# returns self
def update(self, recipename="", recipelink="", recipetype="", recipecuisine=""):
"""only updates values with length"""
if len(recipename) > 0:
self.recipename = recipename
if len(recipelink) > 0:
self.recipelink = recipelink
if len(recipetype) > 0:
self.recipetype = recipetype
if len(recipecuisine) > 0:
self.recipecuisine = recipecuisine
db.session.commit()
return self
# CRUD delete: remove self
# None
def delete(self):
db.session.delete(self)
db.session.commit()
return None
"""Database Creation and Testing """
# Builds working data for testing
def initRecipes():
with app.app_context():
"""Create database and tables"""
db.drop_all()
db.create_all()
"""Tester data for table"""
r1 = Recipe(recipename='Avocado Toast', recipelink='link1', recipetype='Breakfast', recipecuisine='American')
r2 = Recipe(recipename='Scrambled Eggs', recipelink='link2', recipetype='Breakfast', recipecuisine='American')
r3 = Recipe(recipename='Pancake', recipelink='link3', recipetype='Breakfast', recipecuisine='American')
r4 = Recipe(recipename='Mac and Cheese', recipelink='link4', recipetype='Lunch', recipecuisine='American')
r5 = Recipe(recipename='Panini Sandwich', recipelink='link5', recipetype='Lunch', recipecuisine='French')
r6 = Recipe(recipename='Salad', recipelink='link6', recipetype='Lunch', recipecuisine='Mediterranean')
r7 = Recipe(recipename='Minestrone Soup', recipelink='link7', recipetype='Dinner', recipecuisine='Italian')
r8 = Recipe(recipename='Lasagna', recipelink='link8', recipetype='Dinner', recipecuisine='Italian')
r9 = Recipe(recipename='Pasta', recipelink='link9', recipetype='Dinner', recipecuisine='Italian')
r10 = Recipe(recipename='Brownies', recipelink='link10', recipetype='Dessert', recipecuisine='German')
r11 = Recipe(recipename='Chocolate Chip Cookies', recipelink='link11', recipetype='Dessert', recipecuisine='American')
r12 = Recipe(recipename='Custard Pudding', recipelink='link12', recipetype='Dessert', recipecuisine='German')
recipes = [r1, r2, r3, r4, r5, r6, r7, r8, r9, r10, r11, r12]
"""Builds sample user/note(s) data"""
for recipe in recipes:
try:
recipe.create()
except IntegrityError:
'''fails with bad or duplicate data'''
db.session.remove()
print(f"Records exist, duplicate email, or error: {recipe.model}")
initRecipes()
# read function
# SQLAlchemy extracts all users from database, turns each user into JSON
def read():
with app.app_context():
table = Recipe.query.all()
json_ready = [recipe.read() for recipe in table] # "List Comprehensions", for each user add user.read() to list
return json_ready
read()
# Inputs, Try/Except, and SQLAlchemy work together to build a valid database object
def creater():
# request value that ensure creating valid object
recipename = input("Enter your recipe name:")
recipelink = input("Enter your recipe link")
recipetype = input("Enter your recipe type")
recipecuisine = input("Enter your recipe cuisine")
# Initialize User object before date
recipe = Recipe(recipename=recipename,
recipelink=recipelink,
recipetype=recipetype,
recipecuisine=recipecuisine
)
# write object to database
with app.app_context():
try:
object = recipe.create()
print("Created\n", object.read())
except: # error raised if object not created
print("Unknown error recipeid {id}")
creater()
def update_recipe(recipe_id, newrecipename=None, newrecipelink=None, newrecipetype=None, newrecipecuisine=None):
with app.app_context():
# retrieve the recipe object with the specified ID
recipe = Recipe.query.filter_by(id=recipe_id).first()
if recipe:
# update the recipe attributes if new values are provided
if newrecipename:
recipe.recipename = newrecipename
if newrecipelink:
recipe.recipelink = newrecipelink
if newrecipetype:
recipe.recipetype = newrecipetype
if newrecipecuisine:
recipe.recipecuisine = newrecipecuisine
# commit the changes to the database
db.session.commit()
print(f"Recipe with ID {recipe_id} updated successfully")
else:
print(f"Recipe with ID {recipe_id} not found")
# Update the recipe based on the recipe ID
update_recipe(recipe_id=1, newrecipename="Banana Bread", newrecipelink="https://www.yummly.com/recipe/Banana-Chocolate-Bread-1877435", newrecipetype="Breakfast", newrecipecuisine="American")
def delete_recipe(recipe_id):
with app.app_context():
# retrieve the recipe object with the specified ID
recipe = Recipe.query.filter_by(id=recipe_id).first()
if recipe:
# delete the recipe object from the database
db.session.delete(recipe)
db.session.commit()
print(f"Recipe {recipe_id} deleted successfully")
else:
print(f"Recipe {recipe_id} not found")
# Delete the recipe 6
delete_recipe(recipe_id=6)