Browse Source

Make the MH more fault tolerant and add the foreign keys support

Yann Weber 9 years ago
parent
commit
4f8f13e41f
1 changed files with 134 additions and 19 deletions
  1. 134
    19
      EditorialModel/migrationhandler/mysql.py

+ 134
- 19
EditorialModel/migrationhandler/mysql.py View File

@@ -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

Loading…
Cancel
Save