This repository has been archived by the owner on Sep 13, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 4
/
billogram_api.py
893 lines (717 loc) · 28 KB
/
billogram_api.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
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
# encoding=utf-8
# Copyright (c) 2013 Billogram AB
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.
"Library for accessing the Billogram v2 HTTP API"
from __future__ import unicode_literals, print_function, division
import requests
import json
API_URL_BASE = "https://billogram.com/api/v2"
USER_AGENT = "Billogram API Python Library/1.00"
# python 2/3 intercompatibility
try:
unicode # not defined in py3k
def _printable_repr(s):
# python 2.7 can't handle unicode objects from __repr__
# so need this wrapper
return s.encode('utf-8')
except NameError:
basestring = str # py3k has no basestring type so fake one
def _printable_repr(s):
return s
class BillogramAPIError(Exception):
"Base class for errors from the Billogram API"
def __init__(self, message, **kwargs):
super(BillogramAPIError, self).__init__(message)
self.field = kwargs.get('field', None)
self.field_path = kwargs.get('field_path', None)
self.extra_data = kwargs
if 'field' in self.extra_data:
del self.extra_data['field']
if 'field_path' in self.extra_data:
del self.extra_data['field_path']
if not self.extra_data:
self.extra_data = None
class ServiceMalfunctioningError(BillogramAPIError):
"The Billogram API service seems to be malfunctioning"
pass
class RequestFormError(BillogramAPIError):
"Errors caused by malformed requests"
pass
class PermissionDeniedError(BillogramAPIError):
"No permission to perform the requested operation"
pass
class InvalidAuthenticationError(PermissionDeniedError):
"The user/key combination could not be authenticated"
pass
class NotAuthorizedError(PermissionDeniedError):
"The user does not have authorization to perform the requested operation"
pass
class RequestDataError(BillogramAPIError):
"Errors caused by bad data passed to request"
pass
class UnknownFieldError(RequestDataError):
"An unknown field was passed in the request data"
pass
class MissingFieldError(RequestDataError):
"A required field was missing from the request data"
pass
class InvalidFieldCombinationError(RequestDataError):
"Mutually exclusive fields were specified together"
pass
class InvalidFieldValueError(RequestDataError):
"A field was given an out-of-range value or a value of incorrect type"
pass
class ReadOnlyFieldError(RequestDataError):
"Attempt to modify a read-only field"
pass
class InvalidObjectStateError(RequestDataError):
"The request can not be performed on an object in this state"
pass
class ObjectNotFoundError(RequestDataError):
"No object by the requested ID exists"
pass
class ObjectNotAvailableYetError(ObjectNotFoundError):
"No object by the requested ID exists, but is expected to be created soon"
pass
class BillogramAPI(object):
"""Pseudo-connection to the Billogram v2 API
Objects of this class provide a call interface to the Billogram
v2 HTTP API.
"""
def __init__(self, auth_user, auth_key, user_agent=None, api_base=None):
"""Create a Billogram API connection object
Pass the API authentication in the auth_user and auth_key parameters.
API accounts can only be created from the Billogram web interface.
"""
self._auth = (auth_user, auth_key)
self._items = None
self._customers = None
self._billogram = None
self._settings = None
self._logotype = None
self._reports = None
self._user_agent = user_agent or USER_AGENT
self._api_base = api_base or API_URL_BASE
@property
def items(self):
"Provide access to the items database"
if self._items is None:
self._items = SimpleClass(self, 'item', 'item_no')
return self._items
@property
def customers(self):
"Provide access to the customer database"
if self._customers is None:
self._customers = SimpleClass(self, 'customer', 'customer_no')
return self._customers
@property
def billogram(self):
"Provide access to billogram objects and attached invoices"
if self._billogram is None:
self._billogram = BillogramClass(self)
return self._billogram
@property
def settings(self):
"Provide access to settings for the Billogram account"
if self._settings is None:
self._settings = SingletonObject(self, 'settings')
return self._settings
@property
def logotype(self):
"Provide access to the logotype for the Billogram account"
if self._logotype is None:
self._logotype = SingletonObject(self, 'logotype')
return self._logotype
@property
def reports(self):
"Provide access to the reports database"
if self._reports is None:
self._reports = SimpleClass(self, 'report', 'filename')
return self._reports
@classmethod
def _check_api_response(cls, resp, expect_content_type=None):
if not resp.ok or expect_content_type is None:
# if the request failed the response should always be json
expect_content_type = 'application/json'
if resp.status_code in range(500, 600):
# internal error
if resp.headers['content-type'] == expect_content_type and \
expect_content_type == 'application/json':
data = resp.json()
raise ServiceMalfunctioningError(
'Billogram API reported a server error: {} - {}'.format(
data.get('status'),
data.get('data').get('message')
)
)
raise ServiceMalfunctioningError(
'Billogram API reported a server error'
)
if resp.headers['content-type'] != expect_content_type:
# the service returned a different content-type from the expected,
# probably some malfunction on the remote end
if resp.headers['content-type'] == 'application/json':
data = resp.json()
if data.get('status') == 'NOT_AVAILABLE_YET':
raise ObjectNotAvailableYetError(
'Object not available yet'
)
raise ServiceMalfunctioningError(
'Billogram API returned unexpected content type'
)
if expect_content_type == 'application/json':
data = resp.json()
status = data.get('status')
if not status:
raise ServiceMalfunctioningError(
'Response data missing status field'
)
if not 'data' in data:
raise ServiceMalfunctioningError(
'Response data missing data field'
)
else:
# per above, non-json responses are always ok, so just return them
return resp.content
if resp.status_code == 403:
# bad auth
if status == 'PERMISSION_DENIED':
raise NotAuthorizedError(
'Not allowed to perform the requested operation'
)
elif status == 'INVALID_AUTH':
raise InvalidAuthenticationError(
'The user/key combination is wrong, check the credentials \
used and possibly generate a new set'
)
elif status == 'MISSING_AUTH':
raise RequestFormError('No authentication data was given')
else:
raise PermissionDeniedError(
'Permission denied, status={}'.format(
status
)
)
if resp.status_code == 404:
# not found
if data.get('status') == 'NOT_AVAILABLE_YET':
raise ObjectNotFoundError('Object not available yet')
raise ObjectNotFoundError('Object not found')
if resp.status_code == 405:
# bad http method
raise RequestFormError('Invalid HTTP method')
if status == 'OK':
return data
errordata = data.get('data', {})
raise {
'MISSING_QUERY_PARAMETER': RequestFormError,
'INVALID_QUERY_PARAMETER': RequestFormError,
'INVALID_PARAMETER': InvalidFieldValueError,
'INVALID_PARAMETER_COMBINATION': InvalidFieldCombinationError,
'READ_ONLY_PARAMETER': ReadOnlyFieldError,
'UNKNOWN_PARAMETER': UnknownFieldError,
'INVALID_OBJECT_STATE': InvalidObjectStateError,
}.get(status, RequestDataError)(**errordata)
def get(self, obj, params=None, expect_content_type=None):
"Perform a HTTP GET request to the Billogram API"
url = '{}/{}'.format(self._api_base, obj)
return self._check_api_response(
requests.get(
url,
auth=self._auth,
params=params,
headers={'user-agent': self._user_agent}
),
expect_content_type=expect_content_type
)
def post(self, obj, data):
"Perform a HTTP POST request to the Billogram API"
url = '{}/{}'.format(self._api_base, obj)
return self._check_api_response(
requests.post(
url,
auth=self._auth,
data=json.dumps(data),
headers={
'content-type': 'application/json',
'user-agent': self._user_agent
}
)
)
def put(self, obj, data):
"Perform a HTTP PUT request to the Billogram API"
url = '{}/{}'.format(self._api_base, obj)
return self._check_api_response(
requests.put(
url,
auth=self._auth,
data=json.dumps(data),
headers={
'content-type': 'application/json',
'user-agent': self._user_agent
}
)
)
def delete(self, obj):
"Perform a HTTP DELETE request to the Billogram API"
url = '{}/{}'.format(self._api_base, obj)
return self._check_api_response(
requests.delete(
url,
auth=self._auth,
headers={'user-agent': self._user_agent}
)
)
class SingletonObject(object):
"""Represents a remote singleton object on Billogram
Implements __getattr__ for dict-like access to the data of the remote
object, or use the 'data' property to access the backing dict object.
The data in this dict and all sub-objects should be treated as read-only,
the only way to change the remote object is through the 'update' method.
The represented object is initially "lazy" and will only be fetched on the
first access. If the remote data are changed, the local copy can be updated
bythe 'refresh' method.
See the online documentation for the actual structure of remote objects.
"""
def __init__(self, api, url_name):
self._api = api
self._object_class = url_name
self._data = None
__slots__ = ('_api', '_object_class', '_data')
def __getitem__(self, key):
"Dict-like access to object data"
return self.data[key]
def __repr__(self):
return _printable_repr(
"<Billogram object '{}'{}>".format(
self._url,
(self._data is None) and ' (lazy)' or ''
)
)
@property
def _url(self):
return self._object_class
@property
def data(self):
"Access the data of the actual object"
if self._data is None:
self.refresh()
return self._data
def refresh(self):
"Refresh the local copy of the object data from remote"
resp = self._api.get(self._url)
self._data = resp['data']
return self
def update(self, data):
"Modify the remote object with a partial or complete structure"
resp = self._api.put(self._url, data)
self._data = resp['data']
return self
class SimpleObject(SingletonObject):
"""Represents a remote object on the Billogram service
Implements __getattr__ for dict-like access to the data of the remote
object, or use the 'data' property to access the backing dict object.
The data in this dict and all sub-objects should be treated as read-only,
the only way to change the remote object is through the 'update' method.
If the remote data are changed, the local copy can be updated by
the 'refresh' method.
The 'delete' method can be used to remove the backing object.
See the online documentation for the actual structure of remote objects.
"""
def __init__(self, api, object_class, data):
self._api = api
self._object_class = object_class
self._data = data
__slots__ = ()
@property
def _url(self):
return self._object_class._url_of(self)
def __getattr__(self, key):
return self._data[key]
def delete(self):
"Remove the remote object from the database"
self._api.delete(self._url)
return None
class Query(object):
"""Builds queries and fetches pages of remote objects
Due to internal limitations in Billogram it is currently only possible to
filter on a single field or special query at a time. This may change in the
future. When it does the API will continue supporting the old filtering
mechanism, however this client library will be updated to use the new one,
and at that point we will strongly recommend all applications be updated.
The exact fields and special queries available for each object type varies,
see the online documentation for details.
"""
def __init__(self, type_class):
self._type_class = type_class
self._filter = {}
self._count_cached = None
self._page_size = 100
self._order = {}
def _make_query(self, page_number=1, page_size=None):
query_args = {
'page_size': page_size or self._page_size,
'page': page_number,
}
query_args.update(self._get_queryargs())
resp = self._type_class.api.get(self._type_class._url_name, query_args)
self._count_cached = resp['meta']['total_count']
return resp
def _get_queryargs(self):
args = {}
args.update(self.filter)
args.update(self.order)
return args
@property
def count(self):
"""Total amount of objects matched by the current query, reading this
may cause a remote request"""
if self._count_cached is None:
# make a query for a single result,
# this will update the cached count
self._make_query(1, 1)
return self._count_cached
@property
def total_pages(self):
"""Total number of pages required for all objects based on current
pagesize, reading this may cause a remote request"""
return (self.count + self.page_size - 1) // self.page_size
@property
def page_size(self):
"Number of objects to return per page"
return self._page_size
@page_size.setter
def page_size(self, value):
value = int(value)
assert value >= 1
self._page_size = value
return self
@property
def filter(self):
"Filter to apply to query"
return self._filter
@filter.setter
def filter(self, value):
if value == self._filter:
return
if value:
assert 'filter_type' in value and \
'filter_field' in value and \
'filter_value' in value
assert value['filter_type'] in (
'field',
'field-prefix',
'field-search',
'special'
)
self._filter = dict(value)
else:
self._filter = {}
self._count_cached = None
return self
@property
def order(self):
return self._order
@order.setter
def order(self, value):
if value:
assert 'order_field' in value and 'order_direction' in value
assert value['order_direction'] in ('asc', 'desc')
self._order = dict(value)
else:
self._order = {}
return self
def make_filter(self, filter_type=None, filter_field=None,
filter_value=None):
if None in (filter_type, filter_field, filter_value):
self.filter = {}
else:
self.filter = {
'filter_type': filter_type,
'filter_field': filter_field,
'filter_value': filter_value
}
return self
def remove_filter(self):
"Remove any filter currently set"
self.filter = {}
return self
def filter_field(self, filter_field, filter_value):
"Filter on a basic field, look for exact matches"
return self.make_filter('field', filter_field, filter_value)
def filter_prefix(self, filter_field, filter_value):
"Filter on a basic field, look for prefix matches"
return self.make_filter('field-prefix', filter_field, filter_value)
def filter_search(self, filter_field, filter_value):
"Filter on a basic field, look for substring matches"
return self.make_filter('field-search', filter_field, filter_value)
def filter_special(self, filter_field, filter_value):
"Filter on a special query"
return self.make_filter('special', filter_field, filter_value)
def search(self, search_terms):
"Filter by a full data search (exact meaning depends on object type)"
return self.make_filter('special', 'search', search_terms)
def get_page(self, page_number):
"Fetch objects for the one-based page number"
resp = self._make_query(int(page_number))
return [
self._type_class._object_class(
self._type_class.api,
self._type_class,
o
) for o in resp['data']
]
def iter_all(self):
"Iterate over all matched objects"
# make a copy of ourselves so parameters can't be changed behind
# our back
import copy
qry = copy.copy(self)
# iterate over every object on every page
for page_number in range(1, qry.total_pages+1):
page = qry.get_page(page_number)
for obj in page:
yield obj
class SimpleClass(object):
"""Represents a collection of remote objects on the Billogram service
Provides methods to search, fetch and create instances of the object type.
See the online documentation for the actual structure of remote objects.
"""
_object_class = SimpleObject
def __init__(self, api, url_name, object_id_field):
self._api = api
self._url_name = url_name
self._object_id_field = object_id_field
def _url_of(self, obj=None, obj_id=None):
if obj_id is None:
obj_id = obj[self._object_id_field]
return '{}/{}'.format(self.url_name, obj_id)
@property
def url_name(self):
return self._url_name
@property
def api(self):
return self._api
def query(self):
"Create a query for objects of this type"
return Query(self)
def get(self, object_id):
"Fetch a single object by its identification"
resp = self.api.get(self._url_of(obj_id=object_id))
return self._object_class(self.api, self, resp['data'])
def create(self, data):
"Create a new object with the given data"
resp = self.api.post(self.url_name, data)
return self._object_class(self.api, self, resp['data'])
class BillogramObject(SimpleObject):
"""Represents a billogram object on the Billogram service
In addition to the basic methods of the SimpleObject remote object class,
also provides specialized methods to perform events on billogram objects.
See the online documentation for the actual structure of billogram objects.
"""
__slots__ = ()
def perform_event(self, evt_name, evt_data=None):
"""Perform a generic state transition event on billogram object
"""
url = '{}/command/{}'.format(self._url, evt_name)
resp = self._api.post(url, evt_data)
self._data = resp['data']
return self
def create_payment(self, amount):
"""Create a manual payment on billogram
Only possible in "Unpaid" state.
"""
assert amount > 0
return self.perform_event('payment', {'amount': amount})
def credit_amount(self, amount):
"""Credit a specific amount of the billogram
Only possible in states "Unpaid", "Sold" and "Ended".
"""
assert amount > 0
return self.perform_event(
'credit',
{
'mode': 'amount',
'amount': amount
}
)
def credit_full(self):
"""Credit the full, original amount of the billogram
Only possible in states "Unpaid", "Sold" and "Ended".
"""
return self.perform_event('credit', {'mode': 'full'})
def credit_remaining(self):
"""Credit the remaining unpaid amount of the billogram
Only possible in states "Unpaid", "Sold" and "Ended".
"""
return self.perform_event('credit', {'mode': 'remaining'})
def send_message(self, message):
"""Send a message to the recipient of the billogram
Possible from all states, except on deleted billograms.
"""
return self.perform_event('message', {'message': message})
def send_to_collector(self):
"""Send the billogram to the collection agency
Only possible from state "Unpaid".
"""
return self.perform_event('collect')
def send_to_factoring(self):
"""Send the billogram to the factoring agency to be sold
Only possible from state "Unattested".
"""
return self.perform_event('sell')
def send_reminder(self, method=None):
"""Send a reminder to the recipient
'method' is the type of reminder to be sent:
- "Email"
- "Letter".
Only possible from state "Unpaid".
"""
if method:
assert method in ('Email', 'Letter')
return self.perform_event('remind', {'method': method})
return self.perform_event('remind')
def send(self, method):
"""Send the billogram to the recipient
'method' is the medium to send the billogram by:
- "Email"
- "Letter"
- "Email+Letter".
Only possible from state "Unattested".
"""
assert method in ('Email', 'Letter', 'Email+Letter')
return self.perform_event('send', {'method': method})
def resend(self, method=None):
"""Send the billogram to the recipient again
'method' is the medium to send the billogram by:
- "Email"
- "Letter".
Only possible from state "Unpaid".
"""
if method:
assert method in ('Email', 'Letter')
return self.perform_event('resend', {'method': method})
return self.perform_event('resend')
def get_invoice_pdf(self, letter_id=None, invoice_no=None):
"""Fetch the PDF content for a specific invoice on this billogram
"""
import base64
url = '{}.pdf'.format(self._url)
params = {}
if letter_id:
params['letter_id'] = letter_id
if invoice_no:
params['invoice_no'] = invoice_no
resp = self._api.get(
url,
params,
expect_content_type='application/json'
)
return base64.b64decode(resp['data']['content'])
def get_attachment_pdf(self, letter_id=None, invoice_no=None):
"""Fetch the PDF attachment for the billogram
"""
import base64
url = '{}/attachment.pdf'.format(self._url)
resp = self._api.get(url, expect_content_type='application/json')
return base64.b64decode(resp['data']['content'])
def attach_pdf(self, filepath):
"""Attach a PDF to the billogram
"""
import base64
import os
with file(filepath) as f:
content = f.read()
filename = os.path.basename(filepath)
return self.perform_event(
'attach',
{
'content': base64.b64encode(content),
'filename': filename
}
)
def writeoff(self):
"""Write-off remaining fees from a billogram.
"""
return self.perform_event('writeoff')
class BillogramQuery(Query):
"""Represents a query for billogram objects
"""
def filter_state_any(self, *states):
"Find billogram objects with any state of the listed ones"
if len(states) == 1 and (
isinstance(states[0], list) or
isinstance(states[0], tuple) or
isinstance(states[0], set) or
isinstance(states[0], frozenset)
):
states = states[0]
assert all(isinstance(s, basestring) for s in states)
return self.filter_field('state', ','.join(states))
class BillogramClass(SimpleClass):
"""Represents the collection of billogram objects on the Billogram service
In addition to the methods of the SimpleClass collection wrapper, also
provides specialized creation methods to create billogram objects and state
transition them immediately.
"""
_object_class = BillogramObject
def __init__(self, api):
super(BillogramClass, self).__init__(api, 'billogram', 'id')
def query(self):
"Create a query for billogram objects"
return BillogramQuery(self)
def create_and_send(self, data, method):
"""Create the billogram and send it to the recipient in one operation
'method' is the medium to send the billogram by:
- "Email"
- "Letter"
- "Email+Letter".
New billogram will be in state "Unpaid" or "Ended" (if the total sum
would be zero).
"""
assert method in ('Email', 'Letter', 'Email+Letter')
billogram = self.create(data)
try:
billogram.send(method)
except Exception as e:
billogram.delete()
raise e
return billogram
def create_and_sell(self, data):
"""Create the billogram and send it to factoring in one operation
New billogram will be in state "Factoring".
"""
data['_event'] = 'sell'
billogram = self.create(data)
return billogram
# make an exportable namespace-class with all the exceptions
BillogramExceptions = type(
str('BillogramExceptions'),
(),
{
nm: cl for
nm, cl in
globals().items() if
isinstance(cl, type) and issubclass(cl, BillogramAPIError)
}
)
# just the BillogramAPI class and the exceptions are really part
# of the call API of this module
__all__ = ['BillogramAPI', 'BillogramExceptions']