|
@@ -2,6 +2,7 @@
|
2
|
2
|
|
3
|
3
|
import copy
|
4
|
4
|
import _mysql as mysqlclient
|
|
5
|
+import _mysql_exceptions
|
5
|
6
|
|
6
|
7
|
import EditorialModel
|
7
|
8
|
|
|
@@ -12,6 +13,15 @@ import EditorialModel
|
12
|
13
|
#
|
13
|
14
|
# The create_default_table method will call both methods to create the object and relation tables
|
14
|
15
|
#
|
|
16
|
+# Supported operations :
|
|
17
|
+# - EmClass creation
|
|
18
|
+# - EmClass deletion (untested)
|
|
19
|
+# - EmField creation
|
|
20
|
+# - EmField deletion (untested)
|
|
21
|
+#
|
|
22
|
+# Unsupported operations :
|
|
23
|
+# - EmClass rename
|
|
24
|
+# - EmField rename
|
15
|
25
|
|
16
|
26
|
## @brief Modify a MySQL database given editorial model changes
|
17
|
27
|
class MysqlMigrationHandler(EditorialModel.migrationhandler.dummy.DummyMigrationHandler):
|
|
@@ -24,32 +34,45 @@ class MysqlMigrationHandler(EditorialModel.migrationhandler.dummy.DummyMigration
|
24
|
34
|
# @param user str : The db user
|
25
|
35
|
# @param password str : The db password
|
26
|
36
|
# @param db str : The db name
|
27
|
|
- def __init__(self, host, user, password, db, debug = False, dryrun = False):
|
|
37
|
+ def __init__(self, host, user, password, db, db_engine = 'InnoDB', foreign_keys = True, debug = False, dryrun = False, drop_if_exists = False):
|
28
|
38
|
#Connect to MySQL
|
29
|
39
|
self.db = mysqlclient.connect(host=host, user=user, passwd=password, db=db)
|
30
|
40
|
self.debug = debug
|
31
|
41
|
self.dryrun = dryrun
|
|
42
|
+ self.db_engine = db_engine
|
|
43
|
+ self.foreign_keys = foreign_keys if db_engine == 'InnoDB' else False
|
|
44
|
+ self.drop_if_exists = drop_if_exists
|
32
|
45
|
#Create default tables
|
|
46
|
+ self._create_default_tables(self.drop_if_exists)
|
33
|
47
|
pass
|
34
|
|
-
|
35
|
|
- def register_change(self, em, uid, initial_state, new_state):
|
|
48
|
+
|
|
49
|
+ ## @brief Modify the db given an EM change
|
|
50
|
+ def register_change(self, em, uid, initial_state, new_state, engine = None):
|
|
51
|
+ if engine is None:
|
|
52
|
+ engine = self.db_engine
|
36
|
53
|
if isinstance(em.component(uid), EditorialModel.classes.EmClass):
|
37
|
54
|
if initial_state is None:
|
38
|
|
- self.create_emclass_table(em, uid)
|
39
|
|
- pass
|
|
55
|
+ #EmClass creation
|
|
56
|
+ self.create_emclass_table(em, uid, engine)
|
40
|
57
|
elif new_state is None:
|
41
|
|
- #table deletion
|
42
|
|
- pass
|
|
58
|
+ #EmClass deletion
|
|
59
|
+ self.delete_emclass_table(em, uid)
|
43
|
60
|
elif isinstance(em.component(uid), EditorialModel.fields.EmField):
|
|
61
|
+ emfield = em.component(uid)
|
44
|
62
|
if initial_state is None:
|
45
|
|
- emfield = em.component(uid)
|
|
63
|
+ #EmField creation
|
46
|
64
|
if not(emfield.name in EditorialModel.classtypes.common_fields.keys()):
|
47
|
65
|
self.add_col_from_emfield(em,uid)
|
48
|
66
|
elif new_state is None:
|
49
|
|
- #column deletion
|
|
67
|
+ #EmField deletion
|
|
68
|
+ if not (emfield.name in EditorialModel.classtypes.common_fields.keys()):
|
|
69
|
+ self.del_col_from_emfield(em, uid)
|
50
|
70
|
pass
|
51
|
71
|
pass
|
52
|
72
|
|
|
73
|
+ def register_model_state(self, em, state_hash):
|
|
74
|
+ pass
|
|
75
|
+
|
53
|
76
|
## @brief Exec a query
|
54
|
77
|
def _query(self, query):
|
55
|
78
|
if self.debug:
|
|
@@ -69,19 +92,60 @@ class MysqlMigrationHandler(EditorialModel.migrationhandler.dummy.DummyMigration
|
69
|
92
|
emclass = emfield.em_class
|
70
|
93
|
tname = self._emclass2table_name(emclass)
|
71
|
94
|
self._add_column(tname, emfield.name, emfield.fieldtype_instance())
|
72
|
|
-
|
|
95
|
+ # Refresh the table triggers
|
73
|
96
|
cols_l = self._class2cols(emclass)
|
74
|
97
|
self._generate_triggers(tname, cols_l)
|
75
|
98
|
|
76
|
99
|
|
77
|
100
|
## @brief Given a class uid create the coressponding table
|
78
|
|
- def create_emclass_table(self, em, uid):
|
|
101
|
+ def create_emclass_table(self, em, uid, engine):
|
79
|
102
|
emclass = em.component(uid)
|
80
|
103
|
if not isinstance(emclass, EditorialModel.classes.EmClass):
|
81
|
104
|
raise ValueError("The given uid is not an EmClass uid")
|
82
|
|
-
|
83
|
105
|
pkname, pktype = self._common_field_pk
|
84
|
|
- self._create_table(self._emclass2table_name(emclass), pkname, pktype)
|
|
106
|
+ table_name = self._emclass2table_name(emclass)
|
|
107
|
+ self._create_table(table_name, pkname, pktype, engine=engine)
|
|
108
|
+
|
|
109
|
+ if self.foreign_keys:
|
|
110
|
+ self._add_fk(table_name, self._object_tname, pkname, pkname)
|
|
111
|
+
|
|
112
|
+ ## @brief Given an EmClass uid delete the corresponding table
|
|
113
|
+ def delete_emclass_table(self, em, uid):
|
|
114
|
+ emclass = emcomponent(uid)
|
|
115
|
+ if not isinstance(emclass, EditorialModel.classes.EmClass):
|
|
116
|
+ raise ValueError("The give uid is not an EmClass uid")
|
|
117
|
+ tname = self._idname_escape(self._emclass2table_name(emclass.name))
|
|
118
|
+ # Delete the table triggers to prevent errors
|
|
119
|
+ self._generate_triggers(tname, dict())
|
|
120
|
+
|
|
121
|
+ self._query("""DROP TABLE {table_name};""".format(table_name = tname))
|
|
122
|
+
|
|
123
|
+ ## @brief Given an EmField delete the corresponding column
|
|
124
|
+ # @param em Model : an @ref EditorialModel.model.Model instance
|
|
125
|
+ # @param uid int : an EmField uid
|
|
126
|
+ def delete_col_from_emfield(self, em, uid):
|
|
127
|
+ emfield = em.component(uid)
|
|
128
|
+ if not isinstance(emfield, EditorialModel.fields.EmField):
|
|
129
|
+ raise ValueError("The given uid is not an EmField uid")
|
|
130
|
+
|
|
131
|
+ emclass = emfield.em_class
|
|
132
|
+ tname = self._emclass2table_name(emclass)
|
|
133
|
+ # Delete the table triggers to prevent errors
|
|
134
|
+ self._generate_triggers(tname, dict())
|
|
135
|
+
|
|
136
|
+ self._del_column(tname, emfield.name)
|
|
137
|
+ # Refresh the table triggers
|
|
138
|
+ cols_ls = self._class2cols(emclass)
|
|
139
|
+ self._generate_triggers(tname, cols_l)
|
|
140
|
+
|
|
141
|
+ ## @brief Delete a column from a table
|
|
142
|
+ # @param tname str : The table name
|
|
143
|
+ # @param fname str : The column name
|
|
144
|
+ def _del_column(self, tname, fname):
|
|
145
|
+ tname = self._idname_escape(tname)
|
|
146
|
+ fname = self._idname_escape(fname)
|
|
147
|
+
|
|
148
|
+ self._query("""ALTER TABLE {table_name} DROP COLUMN {col_name};""".format(table_name = tname, col_name = fname))
|
85
|
149
|
|
86
|
150
|
## @brief Construct a table name given an EmClass instance
|
87
|
151
|
# @param emclass EmClass : An EmClass instance
|
|
@@ -100,10 +164,11 @@ class MysqlMigrationHandler(EditorialModel.migrationhandler.dummy.DummyMigration
|
100
|
164
|
## @brief Create object and relations tables
|
101
|
165
|
# @param drop_if_exist bool : If true drop tables if exists
|
102
|
166
|
def _create_default_tables(self, drop_if_exist = False):
|
|
167
|
+ if_exists = 'drop' if drop_if_exist else 'nothing'
|
103
|
168
|
#Object tablea
|
104
|
169
|
tname = self._object_tname
|
105
|
170
|
pk_name, pk_ftype = self._common_field_pk
|
106
|
|
- self._create_table(tname, pk_name, pk_ftype)
|
|
171
|
+ self._create_table(tname, pk_name, pk_ftype, engine=self.db_engine, if_exists = if_exists)
|
107
|
172
|
#Adding columns
|
108
|
173
|
cols = { fname: self._common_field_to_ftype(fname) for fname in EditorialModel.classtypes.common_fields }
|
109
|
174
|
for fname, ftype in cols.items():
|
|
@@ -115,7 +180,7 @@ class MysqlMigrationHandler(EditorialModel.migrationhandler.dummy.DummyMigration
|
115
|
180
|
#Relation table
|
116
|
181
|
tname = self._relation_tname
|
117
|
182
|
pk_name, pk_ftype = self._relation_pk
|
118
|
|
- self._create_table(tname, pk_name, pk_ftype)
|
|
183
|
+ self._create_table(tname, pk_name, pk_ftype, engine = self.db_engine, if_exists = if_exists)
|
119
|
184
|
#Adding columns
|
120
|
185
|
for fname, ftype in self._relation_cols.items():
|
121
|
186
|
self._add_column(tname, fname, ftype)
|
|
@@ -134,14 +199,15 @@ class MysqlMigrationHandler(EditorialModel.migrationhandler.dummy.DummyMigration
|
134
|
199
|
# @param charset str : The charset of this table
|
135
|
200
|
# @param if_exist str : takes values in ['nothing', 'drop']
|
136
|
201
|
# @return None
|
137
|
|
- def _create_table(self, table_name, pk_name, pk_ftype, engine = "MyISAM", charset = 'utf8', if_exists = 'nothing'):
|
|
202
|
+ def _create_table(self, table_name, pk_name, pk_ftype, engine, charset = 'utf8', if_exists = 'nothing'):
|
138
|
203
|
#Escaped table name
|
139
|
204
|
etname = self._idname_escape(table_name)
|
140
|
205
|
pk_type = self._field_to_type(pk_ftype)
|
141
|
206
|
pk_specs = self._field_to_specs(pk_ftype)
|
142
|
207
|
|
143
|
208
|
if if_exists == 'drop':
|
144
|
|
- qres = """DROP TABLE IF EXISTS {table_name};
|
|
209
|
+ self._query("""DROP TABLE IF EXISTS {table_name};""".format(table_name = etname))
|
|
210
|
+ qres = """
|
145
|
211
|
CREATE TABLE {table_name} (
|
146
|
212
|
{pk_name} {pk_type} {pk_specs},
|
147
|
213
|
PRIMARY KEY({pk_name})
|
|
@@ -168,7 +234,7 @@ PRIMARY KEY({pk_name})
|
168
|
234
|
# @param col_name str : The columns name
|
169
|
235
|
# @param col_fieldtype EmFieldype the fieldtype
|
170
|
236
|
# @return None
|
171
|
|
- def _add_column(self, table_name, col_name, col_fieldtype):
|
|
237
|
+ def _add_column(self, table_name, col_name, col_fieldtype, drop_if_exists = False):
|
172
|
238
|
add_col = """ALTER TABLE {table_name}
|
173
|
239
|
ADD COLUMN {col_name} {col_type} {col_specs};"""
|
174
|
240
|
|
|
@@ -181,8 +247,57 @@ ADD COLUMN {col_name} {col_type} {col_specs};"""
|
181
|
247
|
col_type = self._field_to_type(col_fieldtype),
|
182
|
248
|
col_specs = self._field_to_specs(col_fieldtype),
|
183
|
249
|
)
|
|
250
|
+ try:
|
|
251
|
+ self._query(add_col)
|
|
252
|
+ except _mysql_exceptions.OperationalError as e:
|
|
253
|
+ if drop_if_exists:
|
|
254
|
+ self._del_column(table_name, col_name)
|
|
255
|
+ self._add_column(table_name, col_name, col_fieldtype, drop_if_exists)
|
|
256
|
+ else:
|
|
257
|
+ #LOG
|
|
258
|
+ print("Aborded, column `%s` exists"%col_name)
|
|
259
|
+
|
|
260
|
+ ## @brief Add a foreign key
|
|
261
|
+ # @param src_table_name str : The name of the table where we will add the FK
|
|
262
|
+ # @param dst_table_name str : The name of the table the FK will point on
|
|
263
|
+ # @param src_col_name str : The name of the concerned column in the src_table
|
|
264
|
+ # @param dst_col_name str : The name of the concerned column in the dst_table
|
|
265
|
+ def _add_fk(self, src_table_name, dst_table_name, src_col_name, dst_col_name):
|
|
266
|
+ stname = self._idname_escape(src_table_name)
|
|
267
|
+ dtname = self._idname_escape(dst_table_name)
|
|
268
|
+ scname = self._idname_escape(src_col_name)
|
|
269
|
+ dcname = self._idname_escape(dst_col_name)
|
|
270
|
+
|
|
271
|
+ fk_name = self._fk_name(src_table_name, dst_table_name)
|
|
272
|
+
|
|
273
|
+ self._del_fk(src_table_name, dst_table_name)
|
|
274
|
+
|
|
275
|
+ self._query("""ALTER TABLE {src_table}
|
|
276
|
+ADD CONSTRAINT {fk_name}
|
|
277
|
+FOREIGN KEY ({src_col}) references {dst_table}({dst_col});""".format(
|
|
278
|
+ fk_name = self._idname_escape(fk_name),
|
|
279
|
+ src_table = stname,
|
|
280
|
+ src_col = scname,
|
|
281
|
+ dst_table = dtname,
|
|
282
|
+ dst_col = dcname
|
|
283
|
+ ))
|
|
284
|
+
|
|
285
|
+ ## @brief Given a source and a destination table, delete the corresponding FK
|
|
286
|
+ # @param src_table_name str : The name of the table where the FK is
|
|
287
|
+ # @param dst_table_name str : The name of the table the FK point on
|
|
288
|
+ # @warning fails silently
|
|
289
|
+ def _del_fk(self, src_table_name, dst_table_name):
|
|
290
|
+ try:
|
|
291
|
+ self._query("""ALTER TABLE {src_table}
|
|
292
|
+DROP FOREIGN KEY {fk_name}""".format(
|
|
293
|
+ src_table = self._idname_escape(src_table_name),
|
|
294
|
+ fk_name = self._idname_escape(self._fk_name(src_table_name, dst_table_name))
|
|
295
|
+ ))
|
|
296
|
+ except _mysql_exceptions.OperationalError: pass
|
|
297
|
+
|
|
298
|
+ def _fk_name(self, src_table_name, dst_table_name):
|
|
299
|
+ return "fk_%s_%s"%(src_table_name, dst_table_name)
|
184
|
300
|
|
185
|
|
- self._query(add_col)
|
186
|
301
|
|
187
|
302
|
## @brief Generate triggers given a table_name and its columns fieldtypes
|
188
|
303
|
# @param table_name str : Table name
|