iqd_convert.py 4.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. from pathlib import Path
  2. import re
  3. import plac
  4. class IqdConverter:
  5. commands = ['convert']
  6. target_schema_by_type = {
  7. 'csv': 'staging',
  8. 'ims': 'ims'
  9. }
  10. output_dir = 'C:\\GlobalCube_LOCOSOFT\\System\\LOCOSOFT\\SQL\\schema\\LOCOSOFT\\views'
  11. def convert(self, iqd_file, target_type='csv'):
  12. query = self.get_query_from_iqdfile(iqd_file)
  13. query = self.cleanup_query(query)
  14. output_file = iqd_file[:-4] + '.sql'
  15. with open(output_file, 'w') as fwh:
  16. fwh.write(query)
  17. table = Path(iqd_file).stem
  18. schema = self.target_schema_by_type[target_type]
  19. create_view = "SET QUOTED_IDENTIFIER ON \nGO\nSET ANSI_NULLS ON \nGO\n" + \
  20. f"CREATE VIEW {schema}.{table} AS\n\n{query}\n" + \
  21. "GO\nSET QUOTED_IDENTIFIER OFF \nGO\nSET ANSI_NULLS OFF \nGO\n\nGO"
  22. view_file = f"{self.output_dir}\\{schema}.{table}.sql"
  23. with open(view_file, 'w') as fwh:
  24. fwh.write(create_view)
  25. def get_query_from_iqdfile(self, iqd_file):
  26. if iqd_file[-4:].lower() == '.imr':
  27. iqd_file = iqd_file[:-4] + '.iqd'
  28. iqd = open(iqd_file, 'r', encoding='latin-1').read()
  29. res = re.findall(r'BEGIN SQL\n(.*)\n\nEND SQL', iqd, re.MULTILINE | re.DOTALL)
  30. query = ''
  31. if len(res) > 0:
  32. query = res[0]
  33. columns = re.findall(r'COLUMN,\d+,(.*)', iqd)
  34. col_keys = [f'c{i}' for i in range(len(columns), 0, -1)]
  35. col_names = [f'"{c}"' for c in reversed(columns)]
  36. for col_key, col_name in zip(col_keys, col_names):
  37. query = re.sub(col_key + r'([^\d])', col_name + r'\1', query)
  38. return query
  39. def cleanup_query(self, query):
  40. query = re.sub(r' from (\d+) for (\d+)\)', r', \1, \2)', query)
  41. query = query.replace(' || ', ' + ')
  42. query = query.replace('truncate(', 'rtrim(')
  43. query = query.replace('char_length(', 'len(')
  44. query = query.replace('ascii(', 'convert(varchar(50), ')
  45. query = query.replace('extract(DAY FROM ', 'day(')
  46. query = query.replace('extract(MONTH FROM ', 'month(')
  47. query = query.replace('extract(YEAR FROM ', 'year(')
  48. query = query.replace('cdatetime(', 'convert(datetime, ')
  49. query = query.replace('cast_float(', 'convert(float, ')
  50. query = query.replace('sy_right(', 'right(')
  51. query = query.replace('od_left(', 'left(')
  52. query = query.replace('length(', 'len(')
  53. query = query.replace('{hour}', 'hour')
  54. query = query.replace('{minute}', 'minute')
  55. query = query.replace('{weekday}', 'weekday')
  56. query = query.replace('@CURRENT_DATE', 'getdate()')
  57. query = query.replace('cdate(', '(')
  58. query = re.sub(r'intdiv\(([^\)]+)\,1\)', r'\1', query)
  59. query = re.sub(r'XCOUNT\(([^\)]+) for ', r'COUNT(\1) OVER (partition by ', query)
  60. query = re.sub(r'XSUM\(([^\)]+) for ', r'SUM(\1) OVER (partition by ', query)
  61. query = re.sub(r'XMIN\(([^\)]+) for ', r'MIN(\1) OVER (partition by ', query)
  62. query = re.sub(r'XMAX\(([^\)]+) for ', r'MAX(\1) OVER (partition by ', query)
  63. query = re.sub(r'QSS\.\"[^\"]+\\([^\\]+)\.ims\"', r'"ims"."\1"', query)
  64. query = re.sub(r"DATE '([\d-]+)'", r"convert(date, '\1')", query)
  65. query = re.sub(r"TIMESTAMP '([\d\s\-\:\.]+)'", r"convert(datetime, '\1')", query)
  66. query = re.sub(r'asciiz\(([^\,]*)\,\d+\)', r'convert(varchar(50), \1)', query)
  67. query = re.sub(r'asciiz\(([^\+]*)\)', r'convert(varchar(50), \1)', query)
  68. query = re.sub(r'convert\(varchar\(50\)\, ([^,]+)\,\d+\)', r'convert(varchar(50), \1)', query)
  69. query = query.replace('cdate((convert(float, ', 'convert(datetime, ((')
  70. query = re.sub(r'(order by .*)', r'-- \1', query)
  71. return query
  72. def run_folder(base_dir: str):
  73. files = sorted([(f.stat().st_mtime, f) for f in Path(base_dir).rglob('*.iqd')])
  74. print(files)
  75. if __name__ == '__main__':
  76. IqdConverter().convert('C:\\GlobalCube_LOCOSOFT\\System\\LOCOSOFT\\IQD\\NW\\LOC_Belege_NW_GW_VK_Stk_FIBU.iqd')
  77. # plac.Interpreter.call(IqdConverter)