Py YAML Fixtures¶
Py YAML Fixtures¶
A (work-in-progress) library for loading database fixtures written in Jinja2-templated YAML files. It comes with support for faker and relationships between fixture objects. Currently it works with the following packages:
- Django 2+
- Flask SQLAlchemy
- Flask Unchained
- Standalone SQLAlchemy
Requires Python 3.5+
Table of Contents¶
Fixture File Syntax¶
First, let’s define some example models to work with:
class Parent(BaseModel):
__tablename__ = 'parent'
id = sa.Column(sa.Integer, primary_key=True)
name = sa.Column(sa.String)
children = relationship('Child', back_populates='parent')
class Child(BaseModel):
__tablename__ = 'child'
id = sa.Column(sa.Integer, primary_key=True)
name = sa.Column(sa.String)
parent_id = sa.Column(sa.Integer, sa.ForeignKey('parent.id'))
parent = relationship('Parent', back_populates='children')
To populate these models with fixtures data, you can either create a single fixtures.yaml
file in your fixtures directory where the top-level keys are the model class names:
# db/fixtures/fixtures.yaml
Child:
alice:
name: Alice
bob:
name: Bob
grace:
name: Grace
judy:
name: Judy
Parent:
parent1:
name: Parent 1
children: ['Child(alice)', 'Child(bob)']
parent2:
name: Parent 2
children:
- 'Child(grace)'
- 'Child(judy)'
Or you can create YAML files named after each model’s class name (Parent
and Child
in our case). For example:
# db/fixtures/Child.yaml
alice:
name: Alice
bob:
name: Bob
grace:
name: Grace
judy:
name: Judy
# db/fixtures/Parent.yaml
parent1:
name: Parent 1
children: ['Child(alice)', 'Child(bob)']
parent2:
name: Parent 2
children:
- 'Child(grace)'
- 'Child(judy)'
Relationships¶
The top-level YAML keys (alice
, bob
, grace
, judy
, parent1
, parent2
) are unique ids used to reference objects in relationships. They must be unique across all model fixtures.
To reference them, we use an identifier string. An identifier string consists of two parts: the class name, and one or more ids. For singular relationships the notation is 'ModelClassName(id)'
. For the many-side of relationships, the notation is the same, just combined with YAML’s list syntax:
# db/fixtures/Parent.yaml
parent1:
name: Parent 1
children: ['Child(alice)', 'Child(bob)']
parent2:
name: Parent 2
children:
- 'Child(grace)'
- 'Child(judy)'
# or in short-hand notation
parent3:
name: Parent 3
children: ['Child(alice, bob)']
# technically, as long as there are at least 2 ids in the identifier string,
# then even the YAML list syntax is optional, and you can write stuff like this:
parent4:
name: Parent 4
children: Child(alice, bob)
# or spanning multiple lines:
parent5:
name: Parent 5
children: >
Child(
grace,
judy,
)
Faker and Jinja Templating¶
All of the YAML fixtures files are rendered by Jinja before getting loaded. This means you have full access to the Jinja environment, and can use things like faker
, range
and random
:
# db/fixtures/Child.yaml
{% for i in range(0, 20) %}
child{{ i }}:
name: {{ faker.name() }}
{% endfor %}
# db/fixtures/Parent.yaml
{% for i in range(0, 10) %}
parent{{ i }}:
name: {{ faker.name() }}
children: {{ random_models('Child', 0, range(0, 4)|random) }}
{% endfor %}
There are also two included Jinja helper functions:
random_model(model_name: str)
- For example, to get one random
Child
model:{{ random_model('Child') }}
- For example, to get one random
random_models(model_name: str, min_count: int = 0, max_count: int = 3)
- For example, to get a list of 0 to 3
Child
models:{{ random_models('Child') }}
- For example, to get a list of 1 to 4
Child
models:{{ random_models('Child', 1, 4) }}
- For example, to get a list of 0 to 3
Installation¶
# to use with django
pip install py-yaml-fixtures[django]
# to use with flask-sqlalchemy
pip install py-yaml-fixtures[flask-sqlalchemy]
# to use with flask-unchained
pip install py-yaml-fixtures[flask-unchained]
# to use with standalone sqlalchemy
pip install py-yaml-fixtures[sqlalchemy]
Configuration¶
With Django¶
Add py_yaml_fixtures
to your settings.INSTALLED_APPS
.
The py_yaml_fixtures
app adds one command: manage.py import_fixtures
. It looks for fixture files in every app configured in your settings.INSTALLED_APPS
that has a fixtures
folder. For example:
# example folder structure:
# app
project-root/app/fixtures/
project-root/app/fixtures/ModelOne.yaml
# blog
project-root/blog/fixtures/
project-root/blog/fixtures/ModelTwo.yaml
# auth
project-root/auth/fixtures/
project-root/auth/fixtures/ModelThree.yaml
project-root/auth/fixtures/ModelFour.yaml
# project-root/app/settings.py
INSTALLED_APPS = [
# ...
'py_yaml_fixtures',
'auth',
'blog',
'app',
]
To load the model fixtures into the database, you would run:
cd your-django-project-root
# to load fixtures from all apps
./manage.py import_fixtures
# or to load fixtures from specific apps
./manage.py import_fixtures app blog
With Flask and Flask-SQLAlchemy¶
This is the minimal setup required to make a Flask cli command available to import fixtures, by default, flask import-fixtures
:
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from py_yaml_fixtures.flask import PyYAMLFixtures
app = Flask(__name__)
db = SQLAlchemy(app)
# optional configuration settings (these are all the defaults):
app.config['FLASK_MODELS_MODULE'] = 'app.models' # where all of your model classes are imported
app.config['PY_YAML_FIXTURES_DIR'] = 'db/fixtures' # where your fixtures file(s) live
app.config['PY_YAML_FIXTURES_COMMAND_NAME'] = 'import-fixtures' # the name of the CLI command
fixtures = PyYAMLFixtures(app) # instantiate the PyYAMLFixtures Flask Extension
After creating fixture files in the configured PY_YAML_FIXTURES_DIR
, you would then be able to run flask import-fixtures
to load the fixtures into the database.
With Flask Unchained¶
Add py_yaml_fixtures
to your unchained_config.BUNDLES
.
The PyYAMLFixtures bundle adds one command to Flask Unchained: flask db import-fixtures
. It looks for fixture files in each bundle’s fixtures
folder (if it exists). For example:
# example folder structure:
# app
project-root/app/fixtures/
project-root/app/fixtures/ModelOne.yaml
# blog_bundle
project-root/bundles/blog/fixtures/
project-root/bundles/blog/fixtures/Post.yaml
# security_bundle
project-root/bundles/security/fixtures/
project-root/bundles/security/fixtures/User.yaml
project-root/bundles/security/fixtures/Role.yaml
# project-root/unchained_config.py
BUNDLES = [
# ...
'flask_unchained.bundles.sqlalchemy',
'py_yaml_fixtures',
'bundles.blog',
'bundles.security',
'app',
]
To load the model fixtures into the database, you would run:
cd your-flask-unchained-project-root
# to load fixtures from all bundles
flask db import-fixtures
# or to load fixtures from specific bundles
flask db import-fixtures app security_bundle
With Standalone SQLAlchemy¶
import sqlalchemy as sa
from py_yaml_fixtures import FixturesLoader
from py_yaml_fixtures.factories.sqlalchemy import SQLAlchemyModelFactory
from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import relationship, sessionmaker
PY_YAML_FIXTURES_DIR = 'db/fixtures'
BaseModel = declarative_base()
class Parent(BaseModel):
__tablename__ = 'parent'
id = sa.Column(sa.Integer, primary_key=True)
name = sa.Column(sa.String)
children = relationship('Child', back_populates='parent')
class Child(BaseModel):
__tablename__ = 'child'
id = sa.Column(sa.Integer, primary_key=True)
name = sa.Column(sa.String)
parent_id = sa.Column(sa.Integer, sa.ForeignKey('parent.id'))
parent = relationship('Parent', back_populates='children')
# first we need a list of our model classes to provide to the factory
model_classes = [Parent, Child]
# and we need a session connected to the database, also for the factory
engine = create_engine('sqlite:///:memory:')
Session = sessionmaker()
Session.configure(bind=engine)
session = Session()
# then we create the factory, and pass it to the fixtures loader
factory = SQLAlchemyModelFactory(session, model_classes)
loader = FixturesLoader(factory, fixture_dirs=[PY_YAML_FIXTURES_DIR])
# to create all the fixtures in the database, we have to call loader.create_all()
if __name__ == '__main__':
# create the tables in the database
BaseModel.metadata.create_all(bind=engine)
# and use the loader to import the fixtures data into the database
loader.create_all(lambda identifier, model, created: print(
'{action} {identifier}: {model}'.format(
action='Creating' if created else 'Updating',
identifier=identifier.key,
model=repr(model)
)))
Contributing¶
Contributions are welcome!
- Please file bug reports as GitHub issues.
- Or even better, open a pull request with the fix!
Adding support for other ORMs¶
You must implement a concrete factory by extending py_yaml_fixtures.FactoryInterface
. There are three abstract methods that must be implemented: create_or_update
, get_relationships
, and maybe_convert_values
(see the DjangoModelFactory and SQLAlchemyModelFactory implementations as examples).
License¶
MIT
API Documentation¶
FixturesLoader¶
-
class
py_yaml_fixtures.
FixturesLoader
(factory: py_yaml_fixtures.factories.factory_interface.FactoryInterface, fixture_dirs: List[str], env: Optional[jinja2.environment.Environment] = None)[source]¶ The factory “driver” class. Does most of the hard work of loading fixtures, leaving the responsibility of model instantiation up to the factory class passed in.
Parameters: - factory – An instance of the concrete factory to use for creating models
- fixture_dirs – A list of directory paths to load fixtures templates from
- env – An optional jinja environment (the default one will include faker as a template global, but if you want to customize its tags/filters/etc, then you need to create an env yourself - the correct loader will be set automatically for you)
-
env
= None¶ The Jinja Environment used for rendering the yaml template files.
-
factory
= None¶ The factory instance.
-
fixture_dirs
= None¶ A list of directories where fixture files should be loaded from.
-
relationships
= None¶ A dict keyed by model name where values are a list of related model names.
-
model_fixtures
= None¶ A dict of models names to their semi-processed data from the yaml files.
-
create_all
(progress_callback: Optional[callable] = None) → Dict[str, object][source]¶ Creates all the models discovered from fixture files in
fixtures_dir
.Parameters: progress_callback – An optional function to track progress. It must take three parameters:
- an
Identifier
- the model instance
- and a boolean specifying whether the model was created
Returns: A dictionary keyed by identifier where the values are model instances. - an
FactoryInterface¶
-
class
py_yaml_fixtures.
FactoryInterface
[source]¶ Abstract base class for ORM factories. Extend this base class to add support for a database ORM.
-
create_or_update
(identifier: py_yaml_fixtures.types.Identifier, data: Dict[str, Any]) → Tuple[object, bool][source]¶ Create or update a model.
Parameters: - identifier – An object with
class_name
andkey
attributes - data – A dictionary keyed by column name, with values being the converted values to set on the model instance
Returns: A two-tuple of model instance and whether or not it was created.
- identifier – An object with
-
get_relationships
(class_name: str) → Set[str][source]¶ Return a list of model attribute names that could have relationships for the given model class name.
Parameters: class_name – The name of the class name to discover relationships for. Returns: A set of model attribute names.
-
maybe_convert_values
(identifier: py_yaml_fixtures.types.Identifier, data: Dict[str, Any]) → Dict[str, Any][source]¶ Takes a dictionary of raw values for a specific identifier, as parsed from the YAML file, and depending upon the type of db column the data is meant for, decides what to do with the value (eg leave it alone, convert a string to a date/time instance, or convert identifiers to model instances by calling
self.loader.convert_identifiers()
)Parameters: - identifier – An object with
class_name
andkey
attributes - data – A dictionary keyed by column name, with values being the raw values as parsed from the YAML
Returns: A dictionary keyed by column name, with values being the converted values meant to be set on the model instance
- identifier – An object with
-
Jinja Helper Functions¶
random_model¶
-
py_yaml_fixtures.utils.
random_model
(ctx, model_class_name)[source]¶ Get a random model identifier by class name. For example:
# db/fixtures/Category.yml {% for i in range(0, 10) %} category{{ i }}: name: {{ faker.name() }} {% endfor %} # db/fixtures/Post.yml a_blog_post: category: {{ random_model('Category') }}
Will render to something like the following:
# db/fixtures/Post.yml (rendered) a blog_post: category: "Category(category7)"
Parameters: - ctx – The context variables of the current template (passed automatically)
- model_class_name – The class name of the model to get.
random_models¶
-
py_yaml_fixtures.utils.
random_models
(ctx, model_class_name, min_count=0, max_count=3)[source]¶ Get a random model identifier by class name. Example usage:
# db/fixtures/Tag.yml {% for i in range(0, 10) %} tag{{ i }}: name: {{ faker.name() }} {% endfor %} # db/fixtures/Post.yml a_blog_post: tags: {{ random_models('Tag') }}
Will render to something like the following:
# db/fixtures/Post.yml (rendered) a blog_post: tags: ["Tag(tag2, tag5)"]
Parameters: - ctx – The context variables of the current template (passed automatically)
- model_class_name – The class name of the models to get.
- min_count – The minimum number of models to return.
- max_count – The maximum number of models to return.