// QtPromise #include // Qt #include #include using namespace QtPromise; using namespace QtPromisePrivate; class tst_qpromise : public QObject { Q_OBJECT private Q_SLOTS: void resolveSync(); void resolveSync_void(); void resolveDelayed(); void rejectSync(); void rejectDelayed(); void rejectThrows(); void thenReturns(); void thenThrows(); void thenNullPtr(); void thenSkipResult(); void thenDelayedResolved(); void thenDelayedRejected(); void failSameType(); void failBaseClass(); void failCatchAll(); void finallyFulfilled(); void finallyFulfilled_void(); void finallyRejected(); void finallyRejected_void(); void finallyThrows(); void finallyThrows_void(); void finallyDelayedResolved(); void finallyDelayedRejected(); void tapFulfilled(); void tapFulfilled_void(); void tapRejected(); void tapRejected_void(); void tapThrows(); void tapThrows_void(); void tapDelayedResolved(); void tapDelayedRejected(); void delayFulfilled(); void delayRejected(); }; // class tst_qpromise QTEST_MAIN(tst_qpromise) #include "tst_qpromise.moc" template T waitForValue(const QPromise& promise, const T& initial) { T value(initial); promise.then([&](const T& res) { value = res; }).wait(); return value; } template T waitForValue(const QPromise& promise, const T& initial, const T& expected) { T value(initial); promise.then([&]() { value = expected; }).wait(); return value; } template E waitForError(const QPromise& promise, const E& initial) { E error(initial); promise.fail([&](const E& err) { error = err; return T(); }).wait(); return error; } template E waitForError(const QPromise& promise, const E& initial) { E error(initial); promise.fail([&](const E& err) { error = err; }).wait(); return error; } void tst_qpromise::resolveSync() { { // resolver(resolve) QPromise p([](const QPromiseResolve& resolve) { resolve(42); }); QCOMPARE(p.isFulfilled(), true); QCOMPARE(waitForError(p, QString()), QString()); QCOMPARE(waitForValue(p, -1), 42); } { // resolver(resolve, reject) QPromise p([](const QPromiseResolve& resolve, const QPromiseReject&) { resolve(42); }); QCOMPARE(p.isFulfilled(), true); QCOMPARE(waitForError(p, QString()), QString()); QCOMPARE(waitForValue(p, -1), 42); } } void tst_qpromise::resolveSync_void() { { // resolver(resolve) QPromise p([](const QPromiseResolve& resolve) { resolve(); }); QCOMPARE(p.isFulfilled(), true); QCOMPARE(waitForError(p, QString()), QString()); QCOMPARE(waitForValue(p, -1, 42), 42); } { // resolver(resolve, reject) QPromise p([](const QPromiseResolve& resolve, const QPromiseReject&) { resolve(); }); QCOMPARE(p.isFulfilled(), true); QCOMPARE(waitForError(p, QString()), QString()); QCOMPARE(waitForValue(p, -1, 42), 42); } } void tst_qpromise::resolveDelayed() { { // resolver(resolve) QPromise p([](const QPromiseResolve& resolve) { QtPromisePrivate::qtpromise_defer([=]() { resolve(42); }); }); QCOMPARE(p.isPending(), true); QCOMPARE(waitForError(p, QString()), QString()); QCOMPARE(waitForValue(p, -1), 42); QCOMPARE(p.isFulfilled(), true); } { // resolver(resolve, reject) QPromise p([](const QPromiseResolve& resolve, const QPromiseReject&) { QtPromisePrivate::qtpromise_defer([=]() { resolve(42); }); }); QCOMPARE(p.isPending(), true); QCOMPARE(waitForError(p, QString()), QString()); QCOMPARE(waitForValue(p, -1), 42); QCOMPARE(p.isFulfilled(), true); } } void tst_qpromise::rejectSync() { QPromise p([](const QPromiseResolve&, const QPromiseReject& reject) { reject(QString("foo")); }); QCOMPARE(p.isRejected(), true); QCOMPARE(waitForValue(p, -1), -1); QCOMPARE(waitForError(p, QString()), QString("foo")); } void tst_qpromise::rejectDelayed() { QPromise p([](const QPromiseResolve&, const QPromiseReject& reject) { QtPromisePrivate::qtpromise_defer([=]() { reject(QString("foo")); }); }); QCOMPARE(p.isPending(), true); QCOMPARE(waitForValue(p, -1), -1); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); } void tst_qpromise::rejectThrows() { { // resolver(resolve) QPromise p([](const QPromiseResolve&) { throw QString("foo"); }); QCOMPARE(p.isRejected(), true); QCOMPARE(waitForValue(p, -1), -1); QCOMPARE(waitForError(p, QString()), QString("foo")); } { // resolver(resolve, reject) QPromise p([](const QPromiseResolve&, const QPromiseReject&) { throw QString("foo"); }); QCOMPARE(p.isRejected(), true); QCOMPARE(waitForValue(p, -1), -1); QCOMPARE(waitForError(p, QString()), QString("foo")); } } void tst_qpromise::thenReturns() { auto p = QPromise::resolve(42); QVariantList values; p.then([&](int res) { values << res; return QString::number(res+1); }).then([&](const QString& res) { values << res; }).then([&]() { values << 44; }).wait(); QCOMPARE(values, QVariantList({42, QString("43"), 44})); } void tst_qpromise::thenThrows() { auto input = QPromise::resolve(42); auto output = input.then([](int res) { throw QString("foo%1").arg(res); return 42; }); QString error; output.then([&](int res) { error += "bar" + QString::number(res); }).fail([&](const QString& err) { error += err; }).wait(); QCOMPARE(input.isFulfilled(), true); QCOMPARE(output.isRejected(), true); QCOMPARE(error, QString("foo42")); } void tst_qpromise::thenNullPtr() { { // resolved auto p = QPromise::resolve(42).then(nullptr); QCOMPARE(waitForValue(p, -1), 42); QCOMPARE(p.isFulfilled(), true); } { // rejected auto p = QPromise::reject(QString("foo")).then(nullptr); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); } } void tst_qpromise::thenSkipResult() { auto p = QPromise::resolve(42); int value = -1; p.then([&]() { value = 43; }).wait(); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(value, 43); } void tst_qpromise::thenDelayedResolved() { auto p = QPromise::resolve(42).then([](int res) { return QPromise([=](const QPromiseResolve& resolve) { QtPromisePrivate::qtpromise_defer([=]() { resolve(QString("foo%1").arg(res)); }); }); }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForValue(p, QString()), QString("foo42")); } void tst_qpromise::thenDelayedRejected() { auto p = QPromise::resolve(42).then([](int res) { return QPromise([=](const QPromiseResolve&, const QPromiseReject& reject) { QtPromisePrivate::qtpromise_defer([=]() { reject(QString("foo%1").arg(res)); }); }); }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForError(p, QString()), QString("foo42")); } void tst_qpromise::failSameType() { // http://en.cppreference.com/w/cpp/error/exception auto p = QPromise::reject(std::out_of_range("foo")); QString error; p.fail([&](const std::domain_error& e) { error += QString(e.what()) + "0"; return -1; }).fail([&](const std::out_of_range& e) { error += QString(e.what()) + "1"; return -1; }).fail([&](const std::exception& e) { error += QString(e.what()) + "2"; return -1; }).wait(); QCOMPARE(error, QString("foo1")); } void tst_qpromise::failBaseClass() { // http://en.cppreference.com/w/cpp/error/exception auto p = QPromise::reject(std::out_of_range("foo")); QString error; p.fail([&](const std::runtime_error& e) { error += QString(e.what()) + "0"; return -1; }).fail([&](const std::logic_error& e) { error += QString(e.what()) + "1"; return -1; }).fail([&](const std::exception& e) { error += QString(e.what()) + "2"; return -1; }).wait(); QCOMPARE(error, QString("foo1")); } void tst_qpromise::failCatchAll() { auto p = QPromise::reject(std::out_of_range("foo")); QString error; p.fail([&](const std::runtime_error& e) { error += QString(e.what()) + "0"; return -1; }).fail([&]() { error += "bar"; return -1; }).fail([&](const std::exception& e) { error += QString(e.what()) + "2"; return -1; }).wait(); QCOMPARE(error, QString("bar")); } void tst_qpromise::finallyFulfilled() { int value = -1; auto p = QPromise::resolve(42).finally([&]() { value = 8; return 16; // ignored! }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForValue(p, -1), 42); QCOMPARE(p.isFulfilled(), true); QCOMPARE(value, 8); } void tst_qpromise::finallyFulfilled_void() { int value = -1; auto p = QPromise::resolve().finally([&]() { value = 8; return 16; // ignored! }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForValue(p, -1, 42), 42); QCOMPARE(p.isFulfilled(), true); QCOMPARE(value, 8); } void tst_qpromise::finallyRejected() { int value = -1; auto p = QPromise::reject(QString("foo")).finally([&]() { value = 8; return 16; // ignored! }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); QCOMPARE(value, 8); } void tst_qpromise::finallyRejected_void() { int value = -1; auto p = QPromise::reject(QString("foo")).finally([&]() { value = 8; return 16; // ignored! }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); QCOMPARE(value, 8); } void tst_qpromise::finallyThrows() { { // fulfilled auto p = QPromise::resolve(42).finally([&]() { throw QString("bar"); }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForError(p, QString()), QString("bar")); QCOMPARE(p.isRejected(), true); } { // rejected auto p = QPromise::reject(QString("foo")).finally([&]() { throw QString("bar"); }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForError(p, QString()), QString("bar")); QCOMPARE(p.isRejected(), true); } } void tst_qpromise::finallyThrows_void() { { // fulfilled auto p = QPromise::resolve().finally([&]() { throw QString("bar"); }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForError(p, QString()), QString("bar")); QCOMPARE(p.isRejected(), true); } { // rejected auto p = QPromise::reject(QString("foo")).finally([&]() { throw QString("bar"); }); Q_STATIC_ASSERT((std::is_same >::value)); QCOMPARE(waitForError(p, QString()), QString("bar")); QCOMPARE(p.isRejected(), true); } } void tst_qpromise::finallyDelayedResolved() { { // fulfilled QVector values; auto p = QPromise::resolve(42).finally([&]() { QPromise p([&](const QPromiseResolve& resolve) { qtpromise_defer([=, &values]() { values << 64; resolve(16); // ignored! }); }); values << 8; return p; }); QCOMPARE(waitForValue(p, -1), 42); QCOMPARE(p.isFulfilled(), true); QCOMPARE(values, QVector({8, 64})); } { // rejected QVector values; auto p = QPromise::reject(QString("foo")).finally([&]() { QPromise p([&](const QPromiseResolve& resolve) { qtpromise_defer([=, &values]() { values << 64; resolve(16); // ignored! }); }); values << 8; return p; }); p.then([&](int r) { values << r; }).wait(); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); QCOMPARE(values, QVector({8, 64})); } } void tst_qpromise::finallyDelayedRejected() { { // fulfilled auto p = QPromise::resolve(42).finally([]() { return QPromise([](const QPromiseResolve&, const QPromiseReject& reject) { qtpromise_defer([=]() { reject(QString("bar")); }); }); }); QCOMPARE(waitForError(p, QString()), QString("bar")); QCOMPARE(p.isRejected(), true); } { // rejected auto p = QPromise::reject(QString("foo")).finally([]() { return QPromise([](const QPromiseResolve&, const QPromiseReject& reject) { qtpromise_defer([=]() { reject(QString("bar")); }); }); }); QCOMPARE(waitForError(p, QString()), QString("bar")); QCOMPARE(p.isRejected(), true); } } void tst_qpromise::tapFulfilled() { int value = -1; auto p = QPromise::resolve(42).tap([&](int res) { value = res + 1; return 8; }); QCOMPARE(waitForValue(p, 42), 42); QCOMPARE(p.isFulfilled(), true); QCOMPARE(value, 43); } void tst_qpromise::tapFulfilled_void() { int value = -1; auto p = QPromise::resolve().tap([&]() { value = 43; return 8; }); QCOMPARE(waitForValue(p, -1, 42), 42); QCOMPARE(p.isFulfilled(), true); QCOMPARE(value, 43); } void tst_qpromise::tapRejected() { int value = -1; auto p = QPromise::reject(QString("foo")).tap([&](int res) { value = res + 1; }); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); QCOMPARE(value, -1); } void tst_qpromise::tapRejected_void() { int value = -1; auto p = QPromise::reject(QString("foo")).tap([&]() { value = 43; }); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); QCOMPARE(value, -1); } void tst_qpromise::tapThrows() { auto p = QPromise::resolve(42).tap([&](int) { throw QString("foo"); }); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); } void tst_qpromise::tapThrows_void() { auto p = QPromise::resolve().tap([&]() { throw QString("foo"); }); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); } void tst_qpromise::tapDelayedResolved() { QVector values; auto p = QPromise::resolve(1).tap([&](int) { QPromise p([&](const QPromiseResolve& resolve) { qtpromise_defer([=, &values]() { values << 3; resolve(4); // ignored! }); }); values << 2; return p; }); p.then([&](int r) { values << r; }).wait(); QCOMPARE(p.isFulfilled(), true); QCOMPARE(values, QVector({2, 3, 1})); } void tst_qpromise::tapDelayedRejected() { QVector values; auto p = QPromise::resolve(1).tap([&](int) { QPromise p([&](const QPromiseResolve&, const QPromiseReject& reject) { qtpromise_defer([=, &values]() { values << 3; reject(QString("foo")); }); }); values << 2; return p; }); p.then([&](int r) { values << r; }).wait(); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); QCOMPARE(values, QVector({2, 3})); } void tst_qpromise::delayFulfilled() { QElapsedTimer timer; qint64 elapsed = -1; timer.start(); auto p = QPromise::resolve(42).delay(1000).finally([&]() { elapsed = timer.elapsed(); }); QCOMPARE(waitForValue(p, -1), 42); QCOMPARE(p.isFulfilled(), true); QVERIFY(elapsed >= 1000 * 0.95); // Qt::CoarseTimer (default) Coarse timers try to QVERIFY(elapsed <= 1000 * 1.05); // keep accuracy within 5% of the desired interval. } void tst_qpromise::delayRejected() { QElapsedTimer timer; qint64 elapsed = -1; timer.start(); auto p = QPromise::reject(QString("foo")).delay(1000).finally([&]() { elapsed = timer.elapsed(); }); QCOMPARE(waitForError(p, QString()), QString("foo")); QCOMPARE(p.isRejected(), true); QVERIFY(elapsed < 5); }