Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/miki725/alchemy-mock
SQLAlchemy mock helpers.
https://github.com/miki725/alchemy-mock
mock sqlalchemy testing
Last synced: 5 days ago
JSON representation
SQLAlchemy mock helpers.
- Host: GitHub
- URL: https://github.com/miki725/alchemy-mock
- Owner: miki725
- License: other
- Created: 2018-01-12T04:00:09.000Z (about 7 years ago)
- Default Branch: master
- Last Pushed: 2021-01-07T23:56:50.000Z (about 4 years ago)
- Last Synced: 2025-01-15T19:14:37.314Z (12 days ago)
- Topics: mock, sqlalchemy, testing
- Language: Python
- Homepage:
- Size: 62.5 KB
- Stars: 81
- Watchers: 6
- Forks: 15
- Open Issues: 17
-
Metadata Files:
- Readme: README.rst
- Changelog: HISTORY.rst
- License: LICENSE.rst
Awesome Lists containing this project
README
===============
SQLAlchemy Mock
===============.. image:: https://badge.fury.io/py/alchemy-mock.png
:target: http://badge.fury.io/py/alchemy-mock.. image:: https://travis-ci.org/miki725/alchemy-mock.png?branch=master
:target: https://travis-ci.org/miki725/alchemy-mock.. image:: https://coveralls.io/repos/miki725/alchemy-mock/badge.png?branch=master
:target: https://coveralls.io/r/miki725/alchemy-mock?branch=masterSQLAlchemy mock helpers.
* Free software: MIT license
* GitHub: https://github.com/miki725/alchemy-mockInstalling
----------You can install ``alchemy-mock`` using pip::
$ pip install alchemy-mock
Why?
----SQLAlchemy is awesome. Unittests are great.
Accessing DB during tests - not so much.
This library provides easy way to mock SQLAlchemy's session
in unittests while preserving ability to do sane asserts.
Normally SQLAlchemy's expressions cannot be easily compared
as comparison on binary expression produces yet another binary expression::>>> type((Model.foo == 5) == (Model.bar == 5))
But they can be compared with this library::
>>> ExpressionMatcher(Model.foo == 5) == (Model.bar == 5)
FalseUsing
-----``ExpressionMatcher`` can be directly used::
>>> from alchemy_mock.comparison import ExpressionMatcher
>>> ExpressionMatcher(Model.foo == 5) == (Model.foo == 5)
TrueAlternatively ``AlchemyMagicMock`` can be used to mock out SQLAlchemy session::
>>> from alchemy_mock.mocking import AlchemyMagicMock
>>> session = AlchemyMagicMock()
>>> session.query(Model).filter(Model.foo == 5).all()>>> session.query.return_value.filter.assert_called_once_with(Model.foo == 5)
In real world though session can be interacted with multiple times to query some data.
In those cases ``UnifiedAlchemyMagicMock`` can be used which combines various calls for easier assertions::>>> from alchemy_mock.mocking import UnifiedAlchemyMagicMock
>>> session = UnifiedAlchemyMagicMock()>>> m = session.query(Model)
>>> q = m.filter(Model.foo == 5)
>>> if condition:
... q = q.filter(Model.bar > 10).all()
>>> data1 = q.all()
>>> data2 = m.filter(Model.note == 'hello world').all()>>> session.filter.assert_has_calls([
... mock.call(Model.foo == 5, Model.bar > 10),
... mock.call(Model.note == 'hello world'),
... ])Also real-data can be stubbed by criteria::
>>> from alchemy_mock.mocking import UnifiedAlchemyMagicMock
>>> session = UnifiedAlchemyMagicMock(data=[
... (
... [mock.call.query(Model),
... mock.call.filter(Model.foo == 5, Model.bar > 10)],
... [Model(foo=5, bar=11)]
... ),
... (
... [mock.call.query(Model),
... mock.call.filter(Model.note == 'hello world')],
... [Model(note='hello world')]
... ),
... (
... [mock.call.query(AnotherModel),
... mock.call.filter(Model.foo == 5, Model.bar > 10)],
... [AnotherModel(foo=5, bar=17)]
... ),
... ])
>>> session.query(Model).filter(Model.foo == 5).filter(Model.bar > 10).all()
[Model(foo=5, bar=11)]
>>> session.query(Model).filter(Model.note == 'hello world').all()
[Model(note='hello world')]
>>> session.query(AnotherModel).filter(Model.foo == 5).filter(Model.bar > 10).all()
[AnotherModel(foo=5, bar=17)]
>>> session.query(AnotherModel).filter(Model.note == 'hello world').all()
[]Finally ``UnifiedAlchemyMagicMock`` can partially fake session mutations
such as ``session.add(instance)``. For example::>>> session = UnifiedAlchemyMagicMock()
>>> session.add(Model(pk=1, foo='bar'))
>>> session.add(Model(pk=2, foo='baz'))
>>> session.query(Model).all()
[Model(foo='bar'), Model(foo='baz')]
>>> session.query(Model).get(1)
Model(foo='bar')
>>> session.query(Model).get(2)
Model(foo='baz')Note that its partially correct since if added models are filtered on,
session is unable to actually apply any filters so it returns everything::>>> session.query(Model).filter(Model.foo == 'bar').all()
[Model(foo='bar'), Model(foo='baz')]