forked from TestInABox/stackInABox
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdecorator.py
132 lines (106 loc) · 4 KB
/
decorator.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
122
123
124
125
126
127
128
129
130
131
132
"""
Stack-In-A-Box: Responses Support via decorator
"""
try:
import collections.abc as collections
except ImportError:
import collections
import functools
import logging
import re
import types
import responses
import six
from stackinabox.services.service import StackInABoxService
from stackinabox.stack import StackInABox
from stackinabox.util.responses.core import (
responses_registration
)
from stackinabox.util import deprecator
from stackinabox.util.tools import CaseInsensitiveDict
logger = logging.getLogger(__name__)
class activate(object):
"""
Decorator class to make use of Responses and Stack-In-A-Box
extremely simple to do.
"""
def __init__(self, uri, *args, **kwargs):
"""
Initialize the decorator instance
:param uri: URI Stack-In-A-Box will use to recognize the HTTP calls
f.e 'localhost'.
:param text_type access_services: name of a keyword parameter in the
test function to assign access to the services created in the
arguments to the decorator.
:param args: A tuple containing all the positional arguments. Any
StackInABoxService arguments are removed before being passed to
the actual function.
:param kwargs: A dictionary of keyword args that are passed to the
actual function.
"""
self.uri = uri
self.services = {}
self.args = []
self.kwargs = kwargs
if "access_services" in self.kwargs:
self.enable_service_access = self.kwargs["access_services"]
del self.kwargs["access_services"]
else:
self.enable_service_access = None
for arg in args:
if self.process_service(arg, raise_on_type=False):
pass
elif (
isinstance(arg, types.GeneratorType) or
isinstance(arg, collections.Iterable)
):
for sub_arg in arg:
self.process_service(sub_arg, raise_on_type=True)
else:
self.args.append(arg)
def process_service(self, arg_based_service, raise_on_type=True):
if isinstance(arg_based_service, StackInABoxService):
logger.debug("Registering {0}".format(arg_based_service.name))
self.services[arg_based_service.name] = arg_based_service
return True
elif raise_on_type:
raise TypeError(
"Generator or Iterable must provide a "
"StackInABoxService in all of its results."
)
return False
def __call__(self, fn):
"""
Call to actually wrap the function call.
"""
@functools.wraps(fn)
def wrapped(*args, **kwargs):
args_copy = list(args)
for arg in self.args:
args_copy.append(arg)
args_finalized = tuple(args_copy)
kwargs.update(self.kwargs)
if self.enable_service_access is not None:
kwargs[self.enable_service_access] = self.services
return_value = None
def run():
responses.mock.start()
StackInABox.reset_services()
for service in self.services.values():
StackInABox.register_service(service)
responses_registration(self.uri)
return_value = fn(*args_finalized, **kwargs)
StackInABox.reset_services()
responses.mock.stop()
responses.mock.reset()
with responses.RequestsMock():
run()
return return_value
return wrapped
class stack_activate(activate):
@deprecator.DeprecatedInterface("stack_activate", "activate")
def __init__(self, *args, **kwargs):
super(stack_activate, self).__init__(*args, **kwargs)
@deprecator.DeprecatedInterface("stack_activate", "activate")
def __call__(self, *args, **kwargs):
super(stack_activate, self).__call__(*args, **kwargs)