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.

ledatasourcesql.py 35KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749
  1. #-*- coding: utf-8 -*-
  2. import pymysql
  3. import copy
  4. import leapi
  5. from leapi.datasources.dummy import DummyDatasource
  6. from leapi.leobject import REL_SUB, REL_SUP
  7. from leapi.lecrud import _LeCrud
  8. from mosql.db import Database, all_to_dicts, one_to_dict
  9. from mosql.query import select, insert, update, delete, join, left_join
  10. from mosql.util import raw, or_
  11. import mosql.mysql
  12. from DataSource.MySQL.MySQL import MySQL
  13. from EditorialModel.classtypes import EmNature, common_fields
  14. ## MySQL DataSource for LeObject
  15. class LeDataSourceSQL(DummyDatasource):
  16. RELATIONS_POSITIONS_FIELDS = {REL_SUP: 'superior_id', REL_SUB: 'subordinate_id'}
  17. def __init__(self, module=pymysql, conn_args=None):
  18. super(LeDataSourceSQL, self).__init__()
  19. self.module = module
  20. self.datasource_utils = MySQL
  21. if conn_args is None:
  22. conn_args = copy.copy(self.datasource_utils.connections['default'])
  23. self.module = conn_args['module']
  24. del conn_args['module']
  25. self.connection = Database(self.module, **conn_args)
  26. ## @brief select lodel editorial components using given filters
  27. # @param target_cls LeCrud(class): The component class concerned by the select (a LeCrud child class (not instance !) )
  28. # @param field_list list: List of field to fetch
  29. # @param filters list: List of filters (see @ref lecrud_filters)
  30. # @param rel_filters list: List of relational filters (see @ref lecrud_filters)
  31. # @return a list of LeCrud child classes
  32. # @todo this only works with LeObject.get(), LeClass.get() and LeType.get()
  33. # @todo for speed get rid of all_to_dicts
  34. # @todo filters: all use cases are not implemented
  35. def select(self, target_cls, field_list, filters, rel_filters=None):
  36. joins = []
  37. # it is a LeObject, query only on main table
  38. if target_cls.__name__ == 'LeObject':
  39. main_table = self.datasource_utils.objects_table_name
  40. fields = [(main_table, common_fields)]
  41. # it is a LeType or a LeClass, query on main table left join class table on lodel_id
  42. elif target_cls.is_letype() or target_cls.is_leclass():
  43. # find main table and main table datas
  44. main_table = self.datasource_utils.objects_table_name
  45. main_class = target_cls._leclass if hasattr(target_cls, '_leclass') else target_cls
  46. class_table = self.datasource_utils.get_table_name_from_class(main_class.__name__)
  47. main_lodel_id = self.datasource_utils.column_prefix(main_table, self.datasource_utils.field_lodel_id)
  48. class_lodel_id = self.datasource_utils.column_prefix(class_table, self.datasource_utils.field_lodel_id)
  49. joins = [left_join(class_table, {main_lodel_id:class_lodel_id})]
  50. fields = [(main_table, common_fields), (class_table, main_class.fieldlist())]
  51. # prefix column name in field list
  52. prefixed_field_list = [self.datasource_utils.find_prefix(name, fields) for name in field_list]
  53. # @todo implement that
  54. #if rel_filters is not None and len(rel_filters) > 0:
  55. #rel_filters = self._prepare_rel_filters(rel_filters)
  56. #for rel_filter in rel_filters:
  57. ## join condition
  58. #relation_table_join_field = "%s.%s" % (self.datasource_utils.relations_table_name, self.RELATIONS_POSITIONS_FIELDS[rel_filter['position']])
  59. #query_table_join_field = "%s.%s" % (query_table_name, self.datasource_utils.relations_field_nature)
  60. #join_fields[query_table_join_field] = relation_table_join_field
  61. ## adding "where" filters
  62. #where_filters['%s.%s' % (self.datasource_utils.relations_table_name, self.datasource_utils.relations_field_nature)] = rel_filter['nature']
  63. #where_filters[rel_filter['condition_key']] = rel_filter['condition_value']
  64. # prefix filters'' column names, and prepare dict for mosql where {(fieldname, op): value}
  65. wheres = {(self.datasource_utils.find_prefix(name, fields), op):value for name,op,value in filters}
  66. query = select(main_table, select=prefixed_field_list, where=wheres, joins=joins)
  67. #print ('SQL', query)
  68. # Executing the query
  69. cur = self.datasource_utils.query(self.connection, query)
  70. results = all_to_dicts(cur)
  71. # instanciate each row to editorial components
  72. results = [target_cls.uid2leobj(datas['type_id'])(**datas) for datas in results]
  73. #print('results', results)
  74. return results
  75. ## @brief delete lodel editorial components given filters
  76. # @param target_cls LeCrud(class): The component class concerned by the delete (a LeCrud child class (not instance !) )
  77. # @param filters list : List of filters (see @ref leobject_filters)
  78. # @param rel_filters list : List of relational filters (see @ref leobject_filters)
  79. # @return the number of deleted components
  80. def delete(self, target_cls, filters, rel_filters):
  81. query_table_name = self.datasource_utils.get_table_name_from_class(target_cls.__name__)
  82. prep_filters = self._prepare_filters(filters, query_table_name)
  83. prep_rel_filters = self._prepare_rel_filters(rel_filters)
  84. if len(prep_rel_filters) > 0:
  85. query = "DELETE %s FROM" % query_table_name
  86. for prep_rel_filter in prep_rel_filters:
  87. query += "%s INNER JOIN %s ON (%s.%s = %s.%s)" % (
  88. self.datasource_utils.relations_table_name,
  89. query_table_name,
  90. self.datasource_utils.relations_table_name,
  91. prep_rel_filter['position'],
  92. query_table_name,
  93. self.datasource_utils.field_lodel_id
  94. )
  95. if prep_rel_filter['condition_key'][0] is not None:
  96. prep_filters[("%s.%s" % (self.datasource_utils.relations_table_name, prep_rel_filter['condition_key'][0]), prep_rel_filter['condition_key'][1])] = prep_rel_filter['condition_value']
  97. if prep_filters is not None and len(prep_filters) > 0:
  98. query += " WHERE "
  99. filter_counter = 0
  100. for filter_item in prep_filters:
  101. if filter_counter > 1:
  102. query += " AND "
  103. query += "%s %s %s" % (filter_item[0][0], filter_item[0][1], filter_item[1])
  104. else:
  105. query = delete(query_table_name, prep_filters)
  106. query_delete_from_object = delete(self.datasource_utils.objects_table_name, {'lodel_id': filters['lodel_id']})
  107. with self.connection as cur:
  108. result = cur.execute(query)
  109. cur.execute(query_delete_from_object)
  110. return result
  111. ## @brief update an existing lodel editorial component
  112. # @param target_cls LeCrud(class) : The component class concerned by the update (a LeCrud child class (not instance !) )
  113. # @param filters list : List of filters (see @ref leobject_filters)
  114. # @param rel_filters list : List of relationnal filters (see @ref leobject_filters)
  115. # @param **datas : Datas in kwargs
  116. # @return the number of updated components
  117. # @todo implement other filters than lodel_id
  118. def update(self, target_cls, filters, rel_filters, **datas):
  119. print(target_cls, filters, rel_filters, datas)
  120. # it is a LeType
  121. if not target_cls.is_letype():
  122. raise AttributeError("'%s' is not a LeType, it is not possible to update it" % target_cls)
  123. # find main table and main table datas
  124. main_table = self.datasource_utils.objects_table_name
  125. main_datas = {self.datasource_utils.field_lodel_id: raw(self.datasource_utils.field_lodel_id)} # be sure to have one SET clause
  126. for main_column_name in common_fields:
  127. if main_column_name in datas:
  128. main_datas[main_column_name] = datas[main_column_name]
  129. del(datas[main_column_name])
  130. wheres = {(name, op):value for name,op,value in filters}
  131. query = update(main_table, wheres, main_datas)
  132. #print(query)
  133. self.datasource_utils.query(self.connection, query)
  134. # update on class table
  135. if datas:
  136. class_table = self.datasource_utils.get_table_name_from_class(target_cls._leclass.__name__)
  137. query = update(class_table, wheres, datas)
  138. #print(query)
  139. self.datasource_utils.query(self.connection, query)
  140. return True
  141. ## @brief inserts a new lodel editorial component
  142. # @param target_cls LeCrud(class) : The component class concerned by the insert (a LeCrud child class (not instance !) )
  143. # @param **datas : The datas to insert
  144. # @return The inserted component's id
  145. # @todo should work with LeType, LeClass, and Relations
  146. def insert(self, target_cls, **datas):
  147. # it is a LeType
  148. if not target_cls.is_letype():
  149. raise AttributeError("'%s' is not a LeType, it is not possible to insert it" % target_cls)
  150. # find main table and main table datas
  151. main_table = self.datasource_utils.objects_table_name
  152. main_datas = {'class_id':target_cls._leclass._class_id, 'type_id':target_cls._type_id}
  153. for main_column_name in common_fields:
  154. if main_column_name in datas:
  155. main_datas[main_column_name] = datas[main_column_name]
  156. del(datas[main_column_name])
  157. cur = self.datasource_utils.query(self.connection, insert(main_table, main_datas))
  158. lodel_id = cur.lastrowid
  159. # insert in class_table
  160. datas[self.datasource_utils.field_lodel_id] = lodel_id
  161. class_table = self.datasource_utils.get_table_name_from_class(target_cls._leclass.__name__)
  162. self.datasource_utils.query(self.connection, insert(class_table, datas))
  163. return lodel_id
  164. ## @brief insert multiple editorial component
  165. # @param target_cls LeCrud(class) : The component class concerned by the insert (a LeCrud child class (not instance !) )
  166. # @param datas_list list : A list of dict representing the datas to insert
  167. # @return int the number of inserted component
  168. def insert_multi(self, target_cls, datas_list):
  169. res = list()
  170. for data in datas_list:
  171. res.append(self.insert(target_cls, data))
  172. return len(res)
  173. ## @brief prepares the relational filters
  174. # @params rel_filters : (("superior"|"subordinate"), operator, value)
  175. # @return list
  176. def _prepare_rel_filters(self, rel_filters):
  177. prepared_rel_filters = []
  178. if rel_filters is not None and len(rel_filters) > 0:
  179. for rel_filter in rel_filters:
  180. rel_filter_dict = {
  181. 'position': REL_SUB if rel_filter[0][0] == REL_SUP else REL_SUB,
  182. 'nature': rel_filter[0][1],
  183. 'condition_key': (self.RELATIONS_POSITIONS_FIELDS[rel_filter[0][0]], rel_filter[1]),
  184. 'condition_value': rel_filter[2]
  185. }
  186. prepared_rel_filters.append(rel_filter_dict)
  187. return prepared_rel_filters
  188. ## @brief prepares the filters to be used by the mosql library's functions
  189. # @params filters : (FIELD, OPERATOR, VALUE) tuples
  190. # @return dict : Dictionnary with (FIELD, OPERATOR):VALUE style elements
  191. def _prepare_filters(self, filters, tablename=None):
  192. prepared_filters = {}
  193. if filters is not None and len(filters) > 0:
  194. for filter_item in filters:
  195. if '.' in filter_item[0]:
  196. prepared_filter_key = (filter_item[0], filter_item[1])
  197. else:
  198. prepared_filter_key = ("%s.%s" % (tablename, filter_item[0]), filter_item[1])
  199. prepared_filter_value = filter_item[2]
  200. prepared_filters[prepared_filter_key] = prepared_filter_value
  201. return prepared_filters
  202. # ================================================================================================================ #
  203. # FONCTIONS A DEPLACER #
  204. # ================================================================================================================ #
  205. ## @brief Make a relation between 2 LeType
  206. # @note rel2type relations. Superior is the LeType from the EmClass and subordinate the LeType for the EmType
  207. # @param lesup LeType : LeType child class instance that is from the EmClass containing the rel2type field
  208. # @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 )
  209. # @return The relation_id if success else return False
  210. def add_related(self, lesup, lesub, rank, **rel_attr):
  211. with self.connection as cur:
  212. #First step : relation table insert
  213. sql = insert(MySQL.relations_table_name, {
  214. 'id_sup': lesup.lodel_id,
  215. 'id_sub': lesub.lodel_id,
  216. 'rank': 0, # default value that will be set latter
  217. })
  218. cur.execute(sql)
  219. relation_id = cur.lastrowid
  220. if len(rel_attr) > 0:
  221. #There is some relation attribute to add in another table
  222. attr_table = get_r2t2table_name(lesup._leclass.__name__, lesub.__class__.__name__)
  223. rel_attr['id_relation'] = relation_id
  224. sql = insert(attr_table, rel_attr)
  225. cur.execute(sql)
  226. self._set_relation_rank(id_relation, rank)
  227. return relation_id
  228. ## @brief Deletes the relation between 2 LeType
  229. # @param lesup LeType
  230. # @param lesub LeType
  231. # @param fields dict
  232. # @return True if success else False
  233. # @todo Add fields parameter to identify relation
  234. # @todo Delete relationnal fields if some exists
  235. def del_related(self, lesup, lesub, fields=None):
  236. with self.connection as cur:
  237. del_params = {
  238. 'id_sup': lesup.lodel_id,
  239. 'id_sub': lesub.lodel_id
  240. }
  241. delete_params = {}
  242. if fields is not None:
  243. delete_params = del_params.copy()
  244. delete_params.update(fields)
  245. else:
  246. delete_params = del_params
  247. sql = delete(
  248. self.datasource_utils.relations_table_name,
  249. delete_params
  250. )
  251. if cur.execute(sql) != 1:
  252. return False
  253. return True
  254. ## @brief Fetch related (rel2type) by LeType
  255. # @param leo LeType : We want related LeObject of this LeType child class instance
  256. # @param letype LeType(class) : We want related LeObject of this LeType child class (not instance)
  257. # @param get_sub bool : If True leo is the superior and we want subordinates, else its the opposite
  258. # @return a list of dict { 'id_relation':.., 'rank':.., 'lesup':.., 'lesub'.., 'rel_attrs': dict() }
  259. def get_related(self, leo, letype, get_sub=True):
  260. if LeCrud.name2class('LeType') not in letype.__bases__:
  261. raise ValueError("letype argument should be a LeType child class, but got %s" % type(letype))
  262. if not isinstance(leo, LeType):
  263. raise ValueError("leo argument should be a LeType child class instance but got %s" % type(leo))
  264. with self.connection as cur:
  265. id_leo, id_type = 'id_sup', 'id_sub' if get_sub else 'id_sub', 'id_sup'
  266. joins = [
  267. join(
  268. (MySQL.objects_table_name, 'o'),
  269. on={'r.' + id_type: 'o.' + MySQL.field_lodel_id}
  270. ),
  271. join(
  272. (MySQL.objects_table_name, 'p'),
  273. on={'r.' + id_leo: 'p.' + MySQL.field_lodel_id}
  274. ),
  275. ]
  276. lesup, lesub = leo.__class__, letype if get_sub else letype, leo.__class__
  277. common_infos = ('r.id_relation', 'r.id_sup', 'r.id_sub', 'r.rank', 'r.depth')
  278. if len(lesup._linked_types[lesub]) > 0:
  279. #relationnal attributes, need to join with r2t table
  280. cls_name = leo.__class__.__name__ if get_sub else letype.__name__
  281. type_name = letype.__name__ if get_sub else leo.__class__.__name__
  282. joins.append(
  283. join(
  284. (MySQL.get_r2t2table_name(cls_name, type_name), 'r2t'),
  285. on={'r.' + MySQL.relations_pkname: 'r2t' + MySQL.relations_pkname}
  286. )
  287. )
  288. select = ('r.id_relation', 'r.id_sup', 'r.id_sub', 'r.rank', 'r.depth', 'r2t.*')
  289. else:
  290. select = common_infos
  291. sql = select(
  292. (MySQL.relations_table_name, 'r'),
  293. select=select,
  294. where={
  295. id_leo: leo.lodel_id,
  296. 'type_id': letype._type_id,
  297. },
  298. joins=joins
  299. )
  300. cur.execute(sql)
  301. res = all_to_dicts(cur)
  302. #Building result
  303. ret = list()
  304. for datas in res:
  305. r_letype = letype(res['r.' + id_type])
  306. ret_item = {
  307. 'id_relation': +datas[MySQL.relations_pkname],
  308. 'lesup': r_leo if get_sub else r_letype,
  309. 'lesub': r_letype if get_sub else r_leo,
  310. 'rank': res['rank']
  311. }
  312. rel_attr = copy.copy(datas)
  313. for todel in common_infos:
  314. del rel_attr[todel]
  315. ret_item['rel_attrs'] = rel_attr
  316. ret.append(ret_item)
  317. return ret
  318. ## @brief Set the rank of a relation identified by its ID
  319. # @param id_relation int : relation ID
  320. # @param rank int|str : 'first', 'last', or an integer value
  321. # @throw ValueError if rank is not valid
  322. # @throw leapi.leapi.LeObjectQueryError if id_relation don't exists
  323. def set_relation_rank(self, id_relation, rank):
  324. self._check_rank(rank)
  325. self._set_relation_rank(id_relation, rank)
  326. ## @brief Set the rank of a relation identified by its ID
  327. #
  328. # @note this solution is not the more efficient solution but it
  329. # garantee that ranks are continuous and starts at 1
  330. # @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
  331. # @param id_relation int : relation ID
  332. # @param rank int|str : 'first', 'last', or an integer value
  333. # @throw leapi.leapi.LeObjectQueryError if id_relation don't exists
  334. def _set_relation_rank(self, id_relation, rank):
  335. ret = self.get_relation(id_relation, no_attr=True)
  336. if not ret:
  337. raise leapi.leapi.LeObjectQueryError("No relation with id_relation = %d" % id_relation)
  338. lesup = ret['lesup']
  339. lesub = ret['lesup']
  340. cur_rank = ret['rank']
  341. rank = 1 if rank == 'first' or rank < 1 else rank
  342. if cur_rank == rank:
  343. return True
  344. relations = self.get_related(lesup, lesub.__class__, get_sub=True)
  345. if not isinstance(rank, int) or rank > len(relations):
  346. rank = len(relations)
  347. if cur_rank == rank:
  348. return True
  349. #insert the relation at the good position
  350. our_relation = relations.pop(cur_rank)
  351. relations.insert(our_relation, rank)
  352. #gathering (relation_id, new_rank)
  353. rdatas = [(attrs['relation_id'], new_rank + 1) for new_rank, (sup, sub, attrs) in enumerate(relations)]
  354. sql = insert(MySQL.relations_table_name, columns=(MySQL.relations_pkname, 'rank'), values=rdatas, on_duplicate_key_update={'rank', mosql.util.raw('VALUES(`rank`)')})
  355. ## @brief Check a rank value
  356. # @param rank int | str : Can be an integer >= 1 , 'first' or 'last'
  357. # @throw ValueError if the rank is not valid
  358. def _check_rank(self, rank):
  359. if isinstance(rank, str) and rank != 'first' and rank != 'last':
  360. raise ValueError("Invalid rank value : %s" % rank)
  361. elif isinstance(rank, int) and rank < 1:
  362. raise ValueError("Invalid rank value : %d" % rank)
  363. else:
  364. raise ValueError("Invalid rank type : %s" % type(rank))
  365. ## @brief Link two object given a relation nature, depth and rank
  366. # @param lesup LeObject : a LeObject
  367. # @param lesub LeObject : a LeObject
  368. # @param nature str|None : The relation nature or None if rel2type
  369. # @param rank int : a rank
  370. def add_relation(self, lesup, lesub, nature=None, depth=None, rank=None, **rel_attr):
  371. if len(rel_attr) > 0 and nature is not None:
  372. #not a rel2type but have some relation attribute
  373. raise AttributeError("No relation attributes allowed for non rel2type relations")
  374. with self.connection as cur:
  375. sql = insert(self.datasource_utils.relations_table_name, {'id_sup': lesup.lodel_id, 'id_sub': lesub.lodel_id, 'nature': nature, 'rank': rank, 'depth': depth})
  376. if cur.execute(sql) != 1:
  377. raise RuntimeError("Unknow SQL error")
  378. if len(rel_attr) > 0:
  379. #a relation table exists
  380. cur.execute('SELECT last_insert_id()')
  381. relation_id, = cur.fetchone()
  382. raise NotImplementedError()
  383. return True
  384. ## @brief Delete a relation
  385. # @warning this method may not be efficient
  386. # @param id_relation int : The relation identifier
  387. # @return bool
  388. def del_relation(self, id_relation):
  389. with self.connection as cur:
  390. pk_where = {MySQL.relations_pkname: id_relation}
  391. if not MySQL.fk_on_delete_cascade and len(lesup._linked_types[lesub.__class__]) > 0:
  392. #Delete the row in the relation attribute table
  393. ret = self.get_relation(id_relation, no_attr=False)
  394. lesup = ret['lesup']
  395. lesub = ret['lesub']
  396. sql = delete(MySQL.relations_table_name, pk_where)
  397. if cur.execute(sql) != 1:
  398. raise RuntimeError("Unknown SQL Error")
  399. sql = delete(MySQL.relations_table_name, pk_where)
  400. if cur.execute(sql) != 1:
  401. raise RuntimeError("Unknown SQL Error")
  402. return True
  403. ## @brief Fetch a relation
  404. # @param id_relation int : The relation identifier
  405. # @param no_attr bool : If true dont fetch rel_attr
  406. # @return a dict{'id_relation':.., 'lesup':.., 'lesub':..,'rank':.., 'depth':.., #if not none#'nature':.., #if exists#'dict_attr':..>}
  407. #
  408. # @todo TESTS
  409. def get_relation(self, id_relation, no_attr=False):
  410. relation = dict()
  411. with self.connection as cur:
  412. sql = select(MySQL.relation_table_name, {MySQL.relations_pkname: id_relation})
  413. if cur.execute(sql) != 1:
  414. raise RuntimeError("Unknow SQL error")
  415. res = all_to_dicts(cur)
  416. if len(res) == 0:
  417. return False
  418. if len(res) > 1:
  419. raise RuntimeError("When selecting on primary key, get more than one result. Bailout")
  420. if res['nature'] is not None:
  421. raise ValueError("The relation with id %d is not a rel2type relation" % id_relation)
  422. leobj = leapi.lefactory.LeFactory.leobj_from_name('LeObject')
  423. lesup = leobj.uid2leobj(res['id_sup'])
  424. lesub = leobj.uid2leobj(res['id_sub'])
  425. relation['id_relation'] = res['id_relation']
  426. relation['lesup'] = lesup
  427. relation['lesub'] = lesub
  428. relation['rank'] = rank
  429. relation['depth'] = depth
  430. if res['nature'] is not None:
  431. relation['nature'] = res['nature']
  432. if not no_attr and res['nature'] is None and len(lesup._linked_types[lesub.__class__]) != 0:
  433. #Fetch relation attributes
  434. rel_attr_table = MySQL.get_r2t2table_name(lesup.__class__.__name__, lesub.__class__.__name__)
  435. sql = select(MySQL.rel_attr_table, {MySQL.relations_pkname: id_relation})
  436. if cur.execute(sql) != 1:
  437. raise RuntimeError("Unknow SQL error")
  438. res = all_to_dicts(cur)
  439. if len(res) == 0:
  440. #Here raising a warning and adding empty (or default) attributes will be better
  441. raise RuntimeError("This relation should have attributes but none found !!!")
  442. if len(res) > 1:
  443. raise RuntimeError("When selecting on primary key, get more than one result. Bailout")
  444. attrs = res[0]
  445. relation['rel_attr'] = attrs
  446. return relation
  447. ## @brief Fetch all relations concerning an object (rel2type relations)
  448. # @param leo LeType : LeType child instance
  449. # @return a list of tuple (lesup, lesub, dict_attr)
  450. def get_relations(self, leo):
  451. sql = select(self.datasource_utils.relations_table_name, where=or_(({'id_sub': leo.lodel_id}, {'id_sup': leo.lodel_id})))
  452. with self.connection as cur:
  453. results = all_to_dicts(cur.execute(sql))
  454. relations = []
  455. for result in results:
  456. id_sup = result['id_sup']
  457. id_sub = result['id_sub']
  458. del result['id_sup']
  459. del result['id_sub']
  460. rel_attr = result
  461. relations.append((id_sup, id_sub, rel_attr))
  462. return relations
  463. ## @brief Add a superior to a LeObject
  464. # @note in the MySQL version the method will have a depth=None argument to allow reccursive calls to add all the path to the root with corresponding depth
  465. # @param lesup LeType : superior LeType child class instance
  466. # @param lesub LeType : subordinate LeType child class instance
  467. # @param nature str : A relation nature @ref EditorialModel.classtypesa
  468. # @param rank int : The rank of this relation
  469. # @param depth None|int : The depth of the relation (used to make reccursive calls in order to link with all superiors)
  470. # @return The relation ID or False if fails
  471. def add_superior(self, lesup, lesub, nature, rank, depth=None):
  472. params = {'id_sup': lesup.lodel_id, 'id_sub': lesub.lodel_id, 'nature': nature, 'rank': rank}
  473. if depth is not None:
  474. params['depth'] = depth
  475. sql_insert = insert(self.datasource_utils.relations_table_name, params)
  476. with self.connection as cur:
  477. if cur.execute(sql_insert) != 1:
  478. return False
  479. cur.execute('SELECT last_insert_id()')
  480. relation_id, = cur.fetchone()
  481. if nature in EmNature.getall():
  482. parent_superiors = lesup.superiors()
  483. for superior in parent_superiors:
  484. depth = depth - 1 if depth is not None else 1
  485. self.add_relation(lesup=superior.lodel_id, lesub=lesub.lodel_id, nature=nature, depth=depth, rank=rank)
  486. return relation_id
  487. ## @brief Fetch a superiors list ordered by depth for a LeType
  488. # @param lesub LeType : subordinate LeType child class instance
  489. # @param nature str : A relation nature @ref EditorialModel.classtypes
  490. # @return A list of LeType ordered by depth (the first is the direct superior)
  491. def get_superiors(self, lesub, nature):
  492. sql = select(
  493. self.datasource_utils.relations_table_name,
  494. columns=('id_sup',),
  495. where={'id_sub': lesub.lodel_id, 'nature': nature},
  496. order_by=('depth desc',)
  497. )
  498. result = []
  499. with self.connection as cur:
  500. results = all_to_dicts(cur.execute(sql))
  501. superiors = [LeType(result['id_sup']) for result in results]
  502. return superiors
  503. ## @brief Fetch the list of the subordinates given a nature
  504. # @param lesup LeType : superior LeType child class instance
  505. # @param nature str : A relation nature @ref EditorialModel.classtypes
  506. # @return A list of LeType ordered by rank that are subordinates of lesup in a "nature" relation
  507. def get_subordinates(self, lesup, nature):
  508. with self.connection as cur:
  509. id_sup = lesup.lodel_id if isinstance(lesup, leapi.letype.LeType) else MySQL.leroot_lodel_id
  510. sql = select(
  511. MySQL.relations_table_name,
  512. columns=('id_sup',),
  513. where={'id_sup': id_sup, 'nature': nature},
  514. order_by=('rank',)
  515. )
  516. cur.execut(sql)
  517. res = all_to_dicts(cur)
  518. return [LeType(r['id_sup']) for r in res]
  519. # ================================================================================================================ #
  520. # FONCTIONS A SUPPRIMER #
  521. # ================================================================================================================ #
  522. ## @brief inserts a new object
  523. # @param letype LeType
  524. # @param leclass LeClass
  525. # @param datas dict : dictionnary of field:value pairs to save
  526. # @return int : lodel_id of the created object
  527. # @todo add the returning clause and the insertion in "object"
  528. # def insert(self, letype, leclass, datas):
  529. # if isinstance(datas, list):
  530. # res = list()
  531. # for data in datas:
  532. # res.append(self.insert(letype, leclass, data))
  533. # return res if len(res)>1 else res[0]
  534. # elif isinstance(datas, dict):
  535. #
  536. # object_datas = {'class_id': leclass._class_id, 'type_id': letype._type_id}
  537. #
  538. # cur = self.datasource_utils.query(self.connection, insert(self.datasource_utils.objects_table_name, object_datas))
  539. # lodel_id = cur.lastrowid
  540. #
  541. # datas[self.datasource_utils.field_lodel_id] = lodel_id
  542. # query_table_name = self.datasource_utils.get_table_name_from_class(leclass.__name__)
  543. # self.datasource_utils.query(self.connection, insert(query_table_name, datas))
  544. #
  545. # return lodel_id
  546. ## @brief search for a collection of objects
  547. # @param leclass LeClass
  548. # @param letype LeType
  549. # @field_list list
  550. # @param filters list : list of tuples formatted as (FIELD, OPERATOR, VALUE)
  551. # @param relation_filters list : list of tuples formatted as (('superior'|'subordinate', FIELD), OPERATOR, VALUE)
  552. # @return list
  553. # def get(self, leclass, letype, field_list, filters, relational_filters=None):
  554. #
  555. # if leclass is None:
  556. # query_table_name = self.datasource_utils.objects_table_name
  557. # else:
  558. # query_table_name = self.datasource_utils.get_table_name_from_class(leclass.__name__)
  559. # where_filters = self._prepare_filters(filters, query_table_name)
  560. # join_fields = {}
  561. #
  562. # if relational_filters is not None and len(relational_filters) > 0:
  563. # rel_filters = self._prepare_rel_filters(relational_filters)
  564. # for rel_filter in rel_filters:
  565. # # join condition
  566. # relation_table_join_field = "%s.%s" % (self.datasource_utils.relations_table_name, self.RELATIONS_POSITIONS_FIELDS[rel_filter['position']])
  567. # query_table_join_field = "%s.%s" % (query_table_name, self.datasource_utils.field_lodel_id)
  568. # join_fields[query_table_join_field] = relation_table_join_field
  569. # # Adding "where" filters
  570. # where_filters['%s.%s' % (self.datasource_utils.relations_table_name, self.datasource_utils.relations_field_nature)] = rel_filter['nature']
  571. # where_filters[rel_filter['condition_key']] = rel_filter['condition_value']
  572. #
  573. # # building the query
  574. # query = select(query_table_name, where=where_filters, select=field_list, joins=join(self.datasource_utils.relations_table_name, join_fields))
  575. # else:
  576. # query = select(query_table_name, where=where_filters, select=field_list)
  577. #
  578. # Executing the query
  579. # cur = self.datasource_utils.query(self.connection, query)
  580. # results = all_to_dicts(cur)
  581. #
  582. # return results
  583. ## @brief delete an existing object
  584. # @param letype LeType
  585. # @param leclass LeClass
  586. # @param filters list : list of tuples formatted as (FIELD, OPERATOR, VALUE)
  587. # @param relational_filters list : list of tuples formatted as (('superior'|'subordinate', FIELD), OPERATOR, VALUE)
  588. # @return bool : True on success
  589. # def delete(self, letype, leclass, filters, relational_filters):
  590. # query_table_name = self.datasource_utils.get_table_name_from_class(leclass.__name__)
  591. # prep_filters = self._prepare_filters(filters, query_table_name)
  592. # prep_rel_filters = self._prepare_rel_filters(relational_filters)
  593. #
  594. # if len(prep_rel_filters) > 0:
  595. # query = "DELETE %s FROM " % query_table_name
  596. #
  597. # for prep_rel_filter in prep_rel_filters:
  598. # query += "%s INNER JOIN %s ON (%s.%s = %s.%s)" % (
  599. # self.datasource_utils.relations_table_name,
  600. # query_table_name,
  601. # self.datasource_utils.relations_table_name,
  602. # prep_rel_filter['position'],
  603. # query_table_name,
  604. # self.datasource_utils.field_lodel_id
  605. # )
  606. #
  607. # if prep_rel_filter['condition_key'][0] is not None:
  608. # prep_filters[("%s.%s" % (self.datasource_utils.relations_table_name, prep_rel_filter['condition_key'][0]), prep_rel_filter['condition_key'][1])] = prep_rel_filter['condition_value']
  609. #
  610. # if prep_filters is not None and len(prep_filters) > 0:
  611. # query += " WHERE "
  612. # filter_counter = 0
  613. # for filter_item in prep_filters:
  614. # if filter_counter > 1:
  615. # query += " AND "
  616. # query += "%s %s %s" % (filter_item[0][0], filter_item[0][1], filter_item[1])
  617. # else:
  618. # query = delete(query_table_name, filters)
  619. #
  620. # query_delete_from_object = delete(self.datasource_utils.objects_table_name, {'lodel_id': filters['lodel_id']})
  621. # with self.connection as cur:
  622. # cur.execute(query)
  623. # cur.execute(query_delete_from_object)
  624. #
  625. # return True
  626. ## @brief update an existing object's data
  627. # @param letype LeType
  628. # @param leclass LeClass
  629. # @param filters list : list of tuples formatted as (FIELD, OPERATOR, VALUE)
  630. # @param rel_filters list : list of tuples formatted as (('superior'|'subordinate', FIELD), OPERATOR, VALUE)
  631. # @param data dict
  632. # @return bool
  633. # @todo prendre en compte les rel_filters
  634. # def update(self, letype, leclass, filters, rel_filters, data):
  635. #
  636. # query_table_name = self.datasource_utils.get_table_name_from_class(leclass.__name__)
  637. # where_filters = filters
  638. # set_data = data
  639. #
  640. # prepared_rel_filters = self._prepare_rel_filters(rel_filters)
  641. #
  642. # Building the query
  643. # query = update(table=query_table_name, where=where_filters, set=set_data)
  644. # Executing the query
  645. # with self.connection as cur:
  646. # cur.execute(query)
  647. # return True