/* Copyright (C) 2015 * swift project Community / Contributors * * This file is part of swift project. It is subject to the license terms in the LICENSE file found in the top-level * directory of this distribution and at http://www.swift-project.org/license.html. No part of swift project, * including this file, may be copied, modified, propagated, or distributed except according to the terms * contained in the LICENSE file. */ #include "blackmisc/db/datastore.h" #include "blackmisc/comparefunctions.h" #include "blackmisc/db/datastoreutility.h" #include "blackmisc/icon.h" #include #include #include namespace BlackMisc { namespace Db { void IDatastoreObject::setTimestampVersionFromDatabaseJson(const QJsonObject &json, const QString &prefix) { // we check 2 formats, the DB format and the backend object format QString timestampString(json.value(prefix % u"lastupdated").toString()); if (timestampString.isEmpty()) { timestampString = json.value(prefix % u"tsLastUpdated").toString(); } const QDateTime ts(CDatastoreUtility::parseTimestamp(timestampString)); this->setUtcTimestamp(ts); // version this->setVersion(json.value(prefix % u"version").toString()); } QString IDatastoreObjectWithIntegerKey::getDbKeyAsString() const { if (m_dbKey < 0) { return {}; } return QString::number(m_dbKey); } QString IDatastoreObjectWithIntegerKey::getDbKeyAsStringInParentheses(const QString &prefix) const { if (m_dbKey < 0) { return {}; } return prefix % u'(' % QString::number(m_dbKey) % u')'; } void IDatastoreObjectWithIntegerKey::setDbKey(const QString &key) { bool ok; const int k = key.toInt(&ok); m_dbKey = ok ? k : -1; } bool IDatastoreObjectWithIntegerKey::isLoadedFromDb() const { return this->hasValidDbKey(); } bool IDatastoreObjectWithIntegerKey::matchesDbKeyState(Db::DbKeyStateFilter filter) const { if (filter == All) { return true; } const bool validKey = this->hasValidDbKey(); return (validKey && filter.testFlag(Valid)) || (!validKey && filter.testFlag(Invalid)); } const CIcon &IDatastoreObjectWithIntegerKey::toDatabaseIcon() const { static const CIcon empty; if (this->hasValidDbKey()) { return CIcon::iconByIndex(CIcons::StandardIconDatabaseKey16); } return empty; } int IDatastoreObjectWithIntegerKey::stringToDbKey(const QString &candidate) { if (candidate.isEmpty()) { return invalidDbKey(); } bool ok; int k = candidate.toInt(&ok); return ok ? k : invalidDbKey(); } QJsonValue IDatastoreObjectWithIntegerKey::getDbKeyAsJsonValue() const { if (this->hasValidDbKey()) { return QJsonValue(m_dbKey); } return QJsonValue(); } void IDatastoreObjectWithIntegerKey::setKeyVersionTimestampFromDatabaseJson(const QJsonObject &json, const QString &prefix) { // this function is performance sensitive, as it is called for all DB data const int dbKey = json.value(prefix % u"id").toInt(-1); this->setDbKey(dbKey); IDatastoreObject::setTimestampVersionFromDatabaseJson(json, prefix); } bool IDatastoreObjectWithIntegerKey::existsKey(const QJsonObject &json, const QString &prefix) { const QJsonValue jv(json.value(prefix % u"id")); return !(jv.isNull() || jv.isUndefined()); } CVariant IDatastoreObjectWithIntegerKey::propertyByIndex(const CPropertyIndex &index) const { if (ITimestampBased::canHandleIndex(index)) { return ITimestampBased::propertyByIndex(index); } const ColumnIndex i = index.frontCasted(); switch (i) { case IndexDbIntegerKey: return CVariant::from(m_dbKey); case IndexDbKeyAsString: return CVariant::from(this->getDbKeyAsString()); case IndexIsLoadedFromDb: return CVariant::from(this->hasValidDbKey()); case IndexDatabaseIcon: return CVariant::from(this->toDatabaseIcon()); case IndexVersion: return CVariant::from(this->getVersion()); default: break; } return CVariant(); } void IDatastoreObjectWithIntegerKey::setPropertyByIndex(const CPropertyIndex &index, const CVariant &variant) { if (ITimestampBased::canHandleIndex(index)) { ITimestampBased::setPropertyByIndex(index, variant); return; } const ColumnIndex i = index.frontCasted(); switch (i) { case IndexDbIntegerKey: m_dbKey = variant.toInt(); break; case IndexDbKeyAsString: m_dbKey = stringToDbKey(variant.toQString()); break; case IndexVersion: this->setVersion(variant.toQString()); break; default: break; } } int IDatastoreObjectWithIntegerKey::comparePropertyByIndex(const CPropertyIndex &index, const IDatastoreObjectWithIntegerKey &compareValue) const { if (ITimestampBased::canHandleIndex(index)) { return ITimestampBased::comparePropertyByIndex(index, compareValue); } const ColumnIndex i = index.frontCasted(); switch (i) { case IndexDbKeyAsString: // fall thru case IndexDbIntegerKey: return Compare::compare(m_dbKey, compareValue.getDbKey()); case IndexDatabaseIcon: return Compare::compare(this->hasValidDbKey(), compareValue.hasValidDbKey()); case IndexVersion: return this->getVersion().compare(compareValue.getVersion()); default: break; } Q_ASSERT_X(false, Q_FUNC_INFO, "Compare failed"); return 0; } bool IDatastoreObjectWithIntegerKey::canHandleIndex(const BlackMisc::CPropertyIndex &index) { if (ITimestampBased::canHandleIndex(index)) { return true; } const int i = index.frontCasted(); return (i >= static_cast(IndexDbIntegerKey)) && (i < static_cast(IndexEndMarker)); } QJsonValue IDatastoreObjectWithStringKey::getDbKeyAsJsonValue() const { if (this->hasValidDbKey()) { return QJsonValue(m_dbKey); } static const QJsonValue null; return null; } QString IDatastoreObjectWithStringKey::getDbKeyAsStringInParentheses(const QString &prefix) const { if (m_dbKey.isEmpty()) { return {}; } return prefix % '(' % m_dbKey % ')'; } bool IDatastoreObjectWithStringKey::matchesDbKeyState(Db::DbKeyStateFilter filter) const { if (filter == All) { return true; } return this->hasValidDbKey() && filter.testFlag(Valid); } const CIcon &IDatastoreObjectWithStringKey::toDatabaseIcon() const { static const CIcon empty; if (this->hasValidDbKey()) { return CIcon::iconByIndex(CIcons::StandardIconDatabaseKey16); } return empty; } void IDatastoreObjectWithStringKey::setKeyVersionTimestampFromDatabaseJson(const QJsonObject &json, const QString &prefix) { QString dbKey = json.value(prefix % u"id").toString(); this->setDbKey(dbKey); IDatastoreObject::setTimestampVersionFromDatabaseJson(json, prefix); } bool IDatastoreObjectWithStringKey::existsKey(const QJsonObject &json, const QString &prefix) { const QJsonValue jv(json.value(prefix % u"id")); return !(jv.isNull() || jv.isUndefined()); } CVariant IDatastoreObjectWithStringKey::propertyByIndex(const CPropertyIndex &index) const { if (ITimestampBased::canHandleIndex(index)) { return ITimestampBased::propertyByIndex(index); } const ColumnIndex i = index.frontCasted(); switch (i) { case IndexDbKeyAsString: // fall thru case IndexDbStringKey: return CVariant::from(m_dbKey); case IndexDatabaseIcon: return CVariant::from(this->toDatabaseIcon()); case IndexIsLoadedFromDb: return CVariant::from(m_loadedFromDb); case IndexVersion: return CVariant::from(this->getVersion()); default: break; } return CVariant(); } void IDatastoreObjectWithStringKey::setPropertyByIndex(const CPropertyIndex &index, const CVariant &variant) { if (ITimestampBased::canHandleIndex(index)) { ITimestampBased::setPropertyByIndex(index, variant); return; } const ColumnIndex i = index.frontCasted(); switch (i) { case IndexDbStringKey: case IndexDbKeyAsString: m_dbKey = variant.value(); break; case IndexIsLoadedFromDb: m_loadedFromDb = variant.toBool(); break; case IndexVersion: this->setVersion(variant.toQString()); break; default: break; } } int IDatastoreObjectWithStringKey::comparePropertyByIndex(const CPropertyIndex &index, const IDatastoreObjectWithStringKey &compareValue) const { if (ITimestampBased::canHandleIndex(index)) { return ITimestampBased::comparePropertyByIndex(index, compareValue); } const ColumnIndex i = index.frontCasted(); switch (i) { case IndexDbKeyAsString: // fall thru case IndexDbStringKey: return m_dbKey.compare(compareValue.getDbKey()); case IndexDatabaseIcon: return Compare::compare(this->hasValidDbKey(), compareValue.hasValidDbKey()); case IndexVersion: return this->getVersion().compare(compareValue.getVersion()); default: break; } Q_ASSERT_X(false, Q_FUNC_INFO, "Compare failed"); return 0; } bool IDatastoreObjectWithStringKey::canHandleIndex(const CPropertyIndex &index) { if (index.isEmpty()) { return false; } if (ITimestampBased::canHandleIndex(index)) { return true;} const int i = index.frontCasted(); return (i >= static_cast(IndexDbStringKey)) && (i < static_cast(IndexEndMarker)); } } } // namespace