forked from 3scale-labs/3scale_ws_api_for_python
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathThreeScalePY.py
executable file
·374 lines (307 loc) · 12.4 KB
/
ThreeScalePY.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
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
"""Python API for ThreeScale account.
The Python API to interact with ThreeScale account. The library has
interface for following APIs.
- authorize()
- report()
Report GET API usage:
---------------------
auth = ThreeScalePY.ThreeScaleAuthorize(provider_key, app_id, app_key)
if auth.authorize():
resp = auth.build_auth_response()
usage_reports = resp.get_usage_reports()
for report in usage_reports:
print " metric => %s" % report.get_metric()
print " period => %s" % report.get_period()
print " start => %s" % report.get_start_period()
print " end => %s" % report.get_end_period()
print " max => %s" % report.get_max_value()
print " current => %s" % report.get_current_value()
Authorize POST API usage:
-------------------------
t1 = {}
trans_usage = {}
trans_usage['hits'] = 1
trans_usage['max_value'] = 5
trans_usage['timestamp'] = time.gmtime(time.time())
t1['app_id'] = app_id
t1['usage'] = trans_usage
transactions = [t1]
resp = report.report(transactions)
"""
import sys
import urllib2
import urllib
import libxml2
import time
__all__ = ['ThreeScale', 'ThreeScaleAuthorize',
'authorize', 'build_auth_response',
'ThreeScaleAuthorizeResponse',
'get_plan', 'get_usage_reports',
'ThreeScaleAuthorizeResponseUsageReport',
'get_period', 'get_metric', 'get_start_period',
'get_end_period', 'get_max_value', 'get_current_value',
'ThreeScaleReport', 'report'
]
class ThreeScale:
"""The base class to initialize the credentials and URLs"""
def __init__(self, provider_key, app_id=None, app_key=None):
"""initialize the following credentials:
- provider key
- application id
- application key
The application id and key are optional. If it is omitted, the
provider key alone is set. This is useful when the class is
inherited by ThreeScaleReport class, for which application id
is passed in transactions data structure and application key
is not necessary."""
self.domain = "su1.3scale.net"
self.protocol = "http"
self.app_id = app_id
self.app_key = app_key
self.provider_key = provider_key
def get_base_url(self):
"""return the base url for using with authorize and report
APIs"""
base_url = "%s://%s" % (self.protocol, self.domain)
return base_url
def get_auth_url(self):
"""return the url for passing authorize GET request"""
auth_url = "%s/transactions/authorize.xml" % self.get_base_url()
return auth_url
def get_report_url(self):
"""return the url for passing report POST request"""
report_url = "%s/transactions.xml" % self.get_base_url()
return report_url
class ThreeScaleAuthorize(ThreeScale):
"""ThreeScaleAuthorize(): The derived class for ThreeScale. It is
main class to invoke authorize GET API."""
def get_query_string(self):
"""get the url encoded query string"""
params = {
'app_id' : self.app_id,
'app_key' : self.app_key,
'provider_key' : self.provider_key,
}
return urllib.urlencode(params)
def validate(self):
"""validate the arguments. If any of following parameters is
missing, exit from the script.
- application id
- application key
- provider key
@throws ThreeScaleException error, if any of the credentials are
invalid.
"""
err = []
if not self.app_id:
err.append("App Id not defined")
if not self.provider_key:
err.append("Provider key not defined")
if len(err):
raise ThreeScaleException(': '.join(err))
def authorize(self):
"""authorize() -- invoke authorize GET request.
The authorize response is stored in a class variable.
returns True, if authorization is successful.
@throws ThreeScaleServerError error, if invalid response is
received.
@throws ThreeScaleConnectionError error, if connection can not be
established.
@throws ThreeScaleException error, if any other unknown error is
occurred while receiving response for authorize GET api.
"""
self.authorized = False
self.auth_xml = None
self.validate()
auth_url = self.get_auth_url()
query_str = self.get_query_string()
query_url = "%s?%s" % (auth_url, query_str)
try:
urlobj = urllib2.urlopen(query_url)
resp = urlobj.read()
self.authorized = True
self.auth_xml = resp
return True
except urllib2.HTTPError, err:
if err.code == 409: # a 409 means correct credentials but authorization failed
self.authorized = False
self.auth_xml = err.read()
return False
raise ThreeScaleServerError("Invalid response for url "
"%s: %s" % (auth_url, err))
except urllib2.URLError, err:
raise ThreeScaleConnectionError("Connection error %s: "
"%s" % (auth_url, err))
except Exception, err:
# handle all other exceptions
raise ThreeScaleException("Unknown error %s: "
"%s" % (auth_url, err))
def build_auth_response(self):
"""
Store the xml response from authorize GET api in a Python
object, ThreeScaleAuthorizeResponse. The values in xml output
can be retrived using the class methods.
@returns ThreeScaleAuthorizeResponse object.
@throws ThreeScaleException error, if xml output received from
the server is not valid.
"""
xml = None
resp = ThreeScaleAuthorizeResponse()
try:
xml = libxml2.parseDoc(self.auth_xml)
except libxml2.parserError, err:
raise ThreeScaleException("Invalid xml %s" % err)
resp.set_plan(xml.xpathEval('/status/plan')[0].getContent())
resp.set_reason(xml.xpathEval('/status/reason')[0].getContent())
reports = xml.xpathEval('/status/usage_reports/usage_report')
for report in reports:
resp.add_usage_report(report)
return resp
class ThreeScaleAuthorizeResponse():
"""The derived class for ThreeScale() class. The object constitutes
the xml data retrived from authorize GET api."""
def __init__(self):
self.reason = None
self.plan = None
self.usage_reports = []
def set_plan(self, plan):
self.plan = plan
def get_plan(self):
return self.plan
def set_reason(self, reason):
self.reason = reason
def get_reason(self):
return self.reason
def add_usage_report(self, xml):
"""
Create the ThreeScaleAuthorizeResponseUsageReport object for
each usage report.
"""
report = ThreeScaleAuthorizeResponseUsageReport()
report.set_metric(xml.xpathEval('@metric')[0].getContent())
report.set_period(xml.xpathEval('@period')[0].getContent())
start = xml.xpathEval('period_start')[0].getContent()
end = xml.xpathEval('period_end')[0].getContent()
report.set_interval(start, end)
report.set_max_value(xml.xpathEval('max_value')[0].getContent())
report.set_current_value(xml.xpathEval(\
'current_value')[0].getContent())
self.usage_reports.append(report)
def get_usage_reports(self):
"""get all usage reports returned by the authorize GET api."""
return self.usage_reports
class ThreeScaleAuthorizeResponseUsageReport():
"""Object to store all information related to the usage report."""
def __init__(self):
self.metric = None
self.period = None
self.start = None
self.end = None
self.max_value = None
self.current_value = None
def set_metric(self, metric):
self.metric = metric
def set_period(self, period):
self.period = period
def set_interval(self, start, end):
self.start_period = start
self.end_period = end
def set_end_period(self, end_period):
self.end_period = end_period
def set_max_value(self, max_value):
self.max_value = max_value
def set_current_value(self, current_value):
self.current_value = current_value
def get_metric(self):
return self.metric
def get_period(self):
return self.period
def get_start_period(self):
return self.start_period
def get_end_period(self):
return self.end_period
def get_max_value(self):
return self.max_value
def get_current_value(self):
return self.current_value
class ThreeScaleReport(ThreeScale):
"""ThreeScaleReport()
The derived class for ThreeScale() base class, for making report
POST request.
"""
def build_post_data(self, transactions):
return "provider_key=%s&%s" % (self.provider_key, self.encode_transactions(transactions))
def encode_transactions(self, transactions):
"""
@throws ThreeScaleException error, if transaction is invalid.
"""
encoded = []
i = 0
if type(transactions).__name__ != 'list':
raise ThreeScaleException("Invalid transaction type")
for trans in transactions:
for part in self.encode_recursive(trans):
encoded.append("transactions[%d]%s" % (i, part))
i += 1
return "&".join(encoded)
def encode_recursive(self, trans):
"""encode every value in transactions
@throws ThreeScaleException error, if the timestamp specified in
transaction is invalid.
"""
result = []
for key in trans.keys():
if key == 'usage': # usage is list
val = self.encode_recursive(trans[key])[0]
new_value = "[usage]%s" % (self.encode_recursive(trans[key])[0])
elif key == 'timestamp': # specially encode the timestamp
ts = trans[key]
try:
new_value = "[%s]=%s" % (key, time.strftime('%Y-%m-%d %H:%M:%S %z', ts))
except Exception, err:
raise ThreeScaleException("Invalid timestamp "
"'%s' specified in "
"transaction" % ts)
else:
new_value = "[%s]=%s" % (key, trans[key])
result.append(new_value)
return result
def report(self, transactions):
"""send the report POST request.
@returns True, if request is sent successfully.
@throws ThreeScaleServerError error, if invalid response is
received.
@throws ThreeScaleConnectionError error, if connection can not be
established.
@throws ThreeScaleException error, if any other unknown error is
occurred while receiving response for report POST api.
"""
report_url = self.get_report_url()
data = self.build_post_data(transactions)
try:
req = urllib2.Request(report_url, data)
resp = urllib2.urlopen(req)
return True
except urllib2.HTTPError, err:
raise ThreeScaleServerError("Invalid response for url "
"%s: %s" % (report_url, err))
return False
except urllib2.URLError, err:
raise ThreeScaleConnectionError("Connection error %s: "
"%s" % (report_url, err))
return False
except Exception, err:
# handle all other exceptions
raise ThreeScaleException("Unknown error %s: "
"%s" % (report_url, err))
return False
class ThreeScaleException(Exception):
"""main exception class. raise this exception for all other errors"""
pass
class ThreeScaleServerError(ThreeScaleException):
"""raise exception if there are any exception during server
interaction"""
pass
class ThreeScaleConnectionError(ThreeScaleException):
"""raise exception if server connection can not be establised"""
pass