|
@@ -198,6 +198,90 @@ class LeDataSourceSQL(DummyDatasource):
|
198
|
198
|
|
199
|
199
|
return prepared_filters
|
200
|
200
|
|
|
201
|
+
|
|
202
|
+ ## @brief Make a relation between 2 LeType
|
|
203
|
+ # @note rel2type relations. Superior is the LeType from the EmClass and subordinate the LeType for the EmType
|
|
204
|
+ # @param lesup LeType : LeType child class instance that is from the EmClass containing the rel2type field
|
|
205
|
+ # @param lesub LeType : LeType child class instance that is from the EmType linked by the rel2type field ( @ref EditorialModel.fieldtypes.rel2type.EmFieldType.rel_to_type_id )
|
|
206
|
+ # @return The relation_id if success else return False
|
|
207
|
+ def add_related(self, lesup, lesub, rank, **rel_attr):
|
|
208
|
+ with self.connection as cur:
|
|
209
|
+ #First step : relation table insert
|
|
210
|
+ sql = insert(MySQL.relations_table_name,{
|
|
211
|
+ 'id_sup': lesup.lodel_id,
|
|
212
|
+ 'id_sub': lesub.lodel_id,
|
|
213
|
+ 'rank': 0, #default value that will be set latter
|
|
214
|
+ })
|
|
215
|
+ cur.execute(sql)
|
|
216
|
+ relation_id = cur.lastrowid
|
|
217
|
+
|
|
218
|
+
|
|
219
|
+ if len(rel_attr) > 0:
|
|
220
|
+ #There is some relation attribute to add in another table
|
|
221
|
+ attr_table = get_r2t2table_name(lesup._leclass.__name__, lesub.__class__.__name__)
|
|
222
|
+ rel_attr['id_relation'] = relation_id
|
|
223
|
+ sql = insert(attr_table, rel_attr)
|
|
224
|
+ cur.execute(sql)
|
|
225
|
+ self._set_relation_rank(id_relation, rank)
|
|
226
|
+ return relation_id
|
|
227
|
+
|
|
228
|
+ ## @brief Set the rank of a relation identified by its ID
|
|
229
|
+ # @param id_relation int : relation ID
|
|
230
|
+ # @param rank int|str : 'first', 'last', or an integer value
|
|
231
|
+ # @throw ValueError if rank is not valid
|
|
232
|
+ # @throw leobject.leobject.LeObjectQueryError if id_relation don't exists
|
|
233
|
+ def set_relation_rank(self, id_relation, rank):
|
|
234
|
+ self._check_rank(rank)
|
|
235
|
+ self._set_relation_rank(id_relation, rank)
|
|
236
|
+
|
|
237
|
+
|
|
238
|
+
|
|
239
|
+ ## @brief Set the rank of a relation identified by its ID
|
|
240
|
+ #
|
|
241
|
+ # @note this solution is not the more efficient solution but it
|
|
242
|
+ # garantee that ranks are continuous and starts at 1
|
|
243
|
+ # @warning there is no way to fail on rank parameters even giving very bad parameters, if you want a method that may fail on rank use set_relation_rank() instead
|
|
244
|
+ # @param id_relation int : relation ID
|
|
245
|
+ # @param rank int|str : 'first', 'last', or an integer value
|
|
246
|
+ # @throw leobject.leobject.LeObjectQueryError if id_relation don't exists
|
|
247
|
+ def _set_relation_rank(self, id_relation, rank):
|
|
248
|
+ ret = self.get_relation(id_relation, no_attr = True)
|
|
249
|
+ if not ret:
|
|
250
|
+ raise leobject.leobject.LeObjectQueryError("No relation with id_relation = %d"%id_relation)
|
|
251
|
+ lesup = ret['lesup']
|
|
252
|
+ lesub = ret['lesup']
|
|
253
|
+ cur_rank = ret['rank']
|
|
254
|
+ rank = 1 if rank == 'first' or rank < 1 else rank
|
|
255
|
+ if cur_rank == rank:
|
|
256
|
+ return True
|
|
257
|
+
|
|
258
|
+ relations = self.get_related(lesup, lesub.__class__, get_sub=True)
|
|
259
|
+
|
|
260
|
+ if not isinstance(rank, int) or rank > len(relations):
|
|
261
|
+ rank = len(relations)
|
|
262
|
+ if cur_rank == rank:
|
|
263
|
+ return True
|
|
264
|
+
|
|
265
|
+ #insert the relation at the good position
|
|
266
|
+ our_relation = relations.pop(cur_rank)
|
|
267
|
+ relations.insert(our_relation, rank)
|
|
268
|
+
|
|
269
|
+ #gathering (relation_id, new_rank)
|
|
270
|
+ rdatas = [ (attrs['relation_id'], new_rank+1) for new_rank,(sup, sub, attrs) in enumerate(relations) ]
|
|
271
|
+ sql = insert(MySQL.relations_table_name, columns=(MySQL.relations_pkname, 'rank'), values = rdatas, on_duplicate_key_update={'rank',mosql.util.raw('VALUES(`rank`)')})
|
|
272
|
+
|
|
273
|
+
|
|
274
|
+ ## @brief Check a rank value
|
|
275
|
+ # @param rank int | str : Can be an integer >= 1 , 'first' or 'last'
|
|
276
|
+ # @throw ValueError if the rank is not valid
|
|
277
|
+ def _check_rank(self, rank):
|
|
278
|
+ if isinstance(rank, str) and rank != 'first' and rank != 'last':
|
|
279
|
+ raise ValueError("Invalid rank value : %s"%rank)
|
|
280
|
+ elif isinstance(rank, int) and rank < 1:
|
|
281
|
+ raise ValueError("Invalid rank value : %d"%rank)
|
|
282
|
+ else:
|
|
283
|
+ raise ValueError("Invalid rank type : %s"%type(rank))
|
|
284
|
+
|
201
|
285
|
## @brief Link two object given a relation nature, depth and rank
|
202
|
286
|
# @param lesup LeObject : a LeObject
|
203
|
287
|
# @param lesub LeObject : a LeObject
|
|
@@ -230,7 +314,9 @@ class LeDataSourceSQL(DummyDatasource):
|
230
|
314
|
pk_where = {MySQL.relations_pkname:id_relation}
|
231
|
315
|
if not MySQL.fk_on_delete_cascade and len(lesup._linked_types[lesub.__class__]) > 0:
|
232
|
316
|
#Delete the row in the relation attribute table
|
233
|
|
- (lesup, lesub, _) = self.get_relation(id_relation, no_attr = False)
|
|
317
|
+ ret = self.get_relation(id_relation, no_attr = False)
|
|
318
|
+ lesup = ret['lesup']
|
|
319
|
+ lesub = ret['lesub']
|
234
|
320
|
sql = delete(MySQL.relations_table_name, pk_where)
|
235
|
321
|
if cur.execute(sql) != 1:
|
236
|
322
|
raise RuntimeError("Unknown SQL Error")
|
|
@@ -242,12 +328,12 @@ class LeDataSourceSQL(DummyDatasource):
|
242
|
328
|
|
243
|
329
|
## @brief Fetch a relation
|
244
|
330
|
# @param id_relation int : The relation identifier
|
245
|
|
- # @param no_attr bool : If true put None in place of relations attributes
|
246
|
|
- # @return a tuple(lesup, lesub, dict_attr) or False if no relation exists with this id
|
247
|
|
- # @throw Exception if the nature is not NULL
|
|
331
|
+ # @param no_attr bool : If true dont fetch rel_attr
|
|
332
|
+ # @return a dict{'id_relation':.., 'lesup':.., 'lesub':..,'rank':.., 'depth':.., #if not none#'nature':.., #if exists#'dict_attr':..>}
|
248
|
333
|
#
|
249
|
334
|
# @todo TESTS
|
250
|
335
|
def get_relation(self, id_relation, no_attr = False):
|
|
336
|
+ relation = dict()
|
251
|
337
|
with self.connection as cur:
|
252
|
338
|
sql = select(MySQL.relation_table_name, {MySQL.relations_pkname: id_relation})
|
253
|
339
|
if cur.execute(sql) != 1:
|
|
@@ -264,13 +350,16 @@ class LeDataSourceSQL(DummyDatasource):
|
264
|
350
|
leobj = leobject.lefactory.LeFactory.leobj_from_name('LeObject')
|
265
|
351
|
lesup = leobj.uid2leobj(res['id_sup'])
|
266
|
352
|
lesub = leobj.uid2leobj(res['id_sub'])
|
|
353
|
+
|
|
354
|
+ relation['id_relation'] = res['id_relation']
|
|
355
|
+ relation['lesup'] = lesup
|
|
356
|
+ relation['lesub'] = lesub
|
|
357
|
+ relation['rank'] = rank
|
|
358
|
+ relation['depth'] = depth
|
|
359
|
+ if not (res['nature'] is None):
|
|
360
|
+ relation['nature'] = res['nature']
|
267
|
361
|
|
268
|
|
- if no_attr:
|
269
|
|
- attrs = None
|
270
|
|
- elif len(lesup._linked_types[lesub.__class__]) == 0:
|
271
|
|
- #No relation attributes
|
272
|
|
- attrs = dict()
|
273
|
|
- else:
|
|
362
|
+ if not no_attr and res['nature'] is None and len(lesup._linked_types[lesub.__class__]) != 0:
|
274
|
363
|
#Fetch relation attributes
|
275
|
364
|
rel_attr_table = MySQL.get_r2t2table_name(lesup.__class__.__name__, lesub.__class__.__name__)
|
276
|
365
|
sql = select(MySQL.rel_attr_table, {MySQL.relations_pkname: id_relation})
|
|
@@ -284,8 +373,9 @@ class LeDataSourceSQL(DummyDatasource):
|
284
|
373
|
if len(res) > 1:
|
285
|
374
|
raise RuntimeError("When selecting on primary key, get more than one result. Bailout")
|
286
|
375
|
attrs = res[0]
|
|
376
|
+ relation['rel_attr'] = attrs
|
287
|
377
|
|
288
|
|
- return (lesup, lesub, attrs)
|
|
378
|
+ return relation
|
289
|
379
|
|
290
|
380
|
## @brief Fetch all relations concerning an object (rel2type relations)
|
291
|
381
|
# @param leo LeType : LeType child instance
|