/**************************************************************************** ** ** https://www.qxorm.com/ ** Copyright (C) 2013 XDL Team (ic-east.com) ** ** This file is part of the QxOrm library ** ** This software is provided 'as-is', without any express or implied ** warranty. In no event will the authors be held liable for any ** damages arising from the use of this software ** ** Commercial Usage ** Licensees holding valid commercial QxOrm licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and XDL Team ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file 'license.gpl3.txt' included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met : http://www.gnu.org/copyleft/gpl.html ** ** If you are unsure which license is appropriate for your use, or ** if you have questions regarding the use of this file, please contact : ** ic-east.com ** ****************************************************************************/ namespace qx { namespace dao { namespace detail { template struct QxDao_FetchById_WithRelation_Generic { typedef qx::dao::detail::QxDao_Helper type_dao_helper; typedef qx::dao::detail::QxSqlQueryHelper_FetchById_WithRelation type_query_helper; static QSqlError fetchById(const QStringList &relation, T &t, QSqlDatabase *pDatabase) { type_dao_helper dao(t, pDatabase, "fetch by id with relation", new qx::QxSqlQueryBuilder_FetchById_WithRelation()); if (!dao.isValid()) { return dao.error(); } if (!dao.isValidPrimaryKey(t)) { return dao.errInvalidId(); } if (!dao.updateSqlRelationX(relation)) { return dao.errInvalidRelation(); } #ifdef _QX_ENABLE_MONGODB if (dao.isMongoDB()) { qx::dao::on_before_fetch((&t), (&dao)); if (!dao.isValid()) { return dao.error(); } QString json = qx::serialization::json::to_string(t, 1, "mongodb:only_id"); qx::dao::mongodb::QxMongoDB_Helper::findOne((&dao), dao.getDataMemberX()->getClass(), json, NULL); if (!dao.isValid()) { return dao.error(); } qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_build_instance); qx::serialization::json::from_string(t, json, 1, "mongodb"); qx::dao::on_after_fetch((&t), (&dao)); if (!dao.isValid()) { return dao.error(); } return dao.error(); } #endif // _QX_ENABLE_MONGODB QStringList columns; QString sql = dao.builder().buildSql(columns, dao.getSqlRelationLinked()).getSqlQuery(); if (!dao.getDataId() || sql.isEmpty()) { return dao.errEmpty(); } if (!dao.prepare(sql)) { return dao.errFailed(true); } { qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_read_instance); type_query_helper::resolveInput(dao.getSqlRelationLinked(), t, dao.query(), dao.builder()); } if (!dao.exec(true)) { return dao.errFailed(); } { qx::dao::on_before_fetch((&t), (&dao)); if (!dao.isValid()) { return dao.error(); } if (dao.getCartesianProduct()) { fetchById_Complex(t, dao); } else { fetchById_Simple(t, dao); } if (!dao.isValid()) { return dao.error(); } qx::dao::on_after_fetch((&t), (&dao)); } return dao.error(); } private: static inline void fetchById_Simple(T &t, type_dao_helper &dao) { if (!dao.nextRecord()) { dao.errNoData(); return; } qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_build_instance); type_query_helper::resolveOutput(dao.getSqlRelationLinked(), t, dao.query(), dao.builder()); } static inline void fetchById_Complex(T &t, type_dao_helper &dao) { if (!dao.nextRecord()) { dao.errNoData(); return; } do { qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_build_instance); type_query_helper::resolveOutput(dao.getSqlRelationLinked(), t, dao.query(), dao.builder()); if (!dao.isValid()) { return; } } while (dao.nextRecord()); } }; template struct QxDao_FetchById_WithRelation_Container { typedef qx::dao::detail::QxDao_Helper_Container type_dao_helper; typedef qx::dao::detail::QxDao_FetchById_WithRelation_Container type_this; typedef qx::trait::generic_container type_generic_container; typedef typename type_generic_container::type_item type_item; typedef typename type_item::type_value_qx type_value_qx; typedef typename type_item::type_value type_value; static QSqlError fetchById(const QStringList &relation, T &t, QSqlDatabase *pDatabase) { if (qx::trait::generic_container::size(t) <= 0) { return QSqlError(); } type_dao_helper dao(t, pDatabase, "fetch by id with relation", new qx::QxSqlQueryBuilder_FetchById_WithRelation()); if (!dao.isValid()) { return dao.error(); } if (!dao.updateSqlRelationX(relation)) { return dao.errInvalidRelation(); } #ifdef _QX_ENABLE_MONGODB if (dao.isMongoDB()) { for (typename T::iterator it = t.begin(); it != t.end(); ++it) { if (!fetchItem((*it), dao)) { return dao.error(); } } QStringList &itemsAsJson = dao.itemsAsJson(); qx::dao::mongodb::QxMongoDB_Helper::findMany((&dao), dao.getDataMemberX()->getClass(), itemsAsJson, NULL, NULL); if (!dao.isValid()) { return dao.error(); } dao.qxQuery().queryAt(2, ""); for (typename T::iterator it = t.begin(); it != t.end(); ++it) { if (!fetchItem((*it), dao)) { return dao.error(); } } return dao.error(); } #endif // _QX_ENABLE_MONGODB QStringList columns; QString sql = dao.builder().buildSql(columns, dao.getSqlRelationLinked()).getSqlQuery(); if (!dao.getDataId() || sql.isEmpty()) { return dao.errEmpty(); } if (!dao.prepare(sql)) { return dao.errFailed(true); } for (typename T::iterator it = t.begin(); it != t.end(); ++it) { if (!fetchItem((*it), dao)) { return dao.error(); } } return dao.error(); } private: template static inline bool fetchItem(U &item, type_dao_helper &dao) { bool bFetchOk = fetchItem_Helper < U, std::is_pointer::value || qx::trait::is_smart_ptr::value > ::fetch(item, dao); if (bFetchOk) { qx::dao::detail::QxDao_Keep_Original::backup(item); } return bFetchOk; } template struct fetchItem_Helper { static inline bool fetch(U &item, type_dao_helper &dao) { return (item ? type_this::fetchItem((*item), dao) : true); } }; template struct fetchItem_Helper, false> { static inline bool fetch(std::pair &item, type_dao_helper &dao) { return type_this::fetchItem(item.second, dao); } }; template struct fetchItem_Helper, false> { static inline bool fetch(const std::pair &item, type_dao_helper &dao) { return type_this::fetchItem(item.second, dao); } }; #if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0)) template struct fetchItem_Helper, false> { static inline bool fetch(QPair &item, type_dao_helper &dao) { return type_this::fetchItem(item.second, dao); } }; template struct fetchItem_Helper, false> { static inline bool fetch(const QPair &item, type_dao_helper &dao) { return type_this::fetchItem(item.second, dao); } }; #endif // (QT_VERSION < QT_VERSION_CHECK(6, 0, 0)) template struct fetchItem_Helper { typedef qx::dao::detail::QxSqlQueryHelper_FetchById_WithRelation type_query_helper; static bool fetch(U &item, type_dao_helper &dao) { if (!dao.isValidPrimaryKey(item)) { dao.errInvalidId(); return false; } #ifdef _QX_ENABLE_MONGODB if (dao.isMongoDB()) { if (dao.qxQuery().queryAt(2) == "") { qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_build_instance); if (!dao.itemsAsJson().isEmpty()) { QString json = dao.itemsAsJson().takeFirst(); if (!json.isEmpty()) { qx::serialization::json::from_string(item, json, 1, "mongodb"); } } qx::dao::on_after_fetch((&item), (&dao)); return dao.isValid(); } { qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_build_instance); qx::dao::on_before_fetch((&item), (&dao)); if (!dao.isValid()) { return false; } QVariant id = (dao.getDataId() ? dao.getDataId()->toVariant(&item) : QVariant()); if (!id.isNull() && !id.toString().isEmpty()) { dao.itemsAsJson().append(id.toString()); } } return dao.isValid(); } #endif // _QX_ENABLE_MONGODB { qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_read_instance); type_query_helper::resolveInput(dao.getSqlRelationLinked(), item, dao.query(), dao.builder()); } if (!dao.exec(true)) { dao.errFailed(); return false; } { qx::dao::on_before_fetch((&item), (&dao)); if (!dao.isValid()) { return false; } if (dao.getCartesianProduct()) { fetch_Complex(item, dao); } else { fetch_Simple(item, dao); } if (!dao.isValid()) { return false; } qx::dao::on_after_fetch((&item), (&dao)); } return dao.isValid(); } static inline void fetch_Simple(U &item, type_dao_helper &dao) { if (!dao.nextRecord()) { dao.errNoData(); return; } qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_build_instance); type_query_helper::resolveOutput(dao.getSqlRelationLinked(), item, dao.query(), dao.builder()); } static inline void fetch_Complex(U &item, type_dao_helper &dao) { if (!dao.nextRecord()) { dao.errNoData(); return; } do { qx::dao::detail::IxDao_Timer timer((&dao), qx::dao::detail::IxDao_Helper::timer_cpp_build_instance); type_query_helper::resolveOutput(dao.getSqlRelationLinked(), item, dao.query(), dao.builder()); if (!dao.isValid()) { return; } } while (dao.nextRecord()); } }; }; template struct QxDao_FetchById_WithRelation_Ptr { static inline QSqlError fetchById(const QStringList &relation, T &t, QSqlDatabase *pDatabase) { if (!t) { qx::trait::construct_ptr::get(t); }; return qx::dao::fetch_by_id_with_relation(relation, (*t), pDatabase); } }; template struct QxDao_FetchById_WithRelation { static inline QSqlError fetchById(const QString &relation, T &t, QSqlDatabase *pDatabase) { QStringList lst; if (!relation.isEmpty()) { lst = relation.split("|"); } return QxDao_FetchById_WithRelation::fetchById(lst, t, pDatabase); } static inline QSqlError fetchById(const QStringList &relation, T &t, QSqlDatabase *pDatabase) { typedef typename std::conditional::value, qx::dao::detail::QxDao_FetchById_WithRelation_Ptr, qx::dao::detail::QxDao_FetchById_WithRelation_Generic>::type type_dao_1; typedef typename std::conditional::value, qx::dao::detail::QxDao_FetchById_WithRelation_Ptr, type_dao_1>::type type_dao_2; typedef typename std::conditional::value, qx::dao::detail::QxDao_FetchById_WithRelation_Container, type_dao_2>::type type_dao_3; QSqlError error = type_dao_3::fetchById(relation, t, pDatabase); if (!error.isValid()) { qx::dao::detail::QxDao_Keep_Original::backup(t); } return error; } }; } // namespace detail } // namespace dao } // namespace qx