-
Notifications
You must be signed in to change notification settings - Fork 43
/
Copy pathhammock.py
111 lines (91 loc) · 3.42 KB
/
hammock.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
import requests
import copy
class Hammock(object):
"""Chainable, magical class helps you make requests to RESTful services"""
HTTP_METHODS = ['get', 'options', 'head', 'post', 'put', 'patch', 'delete']
def __init__(self, name=None, parent=None, append_slash=False, **kwargs):
"""Constructor
Arguments:
name -- name of node
parent -- parent node for chaining
append_slash -- flag if you want a trailing slash in urls
**kwargs -- `requests` session be initiated with if any available
"""
self._name = name
self._parent = parent
self._append_slash = append_slash
self._session = requests.session()
for k, v in kwargs.items():
orig = getattr(self._session, k) # Let it throw exception
if isinstance(orig, dict):
orig.update(v)
else:
setattr(self._session, k, v)
def _spawn(self, name):
"""Returns a shallow copy of current `Hammock` instance as nested child
Arguments:
name -- name of child
"""
child = copy.copy(self)
child._name = name
child._parent = self
return child
def __getattr__(self, name):
"""Here comes some magic. Any absent attribute typed within class
falls here and return a new child `Hammock` instance in the chain.
"""
# Ignore specials (Otherwise shallow copying causes infinite loops)
if name.startswith('__'):
raise AttributeError(name)
return self._spawn(name)
def __iter__(self):
"""Iterator implementation which iterates over `Hammock` chain."""
current = self
while current:
if current._name:
yield current
current = current._parent
def _chain(self, *args):
"""This method converts args into chained Hammock instances
Arguments:
*args -- array of string representable objects
"""
chain = self
for arg in args:
chain = chain._spawn(str(arg))
return chain
def _close_session(self):
"""Closes session if exists"""
if self._session:
self._session.close()
def __call__(self, *args):
"""Here comes second magic. If any `Hammock` instance called it
returns a new child `Hammock` instance in the chain
"""
return self._chain(*args)
def _url(self, *args):
"""Converts current `Hammock` chain into a url string
Arguments:
*args -- extra url path components to tail
"""
path_comps = [mock._name for mock in self._chain(*args)]
url = "/".join(reversed(path_comps))
if self._append_slash:
url = url + "/"
return url
def __repr__(self):
""" String representaion of current `Hammock` chain"""
return self._url()
def _request(self, method, *args, **kwargs):
"""
Makes the HTTP request using requests module
"""
return self._session.request(method, self._url(*args), **kwargs)
def bind_method(method):
"""Bind `requests` module HTTP verbs to `Hammock` class as
static methods."""
def aux(hammock, *args, **kwargs):
return hammock._request(method, *args, **kwargs)
return aux
for method in Hammock.HTTP_METHODS:
setattr(Hammock, method.upper(), bind_method(method))