QxSqlRelation_ManyToMany.h 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. /****************************************************************************
  2. **
  3. ** https://www.qxorm.com/
  4. ** Copyright (C) 2013 Lionel Marty (contact@qxorm.com)
  5. **
  6. ** This file is part of the QxOrm library
  7. **
  8. ** This software is provided 'as-is', without any express or implied
  9. ** warranty. In no event will the authors be held liable for any
  10. ** damages arising from the use of this software
  11. **
  12. ** Commercial Usage
  13. ** Licensees holding valid commercial QxOrm licenses may use this file in
  14. ** accordance with the commercial license agreement provided with the
  15. ** Software or, alternatively, in accordance with the terms contained in
  16. ** a written agreement between you and Lionel Marty
  17. **
  18. ** GNU General Public License Usage
  19. ** Alternatively, this file may be used under the terms of the GNU
  20. ** General Public License version 3.0 as published by the Free Software
  21. ** Foundation and appearing in the file 'license.gpl3.txt' included in the
  22. ** packaging of this file. Please review the following information to
  23. ** ensure the GNU General Public License version 3.0 requirements will be
  24. ** met : http://www.gnu.org/copyleft/gpl.html
  25. **
  26. ** If you are unsure which license is appropriate for your use, or
  27. ** if you have questions regarding the use of this file, please contact :
  28. ** contact@qxorm.com
  29. **
  30. ****************************************************************************/
  31. #ifndef _QX_SQL_RELATION_MANY_TO_MANY_H_
  32. #define _QX_SQL_RELATION_MANY_TO_MANY_H_
  33. #ifdef _MSC_VER
  34. #pragma once
  35. #endif
  36. /*!
  37. * \file QxSqlRelation_ManyToMany.h
  38. * \author Lionel Marty
  39. * \ingroup QxDao
  40. * \brief Manage a relationship many-to-many defined between 2 classes (or between 2 tables in database)
  41. */
  42. #include <QxDao/QxSqlRelation.h>
  43. namespace qx {
  44. /*!
  45. * \ingroup QxDao
  46. * \brief qx::QxSqlRelation_ManyToMany<DataType, Owner> : manage a relationship many-to-many defined between 2 classes (or between 2 tables in database)
  47. */
  48. template <class DataType, class Owner>
  49. class QxSqlRelation_ManyToMany : public QxSqlRelation<DataType, Owner>
  50. {
  51. private:
  52. typedef typename QxSqlRelation<DataType, Owner>::type_owner type_owner;
  53. typedef typename QxSqlRelation<DataType, Owner>::type_data type_data;
  54. typedef typename QxSqlRelation<DataType, Owner>::type_container type_container;
  55. typedef typename QxSqlRelation<DataType, Owner>::type_generic_container type_generic_container;
  56. typedef typename QxSqlRelation<DataType, Owner>::type_item type_item;
  57. typedef typename type_generic_container::type_iterator type_iterator;
  58. typedef typename type_item::type_value type_value;
  59. enum { is_data_container = QxSqlRelation<DataType, Owner>::is_data_container };
  60. public:
  61. QxSqlRelation_ManyToMany(IxDataMember * p, const QString & sExtraTable, const QString & sForeignKeyOwner, const QString & sForeignKeyDataType) : QxSqlRelation<DataType, Owner>(p) { this->setRelationType(qx::IxSqlRelation::many_to_many); this->setExtraTable(sExtraTable); this->setForeignKeyOwner(sForeignKeyOwner); this->setForeignKeyDataType(sForeignKeyDataType); this->verifyParameters(); }
  62. virtual ~QxSqlRelation_ManyToMany() { static_assert(is_data_container, "is_data_container"); }
  63. virtual QString getDescription() const { return "relation many-to-many"; }
  64. virtual bool getCartesianProduct() const { return true; }
  65. virtual void createTable(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  66. virtual void lazySelect(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  67. virtual void lazyFrom(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  68. virtual void eagerFrom(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  69. virtual void lazyJoin(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  70. virtual void lazyWhere(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  71. virtual void eagerWhere(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  72. virtual void lazyWhereSoftDelete(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  73. virtual void lazyFetch_ResolveInput(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  74. virtual void eagerFetch_ResolveInput(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  75. virtual void lazyFetch_ResolveOutput(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  76. virtual void lazyInsert(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  77. virtual void lazyInsert_Values(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  78. virtual void lazyUpdate(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  79. virtual void lazyInsert_ResolveInput(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  80. virtual void lazyUpdate_ResolveInput(QxSqlRelationParams & params) const { Q_UNUSED(params); }
  81. virtual QSqlError onBeforeSave(QxSqlRelationParams & params) const { Q_UNUSED(params); return QSqlError(); }
  82. virtual QVariant getIdFromQuery(bool bEager, QxSqlRelationParams & params) const
  83. { return this->getIdFromQuery_ManyToMany(bEager, params); }
  84. virtual void updateOffset(bool bEager, QxSqlRelationParams & params) const
  85. { this->updateOffset_ManyToMany(bEager, params); }
  86. virtual void eagerSelect(QxSqlRelationParams & params) const
  87. { this->eagerSelect_ManyToMany(params); }
  88. virtual void eagerJoin(QxSqlRelationParams & params) const
  89. { this->eagerJoin_ManyToMany(params); }
  90. virtual void eagerWhereSoftDelete(QxSqlRelationParams & params) const
  91. { this->eagerWhereSoftDelete_ManyToMany(params); }
  92. virtual void * eagerFetch_ResolveOutput(QxSqlRelationParams & params) const
  93. {
  94. if (! this->verifyOffset(params, true)) { return NULL; }
  95. QSqlQuery & query = params.query();
  96. IxDataMember * p = NULL; IxDataMember * pId = this->getDataId(); qAssert(pId);
  97. long lIndex = 0; long lOffsetId = (pId ? pId->getNameCount() : 0); bool bValidId(false);
  98. long lOffsetOld = params.offset(); this->updateOffset(true, params);
  99. long lRelation = 0; IxSqlRelation * pRelation = NULL;
  100. for (int i = 0; i < pId->getNameCount(); i++)
  101. { QVariant v = query.value(lOffsetOld + i); bValidId = (bValidId || qx::trait::is_valid_primary_key(v)); }
  102. if (! bValidId) { return NULL; }
  103. type_item item = this->createItem();
  104. type_data & item_val = item.value_qx();
  105. if (! this->callTriggerBeforeFetch(item_val, params)) { return NULL; }
  106. for (int i = 0; i < pId->getNameCount(); i++)
  107. { QVariant v = query.value(lOffsetOld + i); qx::cvt::from_variant(v, item.key(), "", i, qx::cvt::context::e_database); }
  108. for (int i = 0; i < pId->getNameCount(); i++)
  109. { QVariant v = query.value(lOffsetOld + i); pId->fromVariant((& item_val), v, "", i, qx::cvt::context::e_database); }
  110. long lOffsetRelation = (lOffsetOld + lOffsetId); long lCurrIndex = 0;
  111. while ((p = this->nextData(lIndex)))
  112. { if (params.checkColumns(p->getKey())) { p->fromVariant((& item_val), query.value(lCurrIndex + lOffsetRelation), -1, qx::cvt::context::e_database); lCurrIndex++; } }
  113. if (params.relationX())
  114. {
  115. long lOffsetCurrent = (lCurrIndex + lOffsetRelation);
  116. QString sOldCustomAliasOwner = params.getCustomAliasOwner(); params.setCustomAliasOwner(params.getCustomAlias());
  117. long lIndexOwnerOld = params.indexOwner(); params.setIndexOwner(params.index());
  118. void * pOwnerOld = params.owner(); params.setOwner(& item_val);
  119. lOffsetOld = params.offset(); params.setOffset(lOffsetCurrent);
  120. while ((pRelation = this->nextRelation(lRelation)))
  121. { if (this->addLazyRelation(params, pRelation)) { pRelation->lazyFetch_ResolveOutput(params); } }
  122. params.setOwner(pOwnerOld); params.setOffset(lOffsetOld);
  123. params.setCustomAliasOwner(sOldCustomAliasOwner);
  124. params.setIndexOwner(lIndexOwnerOld);
  125. }
  126. if (! this->callTriggerAfterFetch(item_val, params)) { return NULL; }
  127. type_value * pValue = type_generic_container::insertItem(this->getContainer(params), item);
  128. if (! type_item::is_value_pointer && pValue) { return pValue; }
  129. return (& item_val);
  130. }
  131. virtual QSqlError onAfterSave(QxSqlRelationParams & params) const
  132. {
  133. QSqlError daoError;
  134. if (this->isNullData(params)) { return this->deleteFromExtraTable(params); }
  135. if (! params.recursiveMode()) { daoError = qx::dao::save(this->getContainer(params), (& params.database())); }
  136. else { daoError = qx::dao::save_with_relation_recursive(this->getContainer(params), params.saveMode(), (& params.database()), (& params)); }
  137. if (daoError.isValid()) { return daoError; }
  138. daoError = this->deleteFromExtraTable(params);
  139. if (daoError.isValid()) { return daoError; }
  140. daoError = this->insertIntoExtraTable(params);
  141. if (daoError.isValid()) { return daoError; }
  142. return QSqlError();
  143. }
  144. virtual QString createExtraTable() const
  145. { return this->createExtraTable_ManyToMany(); }
  146. private:
  147. void verifyParameters()
  148. { qAssert(! this->getExtraTable().isEmpty() && ! this->getForeignKeyOwner().isEmpty() && ! this->getForeignKeyDataType().isEmpty() && (this->getForeignKeyOwner() != this->getForeignKeyDataType())); }
  149. QSqlError deleteFromExtraTable(QxSqlRelationParams & params) const
  150. { return this->deleteFromExtraTable_ManyToMany(params); }
  151. QSqlError insertIntoExtraTable(QxSqlRelationParams & params) const
  152. {
  153. IxDataMember * pIdOwner = this->getDataIdOwner(); qAssert(pIdOwner);
  154. IxDataMember * pIdData = this->getDataId(); qAssert(pIdData);
  155. if (! pIdOwner || ! pIdData) { return QSqlError(); }
  156. QStringList lstForeignKeyOwner = this->getForeignKeyOwner().split("|");
  157. QStringList lstForeignKeyDataType = this->getForeignKeyDataType().split("|");
  158. qAssert(pIdOwner->getNameCount() == lstForeignKeyOwner.count());
  159. qAssert(pIdData->getNameCount() == lstForeignKeyDataType.count());
  160. QString sql = "INSERT INTO " + this->getExtraTable() + " (";
  161. sql += pIdOwner->getSqlName(", ", this->getForeignKeyOwner()) + ", " + pIdData->getSqlName(", ", this->getForeignKeyDataType());
  162. sql += ") VALUES (";
  163. sql += pIdOwner->getSqlPlaceHolder("", -1, ", ", this->getForeignKeyOwner()) + ", " + pIdData->getSqlPlaceHolder("", -1, ", ", this->getForeignKeyDataType()) + ")";
  164. if (this->traceSqlQuery()) { qDebug("[QxOrm] sql query (extra-table) : %s", qPrintable(sql)); }
  165. type_item item;
  166. type_container & container = this->getContainer(params);
  167. type_iterator itr = type_generic_container::begin(container, item);
  168. type_iterator itr_end = type_generic_container::end(container);
  169. QSqlQuery queryInsert(params.database());
  170. if (! queryInsert.prepare(sql)) { return queryInsert.lastError(); }
  171. while (itr != itr_end)
  172. {
  173. pIdOwner->setSqlPlaceHolder(queryInsert, params.owner(), "", this->getForeignKeyOwner());
  174. pIdData->setSqlPlaceHolder(queryInsert, (& item.value_qx()), "", this->getForeignKeyDataType());
  175. if (! queryInsert.exec()) { return queryInsert.lastError(); }
  176. itr = type_generic_container::next(container, itr, item);
  177. }
  178. return QSqlError();
  179. }
  180. };
  181. } // namespace qx
  182. #endif // _QX_SQL_RELATION_MANY_TO_MANY_H_