Skip to content

busykoala/sqlalchemy-hero

Repository files navigation

SQLAlchemy Hero

What is SQLAlchemy Hero?

SchemaHero lets you define declarative database schema migrations. In Python schemas are often mapped with SQLAlchemy models.

SQLAlchemy Hero is a package to generate the declarative table YAMLs for SchemaHero from your existing SQLAlchemy Base.

Getting Started

In easy cases the below code is all you need to add to a cli script. The Base is your DeclarativeBase from SQLAlchemy and ./out is the path to output the generated table schemas.

from pathlib import Path

from sqlalchemy_hero.hero_database import HeroDatabase
from sqlalchemy_hero.hero_generator import HeroGenerator

from my_app.models import Base

hero_generator = HeroGenerator(
    base=Base,
    db_type=HeroDatabase.postgres,
    namespace="hero-ns",
    database="hero-db",
)
hero_generator.to_yaml_files(out_path=Path("./out"))

Example with SQLAlchemy models

Let's look at an example how to use it including some example models. There is a Parent and a Child model both inheriting from a abstract base model with some common fields. The goal is to generate the table YAML files for SchemaHero.

The HeroGenerator class implements the methods to extract the model. It is initialized with your declarative base from SQLAlchemy, the database type you're using (currently postgres and mysql), the namespace where the tables should be deployed and the database name.

The to_yaml_files method allowes to specify the output path for the generated YAMLs (defaults to Path("./out")).

from pathlib import Path

from sqlalchemy import Column
from sqlalchemy import ForeignKey
from sqlalchemy import Integer
from sqlalchemy import Text
from sqlalchemy import func
from sqlalchemy.orm import declarative_base
from sqlalchemy.orm import relationship
from sqlalchemy_utils import ArrowType

from sqlalchemy_hero.hero_database import HeroDatabase
from sqlalchemy_hero.hero_generator import HeroGenerator

DeclarativeBase = declarative_base()


class Base(DeclarativeBase):
    __abstract__ = True

    id = Column(Integer, primary_key=True)
    created_on = Column(ArrowType, default=func.now())
    updated_on = Column(ArrowType, default=func.now(), onupdate=func.now())


class Parent(Base):
    __tablename__ = "parent"

    name = Column(Text)
    ss_number = Column(Integer, autoincrement=True)
    children = relationship("Child")


class Child(Base):
    __tablename__ = "child"

    name = Column(Text, nullable=False)
    description = Column(Text, nullable=False, index=True)
    parent_id = Column(Integer, ForeignKey("parent.id"))


hero_generator = HeroGenerator(
    base=Base,
    db_type=HeroDatabase.postgres,
    namespace="hero-ns",
    database="hero-db",
)
hero_generator.to_yaml_files(out_path=Path("./out"))

Type Overrides

The library tries to implement the most common types but it's hard to keep up with all the possiblities for the different databases. If you find a not yet mapped type (commonly used with SQLAlchemy) please open a pull request to add it.

For custom types or quick fixes you can override the types (the dict entries override/add to the current types).

CUSTOM_TYPE_MAPPINGS = {
    MyCustomType: "text",  # add new type mappings
    Integer: "serial",  # override existing mappings
}

hero_generator = HeroGenerator(
    base=Base,
    db_type=HeroDatabase.postgres,
    namespace="hero-ns",
    database="hero-db",
    db_type_override=CUSTOM_TYPE_MAPPINGS,  # add the mappings on init
)
hero_generator.to_yaml_files()

API Version

We try to update the default API version for SchemaHero to the latest. If you wish to use another version or if we haven't updated yet it can be specified on initializing the HeroGenerator.

hero_generator = HeroGenerator(
    base=Base,
    db_type=HeroDatabase.postgres,
    namespace="hero-ns",
    database="hero-db",
    api_version="schemas.schemahero.io/custom-version",
)
hero_generator.to_yaml_files()

QA Commands

The below commands are run in the pipeline and the according checks are expected to pass.

poetry run pytest
poetry run black .
poetry run isort .
poetry run pylint tests sqlalchemy_hero
poetry run bandit -r sqlalchemy_hero

About

SchemaHero table generator from SQLAlchemy models.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages