etl.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. #!/usr/bin/env python
  2. __doc__ = """
  3. (c) 2018 - 2021 data-transport
  4. steve@the-phi.com, The Phi Technology LLC
  5. https://dev.the-phi.com/git/steve/data-transport.git
  6. This program performs ETL between 9 supported data sources : Couchdb, Mongodb, Mysql, Mariadb, PostgreSQL, Netezza,Redshift, Sqlite, File
  7. LICENSE (MIT)
  8. Copyright 2016-2020, The Phi Technology LLC
  9. 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:
  10. The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
  11. 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.
  12. Usage :
  13. transport --config <path-to-file.json> --procs <number-procs>
  14. @TODO: Create tables if they don't exist for relational databases
  15. example of configuration :
  16. 1. Move data from a folder to a data-store
  17. transport [--folder <path> ] --config <config.json> #-- assuming the configuration doesn't have folder
  18. transport --folder <path> --provider <postgresql|mongo|sqlite> --<database|db> <name> --table|doc <document_name>
  19. In this case the configuration should look like :
  20. {folder:..., target:{}}
  21. 2. Move data from one source to another
  22. transport --config <file.json>
  23. {source:{..},target:{..}} or [{source:{..},target:{..}},{source:{..},target:{..}}]
  24. """
  25. import pandas as pd
  26. import numpy as np
  27. import json
  28. import sys
  29. import transport
  30. import time
  31. import os
  32. from multiprocessing import Process
  33. # SYS_ARGS = {}
  34. # if len(sys.argv) > 1:
  35. # N = len(sys.argv)
  36. # for i in range(1,N):
  37. # value = None
  38. # if sys.argv[i].startswith('--'):
  39. # key = sys.argv[i][2:] #.replace('-','')
  40. # SYS_ARGS[key] = 1
  41. # if i + 1 < N:
  42. # value = sys.argv[i + 1] = sys.argv[i+1].strip()
  43. # if key and value and not value.startswith('--'):
  44. # SYS_ARGS[key] = value
  45. # i += 2
  46. class Transporter(Process):
  47. """
  48. The transporter (Jason Stathem) moves data from one persistant store to another
  49. - callback functions
  50. :onFinish callback function when finished
  51. :onError callback function when an error occurs
  52. :source source data specification
  53. :target destination(s) to move the data to
  54. """
  55. def __init__(self,**_args):
  56. super().__init__()
  57. # self.onfinish = _args['onFinish']
  58. # self._onerror = _args['onError']
  59. self._source = _args['source']
  60. self._target = _args['target']
  61. #
  62. # Let's insure we can support multiple targets
  63. self._target = [self._target] if type(self._target) != list else self._target
  64. pass
  65. def run(self):
  66. _reader = transport.get.etl(source=self._source,target=self._target)
  67. #
  68. if 'cmd' in self._source or 'query' in self._source :
  69. _query = self._source['cmd'] if 'cmd' in self._source else self._source['query']
  70. return _reader.read(**_query)
  71. else:
  72. return _reader.read()
  73. # def _read(self,**_args):
  74. # """
  75. # This function
  76. # """
  77. # _reader = transport.factory.instance(**self._source)
  78. # #
  79. # # If arguments are provided then a query is to be executed (not just a table dump)
  80. # if 'cmd' in self._source or 'query' in self._source :
  81. # _query = self._source['cmd'] if 'cmd' in self._source else self._source['query']
  82. # return _reader.read(**_query)
  83. # else:
  84. # return _reader.read()
  85. # # return _reader.read() if 'query' not in self._source else _reader.read(**self._source['query'])
  86. # def _delegate_write(self,_data,**_args):
  87. # """
  88. # This function will write a data-frame to a designated data-store, The function is built around a delegation design pattern
  89. # :data data-frame or object to be written
  90. # """
  91. # if _data.shape[0] > 0 :
  92. # for _target in self._target :
  93. # if 'write' not in _target :
  94. # _target['context'] = 'write'
  95. # # _target['lock'] = True
  96. # else:
  97. # # _target['write']['lock'] = True
  98. # pass
  99. # _writer = transport.factory.instance(**_target)
  100. # _writer.write(_data,**_args)
  101. # if hasattr(_writer,'close') :
  102. # _writer.close()
  103. # def write(self,_df,**_args):
  104. # """
  105. # """
  106. # SEGMENT_COUNT = 6
  107. # MAX_ROWS = 1000000
  108. # # _df = self.read()
  109. # _segments = np.array_split(np.arange(_df.shape[0]),SEGMENT_COUNT) if _df.shape[0] > MAX_ROWS else np.array( [np.arange(_df.shape[0])])
  110. # # _index = 0
  111. # for _indexes in _segments :
  112. # _fwd_args = {} if not _args else _args
  113. # self._delegate_write(_df.iloc[_indexes],**_fwd_args)
  114. # time.sleep(1)
  115. # #
  116. # # @TODO: Perhaps consider writing up each segment in a thread/process (speeds things up?)
  117. # pass
  118. def instance(**_args):
  119. pthread = Transporter (**_args)
  120. pthread.start()
  121. return pthread
  122. pass
  123. # class Post(Process):
  124. # def __init__(self,**args):
  125. # super().__init__()
  126. # self.store = args['target']
  127. # if 'provider' not in args['target'] :
  128. # pass
  129. # self.PROVIDER = args['target']['type']
  130. # # self.writer = transport.factory.instance(**args['target'])
  131. # else:
  132. # self.PROVIDER = args['target']['provider']
  133. # self.store['context'] = 'write'
  134. # # self.store = args['target']
  135. # self.store['lock'] = True
  136. # # self.writer = transport.instance(**args['target'])
  137. # #
  138. # # If the table doesn't exists maybe create it ?
  139. # #
  140. # self.rows = args['rows']
  141. # # self.rows = args['rows'].fillna('')
  142. # def log(self,**_args) :
  143. # if ETL.logger :
  144. # ETL.logger.info(**_args)
  145. # def run(self):
  146. # _info = {"values":self.rows} if 'couch' in self.PROVIDER else self.rows
  147. # writer = transport.factory.instance(**self.store)
  148. # writer.write(_info)
  149. # writer.close()
  150. # class ETL (Process):
  151. # logger = None
  152. # def __init__(self,**_args):
  153. # super().__init__()
  154. # self.name = _args['id'] if 'id' in _args else 'UNREGISTERED'
  155. # # if 'provider' not in _args['source'] :
  156. # # #@deprecate
  157. # # self.reader = transport.factory.instance(**_args['source'])
  158. # # else:
  159. # # #
  160. # # # This is the new interface
  161. # # _args['source']['context'] = 'read'
  162. # # self.reader = transport.instance(**_args['source'])
  163. # #
  164. # # do we have an sql query provided or not ....
  165. # # self.sql = _args['source']['sql'] if 'sql' in _args['source'] else None
  166. # # self.cmd = _args['source']['cmd'] if 'cmd' in _args['source'] else None
  167. # # self._oargs = _args['target'] #transport.factory.instance(**_args['target'])
  168. # self._source = _args ['source']
  169. # self._target = _args['target']
  170. # self._source['context'] = 'read'
  171. # self._target['context'] = 'write'
  172. # self.JOB_COUNT = _args['jobs']
  173. # self.jobs = []
  174. # # self.logger = transport.factory.instance(**_args['logger'])
  175. # def log(self,**_args) :
  176. # if ETL.logger :
  177. # ETL.logger.info(**_args)
  178. # def run(self):
  179. # # if self.cmd :
  180. # # idf = self.reader.read(**self.cmd)
  181. # # else:
  182. # # idf = self.reader.read()
  183. # # idf = pd.DataFrame(idf)
  184. # # # idf = idf.replace({np.nan: None}, inplace = True)
  185. # # idf.columns = [str(name).replace("b'",'').replace("'","").strip() for name in idf.columns.tolist()]
  186. # # self.log(rows=idf.shape[0],cols=idf.shape[1],jobs=self.JOB_COUNT)
  187. # #
  188. # # writing the data to a designated data source
  189. # #
  190. # try:
  191. # _log = {"name":self.name,"rows":{"input":0,"output":0}}
  192. # _reader = transport.factory.instance(**self._source)
  193. # if 'table' in self._source :
  194. # _df = _reader.read()
  195. # else:
  196. # _df = _reader.read(**self._source['cmd'])
  197. # _log['rows']['input'] = _df.shape[0]
  198. # #
  199. # # Let's write the input data-frame to the target ...
  200. # _writer = transport.factory.instance(**self._target)
  201. # _writer.write(_df)
  202. # _log['rows']['output'] = _df.shape[0]
  203. # # self.log(module='write',action='partitioning',jobs=self.JOB_COUNT)
  204. # # rows = np.array_split(np.arange(0,idf.shape[0]),self.JOB_COUNT)
  205. # # #
  206. # # # @TODO: locks
  207. # # for i in np.arange(self.JOB_COUNT) :
  208. # # # _id = ' '.join([str(i),' table ',self.name])
  209. # # indexes = rows[i]
  210. # # segment = idf.loc[indexes,:].copy() #.to_dict(orient='records')
  211. # # _name = "partition-"+str(i)
  212. # # if segment.shape[0] == 0 :
  213. # # continue
  214. # # proc = Post(target = self._oargs,rows = segment,name=_name)
  215. # # self.jobs.append(proc)
  216. # # proc.start()
  217. # # self.log(module='write',action='working',segment=str(self.name),table=self.name,rows=segment.shape[0])
  218. # # while self.jobs :
  219. # # jobs = [job for job in proc if job.is_alive()]
  220. # # time.sleep(1)
  221. # except Exception as e:
  222. # print (e)
  223. # self.log(**_log)
  224. # def is_done(self):
  225. # self.jobs = [proc for proc in self.jobs if proc.is_alive()]
  226. # return len(self.jobs) == 0
  227. # def instance (**_args):
  228. # """
  229. # path to configuration file
  230. # """
  231. # _path = _args['path']
  232. # _config = {}
  233. # jobs = []
  234. # if os.path.exists(_path) :
  235. # file = open(_path)
  236. # _config = json.loads(file.read())
  237. # file.close()
  238. # if _config and type
  239. # def _instance(**_args):
  240. # """
  241. # :path ,index, id
  242. # :param _info list of objects with {source,target}`
  243. # :param logger
  244. # """
  245. # logger = _args['logger'] if 'logger' in _args else None
  246. # if 'path' in _args :
  247. # _info = json.loads((open(_args['path'])).read())
  248. # if 'index' in _args :
  249. # _index = int(_args['index'])
  250. # _info = _info[_index]
  251. # elif 'id' in _args :
  252. # _info = [_item for _item in _info if '_id' in _item and _item['id'] == _args['id']]
  253. # _info = _info[0] if _info else _info
  254. # else:
  255. # _info = _args['info']
  256. # if logger and type(logger) != str:
  257. # ETL.logger = logger
  258. # elif logger == 'console':
  259. # ETL.logger = transport.factory.instance(provider='console',context='write',lock=True)
  260. # if type(_info) in [list,dict] :
  261. # _info = _info if type(_info) != dict else [_info]
  262. # #
  263. # # The assumption here is that the objects within the list are {source,target}
  264. # jobs = []
  265. # for _item in _info :
  266. # _item['jobs'] = 5 if 'procs' not in _args else int(_args['procs'])
  267. # _job = ETL(**_item)
  268. # _job.start()
  269. # jobs.append(_job)
  270. # return jobs
  271. # else:
  272. # return None
  273. # if __name__ == '__main__' :
  274. # _info = json.loads(open (SYS_ARGS['config']).read())
  275. # index = int(SYS_ARGS['index']) if 'index' in SYS_ARGS else None
  276. # procs = []
  277. # for _config in _info :
  278. # if 'source' in SYS_ARGS :
  279. # _config['source'] = {"type":"disk.DiskReader","args":{"path":SYS_ARGS['source'],"delimiter":","}}
  280. # _config['jobs'] = 3 if 'jobs' not in SYS_ARGS else int(SYS_ARGS['jobs'])
  281. # etl = ETL (**_config)
  282. # if index is None:
  283. # etl.start()
  284. # procs.append(etl)
  285. # elif _info.index(_config) == index :
  286. # # print (_config)
  287. # procs = [etl]
  288. # etl.start()
  289. # break
  290. # #
  291. # #
  292. # N = len(procs)
  293. # while procs :
  294. # procs = [thread for thread in procs if not thread.is_done()]
  295. # if len(procs) < N :
  296. # print (["Finished ",(N-len(procs)), " remaining ", len(procs)])
  297. # N = len(procs)
  298. # time.sleep(1)
  299. # # print ("We're done !!")