Mercurial > repos > guerler > springsuite
comparison planemo/lib/python3.7/site-packages/boto/fps/connection.py @ 0:d30785e31577 draft
"planemo upload commit 6eee67778febed82ddd413c3ca40b3183a3898f1"
author | guerler |
---|---|
date | Fri, 31 Jul 2020 00:18:57 -0400 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:d30785e31577 |
---|---|
1 # Copyright (c) 2012 Andy Davidoff http://www.disruptek.com/ | |
2 # Copyright (c) 2010 Jason R. Coombs http://www.jaraco.com/ | |
3 # Copyright (c) 2008 Chris Moyer http://coredumped.org/ | |
4 # | |
5 # Permission is hereby granted, free of charge, to any person obtaining a | |
6 # copy of this software and associated documentation files (the | |
7 # "Software"), to deal in the Software without restriction, including | |
8 # without limitation the rights to use, copy, modify, merge, publish, dis- | |
9 # tribute, sublicense, and/or sell copies of the Software, and to permit | |
10 # persons to whom the Software is furnished to do so, subject to the fol- | |
11 # lowing conditions: | |
12 # | |
13 # The above copyright notice and this permission notice shall be included | |
14 # in all copies or substantial portions of the Software. | |
15 # | |
16 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | |
17 # OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABIL- | |
18 # ITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT | |
19 # SHALL THE AUTHOR BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | |
20 # WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
21 # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS | |
22 # IN THE SOFTWARE. | |
23 | |
24 import urllib | |
25 import uuid | |
26 from boto.connection import AWSQueryConnection | |
27 from boto.fps.exception import ResponseErrorFactory | |
28 from boto.fps.response import ResponseFactory | |
29 import boto.fps.response | |
30 | |
31 __all__ = ['FPSConnection'] | |
32 | |
33 decorated_attrs = ('action', 'response') | |
34 | |
35 | |
36 def add_attrs_from(func, to): | |
37 for attr in decorated_attrs: | |
38 setattr(to, attr, getattr(func, attr, None)) | |
39 return to | |
40 | |
41 | |
42 def complex_amounts(*fields): | |
43 def decorator(func): | |
44 def wrapper(self, *args, **kw): | |
45 for field in filter(kw.has_key, fields): | |
46 amount = kw.pop(field) | |
47 kw[field + '.Value'] = getattr(amount, 'Value', str(amount)) | |
48 kw[field + '.CurrencyCode'] = getattr(amount, 'CurrencyCode', | |
49 self.currencycode) | |
50 return func(self, *args, **kw) | |
51 wrapper.__doc__ = "{0}\nComplex Amounts: {1}".format(func.__doc__, | |
52 ', '.join(fields)) | |
53 return add_attrs_from(func, to=wrapper) | |
54 return decorator | |
55 | |
56 | |
57 def requires(*groups): | |
58 | |
59 def decorator(func): | |
60 | |
61 def wrapper(*args, **kw): | |
62 hasgroup = lambda x: len(x) == len(filter(kw.has_key, x)) | |
63 if 1 != len(filter(hasgroup, groups)): | |
64 message = ' OR '.join(['+'.join(g) for g in groups]) | |
65 message = "{0} requires {1} argument(s)" \ | |
66 "".format(getattr(func, 'action', 'Method'), message) | |
67 raise KeyError(message) | |
68 return func(*args, **kw) | |
69 message = ' OR '.join(['+'.join(g) for g in groups]) | |
70 wrapper.__doc__ = "{0}\nRequired: {1}".format(func.__doc__, | |
71 message) | |
72 return add_attrs_from(func, to=wrapper) | |
73 return decorator | |
74 | |
75 | |
76 def needs_caller_reference(func): | |
77 | |
78 def wrapper(*args, **kw): | |
79 kw.setdefault('CallerReference', uuid.uuid4()) | |
80 return func(*args, **kw) | |
81 wrapper.__doc__ = "{0}\nUses CallerReference, defaults " \ | |
82 "to uuid.uuid4()".format(func.__doc__) | |
83 return add_attrs_from(func, to=wrapper) | |
84 | |
85 | |
86 def api_action(*api): | |
87 | |
88 def decorator(func): | |
89 action = ''.join(api or map(str.capitalize, func.__name__.split('_'))) | |
90 response = ResponseFactory(action) | |
91 if hasattr(boto.fps.response, action + 'Response'): | |
92 response = getattr(boto.fps.response, action + 'Response') | |
93 | |
94 def wrapper(self, *args, **kw): | |
95 return func(self, action, response, *args, **kw) | |
96 wrapper.action, wrapper.response = action, response | |
97 wrapper.__doc__ = "FPS {0} API call\n{1}".format(action, | |
98 func.__doc__) | |
99 return wrapper | |
100 return decorator | |
101 | |
102 | |
103 class FPSConnection(AWSQueryConnection): | |
104 | |
105 APIVersion = '2010-08-28' | |
106 ResponseError = ResponseErrorFactory | |
107 currencycode = 'USD' | |
108 | |
109 def __init__(self, *args, **kw): | |
110 self.currencycode = kw.pop('CurrencyCode', self.currencycode) | |
111 kw.setdefault('host', 'fps.sandbox.amazonaws.com') | |
112 super(FPSConnection, self).__init__(*args, **kw) | |
113 | |
114 def _required_auth_capability(self): | |
115 return ['fps'] | |
116 | |
117 @needs_caller_reference | |
118 @complex_amounts('SettlementAmount') | |
119 @requires(['CreditInstrumentId', 'SettlementAmount.Value', | |
120 'SenderTokenId', 'SettlementAmount.CurrencyCode']) | |
121 @api_action() | |
122 def settle_debt(self, action, response, **kw): | |
123 """ | |
124 Allows a caller to initiate a transaction that atomically transfers | |
125 money from a sender's payment instrument to the recipient, while | |
126 decreasing corresponding debt balance. | |
127 """ | |
128 return self.get_object(action, kw, response) | |
129 | |
130 @requires(['TransactionId']) | |
131 @api_action() | |
132 def get_transaction_status(self, action, response, **kw): | |
133 """ | |
134 Gets the latest status of a transaction. | |
135 """ | |
136 return self.get_object(action, kw, response) | |
137 | |
138 @requires(['StartDate']) | |
139 @api_action() | |
140 def get_account_activity(self, action, response, **kw): | |
141 """ | |
142 Returns transactions for a given date range. | |
143 """ | |
144 return self.get_object(action, kw, response) | |
145 | |
146 @requires(['TransactionId']) | |
147 @api_action() | |
148 def get_transaction(self, action, response, **kw): | |
149 """ | |
150 Returns all details of a transaction. | |
151 """ | |
152 return self.get_object(action, kw, response) | |
153 | |
154 @api_action() | |
155 def get_outstanding_debt_balance(self, action, response): | |
156 """ | |
157 Returns the total outstanding balance for all the credit instruments | |
158 for the given creditor account. | |
159 """ | |
160 return self.get_object(action, {}, response) | |
161 | |
162 @requires(['PrepaidInstrumentId']) | |
163 @api_action() | |
164 def get_prepaid_balance(self, action, response, **kw): | |
165 """ | |
166 Returns the balance available on the given prepaid instrument. | |
167 """ | |
168 return self.get_object(action, kw, response) | |
169 | |
170 @api_action() | |
171 def get_total_prepaid_liability(self, action, response): | |
172 """ | |
173 Returns the total liability held by the given account corresponding to | |
174 all the prepaid instruments owned by the account. | |
175 """ | |
176 return self.get_object(action, {}, response) | |
177 | |
178 @api_action() | |
179 def get_account_balance(self, action, response): | |
180 """ | |
181 Returns the account balance for an account in real time. | |
182 """ | |
183 return self.get_object(action, {}, response) | |
184 | |
185 @needs_caller_reference | |
186 @requires(['PaymentInstruction', 'TokenType']) | |
187 @api_action() | |
188 def install_payment_instruction(self, action, response, **kw): | |
189 """ | |
190 Installs a payment instruction for caller. | |
191 """ | |
192 return self.get_object(action, kw, response) | |
193 | |
194 @needs_caller_reference | |
195 @requires(['returnURL', 'pipelineName']) | |
196 def cbui_url(self, **kw): | |
197 """ | |
198 Generate a signed URL for the Co-Branded service API given arguments as | |
199 payload. | |
200 """ | |
201 sandbox = 'sandbox' in self.host and 'payments-sandbox' or 'payments' | |
202 endpoint = 'authorize.{0}.amazon.com'.format(sandbox) | |
203 base = '/cobranded-ui/actions/start' | |
204 | |
205 validpipelines = ('SingleUse', 'MultiUse', 'Recurring', 'Recipient', | |
206 'SetupPrepaid', 'SetupPostpaid', 'EditToken') | |
207 assert kw['pipelineName'] in validpipelines, "Invalid pipelineName" | |
208 kw.update({ | |
209 'signatureMethod': 'HmacSHA256', | |
210 'signatureVersion': '2', | |
211 }) | |
212 kw.setdefault('callerKey', self.aws_access_key_id) | |
213 | |
214 safestr = lambda x: x is not None and str(x) or '' | |
215 safequote = lambda x: urllib.quote(safestr(x), safe='~') | |
216 payload = sorted([(k, safequote(v)) for k, v in kw.items()]) | |
217 | |
218 encoded = lambda p: '&'.join([k + '=' + v for k, v in p]) | |
219 canonical = '\n'.join(['GET', endpoint, base, encoded(payload)]) | |
220 signature = self._auth_handler.sign_string(canonical) | |
221 payload += [('signature', safequote(signature))] | |
222 payload.sort() | |
223 | |
224 return 'https://{0}{1}?{2}'.format(endpoint, base, encoded(payload)) | |
225 | |
226 @needs_caller_reference | |
227 @complex_amounts('TransactionAmount') | |
228 @requires(['SenderTokenId', 'TransactionAmount.Value', | |
229 'TransactionAmount.CurrencyCode']) | |
230 @api_action() | |
231 def reserve(self, action, response, **kw): | |
232 """ | |
233 Reserve API is part of the Reserve and Settle API conjunction that | |
234 serve the purpose of a pay where the authorization and settlement have | |
235 a timing difference. | |
236 """ | |
237 return self.get_object(action, kw, response) | |
238 | |
239 @needs_caller_reference | |
240 @complex_amounts('TransactionAmount') | |
241 @requires(['SenderTokenId', 'TransactionAmount.Value', | |
242 'TransactionAmount.CurrencyCode']) | |
243 @api_action() | |
244 def pay(self, action, response, **kw): | |
245 """ | |
246 Allows calling applications to move money from a sender to a recipient. | |
247 """ | |
248 return self.get_object(action, kw, response) | |
249 | |
250 @requires(['TransactionId']) | |
251 @api_action() | |
252 def cancel(self, action, response, **kw): | |
253 """ | |
254 Cancels an ongoing transaction and puts it in cancelled state. | |
255 """ | |
256 return self.get_object(action, kw, response) | |
257 | |
258 @complex_amounts('TransactionAmount') | |
259 @requires(['ReserveTransactionId', 'TransactionAmount.Value', | |
260 'TransactionAmount.CurrencyCode']) | |
261 @api_action() | |
262 def settle(self, action, response, **kw): | |
263 """ | |
264 The Settle API is used in conjunction with the Reserve API and is used | |
265 to settle previously reserved transaction. | |
266 """ | |
267 return self.get_object(action, kw, response) | |
268 | |
269 @complex_amounts('RefundAmount') | |
270 @requires(['TransactionId', 'RefundAmount.Value', | |
271 'CallerReference', 'RefundAmount.CurrencyCode']) | |
272 @api_action() | |
273 def refund(self, action, response, **kw): | |
274 """ | |
275 Refunds a previously completed transaction. | |
276 """ | |
277 return self.get_object(action, kw, response) | |
278 | |
279 @requires(['RecipientTokenId']) | |
280 @api_action() | |
281 def get_recipient_verification_status(self, action, response, **kw): | |
282 """ | |
283 Returns the recipient status. | |
284 """ | |
285 return self.get_object(action, kw, response) | |
286 | |
287 @requires(['CallerReference'], ['TokenId']) | |
288 @api_action() | |
289 def get_token_by_caller(self, action, response, **kw): | |
290 """ | |
291 Returns the details of a particular token installed by this calling | |
292 application using the subway co-branded UI. | |
293 """ | |
294 return self.get_object(action, kw, response) | |
295 | |
296 @requires(['UrlEndPoint', 'HttpParameters']) | |
297 @api_action() | |
298 def verify_signature(self, action, response, **kw): | |
299 """ | |
300 Verify the signature that FPS sent in IPN or callback urls. | |
301 """ | |
302 return self.get_object(action, kw, response) | |
303 | |
304 @api_action() | |
305 def get_tokens(self, action, response, **kw): | |
306 """ | |
307 Returns a list of tokens installed on the given account. | |
308 """ | |
309 return self.get_object(action, kw, response) | |
310 | |
311 @requires(['TokenId']) | |
312 @api_action() | |
313 def get_token_usage(self, action, response, **kw): | |
314 """ | |
315 Returns the usage of a token. | |
316 """ | |
317 return self.get_object(action, kw, response) | |
318 | |
319 @requires(['TokenId']) | |
320 @api_action() | |
321 def cancel_token(self, action, response, **kw): | |
322 """ | |
323 Cancels any token installed by the calling application on its own | |
324 account. | |
325 """ | |
326 return self.get_object(action, kw, response) | |
327 | |
328 @needs_caller_reference | |
329 @complex_amounts('FundingAmount') | |
330 @requires(['PrepaidInstrumentId', 'FundingAmount.Value', | |
331 'SenderTokenId', 'FundingAmount.CurrencyCode']) | |
332 @api_action() | |
333 def fund_prepaid(self, action, response, **kw): | |
334 """ | |
335 Funds the prepaid balance on the given prepaid instrument. | |
336 """ | |
337 return self.get_object(action, kw, response) | |
338 | |
339 @requires(['CreditInstrumentId']) | |
340 @api_action() | |
341 def get_debt_balance(self, action, response, **kw): | |
342 """ | |
343 Returns the balance corresponding to the given credit instrument. | |
344 """ | |
345 return self.get_object(action, kw, response) | |
346 | |
347 @needs_caller_reference | |
348 @complex_amounts('AdjustmentAmount') | |
349 @requires(['CreditInstrumentId', 'AdjustmentAmount.Value', | |
350 'AdjustmentAmount.CurrencyCode']) | |
351 @api_action() | |
352 def write_off_debt(self, action, response, **kw): | |
353 """ | |
354 Allows a creditor to write off the debt balance accumulated partially | |
355 or fully at any time. | |
356 """ | |
357 return self.get_object(action, kw, response) | |
358 | |
359 @requires(['SubscriptionId']) | |
360 @api_action() | |
361 def get_transactions_for_subscription(self, action, response, **kw): | |
362 """ | |
363 Returns the transactions for a given subscriptionID. | |
364 """ | |
365 return self.get_object(action, kw, response) | |
366 | |
367 @requires(['SubscriptionId']) | |
368 @api_action() | |
369 def get_subscription_details(self, action, response, **kw): | |
370 """ | |
371 Returns the details of Subscription for a given subscriptionID. | |
372 """ | |
373 return self.get_object(action, kw, response) | |
374 | |
375 @needs_caller_reference | |
376 @complex_amounts('RefundAmount') | |
377 @requires(['SubscriptionId']) | |
378 @api_action() | |
379 def cancel_subscription_and_refund(self, action, response, **kw): | |
380 """ | |
381 Cancels a subscription. | |
382 """ | |
383 message = "If you specify a RefundAmount, " \ | |
384 "you must specify CallerReference." | |
385 assert not 'RefundAmount.Value' in kw \ | |
386 or 'CallerReference' in kw, message | |
387 return self.get_object(action, kw, response) | |
388 | |
389 @requires(['TokenId']) | |
390 @api_action() | |
391 def get_payment_instruction(self, action, response, **kw): | |
392 """ | |
393 Gets the payment instruction of a token. | |
394 """ | |
395 return self.get_object(action, kw, response) |