Skip to content

Commit

Permalink
Update example for latest version of SQLAlchemy
Browse files Browse the repository at this point in the history
Pass argument that disables SQLite thread check when creating the engine
Revise request-processing functions to work within a database session context
Drop TODO from the example README
  • Loading branch information
chrisinmtown committed Dec 5, 2024
1 parent 087f05f commit 2afd62b
Show file tree
Hide file tree
Showing 3 changed files with 39 additions and 44 deletions.
2 changes: 0 additions & 2 deletions examples/sqlalchemy/README.rst
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,6 @@ SQLAlchemy Example

.. note::

TODO: Update this example to work with recent (2024) versions of Python and SQLAlchemy.

A simple example of how one might use SQLAlchemy as a backing store for a
Connexion based application.

Expand Down
65 changes: 30 additions & 35 deletions examples/sqlalchemy/app.py
100755 → 100644
Original file line number Diff line number Diff line change
Expand Up @@ -5,59 +5,54 @@
import orm
from connexion import NoContent

db_session = None


def get_pets(limit, animal_type=None):
q = db_session.query(orm.Pet)
if animal_type:
q = q.filter(orm.Pet.animal_type == animal_type)
return [p.dump() for p in q][:limit]
with db_sess_mkr() as db_session:
q = db_session.query(orm.Pet)
if animal_type:
q = q.filter(orm.Pet.animal_type == animal_type)
return [p.dump() for p in q][:limit]


def get_pet(pet_id):
pet = db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).one_or_none()
return pet.dump() if pet is not None else ("Not found", 404)
with db_sess_mkr() as db_session:
pet = db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).one_or_none()
return pet.dump() if pet is not None else ("Not found", 404)


def put_pet(pet_id, pet):
p = db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).one_or_none()
pet["id"] = pet_id
if p is not None:
logging.info("Updating pet %s..", pet_id)
p.update(**pet)
else:
logging.info("Creating pet %s..", pet_id)
pet["created"] = datetime.datetime.utcnow()
db_session.add(orm.Pet(**pet))
db_session.commit()
return NoContent, (200 if p is not None else 201)
with db_sess_mkr() as db_session:
p = db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).one_or_none()
pet["id"] = pet_id
if p is not None:
logging.info("Updating pet %s..", pet_id)
p.update(**pet)
else:
logging.info("Creating pet %s..", pet_id)
pet["created"] = datetime.datetime.utcnow()
db_session.add(orm.Pet(**pet))
db_session.commit()
return NoContent, (200 if p is not None else 201)


def delete_pet(pet_id):
pet = db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).one_or_none()
if pet is not None:
logging.info("Deleting pet %s..", pet_id)
db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).delete()
db_session.commit()
return NoContent, 204
else:
return NoContent, 404
with db_sess_mkr() as db_session:
pet = db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).one_or_none()
if pet is not None:
logging.info("Deleting pet %s..", pet_id)
db_session.query(orm.Pet).filter(orm.Pet.id == pet_id).delete()
db_session.commit()
return NoContent, 204
else:
return NoContent, 404


logging.basicConfig(level=logging.INFO)
db_session = orm.init_db("sqlite:///:memory:")
db_sess_mkr = orm.init_db()
app = connexion.FlaskApp(__name__, specification_dir="spec")
app.add_api("openapi.yaml")
app.add_api("swagger.yaml")

application = app.app


@application.teardown_appcontext
def shutdown_session(exception=None):
db_session.remove()


if __name__ == "__main__":
app.run(port=8080, reload=False)
16 changes: 9 additions & 7 deletions examples/sqlalchemy/orm.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
from sqlalchemy import Column, DateTime, String, create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import scoped_session, sessionmaker
from sqlalchemy.orm import sessionmaker
from sqlalchemy.pool import StaticPool

Base = declarative_base()

Expand All @@ -24,11 +25,12 @@ def dump(self):
return {k: v for k, v in vars(self).items() if not k.startswith("_")}


def init_db(uri):
engine = create_engine(uri, convert_unicode=True)
db_session = scoped_session(
sessionmaker(autocommit=False, autoflush=False, bind=engine)
def init_db():
# https://stackoverflow.com/questions/6519546/scoped-sessionsessionmaker-or-plain-sessionmaker-in-sqlalchemy
engine = create_engine(
url="sqlite:///:memory:",
connect_args={"check_same_thread": False},
poolclass=StaticPool,
)
Base.query = db_session.query_property()
Base.metadata.create_all(bind=engine)
return db_session
return sessionmaker(autocommit=False, autoflush=False, bind=engine)

0 comments on commit 2afd62b

Please sign in to comment.