From 7a3e5047829974106eeda7dc08b515fcd14e5e2c Mon Sep 17 00:00:00 2001 From: chiscobyte Date: Sun, 25 May 2025 17:38:58 +0300 Subject: [PATCH 1/2] initiate --- lib/__init__.py | 11 + lib/database.py | 27 +++ lib/debug.py | 62 +++++- lib/freebie_tracker.db | Bin 0 -> 16384 bytes lib/freebies.db | Bin 20480 -> 24576 bytes .../versions/xxxx_add_freebies_table.py | 20 ++ lib/models.py | 190 ++++++++++++++++-- lib/seed.py | 85 +++++++- lib/setup_db.py | 19 ++ 9 files changed, 389 insertions(+), 25 deletions(-) create mode 100644 lib/__init__.py create mode 100644 lib/database.py create mode 100644 lib/freebie_tracker.db create mode 100644 lib/migrations/versions/xxxx_add_freebies_table.py create mode 100644 lib/setup_db.py diff --git a/lib/__init__.py b/lib/__init__.py new file mode 100644 index 000000000..c9f43e6f4 --- /dev/null +++ b/lib/__init__.py @@ -0,0 +1,11 @@ +# lib/__init__.py +from sqlalchemy import create_engine +from sqlalchemy.orm import sessionmaker +from .models import Base + +engine = create_engine('sqlite:///freebies.db') +Session = sessionmaker(bind=engine) +session = Session() + + +Base.metadata.create_all(engine) diff --git a/lib/database.py b/lib/database.py new file mode 100644 index 000000000..8fb7d52a7 --- /dev/null +++ b/lib/database.py @@ -0,0 +1,27 @@ +from sqlalchemy import create_engine +from sqlalchemy.orm import sessionmaker, declarative_base + +# Create an engine to your database (SQLite for simplicity) +engine = create_engine('sqlite:///freebie_tracker.db', echo=True) + +# Create a configured "Session" class +Session = sessionmaker(bind=engine) + +# Create a session instance +session = Session() + +# Base class for declarative models +Base = declarative_base() + + + +# from sqlalchemy import create_engine +# from sqlalchemy.orm import sessionmaker + +# from models import Base # Make sure Base is defined in models.py + +# DATABASE_URL = "sqlite:///freebies.db" + +# engine = create_engine(DATABASE_URL) +# Session = sessionmaker(bind=engine) +# session = Session() diff --git a/lib/debug.py b/lib/debug.py index 4f922eb69..f46e2bb25 100644 --- a/lib/debug.py +++ b/lib/debug.py @@ -1,9 +1,61 @@ -#!/usr/bin/env python3 +# #!/usr/bin/env python3 +# lib/debug.py +from models import Dev from sqlalchemy import create_engine +from sqlalchemy.orm import sessionmaker -from models import Company, Dev +engine = create_engine('sqlite:///freebie_tracker.db') +Session = sessionmaker(bind=engine) +session = Session() -if __name__ == '__main__': - engine = create_engine('sqlite:///freebies.db') - import ipdb; ipdb.set_trace() +# List all devs and their freebies +for dev in session.query(Dev).all(): + print(f"{dev.name} has freebies:") + for freebie in dev.freebies: + print(f" - {freebie.item_name} (${freebie.value}) from {freebie.company.name}") + + +# from models import Company, Dev, Freebie +# from database import session +# import ipdb + +# ipdb.set_trace() + +# # Once inside the ipdb console, try commands like: + +# # List all devs and their freebies +# for dev in session.query(Dev).all(): +# print(f"{dev.name} has freebies:") +# for freebie in dev.freebies: +# print(f" - {freebie.print_details()}") + +# # Oldest company +# oldest = Company.oldest_company(session) +# print(f"Oldest company: {oldest.name}") + +# # Give a freebie +# company = session.query(Company).first() +# dev = session.query(Dev).first() +# new_freebie = company.give_freebie(dev, "Notebook", 5) +# print(new_freebie.print_details()) + +# # Check if dev received a specific freebie +# print(dev.received_one("Notebook")) + +# # Give away a freebie +# other_dev = session.query(Dev).filter(Dev.id != dev.id).first() +# success = dev.give_away(other_dev, new_freebie) +# print(f"Give away successful? {success}") + + + + + +# from sqlalchemy import create_engine + +# from models import Company, Dev + +# if __name__ == '__main__': +# engine = create_engine('sqlite:///freebies.db') +# import ipdb; ipdb.set_trace() diff --git a/lib/freebie_tracker.db b/lib/freebie_tracker.db new file mode 100644 index 0000000000000000000000000000000000000000..5d29935d04511a97f9fb549f5792bdd01a0e4c76 GIT binary patch literal 16384 zcmeI&!EVzq7zc1WDa)Fv%Ot3Jn5OwgBLNp)fY5oPqRSu+NIO+o<8Id4W=hjaJ9azq z47?RDfPl-695`~oPD!^G*b%AH-;!(h^VitlFDFZFzir7(uwIf5qKrK!TZB^bf-yqK zy6Pp>i)O0H-S4zmxMy4^&n6!#D#5J$B+f^tq6RbwKmY;|fB*y_009U<00I#B&jMGQ zrd8Y7q1Q4*G)2h2YTng6cUdcssV z6NAn$8VL5@3mUJyKnEvLe=Lf09f-xGbNc`t=_Z4VXn464eG@bfz2KO==EtlqV^?SH z9|gR5;Okg@_8gZ5e4hv0Z}2u#q0wJ+Z+BS8pM_P)`HyUOx?xzg-Ca5rImmo@J~00OrPa&5Outy;B8r*3{2U7PO4!VzW_|MQJ3@BZf|`fhGG zUy1WWeV{=A0uX=z1Rwwb2tWV=5P$##Ah4nWTZTy*&sHU_=2PG0^W46MAI4E((OJNfGQqvc`g2cQ^py=fHd@{)( zU2gtCuAc6G3f``f8W0Vd3PG-Ju0gJT&aS}Wae}lUD+2=q2*UvLL>*&MRtCLzSzi7h3@m(74E((OJNfGQq&5o*EaTlQ I#rMn+05sDJLI3~& diff --git a/lib/migrations/versions/xxxx_add_freebies_table.py b/lib/migrations/versions/xxxx_add_freebies_table.py new file mode 100644 index 000000000..31d4f43d7 --- /dev/null +++ b/lib/migrations/versions/xxxx_add_freebies_table.py @@ -0,0 +1,20 @@ +from alembic import op +import sqlalchemy as sa + +revision = 'xxxx_add_freebies_table' +down_revision = 'previous_revision_id' +branch_labels = None +depends_on = None + +def upgrade(): + op.create_table( + 'freebies', + sa.Column('id', sa.Integer, primary_key=True), + sa.Column('item_name', sa.String, nullable=False), + sa.Column('value', sa.Integer, nullable=False), + sa.Column('dev_id', sa.Integer, sa.ForeignKey('devs.id'), nullable=False), + sa.Column('company_id', sa.Integer, sa.ForeignKey('companies.id'), nullable=False) + ) + +def downgrade(): + op.drop_table('freebies') diff --git a/lib/models.py b/lib/models.py index 2681bee5a..f1b305f37 100644 --- a/lib/models.py +++ b/lib/models.py @@ -1,29 +1,181 @@ -from sqlalchemy import ForeignKey, Column, Integer, String, MetaData -from sqlalchemy.orm import relationship, backref -from sqlalchemy.ext.declarative import declarative_base +# lib/models.py +from sqlalchemy import Column, Integer, String, ForeignKey +from sqlalchemy.orm import relationship, declarative_base -convention = { - "fk": "fk_%(table_name)s_%(column_0_name)s_%(referred_table_name)s", -} -metadata = MetaData(naming_convention=convention) +Base = declarative_base() -Base = declarative_base(metadata=metadata) +class Dev(Base): + __tablename__ = 'devs' + id = Column(Integer, primary_key=True) + name = Column(String) + + freebies = relationship("Freebie", back_populates="dev", overlaps="companies,freebies") + companies = relationship("Company", secondary="freebies", back_populates="devs", overlaps="freebies,devs") class Company(Base): __tablename__ = 'companies' + id = Column(Integer, primary_key=True) + name = Column(String) + founding_year = Column(Integer) - id = Column(Integer(), primary_key=True) - name = Column(String()) - founding_year = Column(Integer()) + freebies = relationship("Freebie", back_populates="company", overlaps="companies,devs") + devs = relationship("Dev", secondary="freebies", back_populates="companies", overlaps="freebies,companies") - def __repr__(self): - return f'' +class Freebie(Base): + __tablename__ = 'freebies' + id = Column(Integer, primary_key=True) + item_name = Column(String) + value = Column(Integer) + dev_id = Column(Integer, ForeignKey('devs.id')) + company_id = Column(Integer, ForeignKey('companies.id')) -class Dev(Base): - __tablename__ = 'devs' + dev = relationship("Dev", back_populates="freebies", overlaps="companies,devs") + company = relationship("Company", back_populates="freebies", overlaps="companies,devs") + + + +# from sqlalchemy import Column, Integer, String, ForeignKey +# from sqlalchemy.orm import relationship +# from database import Base, session # Adjust import as per your setup + +# # class Dev(Base): +# # __tablename__ = 'devs' +# # id = Column(Integer, primary_key=True) +# # name = Column(String) +# # freebies = relationship("Freebie", back_populates="dev") +# # companies = relationship("Company", secondary="freebies", back_populates="devs") + +# class Company(Base): +# __tablename__ = 'companies' + +# id = Column(Integer, primary_key=True) +# name = Column(String) +# founding_year = Column(Integer) + +# freebies = relationship('Freebie', back_populates='company', cascade='all, delete-orphan') +# devs = relationship('Dev', secondary='freebies', back_populates='companies') + +# def give_freebie(self, dev, item_name, value): +# freebie = Freebie(item_name=item_name, value=value, dev=dev, company=self) +# session.add(freebie) +# session.commit() +# return freebie + +# @classmethod +# def oldest_company(cls, session): +# return session.query(cls).order_by(cls.founding_year).first() + +# class Dev(Base): +# __tablename__ = 'devs' + +# id = Column(Integer, primary_key=True) +# name = Column(String) + +# freebies = relationship('Freebie', back_populates='dev', cascade='all, delete-orphan') +# companies = relationship('Company', secondary='freebies', back_populates='devs') + +# def received_one(self, item_name): +# return any(freebie.item_name == item_name for freebie in self.freebies) + +# def give_away(self, new_dev, freebie): +# if freebie in self.freebies: +# freebie.dev = new_dev +# session.commit() +# return True +# return False + +# class Freebie(Base): +# __tablename__ = 'freebies' + +# id = Column(Integer, primary_key=True) +# item_name = Column(String) +# value = Column(Integer) +# dev_id = Column(Integer, ForeignKey('devs.id')) +# company_id = Column(Integer, ForeignKey('companies.id')) + +# dev = relationship('Dev', back_populates='freebies') +# company = relationship('Company', back_populates='freebies') + +# def print_details(self): +# return f"{self.dev.name} owns a {self.item_name} from {self.company.name}." + + +# from sqlalchemy import Column, Integer, String, ForeignKey, MetaData +# from sqlalchemy.orm import relationship, backref +# from sqlalchemy.ext.declarative import declarative_base +# from sqlalchemy.ext.associationproxy import association_proxy + +# # Custom metadata naming convention +# metadata = MetaData(naming_convention={ +# "fk": "fk_%(table_name)s_%(column_0_name)s_%(referred_table_name)s" +# }) + +# Base = declarative_base(metadata=metadata) + +# class Company(Base): +# __tablename__ = 'companies' + +# id = Column(Integer, primary_key=True) +# name = Column(String) +# founding_year = Column(Integer) + +# # Each company can hand out many freebies +# freebies = relationship('Freebie', backref='company', cascade="all, delete-orphan") +# devs = association_proxy('freebies', 'dev') # Developers who got freebies from this company + +# def __repr__(self): +# return f"" + +# def give_freebie(self, dev, item_name, value): +# """Hand a freebie to a developer.""" +# return Freebie(item_name=item_name, value=value, dev=dev, company=self) + +# @classmethod +# def oldest_company(cls, session): +# """Return the company founded first.""" +# return session.query(cls).order_by(cls.founding_year).first() + + +# class Dev(Base): +# __tablename__ = 'devs' + +# id = Column(Integer, primary_key=True) +# name = Column(String) + +# # One dev can receive multiple freebies +# freebies = relationship('Freebie', backref='dev', cascade="all, delete-orphan") +# companies = association_proxy('freebies', 'company') + +# def __repr__(self): +# return f"" + +# def received_one(self, item_name): +# """Check if the dev has received a specific item.""" +# return any(f.item_name == item_name for f in self.freebies) + +# def give_away(self, other_dev, freebie): +# """Give a freebie to another dev if this dev owns it.""" +# if freebie in self.freebies: +# freebie.dev = other_dev + + +# class Freebie(Base): +# __tablename__ = 'freebies' + +# id = Column(Integer, primary_key=True) +# item_name = Column(String) +# value = Column(Integer) + +# dev_id = Column(Integer, ForeignKey('devs.id')) +# company_id = Column(Integer, ForeignKey('companies.id')) + +# def __repr__(self): +# return f"" - id = Column(Integer(), primary_key=True) - name= Column(String()) +# def print_details(self): +# """Nicely formatted description of who owns what.""" +# dev_name = self.dev.name if self.dev else "Unknown Dev" +# company_name = self.company.name if self.company else "Unknown Company" +# return f"{dev_name} owns a {self.item_name} from {company_name}" - def __repr__(self): - return f'' +# Base = Base diff --git a/lib/seed.py b/lib/seed.py index b16becbbb..034e8ba7f 100644 --- a/lib/seed.py +++ b/lib/seed.py @@ -1,3 +1,86 @@ #!/usr/bin/env python3 -# Script goes here! +# lib/seed.py +from sqlalchemy import create_engine +from sqlalchemy.orm import sessionmaker +from models import Dev, Company, Freebie, Base + +engine = create_engine('sqlite:///freebie_tracker.db') +Session = sessionmaker(bind=engine) +session = Session() + +# Clear existing data +session.query(Freebie).delete() +session.query(Dev).delete() +session.query(Company).delete() + +# Create devs +dev1 = Dev(name="Chisco ") +dev2 = Dev(name="Ada Developer") + +# Create companies +company1 = Company(name="Nike", founding_year=2015) +company2 = Company(name="Dior", founding_year=2017) + +# Create freebies +freebie1 = Freebie(item_name="T-shirt", value=20, dev=dev1, company=company1) +freebie2 = Freebie(item_name="Sticker", value=5, dev=dev1, company=company2) +freebie3 = Freebie(item_name="Mug", value=15, dev=dev2, company=company1) + +# Add and commit +session.add_all([dev1, dev2, company1, company2, freebie1, freebie2, freebie3]) +session.commit() + +print("✅ Seeded the database!") + + +# from models import Company, Dev, Freebie +# from database import session + +# # Clear existing data (optional, careful in prod!) +# session.query(Freebie).delete() +# session.query(Dev).delete() +# session.query(Company).delete() +# session.commit() + +# # Create companies +# company1 = Company(name="Code Corp", founding_year=2001) +# company2 = Company(name="Dev Supplies", founding_year=1995) + +# # Create devs +# dev1 = Dev(name="Alice") +# dev2 = Dev(name="Bob") + +# session.add_all([company1, company2, dev1, dev2]) +# session.commit() + +# # Create freebies +# freebie1 = Freebie(item_name="Sticker", value=1, dev=dev1, company=company1) +# freebie2 = Freebie(item_name="T-shirt", value=20, dev=dev1, company=company2) +# freebie3 = Freebie(item_name="Mug", value=10, dev=dev2, company=company1) + +# session.add_all([freebie1, freebie2, freebie3]) +# session.commit() + + +# # from models import Company, Dev, Freebie +# # from sqlalchemy.orm import sessionmaker +# # from sqlalchemy import create_engine + +# # engine = create_engine('sqlite:///freebies.db') +# # Session = sessionmaker(bind=engine) +# # session = Session() + +# # # Clear existing data +# # session.query(Freebie).delete() +# # session.query(Dev).delete() +# # session.query(Company).delete() + +# # # Create records +# # c1 = Company(name="OpenAI", founding_year=2015) +# # d1 = Dev(name="Alice") +# # f1 = Freebie(item_name="T-shirt", value=20, company=c1, dev=d1) + +# # session.add_all([c1, d1, f1]) +# # session.commit() + diff --git a/lib/setup_db.py b/lib/setup_db.py new file mode 100644 index 000000000..16a0b38d4 --- /dev/null +++ b/lib/setup_db.py @@ -0,0 +1,19 @@ +# lib/setup_db.py + +from sqlalchemy import create_engine +from models import Base + +engine = create_engine('sqlite:///freebie_tracker.db') +Base.metadata.create_all(engine) + +print("✅ Tables created!") + + +# from sqlalchemy import create_engine +# from sqlalchemy.orm import sessionmaker +# from models import Base + +# engine = create_engine('sqlite:///freebies.db') +# Base.metadata.create_all(engine) + +# print("✅ Tables created!") From b6015d6cf6aedb7e6145e08b882ea0d47cc8598e Mon Sep 17 00:00:00 2001 From: chiscobyte Date: Sun, 25 May 2025 17:41:22 +0300 Subject: [PATCH 2/2] initiate --- Pipfile | 3 +- Pipfile.lock | 311 ++++++++++++++++++++++++++++------------- alembic.ini | 106 ++++++++++++++ alembic/README | 1 + alembic/env.py | 78 +++++++++++ alembic/script.py.mako | 24 ++++ freebies.db | Bin 0 -> 16384 bytes 7 files changed, 428 insertions(+), 95 deletions(-) create mode 100644 alembic.ini create mode 100644 alembic/README create mode 100644 alembic/env.py create mode 100644 alembic/script.py.mako create mode 100644 freebies.db diff --git a/Pipfile b/Pipfile index 63c79cd98..b04c69778 100644 --- a/Pipfile +++ b/Pipfile @@ -8,7 +8,8 @@ alembic = "1.8.1" importlib-metadata = "6.0.0" importlib-resources = "5.10.0" ipdb = "0.13.9" -sqlalchemy = "1.4.42" +sqlalchemy = "*" +ipython = "*" [dev-packages] diff --git a/Pipfile.lock b/Pipfile.lock index 3923f4a97..198b9ec2d 100644 --- a/Pipfile.lock +++ b/Pipfile.lock @@ -1,7 +1,7 @@ { "_meta": { "hash": { - "sha256": "fa62c636b7ae5acb4993fc7b007a651c55b562aa29962a9a8585ec314b352ae6" + "sha256": "28ea6a9577ce511fd5a6915dfe5f22db0b81b01f50fc144f159a6ddd51753174" }, "pipfile-spec": 6, "requires": { @@ -34,10 +34,11 @@ }, "asttokens": { "hashes": [ - "sha256:4622110b2a6f30b77e1473affaa97e711bc2f07d3f10848420ff1898edbe94f3", - "sha256:6b0ac9e93fb0335014d382b8fa9b3afa7df546984258005da0b9e7095b3deb1c" + "sha256:0dcd8baa8d62b0c1d118b399b2ddba3c4aff271d0d7a9e0d4c1681c79035bbc7", + "sha256:e3078351a059199dd5138cb1c706e6430c05eff2ff136af5eb4790f9d28932e2" ], - "version": "==2.2.1" + "markers": "python_version >= '3.8'", + "version": "==3.0.0" }, "backcall": { "hashes": [ @@ -48,18 +49,98 @@ }, "decorator": { "hashes": [ - "sha256:637996211036b6385ef91435e4fae22989472f9d571faba8927ba8253acbc330", - "sha256:b8c3f85900b9dc423225913c5aace94729fe1fa9763b38939a95226f02d37186" + "sha256:65f266143752f734b0a7cc83c46f4618af75b8c5911b00ccb61d0ac9b6da0360", + "sha256:d316bb415a2d9e2d2b3abcc4084c6502fc09240e292cd76a76afc106a1c8e04a" ], - "markers": "python_version < '3.11' and python_version >= '3.7'", - "version": "==5.1.1" + "markers": "python_version >= '3.8'", + "version": "==5.2.1" }, "executing": { "hashes": [ - "sha256:0314a69e37426e3608aada02473b4161d4caf5a4b244d1d0c48072b8fee7bacc", - "sha256:19da64c18d2d851112f09c287f8d3dbbdf725ab0e569077efb6cdcbd3497c107" + "sha256:11387150cad388d62750327a53d3339fad4888b39a6fe233c3afbb54ecffd3aa", + "sha256:5d108c028108fe2551d1a7b2e8b713341e2cb4fc0aa7dcf966fa4327a5226755" ], - "version": "==1.2.0" + "markers": "python_version >= '3.8'", + "version": "==2.2.0" + }, + "greenlet": { + "hashes": [ + "sha256:0153404a4bb921f0ff1abeb5ce8a5131da56b953eda6e14b88dc6bbc04d2049e", + "sha256:03a088b9de532cbfe2ba2034b2b85e82df37874681e8c470d6fb2f8c04d7e4b7", + "sha256:04b013dc07c96f83134b1e99888e7a79979f1a247e2a9f59697fa14b5862ed01", + "sha256:05175c27cb459dcfc05d026c4232f9de8913ed006d42713cb8a5137bd49375f1", + "sha256:09fc016b73c94e98e29af67ab7b9a879c307c6731a2c9da0db5a7d9b7edd1159", + "sha256:0bbae94a29c9e5c7e4a2b7f0aae5c17e8e90acbfd3bf6270eeba60c39fce3563", + "sha256:0fde093fb93f35ca72a556cf72c92ea3ebfda3d79fc35bb19fbe685853869a83", + "sha256:1443279c19fca463fc33e65ef2a935a5b09bb90f978beab37729e1c3c6c25fe9", + "sha256:1776fd7f989fc6b8d8c8cb8da1f6b82c5814957264d1f6cf818d475ec2bf6395", + "sha256:1d3755bcb2e02de341c55b4fca7a745a24a9e7212ac953f6b3a48d117d7257aa", + "sha256:23f20bb60ae298d7d8656c6ec6db134bca379ecefadb0b19ce6f19d1f232a942", + "sha256:275f72decf9932639c1c6dd1013a1bc266438eb32710016a1c742df5da6e60a1", + "sha256:2846930c65b47d70b9d178e89c7e1a69c95c1f68ea5aa0a58646b7a96df12441", + "sha256:3319aa75e0e0639bc15ff54ca327e8dc7a6fe404003496e3c6925cd3142e0e22", + "sha256:346bed03fe47414091be4ad44786d1bd8bef0c3fcad6ed3dee074a032ab408a9", + "sha256:36b89d13c49216cadb828db8dfa6ce86bbbc476a82d3a6c397f0efae0525bdd0", + "sha256:37b9de5a96111fc15418819ab4c4432e4f3c2ede61e660b1e33971eba26ef9ba", + "sha256:396979749bd95f018296af156201d6211240e7a23090f50a8d5d18c370084dc3", + "sha256:3b2813dc3de8c1ee3f924e4d4227999285fd335d1bcc0d2be6dc3f1f6a318ec1", + "sha256:411f015496fec93c1c8cd4e5238da364e1da7a124bcb293f085bf2860c32c6f6", + "sha256:47da355d8687fd65240c364c90a31569a133b7b60de111c255ef5b606f2ae291", + "sha256:48ca08c771c268a768087b408658e216133aecd835c0ded47ce955381105ba39", + "sha256:4afe7ea89de619adc868e087b4d2359282058479d7cfb94970adf4b55284574d", + "sha256:4ce3ac6cdb6adf7946475d7ef31777c26d94bccc377e070a7986bd2d5c515467", + "sha256:4ead44c85f8ab905852d3de8d86f6f8baf77109f9da589cb4fa142bd3b57b475", + "sha256:54558ea205654b50c438029505def3834e80f0869a70fb15b871c29b4575ddef", + "sha256:5e06afd14cbaf9e00899fae69b24a32f2196c19de08fcb9f4779dd4f004e5e7c", + "sha256:62ee94988d6b4722ce0028644418d93a52429e977d742ca2ccbe1c4f4a792511", + "sha256:63e4844797b975b9af3a3fb8f7866ff08775f5426925e1e0bbcfe7932059a12c", + "sha256:6510bf84a6b643dabba74d3049ead221257603a253d0a9873f55f6a59a65f822", + "sha256:667a9706c970cb552ede35aee17339a18e8f2a87a51fba2ed39ceeeb1004798a", + "sha256:6ef9ea3f137e5711f0dbe5f9263e8c009b7069d8a1acea822bd5e9dae0ae49c8", + "sha256:7017b2be767b9d43cc31416aba48aab0d2309ee31b4dbf10a1d38fb7972bdf9d", + "sha256:7124e16b4c55d417577c2077be379514321916d5790fa287c9ed6f23bd2ffd01", + "sha256:73aaad12ac0ff500f62cebed98d8789198ea0e6f233421059fa68a5aa7220145", + "sha256:77c386de38a60d1dfb8e55b8c1101d68c79dfdd25c7095d51fec2dd800892b80", + "sha256:7876452af029456b3f3549b696bb36a06db7c90747740c5302f74a9e9fa14b13", + "sha256:7939aa3ca7d2a1593596e7ac6d59391ff30281ef280d8632fa03d81f7c5f955e", + "sha256:8320f64b777d00dd7ccdade271eaf0cad6636343293a25074cc5566160e4de7b", + "sha256:85f3ff71e2e60bd4b4932a043fbbe0f499e263c628390b285cb599154a3b03b1", + "sha256:8b8b36671f10ba80e159378df9c4f15c14098c4fd73a36b9ad715f057272fbef", + "sha256:93147c513fac16385d1036b7e5b102c7fbbdb163d556b791f0f11eada7ba65dc", + "sha256:935e943ec47c4afab8965954bf49bfa639c05d4ccf9ef6e924188f762145c0ff", + "sha256:94b6150a85e1b33b40b1464a3f9988dcc5251d6ed06842abff82e42632fac120", + "sha256:94ebba31df2aa506d7b14866fed00ac141a867e63143fe5bca82a8e503b36437", + "sha256:95ffcf719966dd7c453f908e208e14cde192e09fde6c7186c8f1896ef778d8cd", + "sha256:98884ecf2ffb7d7fe6bd517e8eb99d31ff7855a840fa6d0d63cd07c037f6a981", + "sha256:99cfaa2110534e2cf3ba31a7abcac9d328d1d9f1b95beede58294a60348fba36", + "sha256:9e8f8c9cb53cdac7ba9793c276acd90168f416b9ce36799b9b885790f8ad6c0a", + "sha256:a0dfc6c143b519113354e780a50381508139b07d2177cb6ad6a08278ec655798", + "sha256:b2795058c23988728eec1f36a4e5e4ebad22f8320c85f3587b539b9ac84128d7", + "sha256:b42703b1cf69f2aa1df7d1030b9d77d3e584a70755674d60e710f0af570f3761", + "sha256:b7cede291382a78f7bb5f04a529cb18e068dd29e0fb27376074b6d0317bf4dd0", + "sha256:b8a678974d1f3aa55f6cc34dc480169d58f2e6d8958895d68845fa4ab566509e", + "sha256:b8da394b34370874b4572676f36acabac172602abf054cbc4ac910219f3340af", + "sha256:c3a701fe5a9695b238503ce5bbe8218e03c3bcccf7e204e455e7462d770268aa", + "sha256:c4aab7f6381f38a4b42f269057aee279ab0fc7bf2e929e3d4abfae97b682a12c", + "sha256:ca9d0ff5ad43e785350894d97e13633a66e2b50000e8a183a50a88d834752d42", + "sha256:d0028e725ee18175c6e422797c407874da24381ce0690d6b9396c204c7f7276e", + "sha256:d21e10da6ec19b457b82636209cbe2331ff4306b54d06fa04b7c138ba18c8a81", + "sha256:d5e975ca70269d66d17dd995dafc06f1b06e8cb1ec1e9ed54c1d1e4a7c4cf26e", + "sha256:da7a9bff22ce038e19bf62c4dd1ec8391062878710ded0a845bcf47cc0200617", + "sha256:db32b5348615a04b82240cc67983cb315309e88d444a288934ee6ceaebcad6cc", + "sha256:dcc62f31eae24de7f8dce72134c8651c58000d3b1868e01392baea7c32c247de", + "sha256:dfc59d69fc48664bc693842bd57acfdd490acafda1ab52c7836e3fc75c90a111", + "sha256:e347b3bfcf985a05e8c0b7d462ba6f15b1ee1c909e2dcad795e49e91b152c383", + "sha256:e4d333e558953648ca09d64f13e6d8f0523fa705f51cae3f03b5983489958c70", + "sha256:ed10eac5830befbdd0c32f83e8aa6288361597550ba669b04c48f0f9a2c843c6", + "sha256:efc0f674aa41b92da8c49e0346318c6075d734994c3c4e4430b1c3f853e498e4", + "sha256:f1695e76146579f8c06c1509c7ce4dfe0706f49c6831a817ac04eebb2fd02011", + "sha256:f1d4aeb8891338e60d1ab6127af1fe45def5259def8094b9c7e34690c8858803", + "sha256:f406b22b7c9a9b4f8aa9d2ab13d6ae0ac3e85c9a809bd590ad53fed2bf70dc79", + "sha256:f6ff3b14f2df4c41660a7dec01045a045653998784bf8cfcb5a525bdffffbc8f" + ], + "markers": "python_version >= '3.7'", + "version": "==3.1.1" }, "importlib-metadata": { "hashes": [ @@ -79,27 +160,28 @@ }, "ipdb": { "hashes": [ - "sha256:c23b6736f01fd4586cc2ecbebdf79a5eb454796853e1cd8f2ed3b7b91d4a3e93", - "sha256:f74c2f741c18b909eaf89f19fde973f745ac721744aa1465888ce45813b63a9c" + "sha256:951bd9a64731c444fd907a5ce268543020086a697f6be08f7cc2c9a752a278c5" ], "index": "pypi", - "version": "==0.13.11" + "markers": "python_version >= '2.7'", + "version": "==0.13.9" }, "ipython": { "hashes": [ - "sha256:da01e6df1501e6e7c32b5084212ddadd4ee2471602e2cf3e0190f4de6b0ea481", - "sha256:f3bf2c08505ad2c3f4ed5c46ae0331a8547d36bf4b21a451e8ae80c0791db95b" + "sha256:3910c4b54543c2ad73d06579aa771041b7d5707b033bd488669b4cf544e3b363", + "sha256:b0340d46a933d27c657b211a329d0be23793c36595acf9e6ef4164bc01a1804c" ], - "markers": "python_version < '3.11' and python_version >= '3.7'", - "version": "==8.8.0" + "index": "pypi", + "markers": "python_version >= '3.8'", + "version": "==8.12.3" }, "jedi": { "hashes": [ - "sha256:203c1fd9d969ab8f2119ec0a3342e0b49910045abe6af0a3ae83a5764d54639e", - "sha256:bae794c30d07f6d910d32a7048af09b5a39ed740918da923c6b780790ebac612" + "sha256:4770dc3de41bde3966b02eb84fbcf557fb33cce26ad23da12c742fb50ecb11f0", + "sha256:a8ef22bde8490f57fe5c7681a3c83cb58874daf72b4784de3cce5b6ef6edb5b9" ], "markers": "python_version >= '3.6'", - "version": "==0.18.2" + "version": "==0.19.2" }, "mako": { "hashes": [ @@ -167,27 +249,27 @@ }, "matplotlib-inline": { "hashes": [ - "sha256:f1f41aab5328aa5aaea9b16d083b128102f8712542f819fe7e6a420ff581b311", - "sha256:f887e5f10ba98e8d2b150ddcf4702c1e5f8b3a20005eb0f74bfdbd360ee6f304" + "sha256:8423b23ec666be3d16e16b60bdd8ac4e86e840ebd1dd11a30b9f117f2fa0ab90", + "sha256:df192d39a4ff8f21b1895d72e6a13f5fcc5099f00fa84384e0ea28c2cc0653ca" ], - "markers": "python_version >= '3.5'", - "version": "==0.1.6" + "markers": "python_version >= '3.8'", + "version": "==0.1.7" }, "parso": { "hashes": [ - "sha256:8c07be290bb59f03588915921e29e8a50002acaf2cdc5fa0e0114f91709fafa0", - "sha256:c001d4636cd3aecdaf33cbb40aebb59b094be2a74c556778ef5576c175e19e75" + "sha256:a418670a20291dacd2dddc80c377c5c3791378ee1e8d12bffc35420643d43f18", + "sha256:eb3a7b58240fb99099a345571deecc0f9540ea5f4dd2fe14c2a99d6b281ab92d" ], "markers": "python_version >= '3.6'", - "version": "==0.8.3" + "version": "==0.8.4" }, "pexpect": { "hashes": [ - "sha256:0b48a55dcb3c05f3329815901ea4fc1537514d6ba867a152b581d69ae3710937", - "sha256:fc65a43959d153d0114afe13997d439c22823a27cefceb5ff35c2178c6784c0c" + "sha256:7236d1e080e4936be2dc3e326cec0af72acf9212a7e1d060210e70a47e253523", + "sha256:ee7d41123f3c9911050ea2c2dac107568dc43b2d3b0c7557a33212c398ead30f" ], "markers": "sys_platform != 'win32'", - "version": "==4.8.0" + "version": "==4.9.0" }, "pickleshare": { "hashes": [ @@ -198,11 +280,11 @@ }, "prompt-toolkit": { "hashes": [ - "sha256:3e163f254bef5a03b146397d7c1963bd3e2812f0964bb9a24e6ec761fd28db63", - "sha256:aa64ad242a462c5ff0363a7b9cfe696c20d55d9fc60c11fd8e632d064804d305" + "sha256:52742911fde84e2d423e2f9a4cf1de7d7ac4e51958f648d9540e0fb8db077b07", + "sha256:931a162e3b27fc90c86f1b48bb1fb2c528c2761475e57c9c06de13311c7b54ed" ], - "markers": "python_full_version >= '3.6.2'", - "version": "==3.0.36" + "markers": "python_version >= '3.8'", + "version": "==3.0.51" }, "ptyprocess": { "hashes": [ @@ -213,18 +295,26 @@ }, "pure-eval": { "hashes": [ - "sha256:01eaab343580944bc56080ebe0a674b39ec44a945e6d09ba7db3cb8cec289350", - "sha256:2b45320af6dfaa1750f543d714b6d1c520a1688dec6fd24d339063ce0aaa9ac3" + "sha256:1db8e35b67b3d218d818ae653e27f06c3aa420901fa7b081ca98cbedc874e0d0", + "sha256:5f4e983f40564c576c7c8635ae88db5956bb2229d7e9237d03b3c0b0190eaf42" ], - "version": "==0.2.2" + "version": "==0.2.3" }, "pygments": { "hashes": [ - "sha256:b3ed06a9e8ac9a9aae5a6f5dbe78a8a58655d17b43b93c078f094ddc476ae297", - "sha256:fa7bd7bd2771287c0de303af8bfdfc731f51bd2c6a47ab69d117138893b82717" + "sha256:61c16d2a8576dc0649d9f39e089b5f02bcd27fba10d8fb4dcc28173f7a45151f", + "sha256:9ea1544ad55cecf4b8242fab6dd35a93bbce657034b0611ee383099054ab6d8c" ], - "markers": "python_version >= '3.6'", - "version": "==2.14.0" + "markers": "python_version >= '3.8'", + "version": "==2.19.1" + }, + "setuptools": { + "hashes": [ + "sha256:3c1383e1038b68556a382c1e8ded8887cd20141b0eb5708a6c8d277de49364f5", + "sha256:90ab613b6583fc02d5369cbca13ea26ea0e182d1df2d943ee9cbe81d4c61add9" + ], + "markers": "python_version >= '3.8'", + "version": "==75.3.2" }, "six": { "hashes": [ @@ -236,57 +326,82 @@ }, "sqlalchemy": { "hashes": [ - "sha256:07e48cbcdda6b8bc7a59d6728bd3f5f574ffe03f2c9fb384239f3789c2d95c2e", - "sha256:18cafdb27834fa03569d29f571df7115812a0e59fd6a3a03ccb0d33678ec8420", - "sha256:1b1e5e96e2789d89f023d080bee432e2fef64d95857969e70d3cadec80bd26f0", - "sha256:315676344e3558f1f80d02535f410e80ea4e8fddba31ec78fe390eff5fb8f466", - "sha256:31de1e2c45e67a5ec1ecca6ec26aefc299dd5151e355eb5199cd9516b57340be", - "sha256:3d94682732d1a0def5672471ba42a29ff5e21bb0aae0afa00bb10796fc1e28dd", - "sha256:3ec187acf85984263299a3f15c34a6c0671f83565d86d10f43ace49881a82718", - "sha256:4847f4b1d822754e35707db913396a29d874ee77b9c3c3ef3f04d5a9a6209618", - "sha256:4d112b0f3c1bc5ff70554a97344625ef621c1bfe02a73c5d97cac91f8cd7a41e", - "sha256:51e1ba2884c6a2b8e19109dc08c71c49530006c1084156ecadfaadf5f9b8b053", - "sha256:535377e9b10aff5a045e3d9ada8a62d02058b422c0504ebdcf07930599890eb0", - "sha256:5dbf17ac9a61e7a3f1c7ca47237aac93cabd7f08ad92ac5b96d6f8dea4287fc1", - "sha256:5f752676fc126edc1c4af0ec2e4d2adca48ddfae5de46bb40adbd3f903eb2120", - "sha256:64cb0ad8a190bc22d2112001cfecdec45baffdf41871de777239da6a28ed74b6", - "sha256:6913b8247d8a292ef8315162a51931e2b40ce91681f1b6f18f697045200c4a30", - "sha256:69fac0a7054d86b997af12dc23f581cf0b25fb1c7d1fed43257dee3af32d3d6d", - "sha256:7001f16a9a8e06488c3c7154827c48455d1c1507d7228d43e781afbc8ceccf6d", - "sha256:7b81b1030c42b003fc10ddd17825571603117f848814a344d305262d370e7c34", - "sha256:7f8267682eb41a0584cf66d8a697fef64b53281d01c93a503e1344197f2e01fe", - "sha256:887865924c3d6e9a473dc82b70977395301533b3030d0f020c38fd9eba5419f2", - "sha256:9167d4227b56591a4cc5524f1b79ccd7ea994f36e4c648ab42ca995d28ebbb96", - "sha256:939f9a018d2ad04036746e15d119c0428b1e557470361aa798e6e7d7f5875be0", - "sha256:955162ad1a931fe416eded6bb144ba891ccbf9b2e49dc7ded39274dd9c5affc5", - "sha256:984ee13543a346324319a1fb72b698e521506f6f22dc37d7752a329e9cd00a32", - "sha256:9883f5fae4fd8e3f875adc2add69f8b945625811689a6c65866a35ee9c0aea23", - "sha256:a1ad90c97029cc3ab4ffd57443a20fac21d2ec3c89532b084b073b3feb5abff3", - "sha256:a3714e5b33226131ac0da60d18995a102a17dddd42368b7bdd206737297823ad", - "sha256:ae067ab639fa499f67ded52f5bc8e084f045d10b5ac7bb928ae4ca2b6c0429a5", - "sha256:b33ffbdbbf5446cf36cd4cc530c9d9905d3c2fe56ed09e25c22c850cdb9fac92", - "sha256:b6e4cb5c63f705c9d546a054c60d326cbde7421421e2d2565ce3e2eee4e1a01f", - "sha256:b7f4b6aa6e87991ec7ce0e769689a977776db6704947e562102431474799a857", - "sha256:c04144a24103135ea0315d459431ac196fe96f55d3213bfd6d39d0247775c854", - "sha256:c522e496f9b9b70296a7675272ec21937ccfc15da664b74b9f58d98a641ce1b6", - "sha256:c5a99282848b6cae0056b85da17392a26b2d39178394fc25700bcf967e06e97a", - "sha256:c7a46639ba058d320c9f53a81db38119a74b8a7a1884df44d09fbe807d028aaf", - "sha256:d4b1cc7835b39835c75cf7c20c926b42e97d074147c902a9ebb7cf2c840dc4e2", - "sha256:d4d164df3d83d204c69f840da30b292ac7dc54285096c6171245b8d7807185aa", - "sha256:d61e9ecc849d8d44d7f80894ecff4abe347136e9d926560b818f6243409f3c86", - "sha256:d68e1762997bfebf9e5cf2a9fd0bcf9ca2fdd8136ce7b24bbd3bbfa4328f3e4a", - "sha256:e3c1808008124850115a3f7e793a975cfa5c8a26ceeeb9ff9cbb4485cac556df", - "sha256:f8cb80fe8d14307e4124f6fad64dfd87ab749c9d275f82b8b4ec84c84ecebdbe" + "sha256:023b3ee6169969beea3bb72312e44d8b7c27c75b347942d943cf49397b7edeb5", + "sha256:03968a349db483936c249f4d9cd14ff2c296adfa1290b660ba6516f973139582", + "sha256:05132c906066142103b83d9c250b60508af556982a385d96c4eaa9fb9720ac2b", + "sha256:087b6b52de812741c27231b5a3586384d60c353fbd0e2f81405a814b5591dc8b", + "sha256:0b3dbf1e7e9bc95f4bac5e2fb6d3fb2f083254c3fdd20a1789af965caf2d2348", + "sha256:118c16cd3f1b00c76d69343e38602006c9cfb9998fa4f798606d28d63f23beda", + "sha256:1936af879e3db023601196a1684d28e12f19ccf93af01bf3280a3262c4b6b4e5", + "sha256:1e3f196a0c59b0cae9a0cd332eb1a4bda4696e863f4f1cf84ab0347992c548c2", + "sha256:23a8825495d8b195c4aa9ff1c430c28f2c821e8c5e2d98089228af887e5d7e29", + "sha256:293cd444d82b18da48c9f71cd7005844dbbd06ca19be1ccf6779154439eec0b8", + "sha256:32f9dc8c44acdee06c8fc6440db9eae8b4af8b01e4b1aee7bdd7241c22edff4f", + "sha256:34ea30ab3ec98355235972dadc497bb659cc75f8292b760394824fab9cf39826", + "sha256:3d3549fc3e40667ec7199033a4e40a2f669898a00a7b18a931d3efb4c7900504", + "sha256:41836fe661cc98abfae476e14ba1906220f92c4e528771a8a3ae6a151242d2ae", + "sha256:4d44522480e0bf34c3d63167b8cfa7289c1c54264c2950cc5fc26e7850967e45", + "sha256:4eeb195cdedaf17aab6b247894ff2734dcead6c08f748e617bfe05bd5a218443", + "sha256:4f67766965996e63bb46cfbf2ce5355fc32d9dd3b8ad7e536a920ff9ee422e23", + "sha256:57df5dc6fdb5ed1a88a1ed2195fd31927e705cad62dedd86b46972752a80f576", + "sha256:598d9ebc1e796431bbd068e41e4de4dc34312b7aa3292571bb3674a0cb415dd1", + "sha256:5b14e97886199c1f52c14629c11d90c11fbb09e9334fa7bb5f6d068d9ced0ce0", + "sha256:5e22575d169529ac3e0a120cf050ec9daa94b6a9597993d1702884f6954a7d71", + "sha256:60c578c45c949f909a4026b7807044e7e564adf793537fc762b2489d522f3d11", + "sha256:6145afea51ff0af7f2564a05fa95eb46f542919e6523729663a5d285ecb3cf5e", + "sha256:6375cd674fe82d7aa9816d1cb96ec592bac1726c11e0cafbf40eeee9a4516b5f", + "sha256:6854175807af57bdb6425e47adbce7d20a4d79bbfd6f6d6519cd10bb7109a7f8", + "sha256:6ab60a5089a8f02009f127806f777fca82581c49e127f08413a66056bd9166dd", + "sha256:725875a63abf7c399d4548e686debb65cdc2549e1825437096a0af1f7e374814", + "sha256:7492967c3386df69f80cf67efd665c0f667cee67032090fe01d7d74b0e19bb08", + "sha256:81965cc20848ab06583506ef54e37cf15c83c7e619df2ad16807c03100745dea", + "sha256:81c24e0c0fde47a9723c81d5806569cddef103aebbf79dbc9fcbb617153dea30", + "sha256:81eedafa609917040d39aa9332e25881a8e7a0862495fcdf2023a9667209deda", + "sha256:81f413674d85cfd0dfcd6512e10e0f33c19c21860342a4890c3a2b59479929f9", + "sha256:8280856dd7c6a68ab3a164b4a4b1c51f7691f6d04af4d4ca23d6ecf2261b7923", + "sha256:82ca366a844eb551daff9d2e6e7a9e5e76d2612c8564f58db6c19a726869c1df", + "sha256:8b4af17bda11e907c51d10686eda89049f9ce5669b08fbe71a29747f1e876036", + "sha256:90144d3b0c8b139408da50196c5cad2a6909b51b23df1f0538411cd23ffa45d3", + "sha256:906e6b0d7d452e9a98e5ab8507c0da791856b2380fdee61b765632bb8698026f", + "sha256:90c11ceb9a1f482c752a71f203a81858625d8df5746d787a4786bca4ffdf71c6", + "sha256:911cc493ebd60de5f285bcae0491a60b4f2a9f0f5c270edd1c4dbaef7a38fc04", + "sha256:9a420a91913092d1e20c86a2f5f1fc85c1a8924dbcaf5e0586df8aceb09c9cc2", + "sha256:9f8c9fdd15a55d9465e590a402f42082705d66b05afc3ffd2d2eb3c6ba919560", + "sha256:a104c5694dfd2d864a6f91b0956eb5d5883234119cb40010115fd45a16da5e70", + "sha256:a373a400f3e9bac95ba2a06372c4fd1412a7cee53c37fc6c05f829bf672b8769", + "sha256:a62448526dd9ed3e3beedc93df9bb6b55a436ed1474db31a2af13b313a70a7e1", + "sha256:a8808d5cf866c781150d36a3c8eb3adccfa41a8105d031bf27e92c251e3969d6", + "sha256:b1f09b6821406ea1f94053f346f28f8215e293344209129a9c0fcc3578598d7b", + "sha256:b2ac41acfc8d965fb0c464eb8f44995770239668956dc4cdf502d1b1ffe0d747", + "sha256:b46fa6eae1cd1c20e6e6f44e19984d438b6b2d8616d21d783d150df714f44078", + "sha256:b50eab9994d64f4a823ff99a0ed28a6903224ddbe7fef56a6dd865eec9243440", + "sha256:bfc9064f6658a3d1cadeaa0ba07570b83ce6801a1314985bf98ec9b95d74e15f", + "sha256:c0b0e5e1b5d9f3586601048dd68f392dc0cc99a59bb5faf18aab057ce00d00b2", + "sha256:c153265408d18de4cc5ded1941dcd8315894572cddd3c58df5d5b5705b3fa28d", + "sha256:d4ae769b9c1c7757e4ccce94b0641bc203bbdf43ba7a2413ab2523d8d047d8dc", + "sha256:dc56c9788617b8964ad02e8fcfeed4001c1f8ba91a9e1f31483c0dffb207002a", + "sha256:dd5ec3aa6ae6e4d5b5de9357d2133c07be1aff6405b136dad753a16afb6717dd", + "sha256:edba70118c4be3c2b1f90754d308d0b79c6fe2c0fdc52d8ddf603916f83f4db9", + "sha256:ff8e80c4c4932c10493ff97028decfdb622de69cae87e0f127a7ebe32b4069c6" ], "index": "pypi", - "version": "==1.4.46" + "markers": "python_version >= '3.7'", + "version": "==2.0.41" }, "stack-data": { "hashes": [ - "sha256:32d2dd0376772d01b6cb9fc996f3c8b57a357089dec328ed4b6553d037eaf815", - "sha256:cbb2a53eb64e5785878201a97ed7c7b94883f48b87bfb0bbe8b623c74679e4a8" + "sha256:836a778de4fec4dcd1dcd89ed8abff8a221f58308462e1c4aa2a3cf30148f0b9", + "sha256:d5558e0c25a4cb0853cddad3d77da9891a08cb85dd9f9f91b9f8cd66e511e695" + ], + "version": "==0.6.3" + }, + "toml": { + "hashes": [ + "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b", + "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f" ], - "version": "==0.6.2" + "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2'", + "version": "==0.10.2" }, "tomli": { "hashes": [ @@ -298,18 +413,26 @@ }, "traitlets": { "hashes": [ - "sha256:32500888f5ff7bbf3b9267ea31748fa657aaf34d56d85e60f91dda7dc7f5785b", - "sha256:a1ca5df6414f8b5760f7c5f256e326ee21b581742114545b462b35ffe3f04861" + "sha256:9ed0579d3502c94b4b3732ac120375cda96f923114522847de4b3bb98b96b6b7", + "sha256:b74e89e397b1ed28cc831db7aea759ba6640cb3de13090ca145426688ff1ac4f" ], - "markers": "python_version >= '3.7'", - "version": "==5.8.1" + "markers": "python_version >= '3.8'", + "version": "==5.14.3" + }, + "typing-extensions": { + "hashes": [ + "sha256:a439e7c04b49fec3e5d3e2beaa21755cadbbdc391694e28ccdd36ca4a1408f8c", + "sha256:e6c81219bd689f51865d9e372991c540bda33a0379d5573cddb9a3a23f7caaef" + ], + "markers": "python_version >= '3.8'", + "version": "==4.13.2" }, "wcwidth": { "hashes": [ - "sha256:795b138f6875577cd91bba52baf9e445cd5118fd32723b460e30a0af30ea230e", - "sha256:a5220780a404dbe3353789870978e472cfe477761f06ee55077256e509b156d0" + "sha256:3da69048e4540d84af32131829ff948f1e022c1c6bdb8d6102117aac784f6859", + "sha256:72ea0c06399eb286d978fdedb6923a9eb47e1c486ce63e9b4e64fc18303972b5" ], - "version": "==0.2.6" + "version": "==0.2.13" }, "zipp": { "hashes": [ diff --git a/alembic.ini b/alembic.ini new file mode 100644 index 000000000..08a39b2ed --- /dev/null +++ b/alembic.ini @@ -0,0 +1,106 @@ +# A generic, single database configuration. + +[alembic] +# path to migration scripts +script_location = alembic + +# template used to generate migration file names; The default value is %%(rev)s_%%(slug)s +# Uncomment the line below if you want the files to be prepended with date and time +# see https://alembic.sqlalchemy.org/en/latest/tutorial.html#editing-the-ini-file +# for all available tokens +# file_template = %%(year)d_%%(month).2d_%%(day).2d_%%(hour).2d%%(minute).2d-%%(rev)s_%%(slug)s + +# sys.path path, will be prepended to sys.path if present. +# defaults to the current working directory. +prepend_sys_path = . + +# timezone to use when rendering the date within the migration file +# as well as the filename. +# If specified, requires the python-dateutil library that can be +# installed by adding `alembic[tz]` to the pip requirements +# string value is passed to dateutil.tz.gettz() +# leave blank for localtime +# timezone = + +# max length of characters to apply to the +# "slug" field +# truncate_slug_length = 40 + +# set to 'true' to run the environment during +# the 'revision' command, regardless of autogenerate +# revision_environment = false + +# set to 'true' to allow .pyc and .pyo files without +# a source .py file to be detected as revisions in the +# versions/ directory +# sourceless = false + +# version location specification; This defaults +# to alembic/versions. When using multiple version +# directories, initial revisions must be specified with --version-path. +# The path separator used here should be the separator specified by "version_path_separator" below. +# version_locations = %(here)s/bar:%(here)s/bat:alembic/versions + +# version path separator; As mentioned above, this is the character used to split +# version_locations. The default within new alembic.ini files is "os", which uses os.pathsep. +# If this key is omitted entirely, it falls back to the legacy behavior of splitting on spaces and/or commas. +# Valid values for version_path_separator are: +# +# version_path_separator = : +# version_path_separator = ; +# version_path_separator = space +version_path_separator = os # Use os.pathsep. Default configuration used for new projects. + +# the output encoding used when revision files +# are written from script.py.mako +# output_encoding = utf-8 + +sqlalchemy.url = mysql+pymysql://username:password@localhost/dbname + + + +[post_write_hooks] +# post_write_hooks defines scripts or Python functions that are run +# on newly generated revision scripts. See the documentation for further +# detail and examples + +# format using "black" - use the console_scripts runner, against the "black" entrypoint +# hooks = black +# black.type = console_scripts +# black.entrypoint = black +# black.options = -l 79 REVISION_SCRIPT_FILENAME + +# Logging configuration +[loggers] +keys = root,sqlalchemy,alembic + +[handlers] +keys = console + +[formatters] +keys = generic + +[logger_root] +level = WARN +handlers = console +qualname = + +[logger_sqlalchemy] +level = WARN +handlers = +qualname = sqlalchemy.engine + +[logger_alembic] +level = INFO +handlers = +qualname = alembic + +[handler_console] +class = StreamHandler +args = (sys.stderr,) +level = NOTSET +formatter = generic + +[formatter_generic] +format = %(levelname)-5.5s [%(name)s] %(message)s +datefmt = %H:%M:%S diff --git a/alembic/README b/alembic/README new file mode 100644 index 000000000..98e4f9c44 --- /dev/null +++ b/alembic/README @@ -0,0 +1 @@ +Generic single-database configuration. \ No newline at end of file diff --git a/alembic/env.py b/alembic/env.py new file mode 100644 index 000000000..6626bfd0c --- /dev/null +++ b/alembic/env.py @@ -0,0 +1,78 @@ +from logging.config import fileConfig + +from sqlalchemy import engine_from_config +from sqlalchemy import pool + +from alembic import context + +# this is the Alembic Config object, which provides +# access to the values within the .ini file in use. +config = context.config + +# Interpret the config file for Python logging. +# This line sets up loggers basically. +if config.config_file_name is not None: + fileConfig(config.config_file_name) + +# add your model's MetaData object here +# for 'autogenerate' support +# from myapp import mymodel +# target_metadata = mymodel.Base.metadata +target_metadata = None + +# other values from the config, defined by the needs of env.py, +# can be acquired: +# my_important_option = config.get_main_option("my_important_option") +# ... etc. + + +def run_migrations_offline() -> None: + """Run migrations in 'offline' mode. + + This configures the context with just a URL + and not an Engine, though an Engine is acceptable + here as well. By skipping the Engine creation + we don't even need a DBAPI to be available. + + Calls to context.execute() here emit the given string to the + script output. + + """ + url = config.get_main_option("sqlalchemy.url") + context.configure( + url=url, + target_metadata=target_metadata, + literal_binds=True, + dialect_opts={"paramstyle": "named"}, + ) + + with context.begin_transaction(): + context.run_migrations() + + +def run_migrations_online() -> None: + """Run migrations in 'online' mode. + + In this scenario we need to create an Engine + and associate a connection with the context. + + """ + connectable = engine_from_config( + config.get_section(config.config_ini_section), + prefix="sqlalchemy.", + poolclass=pool.NullPool, + ) + + with connectable.connect() as connection: + context.configure( + connection=connection, target_metadata=target_metadata + ) + + with context.begin_transaction(): + context.run_migrations() + + +if context.is_offline_mode(): + run_migrations_offline() +else: + run_migrations_online() diff --git a/alembic/script.py.mako b/alembic/script.py.mako new file mode 100644 index 000000000..55df2863d --- /dev/null +++ b/alembic/script.py.mako @@ -0,0 +1,24 @@ +"""${message} + +Revision ID: ${up_revision} +Revises: ${down_revision | comma,n} +Create Date: ${create_date} + +""" +from alembic import op +import sqlalchemy as sa +${imports if imports else ""} + +# revision identifiers, used by Alembic. +revision = ${repr(up_revision)} +down_revision = ${repr(down_revision)} +branch_labels = ${repr(branch_labels)} +depends_on = ${repr(depends_on)} + + +def upgrade() -> None: + ${upgrades if upgrades else "pass"} + + +def downgrade() -> None: + ${downgrades if downgrades else "pass"} diff --git a/freebies.db b/freebies.db new file mode 100644 index 0000000000000000000000000000000000000000..c1d2b6be2f890ce93ecd058fed219d0d8d09038b GIT binary patch literal 16384 zcmeI#&1%9x5C`zY2vUpKi+B>}XrT1c7icQ0m8P*>(}IT*qFGC*(c%|)w9rEzrtjFS zX-tz+y~}@L**K0n<8O1>-h=5c1dXP%z+KR#a;B)Naz&&liWbg%IOCE1tj1sLSvBQi z^;rrNw9=PSo|Q}Cf&l>tKmY;|fB*y_009U<00M^u-k-HXwN_ICLwaF-sQlog{d(^RP)@Gcnu1(gXX`WLUUWmY%xPhQ2gEwysE`uv~ycBV| z40vMYBpb-ca2mY1lXWI~&pUSp@6#>oQ^WUknc3`GJ)0Y0aT>ikaZ66LrCjFJ>T=d; zTk>a(D6SJ{EykJEWIfvE9M~@D$B7<8l+?aSLTc|MSr#bj!(5?Ssi+%$%PT9(sijz> ze*PuRYGzT2`sP$ERO@whBeqp`J?tKGsd~Hco^#9=9Zi=L&!4