adding sqlalchemy models

master
Drew Bednar 3 years ago
parent 804dda48ea
commit a5d58dfb18

@ -6,6 +6,7 @@ from flask_sqlalchemy import SQLAlchemy
from api.settings import app_settings from api.settings import app_settings
from api.util import CustomJSONEncoder
# instantiate the extensions # instantiate the extensions
db = SQLAlchemy() db = SQLAlchemy()
@ -15,6 +16,7 @@ cors = CORS()
def create_app(script_info=None): def create_app(script_info=None):
app = Flask(__name__) app = Flask(__name__)
app.json_encoder = CustomJSONEncoder
# set config # set config
config_engine = app_settings[os.getenv("APP_SETTINGS", "dev")]() config_engine = app_settings[os.getenv("APP_SETTINGS", "dev")]()
@ -34,5 +36,9 @@ def create_app(script_info=None):
# app.register_blueprint(users_blueprint) # app.register_blueprint(users_blueprint)
# shell context for flask cli # shell context for flask cli
app.shell_context_processor({"app": app, "db": db}) app.shell_context_processor(lambda: {"app": app, "db": db})
return app return app
# Model import occurs here to allow flask_sqlalchemy to import metadata
import api.models

@ -1,10 +1,10 @@
import os import datetime as dt
from flask import Blueprint from flask import Blueprint, jsonify
from .models import Cookie
main_blueprint = Blueprint("main", __name__) main_blueprint = Blueprint("main", __name__)
__version__ = "1.2.0" __version__ = "1.2.0"
@ -15,7 +15,13 @@ def pong():
return {"message": "pong"} return {"message": "pong"}
@main_blueprint.route("/env") @main_blueprint.route("/date")
def env(): def current_date():
"""Exposes the environment variables.""" """A route that simply returns the current datetime of the system"""
return dict(os.environ) return {"current_date": dt.datetime.now(dt.timezone.utc).isoformat()}
@main_blueprint.route("/cookies")
def get_cookies():
cookies = Cookie.query.all()
return jsonify([cookie.to_dict() for cookie in cookies])

@ -0,0 +1,58 @@
from sqlalchemy import (
Column,
Identity, # Identity replaced Serial
Integer,
BigInteger,
Numeric,
String,
ForeignKey,
DateTime,
Boolean,
Enum,
)
from sqlalchemy.orm import relationship
from sqlalchemy.sql import expression
from sqlalchemy.ext.compiler import compiles
from sqlalchemy.types import DateTime as DatetimeType
from . import db
class utcnow(expression.FunctionElement):
type = DatetimeType()
@compiles(utcnow, "postgresql")
def pg_utcnow(element, compiler, **kw):
return "TIMEZONE('utc', CURRENT_TIMESTAMP)"
def ReferenceCol(tablename, nullable=False, **kw):
"""A utility function for declaring foreign key relationships."""
return Column(ForeignKey("{}.id".format(tablename)), nullable=nullable, **kw)
class DBMixin:
id = Column(BigInteger, Identity(start=100000), primary_key=True)
created_date = Column(DateTime(timezone=True), server_default=utcnow())
modified_date = Column(
DateTime(timezone=True), server_default=utcnow(), onupdate=utcnow()
)
def to_dict(self):
d = self.__dict__.copy()
if "_sa_instance_state" in d:
d.pop("_sa_instance_state")
return d
class Cookie(DBMixin, db.Model):
"""A Cookie Class"""
__tablename__ = "cookies"
name = Column(String(50), index=True)
recipe_url = Column(String(255))
sku = Column(String(55))
qoh = Column(Integer)
unit_cost = Column(Numeric(12, 2))

@ -1,5 +1,15 @@
import os import os
# TODO Find an alternative to injection envvars into config.
if os.getenv("FLASK_ENV") == "development":
try:
import dotenv
dotenv.load_dotenv()
except Exception:
print("Didn't load a .env file")
pass
# TODO Add postgres params option # TODO Add postgres params option
def build_postgres_uri(host, user, dbname, passwd=None, port="5432", sslmode=None): def build_postgres_uri(host, user, dbname, passwd=None, port="5432", sslmode=None):
@ -44,6 +54,7 @@ class BaseConfigEngine:
class DevConfigEngine(BaseConfigEngine): class DevConfigEngine(BaseConfigEngine):
DEBUG = True DEBUG = True
SQLALCHEMY_ECHO = True
class TestingConfigEngine(BaseConfigEngine): class TestingConfigEngine(BaseConfigEngine):

@ -0,0 +1,19 @@
import datetime as dt
from decimal import Decimal
from flask.json import JSONEncoder
class CustomJSONEncoder(JSONEncoder):
def default(self, obj):
try:
if isinstance(obj, dt.datetime):
return obj.isoformat()
if isinstance(obj, Decimal):
return float(obj)
iterable = iter(obj)
except TypeError:
pass
else:
return list(iterable)
return JSONEncoder.default(self, obj)
Loading…
Cancel
Save