-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathquack.py
121 lines (89 loc) · 3.2 KB
/
quack.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
"""
Quack
Mock and SQLAlchemy but not horrible.
"""
import mock
from sqlalchemy.orm.session import Session
from sqlalchemy.orm.query import Query
from sqlalchemy.orm.attributes import InstrumentedAttribute
def overridable(func):
"""Decorator used to create overridable methods.
Creates a property that returns a MagicMock. When called, the mock
returns either the result of calling the decorated method or
mock.return_value if it was overridden.
"""
instances = {}
@property
def prop(self):
"""Property that associates mocks with instances."""
try:
return instances[self]
except KeyError:
attr = mock.MagicMock()
def wrapper(*args, **kwargs):
"""Either call the decorated method or return reuturn_value."""
if attr._mock_return_value is mock.DEFAULT:
return func(self, *args, **kwargs)
else:
return mock.DEFAULT
attr.side_effect = wrapper
instances[self] = attr
return attr
return prop
class MockSession(mock.MagicMock):
"""A mock SQLAlchemy session.
The query method returns a special mock Query.
"""
def __init__(self, spec=Session, *args, **kwargs):
"""Use Session as the default spec."""
super(MockSession, self).__init__(spec=spec, *args, **kwargs)
self._mock_queries = []
@overridable
def query(self, *args):
query = MockQuery(entities=args)
self._mock_queries.append(query)
return query
@property
def mock_queries(self):
return self._mock_queries
class MockQuery(mock.MagicMock):
"""A mock SQLAlchemy query.
All methods return the query to allow chaining.
"""
def __init__(self, entities=None, spec=Query, *args, **kwargs):
"""Use Query as the default spec."""
super(MockQuery, self).__init__(spec=spec, *args, **kwargs)
self._mock_entities = entities
def __getattr__(self, name):
"""Get attributes that return self when called.
This can be overridden by setting the attribute's return_value.
"""
attr = super(MockQuery, self).__getattr__(name)
if attr._mock_return_value is mock.DEFAULT:
attr.return_value = self
return attr
@overridable
def all(self):
"""Get a list of all items."""
if not self._mock_entities:
return self.all.return_value
return [mock.MagicMock(spec=self._mock_entities)
for i in range(10)]
@overridable
def first(self):
"""Get the first item from a query result."""
if not self._mock_entities:
return self.first.return_value
return mock.MagicMock(spec=self._mock_entities)
@overridable
def get(self, pk):
"""Get a single item by its primary key."""
if not self._mock_entities:
return self.get.return_value
return mock.MagicMock(spec=self._mock_entities)
@overridable
def one(self):
"""Get a single item."""
if not self._mock_entities:
return self.one.return_value
return mock.MagicMock(spec=self._mock_entities)