#include #include #include #include "store.h" #include "facade.h" #include "resourceconfig.h" #include "modelresult.h" #include "resultprovider.h" #include "facadefactory.h" SINK_DEBUG_AREA("clientapitest") template class TestDummyResourceFacade : public Sink::StoreFacade { public: static std::shared_ptr> registerFacade(const QByteArray &instanceIdentifier = QByteArray()) { static QMap>> map; auto facade = std::make_shared>(); map.insert(instanceIdentifier, facade); bool alwaysReturnFacade = instanceIdentifier.isEmpty(); Sink::FacadeFactory::instance().registerFacade>("dummyresource", [alwaysReturnFacade](const QByteArray &instanceIdentifier) { if (alwaysReturnFacade) { return map.value(QByteArray()); } return map.value(instanceIdentifier); }); return facade; } ~TestDummyResourceFacade(){}; KAsync::Job create(const T &domainObject) Q_DECL_OVERRIDE { return KAsync::null(); }; KAsync::Job modify(const T &domainObject) Q_DECL_OVERRIDE { return KAsync::null(); }; KAsync::Job remove(const T &domainObject) Q_DECL_OVERRIDE { return KAsync::null(); }; QPair, typename Sink::ResultEmitter::Ptr> load(const Sink::Query &query) Q_DECL_OVERRIDE { auto resultProvider = new Sink::ResultProvider(); resultProvider->onDone([resultProvider]() { SinkTrace() << "Result provider is done"; delete resultProvider; }); // We have to do it this way, otherwise we're not setting the fetcher right auto emitter = resultProvider->emitter(); resultProvider->setFetcher([query, resultProvider, this](const typename T::Ptr &parent) { if (parent) { SinkTrace() << "Running the fetcher " << parent->identifier(); } else { SinkTrace() << "Running the fetcher."; } SinkTrace() << "-------------------------."; for (const auto &res : results) { qDebug() << "Parent filter " << query.propertyFilter.value("parent").value.toByteArray() << res->identifier() << res->getProperty("parent").toByteArray(); auto parentProperty = res->getProperty("parent").toByteArray(); if ((!parent && parentProperty.isEmpty()) || (parent && parentProperty == parent->identifier()) || query.parentProperty.isEmpty()) { qDebug() << "Found a hit" << res->identifier(); resultProvider->add(res); } } resultProvider->initialResultSetComplete(parent); }); auto job = KAsync::start([query, resultProvider]() {}); mResultProvider = resultProvider; return qMakePair(job, emitter); } QList results; Sink::ResultProviderInterface *mResultProvider; }; /** * Test of the client api implementation. * * This test works with injected dummy facades and thus doesn't write to storage. */ class ClientAPITest : public QObject { Q_OBJECT private slots: void initTestCase() { Sink::FacadeFactory::instance().resetFactory(); ResourceConfig::clear(); Sink::Log::setDebugOutputLevel(Sink::Log::Trace); } void testLoad() { auto facade = TestDummyResourceFacade::registerFacade(); facade->results << QSharedPointer::create("resource", "id", 0, QSharedPointer::create()); ResourceConfig::addResource("dummyresource.instance1", "dummyresource"); Sink::Query query; query.resources << "dummyresource.instance1"; query.liveQuery = false; auto model = Sink::Store::loadModel(query); QTRY_VERIFY(model->data(QModelIndex(), Sink::Store::ChildrenFetchedRole).toBool()); QCOMPARE(model->rowCount(QModelIndex()), 1); } void testLoadWithoutResource() { Sink::Query query; query.resources << "nonexisting.resource"; query.liveQuery = false; auto model = Sink::Store::loadModel(query); QTRY_VERIFY(model->data(QModelIndex(), Sink::Store::ChildrenFetchedRole).toBool()); } void testModelSingle() { auto facade = TestDummyResourceFacade::registerFacade(); facade->results << QSharedPointer::create("resource", "id", 0, QSharedPointer::create()); ResourceConfig::addResource("dummyresource.instance1", "dummyresource"); Sink::Query query; query.resources << "dummyresource.instance1"; query.liveQuery = false; auto model = Sink::Store::loadModel(query); QTRY_COMPARE(model->rowCount(), 1); } void testModelNested() { auto facade = TestDummyResourceFacade::registerFacade(); auto folder = QSharedPointer::create("resource", "id", 0, QSharedPointer::create()); auto subfolder = QSharedPointer::create("resource", "subId", 0, QSharedPointer::create()); subfolder->setProperty("parent", "id"); facade->results << folder << subfolder; ResourceConfig::addResource("dummyresource.instance1", "dummyresource"); // Test Sink::Query query; query.resources << "dummyresource.instance1"; query.liveQuery = false; query.parentProperty = "parent"; auto model = Sink::Store::loadModel(query); QTRY_VERIFY(model->data(QModelIndex(), Sink::Store::ChildrenFetchedRole).toBool()); QCOMPARE(model->rowCount(), 1); model->fetchMore(model->index(0, 0)); QTRY_VERIFY(model->data(model->index(0, 0), Sink::Store::ChildrenFetchedRole).toBool()); QCOMPARE(model->rowCount(model->index(0, 0)), 1); } void testModelSignals() { auto facade = TestDummyResourceFacade::registerFacade(); auto folder = QSharedPointer::create("resource", "id", 0, QSharedPointer::create()); auto subfolder = QSharedPointer::create("resource", "subId", 0, QSharedPointer::create()); subfolder->setProperty("parent", "id"); facade->results << folder << subfolder; ResourceConfig::addResource("dummyresource.instance1", "dummyresource"); // Test Sink::Query query; query.resources << "dummyresource.instance1"; query.liveQuery = false; query.parentProperty = "parent"; auto model = Sink::Store::loadModel(query); QSignalSpy spy(model.data(), SIGNAL(rowsInserted(const QModelIndex &, int, int))); QVERIFY(spy.isValid()); model->fetchMore(model->index(0, 0)); QTRY_VERIFY(spy.count() >= 1); } void testModelNestedLive() { auto facade = TestDummyResourceFacade::registerFacade(); auto folder = QSharedPointer::create("dummyresource.instance1", "id", 0, QSharedPointer::create()); auto subfolder = QSharedPointer::create("dummyresource.instance1", "subId", 0, QSharedPointer::create()); subfolder->setProperty("parent", "id"); facade->results << folder << subfolder; ResourceConfig::addResource("dummyresource.instance1", "dummyresource"); // Test Sink::Query query; query.resources << "dummyresource.instance1"; query.liveQuery = true; query.parentProperty = "parent"; auto model = Sink::Store::loadModel(query); QTRY_COMPARE(model->rowCount(), 1); model->fetchMore(model->index(0, 0)); QTRY_COMPARE(model->rowCount(model->index(0, 0)), 1); auto resultProvider = facade->mResultProvider; // Test new toplevel folder { QSignalSpy rowsInsertedSpy(model.data(), SIGNAL(rowsInserted(const QModelIndex &, int, int))); auto folder2 = QSharedPointer::create("resource", "id2", 0, QSharedPointer::create()); resultProvider->add(folder2); QTRY_COMPARE(model->rowCount(), 2); QTRY_COMPARE(rowsInsertedSpy.count(), 1); QCOMPARE(rowsInsertedSpy.at(0).at(0).value(), QModelIndex()); } // Test changed name { QSignalSpy dataChanged(model.data(), SIGNAL(dataChanged(const QModelIndex &, const QModelIndex &, const QVector &))); folder->setProperty("subject", "modifiedSubject"); resultProvider->modify(folder); QTRY_COMPARE(model->rowCount(), 2); QTRY_COMPARE(dataChanged.count(), 1); } // Test removal { QSignalSpy rowsRemovedSpy(model.data(), SIGNAL(rowsRemoved(const QModelIndex &, int, int))); folder->setProperty("subject", "modifiedSubject"); resultProvider->remove(subfolder); QTRY_COMPARE(model->rowCount(model->index(0, 0)), 0); QTRY_COMPARE(rowsRemovedSpy.count(), 1); } // TODO: A modification can also be a move } void testLoadMultiResource() { auto facade1 = TestDummyResourceFacade::registerFacade("dummyresource.instance1"); facade1->results << QSharedPointer::create("resource1", "id", 0, QSharedPointer::create()); auto facade2 = TestDummyResourceFacade::registerFacade("dummyresource.instance2"); facade2->results << QSharedPointer::create("resource2", "id", 0, QSharedPointer::create()); ResourceConfig::addResource("dummyresource.instance1", "dummyresource"); ResourceConfig::addResource("dummyresource.instance2", "dummyresource"); Sink::Query query; query.liveQuery = false; int childrenFetchedCount = 0; auto model = Sink::Store::loadModel(query); QObject::connect(model.data(), &QAbstractItemModel::dataChanged, [&childrenFetchedCount](const QModelIndex &, const QModelIndex &, const QVector &roles) { if (roles.contains(Sink::Store::ChildrenFetchedRole)) { childrenFetchedCount++; } }); QTRY_VERIFY(model->data(QModelIndex(), Sink::Store::ChildrenFetchedRole).toBool()); QCOMPARE(model->rowCount(QModelIndex()), 2); // Ensure children fetched is only emitted once (when all resources are done) QTest::qWait(50); QVERIFY(childrenFetchedCount <= 1); } void testImperativeLoad() { auto facade = TestDummyResourceFacade::registerFacade(); facade->results << QSharedPointer::create("resource", "id", 0, QSharedPointer::create()); ResourceConfig::addResource("dummyresource.instance1", "dummyresource"); Sink::Query query; query.resources << "dummyresource.instance1"; query.liveQuery = false; bool gotValue = false; auto result = Sink::Store::fetchOne(query) .then([&gotValue](const Sink::ApplicationDomain::Event &event) { gotValue = true; }) .exec(); result.waitForFinished(); QVERIFY(!result.errorCode()); QVERIFY(gotValue); } }; QTEST_MAIN(ClientAPITest) #include "clientapitest.moc"