1
- import requests
2
- import json
3
1
import configparser
4
- from uuid import UUID
5
- from pathlib import Path , PurePath
2
+ import csv
3
+ import io
4
+ import json
6
5
import logging
6
+ from pathlib import Path , PurePath
7
+ import re
8
+ from uuid import UUID
9
+
10
+ import requests
11
+
7
12
8
13
class Api :
9
-
10
- def __init__ (self , api_key : str = None , base : str = None ):
14
+ """
15
+ TODO:
16
+ - Add information about what an API key is, how to get one, and how to use Swagger
17
+ - Reference PEP standards for class docstrings
18
+ """
19
+ def __init__ (self , api_key : str = None , base : str = None , log : bool = None ):
11
20
"""
12
21
API can be initialized directly by passing string, if not it looks for prevedere_api.ini in current working directory.
13
22
Copy the prevedere_api.ini.example file and remove `.example` from the end.
14
23
Change the api key to your key.
15
24
"""
25
+ if log :
26
+ if type (log ) == int :
27
+ level = log
28
+ else :
29
+ level = logging .INFO
30
+ self .log = log
31
+ else :
32
+ self .log = False
33
+ level = logging .WARNING
34
+ logging .basicConfig (format = '%(levelname)s-%(message)s' , level = level )
35
+
36
+
16
37
if api_key is None :
17
38
try :
18
- assert PurePath (__file__ ).name == 'prevedere .py'
39
+ assert PurePath (__file__ ).name == 'api .py'
19
40
cwd = PurePath (__file__ ).parent
20
41
except AssertionError as e :
21
42
logging .exception ('Prevedere.Api not initialized from prevedere.py' )
@@ -28,6 +49,8 @@ def __init__(self, api_key: str = None, base: str = None):
28
49
config .read (filepath )
29
50
try :
30
51
api_key = config ['keys' ]['api key' ]
52
+ if 'base' in config ['keys' ]:
53
+ base = config ['keys' ]['base' ]
31
54
assert api_key != "1234567890abcdef1234567890abcdef"
32
55
except KeyError as e :
33
56
logging .exception (f'API key not found in { filepath } : ' + repr (e ))
@@ -47,34 +70,46 @@ def __init__(self, api_key: str = None, base: str = None):
47
70
try :
48
71
self .api_key = str (UUID (api_key ))
49
72
self .context = self .fetch ('/context' )
50
- logging .debug (f"Hello { self .context ['User' ]['FirstName' ]} , you're now connected to the { self .context ['Company' ]['Prefix' ]} instance." )
51
73
except (TypeError , ValueError , requests .exceptions .HTTPError ) as e :
52
74
raise ApiKeyError (f"'{ api_key } ' is not a valid API Key. " + \
53
75
"Please check the config file or string that was passed to the constructor and try again." ) from e
54
76
logging .exception ()
77
+
78
+ @property
79
+ def url (self ):
80
+ return f'https://{ self .base } .prevedere.com'
55
81
56
- def fetch (self , path : str , payload : dict = None ) -> dict :
82
+ def fetch (self , path : str , payload : dict = None , method : str = 'GET' , data : str = None ) -> dict :
57
83
if payload is None :
58
84
payload = {}
59
85
payload ['ApiKey' ] = self .api_key
60
- url = f'https:// { self . base } .prevedere.com { path } '
86
+
61
87
try :
62
- r = requests .get (url , params = payload )
88
+ if method == 'GET' :
89
+ r = requests .get (f'{ self .url } { path } ' , params = payload )
90
+ elif method == 'POST' :
91
+ r = requests .post (f'{ self .url } { path } ' , params = payload , data = data )
63
92
r .raise_for_status ()
93
+
94
+ if self .log :
95
+ try :
96
+ endpoint = re .search ('^\/\w*\/?' , path )[0 ]
97
+ except :
98
+ endpoint = 'endpoint'
99
+ else :
100
+ logging .info (f'{ method } request to { endpoint } took { r .elapsed .total_seconds ():.2f} seconds (status code: { r .status_code } )' )
101
+ return r .json ()
102
+
64
103
except requests .exceptions .HTTPError as e :
65
- logging .warn ('HTTP Error: ' + repr (r .json ()))
66
- raise
104
+ logging .exception (r .text )
67
105
except requests .exceptions .ConnectionError as e :
68
106
logging .exception ('Connection Error' )
69
107
except requests .exceptions .Timeout as e :
70
108
logging .exception ('Timeout Error' )
71
109
except requests .exceptions .RequestException as e :
72
- logging .exception ('Requests Error' )
73
-
74
- try :
75
- return r .json ()
110
+ logging .exception ('Requests Error' )
76
111
except json .decoder .JSONDecodeError as e :
77
- logging .exception ("JSON Error " )
112
+ logging .exception ("Could not read response as JSON " )
78
113
79
114
def indicator (self , provider : str , provider_id : str ) -> dict :
80
115
path = f'/indicator/{ provider } /{ provider_id } '
@@ -182,13 +217,128 @@ def workbench(self, workbench_id: str) -> dict:
182
217
path = f'/workbench/{ workbench_id } '
183
218
return self .fetch (path )
184
219
220
+ # POST
221
+ def get_integrations (self ):
222
+ return self .fetch ('/clientdimensions' )
223
+
224
+ def get_client_dimensions (self , client_dimension_group_id ):
225
+ integrations = self .get_integrations ()
226
+ for i in integrations :
227
+ if i ['Id' ] == client_dimension_group_id :
228
+ return i
229
+ raise Exception (f'ClientDimensionGroupId not found: { client_dimension_group_id } ' )
230
+
231
+ def get_fields (self , client_dimension_group_id ):
232
+ dimensions = self .get_client_dimensions (client_dimension_group_id )
233
+ fields = list (dimensions ['Mapping' ].values ()) + ['Measure' , 'Date' , 'Value' ]
234
+ return set (fields )
235
+
236
+ @staticmethod
237
+ def make_csv (data : list , fields : set ) -> str :
238
+ """
239
+ Turns data into a CSV string to be uploaded.
240
+ Data must contain the **specific dimensions** for the integration job,
241
+ as well as the fields, "Measure", "Date", and "Value".
242
+ Date is in format 'YYYY-MM-DD'
243
+ :param data: A list of records with {'key':'value'} entries.
244
+ e.g. [
245
+ {
246
+ 'Region':'East',
247
+ 'Product':'Product 1',
248
+ 'Date': '2019-09-01',
249
+ 'Measure':'Sales',
250
+ 'Value':100
251
+ },
252
+ {
253
+ 'Region':'East',
254
+ 'Product':'Product 1',
255
+ 'Date': '2019-10-01',
256
+ 'Measure':'Sales',
257
+ 'Value':200
258
+ },
259
+ ]
260
+ :type fields: A list or set of keys in each record.
261
+ e.g. set(['Region', 'Product', 'Date', 'Measure', 'Value'])
262
+
263
+ returns: CSV string
264
+
265
+ example: Api.make_csv(data=[], fields=[])
266
+ """
267
+ s = io .StringIO (newline = '' )
268
+ writer = csv .DictWriter (s , fieldnames = fields )
269
+ writer .writeheader ()
270
+ for d in data :
271
+ writer .writerow (d )
272
+ return s .getvalue ()
273
+
274
+ @staticmethod
275
+ def get_csv_fields (csv_data ):
276
+ reader = csv .DictReader (io .StringIO (csv_data ))
277
+ return set (reader .fieldnames )
278
+
279
+ @staticmethod
280
+ def check_post_response (response ):
281
+ if response ['Success' ] == True :
282
+ logging .info ('POST suceeded.' )
283
+ else :
284
+ raise requests .exceptions .RequestException (f"""
285
+ POST Failed:
286
+ { response ['Message' ]}
287
+ """
288
+ )
289
+
290
+ def validate_data (
291
+ self ,
292
+ data : str ,
293
+ client_dimension_group_id : str ,
294
+ ):
295
+ # Make sure fields match up
296
+ app_fields = self .get_fields (client_dimension_group_id )
297
+ csv_fields = self .get_csv_fields (data )
298
+ assert app_fields == csv_fields , f"""
299
+ Fields do not match.
300
+ App Fields: { app_fields }
301
+ CSV Fields: { csv_fields }
302
+ """
303
+
304
+ response = self .fetch (
305
+ method = 'POST' ,
306
+ path = f'/validateclientdata/{ client_dimension_group_id } ' ,
307
+ data = {'InlineData' : data }
308
+ )
309
+
310
+ self .check_post_response (response )
311
+
312
+ def upload_data (
313
+ self ,
314
+ data : str ,
315
+ client_dimension_group_id : str ,
316
+ should_delete_existing_records : bool = True ,
317
+ should_replace_if_record_exists : bool = True
318
+ ):
319
+ payload = {
320
+ # If false, becomes additive to existing records
321
+ 'ShouldReplaceRecordIfExists' : should_replace_if_record_exists ,
322
+ }
323
+
324
+ response = self .fetch (
325
+ method = 'POST' ,
326
+ path = f'/importclientdata/{ client_dimension_group_id } /{ should_delete_existing_records } ' ,
327
+ data = {'InlineData' :data },
328
+ payload = payload
329
+ )
330
+
331
+ self .check_post_response (response )
332
+
333
+
185
334
class ApiKeyError (ValueError ):
186
335
'''Raise when API is improperly formatted or invalid'''
187
336
def __init__ (self , message = None ):
188
337
if message is None :
189
338
message = "An error occured with the provided API Key."
190
339
self .message = message
191
340
341
+
192
342
def main ():
193
343
pass
194
344
0 commit comments