No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

sqlsetup.py 7.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. # -*- coding: utf-8 -*-
  2. import sqlalchemy as sql
  3. import re # Converting string to sqlalchemy types
  4. from Database import sqlutils
  5. def init_db(dbconfname='default', alchemy_logs=None, schema=None):
  6. dbe = sqlutils.get_engine(dbconfname, alchemy_logs)
  7. meta = sqlutils.meta(dbe)
  8. meta.reflect()
  9. meta.drop_all(dbe)
  10. # refresh meta (maybe useless)
  11. meta = sqlutils.meta(dbe)
  12. meta.reflect()
  13. if schema is None:
  14. schema = get_schema()
  15. for table in schema:
  16. topt = table.copy()
  17. del topt['columns']
  18. name = topt['name']
  19. del topt['name']
  20. cur_table = sql.Table(name, meta, **topt)
  21. for col in table['columns']:
  22. cur_col = create_column(**col)
  23. cur_table.append_column(cur_col)
  24. meta.create_all(bind=dbe)
  25. def get_schema():
  26. tables = []
  27. default_columns = [
  28. {"name": "uid", "type": "INTEGER", "extra": {"foreignkey": "uids.uid", "nullable": False, "primarykey": True}},
  29. {"name": "name", "type": "VARCHAR(50)", "extra": {"nullable": False, "unique": True}},
  30. {"name": "string", "type": "TEXT"},
  31. {"name": "help", "type": "TEXT"},
  32. {"name": "rank", "type": "INTEGER"},
  33. {"name": "date_create", "type": "DATETIME"},
  34. {"name": "date_update", "type": "DATETIME"},
  35. ]
  36. # Table listing all objects created by lodel, giving them an unique id
  37. uids = {
  38. "name": "uids",
  39. "columns": [
  40. {"name": "uid", "type": "INTEGER", "extra": {"nullable": False, "primarykey": True, 'autoincrement': True}},
  41. {"name": "table", "type": "VARCHAR(50)"}
  42. ]
  43. }
  44. tables.append(uids)
  45. # Table listing the classes
  46. em_class = {"name": "em_class"}
  47. em_class['columns'] = default_columns + [
  48. {"name": "classtype", "type": "VARCHAR(50)"},
  49. {"name": "sortcolumn", "type": "VARCHAR(50)", "extra": {"default": "rank"}},
  50. {"name": "icon", "type": "INTEGER"},
  51. ]
  52. tables.append(em_class)
  53. # Table listing the types
  54. em_type = {"name": "em_type"}
  55. em_type['columns'] = default_columns + [
  56. {"name": "class_id", "type": "INTEGER", "extra": {"foreignkey": "em_class.uid", "nullable": False}},
  57. {"name": "sortcolumn", "type": "VARCHAR(50)", "extra": {"default": "rank"}},
  58. {"name": "icon", "type": "INTEGER"},
  59. ]
  60. tables.append(em_type)
  61. # relation between types: which type can be a child of another
  62. em_type_hierarchy = {"name": "em_type_hierarchy"}
  63. em_type_hierarchy['columns'] = [
  64. {"name": "superior_id", "type": "INTEGER", "extra": {"foreignkey": "em_type.uid", "nullable": False, "primarykey": True}},
  65. {"name": "subordinate_id", "type": "INTEGER", "extra": {"foreignkey": "em_type.uid", "nullable": False, "primarykey": True}},
  66. {"name": "nature", "type": "VARCHAR(50)", "extra": {"primarykey": True}},
  67. ]
  68. tables.append(em_type_hierarchy)
  69. # Table listing the fieldgroups of a class
  70. em_fieldgroup = {"name": "em_fieldgroup"}
  71. em_fieldgroup['columns'] = default_columns + [
  72. {"name": "class_id", "type": "INTEGER", "extra": {"foreignkey": "em_class.uid", "nullable": False}},
  73. ]
  74. tables.append(em_fieldgroup)
  75. # Table listing the fields of a fieldgroup
  76. em_field = {"name": "em_field"}
  77. em_field['columns'] = default_columns + [
  78. {"name": "fieldtype", "type": "VARCHAR(50)", "extra": {"nullable": False}},
  79. {"name":"fieldtype_opt", "type":"VARCHAR(200)"},
  80. {"name": "fieldgroup_id", "type": "INTEGER", "extra": {"foreignkey": "em_fieldgroup.uid", "nullable": False}},
  81. {"name": "rel_to_type_id", "type": "INTEGER", "extra": {"foreignkey": "em_type.uid", "nullable": True, "server_default": sql.text('NULL')}}, # if relational: type this field refer to
  82. {"name": "rel_field_id", "type": "INTEGER", "extra": {"foreignkey": "em_type.uid", "nullable": True, "server_default": sql.text('NULL')}}, # if relational: field that specify the rel_to_type_id
  83. {"name": "optional", "type": "BOOLEAN"},
  84. {"name": "internal", "type": "BOOLEAN"},
  85. {"name": "icon", "type": "INTEGER"},
  86. ]
  87. tables.append(em_field)
  88. # selected field for each type
  89. em_field_type = {"name": "em_field_type"}
  90. em_field_type['columns'] = [
  91. {"name": "type_id", "type": "INTEGER", "extra": {"foreignkey": "em_type.uid", "nullable": False, "primarykey": True}},
  92. {"name": "field_id", "type": "INTEGER", "extra": {"foreignkey": "em_field.uid", "nullable": False, "primarykey": True}},
  93. ]
  94. tables.append(em_field_type)
  95. # Table of the objects created by the user (instance of the types)
  96. objects = {
  97. "name": "objects",
  98. "columns": [
  99. {"name": "uid", "type": "INTEGER", "extra": {"foreignkey": "uids.uid", "nullable": False, "primarykey": True}},
  100. {"name": "string", "type": "VARCHAR(50)"},
  101. {"name": "class_id", "type": "INTEGER", "extra": {"foreignkey": "em_class.uid"}},
  102. {"name": "type_id", "type": "INTEGER", "extra": {"foreignkey": "em_type.uid"}},
  103. {"name": "date_create", "type": "DATETIME"},
  104. {"name": "date_update", "type": "DATETIME"},
  105. {"name": "history", "type": "TEXT"}
  106. ]
  107. }
  108. tables.append(objects)
  109. # Table listing all files
  110. files = {
  111. "name": "files",
  112. "columns": [
  113. {"name": "uid", "type": "INTEGER", "extra": {"foreignkey": "uids.uid", "nullable": False, "primarykey": True}},
  114. {"name": "field1", "type": "VARCHAR(50)"}
  115. ]
  116. }
  117. tables.append(files)
  118. return tables
  119. def create_column(**kwargs):
  120. #Converting parameters
  121. if 'type_' not in kwargs and 'type' in kwargs:
  122. kwargs['type_'] = string_to_sqla_type(kwargs['type'])
  123. del kwargs['type']
  124. if 'extra' in kwargs:
  125. # put the extra keys in kwargs
  126. for exname in kwargs['extra']:
  127. kwargs[exname] = kwargs['extra'][exname]
  128. del kwargs['extra']
  129. if 'foreignkey' in kwargs:
  130. # Instanciate a fk
  131. foreignkey = sql.ForeignKey(kwargs['foreignkey'])
  132. del kwargs['foreignkey']
  133. else:
  134. foreignkey = None
  135. if 'primarykey' in kwargs:
  136. # renaming primary_key in primarykey in kwargs
  137. kwargs['primary_key'] = kwargs['primarykey']
  138. del kwargs['primarykey']
  139. col = sql.Column(**kwargs)
  140. if foreignkey is not None:
  141. col.append_foreign_key(foreignkey)
  142. return col
  143. def string_to_sqla_type(strtype):
  144. """ Convert a string to an sqlAlchemy column type """
  145. if 'VARCHAR' in strtype:
  146. return string_to_varchar(strtype)
  147. else:
  148. try:
  149. return getattr(sql, strtype)
  150. except AttributeError:
  151. raise NameError("Unknown type '" + strtype + "'")
  152. def string_to_varchar(vstr):
  153. """ Convert a string like 'VARCHAR(XX)' (with XX an integer) to a SqlAlchemy varchar type"""
  154. check_length = re.search(re.compile('VARCHAR\(([\d]+)\)', re.IGNORECASE), vstr)
  155. column_length = int(check_length.groups()[0]) if check_length else None
  156. return sql.VARCHAR(length=column_length)