mirror of
https://github.com/simonbrunel/qtpromise.git
synced 2025-04-04 15:35:06 +08:00
Compare commits
31 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
1a905cbd4f | ||
|
14031392ac | ||
|
f382ad25fc | ||
|
ac9b936959 | ||
|
21faa67b58 | ||
|
9f01c130cd | ||
|
461f09bef8 | ||
|
bac405febf | ||
|
0c3955cca5 | ||
|
60b36e7a70 | ||
|
f7639e921e | ||
|
1752255e7b | ||
|
13b4bb65d2 | ||
|
b7ecd95b04 | ||
|
88289a7635 | ||
|
6deec9f51f | ||
|
d43657fbd5 | ||
|
b99e468c84 | ||
|
be5455a8c8 | ||
|
0bfdddd887 | ||
|
2c1e631aed | ||
|
d5a82518f9 | ||
|
1ad99391a3 | ||
|
78417b5813 | ||
|
58738a5604 | ||
|
7ee51de987 | ||
|
6639ea52db | ||
|
7f9013a878 | ||
|
cc29ef3512 | ||
|
3c1461b8d0 | ||
|
815dc443b9 |
@ -1,27 +1,34 @@
|
|||||||
image: Visual Studio 2015
|
# https://www.appveyor.com/docs/build-configuration/
|
||||||
|
# https://www.appveyor.com/docs/lang/cpp/#visual-studio
|
||||||
init:
|
|
||||||
- cmd: call "C:\Program Files (x86)\Microsoft Visual Studio "%VSVER%".0\VC\vcvarsall.bat" %ARCH%
|
|
||||||
- cmd: qmake --version
|
|
||||||
|
|
||||||
# https://www.appveyor.com/docs/windows-images-software/#qt
|
# https://www.appveyor.com/docs/windows-images-software/#qt
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
PATH: '%PATH%;%QTDIR%\bin'
|
PATH: '%PATH%;%QTDIR%\bin'
|
||||||
matrix:
|
matrix:
|
||||||
- QTDIR: C:\Qt\5.6\msvc2013
|
- APPVEYOR_BUILD_WORKER_IMAGE: Visual Studio 2015
|
||||||
VSVER: 12
|
QTDIR: C:\Qt\5.6\msvc2013
|
||||||
ARCH: x86
|
SETUP_CMD: C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\vcvarsall.bat
|
||||||
- QTDIR: C:\Qt\5.9\msvc2013_64
|
SETUP_ARG: x86
|
||||||
VSVER: 12
|
|
||||||
ARCH: x64
|
- APPVEYOR_BUILD_WORKER_IMAGE: Visual Studio 2019
|
||||||
- QTDIR: C:\Qt\latest\msvc2015_64
|
QTDIR: C:\Qt\5.15\msvc2019_64
|
||||||
VSVER: 14
|
SETUP_CMD: C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\VC\Auxiliary\Build\vcvars64.bat
|
||||||
ARCH: x64
|
|
||||||
|
- APPVEYOR_BUILD_WORKER_IMAGE: Visual Studio 2019
|
||||||
|
QTDIR: C:\Qt\6.2\msvc2019_64
|
||||||
|
SETUP_CMD: C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\VC\Auxiliary\Build\vcvars64.bat
|
||||||
|
|
||||||
|
before_build:
|
||||||
|
- call "%SETUP_CMD%" %SETUP_ARG%
|
||||||
|
- cmake --version
|
||||||
|
- qmake --version
|
||||||
|
|
||||||
build_script:
|
build_script:
|
||||||
- cmd: qmake qtpromise.pro
|
- cmake -G "NMake Makefiles"
|
||||||
- cmd: nmake
|
- cmake --build .
|
||||||
|
|
||||||
test_script:
|
test_script:
|
||||||
- cmd: nmake check
|
- cmake --build . --target test
|
||||||
|
|
||||||
|
matrix:
|
||||||
|
fast_finish: true
|
||||||
|
66
.clang-format
Normal file
66
.clang-format
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
---
|
||||||
|
Language: Cpp
|
||||||
|
Standard: Cpp11
|
||||||
|
BasedOnStyle: WebKit
|
||||||
|
AlignAfterOpenBracket: true
|
||||||
|
AlignTrailingComments: true
|
||||||
|
AllowAllArgumentsOnNextLine: false
|
||||||
|
AllowShortFunctionsOnASingleLine: Inline
|
||||||
|
AllowShortLambdasOnASingleLine: Empty
|
||||||
|
AlwaysBreakTemplateDeclarations: Yes
|
||||||
|
BinPackArguments: false
|
||||||
|
BinPackParameters: false
|
||||||
|
BraceWrapping:
|
||||||
|
AfterCaseLabel: false
|
||||||
|
AfterClass: true
|
||||||
|
AfterControlStatement: false
|
||||||
|
AfterEnum: false
|
||||||
|
AfterFunction: true
|
||||||
|
AfterNamespace: false
|
||||||
|
AfterStruct: true
|
||||||
|
AfterUnion: false
|
||||||
|
AfterExternBlock: false
|
||||||
|
BeforeCatch: false
|
||||||
|
BeforeElse: false
|
||||||
|
IndentBraces: false
|
||||||
|
SplitEmptyFunction: false
|
||||||
|
SplitEmptyRecord: false
|
||||||
|
SplitEmptyNamespace: false
|
||||||
|
BreakBeforeBinaryOperators: NonAssignment
|
||||||
|
BreakBeforeBraces: Custom
|
||||||
|
BreakInheritanceList: BeforeComma
|
||||||
|
BreakConstructorInitializers: BeforeColon
|
||||||
|
ColumnLimit: 100
|
||||||
|
CommentPragmas: "^!|^:"
|
||||||
|
ConstructorInitializerAllOnOneLineOrOnePerLine: true
|
||||||
|
ConstructorInitializerIndentWidth: 4
|
||||||
|
Cpp11BracedListStyle: true
|
||||||
|
FixNamespaceComments: true
|
||||||
|
ForEachMacros:
|
||||||
|
- BOOST_FOREACH
|
||||||
|
- foreach
|
||||||
|
- forever
|
||||||
|
- Q_FOREACH
|
||||||
|
- Q_FOREVER
|
||||||
|
- QBENCHMARK
|
||||||
|
- QBENCHMARK_ONCE
|
||||||
|
IncludeBlocks: Regroup
|
||||||
|
IncludeCategories:
|
||||||
|
- Regex: '^<Q.*>'
|
||||||
|
Priority: 2
|
||||||
|
SortPriority: 0
|
||||||
|
- Regex: '^<.*\.h>'
|
||||||
|
Priority: 3
|
||||||
|
SortPriority: 0
|
||||||
|
- Regex: '^<.*>'
|
||||||
|
Priority: 4
|
||||||
|
SortPriority: 0
|
||||||
|
- Regex: '.*'
|
||||||
|
Priority: 1
|
||||||
|
SortPriority: 0
|
||||||
|
IndentPPDirectives: AfterHash
|
||||||
|
NamespaceIndentation: None
|
||||||
|
PenaltyReturnTypeOnItsOwnLine: 10
|
||||||
|
SpaceAfterTemplateKeyword: false
|
||||||
|
SpaceBeforeCpp11BracedList: false
|
||||||
|
...
|
43
.github/workflows/ci.yaml
vendored
Normal file
43
.github/workflows/ci.yaml
vendored
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
# https://docs.github.com/actions/reference/workflow-syntax-for-github-actions
|
||||||
|
# https://doc.qt.io/qt-6/supported-platforms.html
|
||||||
|
# https://ddalcino.github.io/aqt-list-server/
|
||||||
|
|
||||||
|
name: CI
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
qt:
|
||||||
|
- 5.9.9
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- uses: jurplel/install-qt-action@v4
|
||||||
|
with:
|
||||||
|
version: ${{ matrix.qt }}
|
||||||
|
archives: qtbase icu
|
||||||
|
tools: tools_cmake
|
||||||
|
arch: gcc_64
|
||||||
|
host: linux
|
||||||
|
- run: |
|
||||||
|
qmake --version
|
||||||
|
cmake --version
|
||||||
|
gcc --version
|
||||||
|
g++ --version
|
||||||
|
- run: |
|
||||||
|
cmake -G "Unix Makefiles"
|
||||||
|
cmake --build . -- -j12
|
||||||
|
cmake --build . --target test
|
||||||
|
- run: |
|
||||||
|
sudo apt-get install lcov > /dev/null
|
||||||
|
lcov --version
|
||||||
|
lcov --capture --directory . -o coverage.info
|
||||||
|
lcov -e coverage.info '**/src/**/*' -o coverage.info
|
||||||
|
- uses: codecov/codecov-action@v3
|
||||||
|
env:
|
||||||
|
CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
|
||||||
|
with:
|
||||||
|
files: coverage.info
|
11
.gitignore
vendored
11
.gitignore
vendored
@ -1,8 +1,10 @@
|
|||||||
_book
|
/build
|
||||||
dist
|
/dist
|
||||||
node_modules
|
/node_modules
|
||||||
|
*.autosave
|
||||||
*.gcno
|
*.gcno
|
||||||
*.gcda
|
*.gcda
|
||||||
|
*.info
|
||||||
*.moc
|
*.moc
|
||||||
*.o
|
*.o
|
||||||
*.obj
|
*.obj
|
||||||
@ -11,6 +13,3 @@ node_modules
|
|||||||
Makefile*
|
Makefile*
|
||||||
moc_*.cpp
|
moc_*.cpp
|
||||||
moc_*.h
|
moc_*.h
|
||||||
coverage.info
|
|
||||||
package-lock.json
|
|
||||||
target_wrapper.bat
|
|
||||||
|
6
.remarkrc.yml
Normal file
6
.remarkrc.yml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
plugins:
|
||||||
|
- frontmatter
|
||||||
|
- validate-links
|
||||||
|
- preset-lint-recommended
|
||||||
|
- preset-lint-markdown-style-guide
|
||||||
|
- [lint-maximum-line-length, 100]
|
37
.travis.yml
37
.travis.yml
@ -1,37 +0,0 @@
|
|||||||
sudo: required
|
|
||||||
dist: trusty
|
|
||||||
language: cpp
|
|
||||||
compiler: gcc
|
|
||||||
|
|
||||||
before_install:
|
|
||||||
- sudo add-apt-repository -y ppa:ubuntu-toolchain-r/test
|
|
||||||
- sudo add-apt-repository -y ppa:beineri/opt-qt563-trusty
|
|
||||||
- sudo apt-get update -qq
|
|
||||||
|
|
||||||
install:
|
|
||||||
- sudo apt-get install -qq gcc-4.9 g++-4.9
|
|
||||||
- sudo update-alternatives --install /usr/bin/gcc gcc /usr/bin/gcc-4.9 90
|
|
||||||
- sudo update-alternatives --install /usr/bin/g++ g++ /usr/bin/g++-4.9 90
|
|
||||||
- sudo update-alternatives --install /usr/bin/gcov gcov /usr/bin/gcov-4.9 90
|
|
||||||
- sudo apt-get install -qq qt56base
|
|
||||||
- source /opt/qt56/bin/qt56-env.sh
|
|
||||||
- wget http://archive.ubuntu.com/ubuntu/pool/universe/l/lcov/lcov_1.13.orig.tar.gz
|
|
||||||
- tar xf lcov_1.13.orig.tar.gz
|
|
||||||
- cd lcov-1.13/
|
|
||||||
- sudo make install
|
|
||||||
- cd ..
|
|
||||||
|
|
||||||
before_script:
|
|
||||||
- qmake --version
|
|
||||||
- lcov --version && gcov --version
|
|
||||||
- gcc --version && g++ --version
|
|
||||||
|
|
||||||
script:
|
|
||||||
- qmake qtpromise.pro CONFIG+=coverage
|
|
||||||
- make -j4
|
|
||||||
- make check --quiet
|
|
||||||
- lcov -capture --directory . --o coverage.info
|
|
||||||
- lcov -e coverage.info '**/src/**/*' -o coverage.info
|
|
||||||
|
|
||||||
after_success:
|
|
||||||
- bash <(curl -s https://codecov.io/bash) -f coverage.info
|
|
82
CMakeLists.txt
Normal file
82
CMakeLists.txt
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
cmake_minimum_required(VERSION 3.8)
|
||||||
|
|
||||||
|
if(DEFINED PROJECT_NAME)
|
||||||
|
set(SUBPROJECT ON)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
project(qtpromise VERSION 0.7.0 LANGUAGES CXX)
|
||||||
|
|
||||||
|
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_LIST_DIR}/cmake")
|
||||||
|
|
||||||
|
find_package(QT 5.6.0 NAMES Qt6 Qt5 REQUIRED)
|
||||||
|
find_package(Qt${QT_VERSION_MAJOR} REQUIRED COMPONENTS Core)
|
||||||
|
|
||||||
|
set(CMAKE_AUTOMOC ON)
|
||||||
|
set(CMAKE_CXX_STANDARD 11)
|
||||||
|
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
||||||
|
|
||||||
|
add_library(qtpromise INTERFACE)
|
||||||
|
add_library(qtpromise::qtpromise ALIAS qtpromise)
|
||||||
|
|
||||||
|
target_link_libraries(qtpromise INTERFACE Qt${QT_VERSION_MAJOR}::Core)
|
||||||
|
target_include_directories(qtpromise INTERFACE "${CMAKE_CURRENT_LIST_DIR}/include")
|
||||||
|
|
||||||
|
add_definitions(
|
||||||
|
-DQT_DEPRECATED_WARNINGS
|
||||||
|
-DQT_NO_KEYWORDS
|
||||||
|
)
|
||||||
|
|
||||||
|
# https://github.com/simonbrunel/qtpromise/issues/10
|
||||||
|
if(CMAKE_CXX_COMPILER_ID MATCHES "GNU")
|
||||||
|
# https://gcc.gnu.org/onlinedocs/gcc/Warning-Options.html
|
||||||
|
add_compile_options(
|
||||||
|
-Werror
|
||||||
|
-Wpedantic
|
||||||
|
-Wall
|
||||||
|
-Wextra
|
||||||
|
-Wconversion
|
||||||
|
-Wdouble-promotion
|
||||||
|
-Wformat=2
|
||||||
|
-Wlogical-op
|
||||||
|
-Wmissing-noreturn
|
||||||
|
-Wold-style-cast
|
||||||
|
# -Wshadow # disabled due to many findings in the current code
|
||||||
|
-Wsign-conversion
|
||||||
|
-Wswitch-default
|
||||||
|
-Wunused-local-typedefs
|
||||||
|
-pedantic-errors
|
||||||
|
)
|
||||||
|
|
||||||
|
# https://github.com/Barro/compiler-warnings/blob/master/gcc/warnings-gcc-6.txt
|
||||||
|
if(CMAKE_CXX_COMPILER_VERSION VERSION_GREATER_EQUAL 6)
|
||||||
|
add_compile_options(
|
||||||
|
-Wduplicated-cond
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# https://github.com/Barro/compiler-warnings/blob/master/gcc/warnings-gcc-7.txt
|
||||||
|
if(CMAKE_CXX_COMPILER_VERSION VERSION_GREATER_EQUAL 7)
|
||||||
|
add_compile_options(
|
||||||
|
-Wduplicated-branches
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
elseif(CMAKE_CXX_COMPILER_ID MATCHES "Clang")
|
||||||
|
# https://clang.llvm.org/docs/DiagnosticsReference.html
|
||||||
|
add_compile_options(
|
||||||
|
-Wall
|
||||||
|
-Wextra
|
||||||
|
-Wpedantic
|
||||||
|
-Wsuggest-destructor-override
|
||||||
|
-Wsuggest-override
|
||||||
|
)
|
||||||
|
elseif(CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
|
||||||
|
# https://docs.microsoft.com/en-us/cpp/build/reference/compiler-option-warning-level
|
||||||
|
add_compile_options(
|
||||||
|
/WX
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(NOT SUBPROJECT)
|
||||||
|
enable_testing()
|
||||||
|
add_subdirectory(tests)
|
||||||
|
endif()
|
20
LICENSE
20
LICENSE
@ -1,9 +1,21 @@
|
|||||||
The MIT License (MIT)
|
The MIT License (MIT)
|
||||||
|
|
||||||
Copyright (c) 2019 Simon Brunel
|
Copyright (c) 2017-present Simon Brunel, https://github.com/simonbrunel
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
||||||
|
25
README.md
25
README.md
@ -1,20 +1,27 @@
|
|||||||
<a href="https://promisesaplus.com/" title="Promises/A+ 1.1"><img src="https://promisesaplus.com/assets/logo-small.png" alt="Promises/A+" align="right"/></a>
|
<p align="center">
|
||||||
|
<img width="256" src="docs/.vuepress/public/hero.svg?sanitize=true">
|
||||||
|
</p>
|
||||||
|
|
||||||
# QtPromise
|
<p align="center">
|
||||||
|
<a href="https://www.qpm.io/packages/com.github.simonbrunel.qtpromise/index.html"><img src="https://img.shields.io/github/release/simonbrunel/qtpromise.svg?style=flat-square&label=qpm&colorB=4CAF50&maxAge=600" alt="Install"></a>
|
||||||
|
<a href="https://travis-ci.com/simonbrunel/qtpromise"><img src="https://img.shields.io/travis/simonbrunel/qtpromise/master.svg?style=flat-square&maxAge=600" alt="Builds"></a>
|
||||||
|
<a href="https://codecov.io/gh/simonbrunel/qtpromise"><img src="https://img.shields.io/codecov/c/github/simonbrunel/qtpromise.svg?style=flat-square&maxAge=600" alt="Coverage"></a>
|
||||||
|
<a href="https://marketplace.qt.io/products/qtpromise"><img src="https://img.shields.io/static/v1?style=flat-square&label=Qt&message=Marketplace&colorB=40cd52&maxAge=600" alt="Marketplace"></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
[](https://www.qpm.io/packages/com.github.simonbrunel.qtpromise/index.html) [](https://travis-ci.org/simonbrunel/qtpromise) [](https://codecov.io/gh/simonbrunel/qtpromise)
|
## Overview
|
||||||
|
|
||||||
[Promises/A+](https://promisesaplus.com/) implementation for [Qt/C++](https://www.qt.io/).
|
[Promises/A+](https://promisesaplus.com/) implementation for [Qt/C++](https://www.qt.io/).
|
||||||
|
|
||||||
Requires [Qt 5.6](https://www.qt.io/download/) (or later) with [C++11 support enabled](https://wiki.qt.io/How_to_use_C++11_in_your_Qt_Projects).
|
Requires [Qt 5.6](https://doc.qt.io/qt-5/) (or later) with [C++11 support enabled](https://wiki.qt.io/How_to_use_C++11_in_your_Qt_Projects) or [Qt 6](https://doc.qt.io/qt-6/).
|
||||||
|
|
||||||
## Documentation
|
## Documentation
|
||||||
|
|
||||||
* [Getting Started](https://qtpromise.netlify.com/qtpromise/getting-started.html)
|
- [Getting Started](https://qtpromise.netlify.com/qtpromise/getting-started.html)
|
||||||
* [Qt Concurrent](https://qtpromise.netlify.com/qtpromise/qtconcurrent.html)
|
- [Qt Concurrent](https://qtpromise.netlify.com/qtpromise/qtconcurrent.html)
|
||||||
* [Qt Signals](https://qtpromise.netlify.com/qtpromise/qtsignals.html)
|
- [Qt Signals](https://qtpromise.netlify.com/qtpromise/qtsignals.html)
|
||||||
* [Thread-Safety](https://qtpromise.netlify.com/qtpromise/thread-safety.html)
|
- [Thread-Safety](https://qtpromise.netlify.com/qtpromise/thread-safety.html)
|
||||||
* [API Reference](https://qtpromise.netlify.com/qtpromise/api-reference.html)
|
- [API Reference](https://qtpromise.netlify.com/qtpromise/api-reference.html)
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
44
cmake/QtPromiseAddTest.cmake
Normal file
44
cmake/QtPromiseAddTest.cmake
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
function(qtpromise_add_test NAME)
|
||||||
|
cmake_parse_arguments(_ARG "" "" "SOURCES;LIBRARIES" ${ARGN})
|
||||||
|
|
||||||
|
set(_TARGET qtpromise.tests.auto.${NAME})
|
||||||
|
|
||||||
|
add_executable(${_TARGET} ${_ARG_SOURCES})
|
||||||
|
|
||||||
|
if(CMAKE_CXX_COMPILER_ID MATCHES "GNU")
|
||||||
|
target_link_libraries(${_TARGET} gcov)
|
||||||
|
target_compile_options(${_TARGET}
|
||||||
|
PRIVATE
|
||||||
|
-fprofile-arcs
|
||||||
|
-ftest-coverage
|
||||||
|
-O0
|
||||||
|
-g
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
target_link_libraries(${_TARGET}
|
||||||
|
Qt${QT_VERSION_MAJOR}::Concurrent
|
||||||
|
Qt${QT_VERSION_MAJOR}::Test
|
||||||
|
qtpromise
|
||||||
|
qtpromise.tests.utils
|
||||||
|
${_ARG_LIBRARIES}
|
||||||
|
)
|
||||||
|
|
||||||
|
add_test(NAME ${_TARGET}
|
||||||
|
COMMAND ${_TARGET}
|
||||||
|
WORKING_DIRECTORY ${CMAKE_RUNTIME_OUTPUT_DIRECTORY}
|
||||||
|
)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function(qtpromise_add_tests GROUP)
|
||||||
|
cmake_parse_arguments(_ARG "" "" "SOURCES" ${ARGN})
|
||||||
|
|
||||||
|
foreach(_FILE ${_ARG_SOURCES})
|
||||||
|
get_filename_component(_FILE_NAME ${_FILE} NAME)
|
||||||
|
if (_FILE_NAME MATCHES "^tst_(.+)\.cpp$")
|
||||||
|
string(REGEX REPLACE "^tst_(.+)\.cpp$" "\\1" _TEST_NAME ${_FILE_NAME})
|
||||||
|
qtpromise_add_test(${GROUP}.${_TEST_NAME} SOURCES ${_FILE} ${_ARG_UNPARSED_ARGUMENTS})
|
||||||
|
endif()
|
||||||
|
endforeach()
|
||||||
|
endfunction()
|
||||||
|
|
@ -1,68 +1,88 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
title: 'QtPromise',
|
title: 'QtPromise',
|
||||||
description: 'Promises/A+ implementation for Qt/C++',
|
description: 'Promises/A+ implementation for Qt/C++',
|
||||||
ga: 'UA-113899811-1',
|
dest: 'dist/docs',
|
||||||
head: [
|
head: [
|
||||||
['link', { rel: 'icon', href: `/favicon.png` }],
|
['link', { rel: 'icon', href: `/favicon.png` }],
|
||||||
],
|
],
|
||||||
|
plugins: [
|
||||||
|
['@vuepress/google-analytics', {
|
||||||
|
ga: 'UA-113899811-1'
|
||||||
|
}]
|
||||||
|
],
|
||||||
themeConfig: {
|
themeConfig: {
|
||||||
repo: 'simonbrunel/qtpromise',
|
repo: 'simonbrunel/qtpromise',
|
||||||
lastUpdated: 'Last Updated',
|
lastUpdated: 'Last Updated',
|
||||||
|
smoothScroll: false,
|
||||||
editLinks: true,
|
editLinks: true,
|
||||||
|
sidebarDepth: 2,
|
||||||
docsDir: 'docs',
|
docsDir: 'docs',
|
||||||
algolia: {
|
algolia: {
|
||||||
apiKey: '0e6e9cccb8c2c360a5543e28c4e31cb8',
|
apiKey: '0e6e9cccb8c2c360a5543e28c4e31cb8',
|
||||||
indexName: 'qtpromise'
|
indexName: 'qtpromise'
|
||||||
},
|
},
|
||||||
|
nav: [
|
||||||
|
{ text: 'Home', link: '/' },
|
||||||
|
{ text: 'Guide', link: '/qtpromise/getting-started' },
|
||||||
|
{ text: 'API Reference', link: '/qtpromise/api-reference' },
|
||||||
|
],
|
||||||
sidebar: [
|
sidebar: [
|
||||||
'qtpromise/getting-started',
|
'/qtpromise/getting-started',
|
||||||
'qtpromise/qtconcurrent',
|
'/qtpromise/qtconcurrent',
|
||||||
'qtpromise/qtsignals',
|
'/qtpromise/qtsignals',
|
||||||
'qtpromise/thread-safety',
|
'/qtpromise/thread-safety',
|
||||||
'qtpromise/api-reference',
|
|
||||||
{
|
{
|
||||||
title: 'QPromise',
|
title: 'API Reference',
|
||||||
|
path: '/qtpromise/api-reference',
|
||||||
children: [
|
children: [
|
||||||
'qtpromise/qpromise/constructor',
|
//['/qtpromise/api-reference', 'Overview'],
|
||||||
'qtpromise/qpromise/delay',
|
{
|
||||||
'qtpromise/qpromise/each',
|
title: 'QPromise',
|
||||||
'qtpromise/qpromise/fail',
|
children: [
|
||||||
'qtpromise/qpromise/filter',
|
'/qtpromise/qpromise/constructor',
|
||||||
'qtpromise/qpromise/finally',
|
'/qtpromise/qpromise/convert',
|
||||||
'qtpromise/qpromise/isfulfilled',
|
'/qtpromise/qpromise/delay',
|
||||||
'qtpromise/qpromise/ispending',
|
'/qtpromise/qpromise/each',
|
||||||
'qtpromise/qpromise/isrejected',
|
'/qtpromise/qpromise/fail',
|
||||||
'qtpromise/qpromise/map',
|
'/qtpromise/qpromise/filter',
|
||||||
'qtpromise/qpromise/reduce',
|
'/qtpromise/qpromise/finally',
|
||||||
'qtpromise/qpromise/tap',
|
'/qtpromise/qpromise/isfulfilled',
|
||||||
'qtpromise/qpromise/tapfail',
|
'/qtpromise/qpromise/ispending',
|
||||||
'qtpromise/qpromise/then',
|
'/qtpromise/qpromise/isrejected',
|
||||||
'qtpromise/qpromise/timeout',
|
'/qtpromise/qpromise/map',
|
||||||
'qtpromise/qpromise/wait',
|
'/qtpromise/qpromise/reduce',
|
||||||
'qtpromise/qpromise/reject.md',
|
'/qtpromise/qpromise/tap',
|
||||||
'qtpromise/qpromise/resolve.md'
|
'/qtpromise/qpromise/tapfail',
|
||||||
]
|
'/qtpromise/qpromise/then',
|
||||||
},
|
'/qtpromise/qpromise/timeout',
|
||||||
{
|
'/qtpromise/qpromise/wait',
|
||||||
title: 'Helpers',
|
'/qtpromise/qpromise/reject',
|
||||||
children: [
|
'/qtpromise/qpromise/resolve'
|
||||||
'qtpromise/helpers/all',
|
]
|
||||||
'qtpromise/helpers/attempt',
|
},
|
||||||
'qtpromise/helpers/connect',
|
{
|
||||||
'qtpromise/helpers/each',
|
title: 'Helpers',
|
||||||
'qtpromise/helpers/filter',
|
children: [
|
||||||
'qtpromise/helpers/map',
|
'/qtpromise/helpers/all',
|
||||||
'qtpromise/helpers/reduce',
|
'/qtpromise/helpers/attempt',
|
||||||
'qtpromise/helpers/resolve'
|
'/qtpromise/helpers/connect',
|
||||||
]
|
'/qtpromise/helpers/each',
|
||||||
},
|
'/qtpromise/helpers/filter',
|
||||||
{
|
'/qtpromise/helpers/map',
|
||||||
title: 'Exceptions',
|
'/qtpromise/helpers/reduce',
|
||||||
children: [
|
'/qtpromise/helpers/resolve'
|
||||||
'qtpromise/exceptions/canceled',
|
]
|
||||||
'qtpromise/exceptions/context',
|
},
|
||||||
'qtpromise/exceptions/timeout',
|
{
|
||||||
'qtpromise/exceptions/undefined'
|
title: 'Exceptions',
|
||||||
|
children: [
|
||||||
|
'/qtpromise/exceptions/canceled',
|
||||||
|
'/qtpromise/exceptions/context',
|
||||||
|
'/qtpromise/exceptions/conversion',
|
||||||
|
'/qtpromise/exceptions/timeout',
|
||||||
|
'/qtpromise/exceptions/undefined'
|
||||||
|
]
|
||||||
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
47
docs/.vuepress/public/hero.svg
Normal file
47
docs/.vuepress/public/hero.svg
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.0//EN" "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd">
|
||||||
|
<svg version="1.0" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
|
||||||
|
width="512px" height="512px" viewBox="0 0 512 512" enable-background="new 0 0 512 512" xml:space="preserve">
|
||||||
|
<polygon fill="#F7E02F" points="440.999,446 0,446 0,135.382 69,66 511.999,66 511.999,374.906 "/>
|
||||||
|
<g>
|
||||||
|
<path d="M209.116,301.104c-27.241,0-46.231-7.334-56.971-22.002c-10.739-14.669-16.109-37.85-16.109-69.544
|
||||||
|
c0-31.694,5.456-55.223,16.371-70.591c10.911-15.364,29.816-23.05,56.709-23.05c26.889,0,45.749,7.641,56.578,22.919
|
||||||
|
c10.825,15.282,16.24,38.812,16.24,70.591c0,20.955-2.227,37.895-6.68,50.816c-4.452,12.925-11.746,22.788-21.871,29.599
|
||||||
|
l22.002,35.361l-26.979,12.572l-23.312-38.242C221.775,300.581,216.45,301.104,209.116,301.104z M175.065,260.374
|
||||||
|
c5.934,10.216,17.288,15.323,34.051,15.323s28.068-5.018,33.92-15.062c5.849-10.04,8.775-27.021,8.775-50.947
|
||||||
|
c0-23.922-3.012-41.295-9.037-52.125c-6.024-10.825-17.247-16.24-33.659-16.24c-16.416,0-27.679,5.415-33.79,16.24
|
||||||
|
c-6.115,10.83-9.167,28.117-9.167,51.863C166.159,233.177,169.126,250.158,175.065,260.374z"/>
|
||||||
|
<path d="M382.776,191.616h-36.146v57.625c0,10.654,0.785,17.727,2.357,21.218c1.571,3.495,5.586,5.238,12.049,5.238l21.479-0.786
|
||||||
|
l1.31,22.789c-11.701,2.267-20.606,3.404-26.718,3.404c-14.844,0-25.015-3.404-30.515-10.215
|
||||||
|
c-5.501-6.811-8.251-19.646-8.251-38.505v-60.769h-16.764v-24.36h16.764v-37.98h28.289v37.98h36.146V191.616z"/>
|
||||||
|
<path d="M152.877,382.938H142.83v15.285h-7.965v-49.157h18.012c11.147,0,16.721,5.502,16.721,16.505
|
||||||
|
c0,5.646-1.399,9.952-4.198,12.918C162.601,381.456,158.426,382.938,152.877,382.938z M142.83,376.049h9.975
|
||||||
|
c5.741,0,8.612-3.491,8.612-10.478c0-3.348-0.694-5.788-2.081-7.319c-1.388-1.531-3.565-2.297-6.531-2.297h-9.975V376.049z"/>
|
||||||
|
<path d="M176.343,398.224v-35.882h7.75v4.306c4.066-2.63,8.133-4.329,12.2-5.095v7.822c-4.115,0.814-7.632,1.866-10.549,3.157
|
||||||
|
l-1.579,0.646v25.045H176.343z"/>
|
||||||
|
<path d="M203.936,366.218c2.463-3.109,6.566-4.665,12.307-4.665c5.741,0,9.843,1.556,12.308,4.665
|
||||||
|
c2.463,3.11,3.696,7.774,3.696,13.993c0,6.22-1.197,10.908-3.588,14.065c-2.393,3.158-6.53,4.736-12.415,4.736
|
||||||
|
c-5.884,0-10.023-1.578-12.415-4.736c-2.393-3.157-3.588-7.846-3.588-14.065C200.24,373.992,201.471,369.328,203.936,366.218z
|
||||||
|
M209.784,389.54c1.1,1.914,3.253,2.871,6.458,2.871c3.205,0,5.357-0.957,6.459-2.871c1.1-1.913,1.65-5.047,1.65-9.4
|
||||||
|
s-0.586-7.438-1.758-9.258c-1.173-1.817-3.29-2.727-6.351-2.727c-3.062,0-5.179,0.909-6.351,2.727
|
||||||
|
c-1.173,1.819-1.758,4.904-1.758,9.258S208.683,387.627,209.784,389.54z"/>
|
||||||
|
<path d="M247.818,398.224h-7.822v-35.882h7.75v2.225c3.396-2.009,6.506-3.014,9.329-3.014c4.162,0,7.199,1.173,9.113,3.517
|
||||||
|
c4.354-2.344,8.684-3.517,12.99-3.517c4.305,0,7.344,1.328,9.113,3.982c1.77,2.655,2.656,7.141,2.656,13.456v19.232h-7.752v-19.018
|
||||||
|
c0-3.875-0.395-6.625-1.184-8.252c-0.789-1.626-2.428-2.44-4.916-2.44c-2.152,0-4.473,0.479-6.961,1.436l-1.219,0.502
|
||||||
|
c0.381,0.958,0.572,4.019,0.572,9.186v18.587h-7.75V379.78c0-4.257-0.383-7.199-1.148-8.826c-0.765-1.626-2.438-2.44-5.022-2.44
|
||||||
|
c-2.393,0-4.618,0.479-6.674,1.436l-1.076,0.431V398.224z"/>
|
||||||
|
<path d="M300.133,356.242v-8.253h7.822v8.253H300.133z M300.133,398.224v-35.882h7.822v35.882H300.133z"/>
|
||||||
|
<path d="M342.33,369.733c-5.646-0.765-9.735-1.147-12.271-1.147s-4.294,0.299-5.274,0.896c-0.98,0.599-1.471,1.543-1.471,2.835
|
||||||
|
s0.538,2.201,1.614,2.727c1.077,0.527,3.612,1.138,7.607,1.83c3.994,0.694,6.828,1.783,8.504,3.266
|
||||||
|
c1.674,1.483,2.512,4.115,2.512,7.894c0,3.78-1.209,6.556-3.624,8.324c-2.417,1.771-5.945,2.655-10.585,2.655
|
||||||
|
c-2.919,0-6.603-0.406-11.052-1.22l-2.225-0.358l0.287-6.53c5.741,0.766,9.878,1.147,12.415,1.147c2.535,0,4.342-0.311,5.418-0.933
|
||||||
|
c1.076-0.621,1.614-1.65,1.614-3.086s-0.515-2.428-1.543-2.979c-1.029-0.549-3.492-1.147-7.392-1.794
|
||||||
|
c-3.899-0.646-6.758-1.661-8.575-3.05c-1.819-1.387-2.728-3.922-2.728-7.606c0-3.684,1.256-6.435,3.768-8.253
|
||||||
|
c2.512-1.817,5.729-2.727,9.652-2.727c3.062,0,6.817,0.383,11.267,1.147l2.225,0.431L342.33,369.733z"/>
|
||||||
|
<path d="M376.632,391.765l2.009-0.215l0.144,5.813c-5.454,1.101-10.286,1.65-14.496,1.65c-5.311,0-9.126-1.458-11.446-4.377
|
||||||
|
c-2.32-2.918-3.48-7.582-3.48-13.994c0-12.726,5.19-19.089,15.573-19.089c10.047,0,15.07,5.479,15.07,16.434l-0.503,5.598h-22.246
|
||||||
|
c0.047,2.967,0.692,5.144,1.938,6.53c1.243,1.389,3.563,2.081,6.961,2.081C369.551,392.195,373.044,392.052,376.632,391.765z
|
||||||
|
M372.327,377.556c0-3.54-0.563-6.016-1.687-7.427c-1.125-1.411-3.026-2.117-5.705-2.117c-2.681,0-4.629,0.742-5.849,2.225
|
||||||
|
c-1.221,1.483-1.855,3.923-1.902,7.319H372.327z"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 4.6 KiB |
@ -1,4 +0,0 @@
|
|||||||
@import 'override.styl'
|
|
||||||
|
|
||||||
.content a code
|
|
||||||
color: $accentColor
|
|
37
docs/.vuepress/styles/index.styl
Normal file
37
docs/.vuepress/styles/index.styl
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
@import 'palette.styl'
|
||||||
|
|
||||||
|
a
|
||||||
|
svg.icon.outbound
|
||||||
|
margin-bottom -2px
|
||||||
|
margin-left 2px
|
||||||
|
|
||||||
|
img + svg.icon.outbound
|
||||||
|
display none
|
||||||
|
|
||||||
|
sup
|
||||||
|
font-size 0.75em !important
|
||||||
|
|
||||||
|
.home .hero
|
||||||
|
img
|
||||||
|
margin 1.5rem auto !important
|
||||||
|
.description
|
||||||
|
display none !important
|
||||||
|
#main-title
|
||||||
|
display none !important
|
||||||
|
|
||||||
|
.page a code
|
||||||
|
color $accentColor
|
||||||
|
|
||||||
|
.navbar
|
||||||
|
-webkit-box-shadow 0 4px 4px -4px rgba(0,0,0,0.1)
|
||||||
|
-moz-box-shadow 0 4px 4px -4px rgba(0,0,0,0.1)
|
||||||
|
box-shadow 0 4px 4px -4px rgba(0,0,0,0.1)
|
||||||
|
|
||||||
|
.logo
|
||||||
|
margin-right: 0.5rem;
|
||||||
|
|
||||||
|
.site-name
|
||||||
|
font-size 1.25rem
|
||||||
|
|
||||||
|
.repo-link
|
||||||
|
margin-left 1rem
|
@ -1,9 +1,30 @@
|
|||||||
<a href="https://promisesaplus.com/" title="Promises/A+ 1.1"><img src="https://promisesaplus.com/assets/logo-small.png" alt="Promises/A+" align="right"/></a>
|
---
|
||||||
|
home: true
|
||||||
|
heroImage: /hero.svg
|
||||||
|
footer: MIT Licensed | Copyright © Simon Brunel, https://github.com/simonbrunel
|
||||||
|
---
|
||||||
|
|
||||||
|
<p align="center">
|
||||||
|
<a href="https://www.qpm.io/packages/com.github.simonbrunel.qtpromise/index.html"><img src="https://img.shields.io/github/release/simonbrunel/qtpromise.svg?style=flat-square&label=qpm&colorB=4CAF50&maxAge=600" alt="Install"></a>
|
||||||
|
<a href="https://travis-ci.com/simonbrunel/qtpromise"><img src="https://img.shields.io/travis/simonbrunel/qtpromise/master.svg?style=flat-square&maxAge=600" alt="Builds"></a>
|
||||||
|
<a href="https://codecov.io/gh/simonbrunel/qtpromise"><img src="https://img.shields.io/codecov/c/github/simonbrunel/qtpromise.svg?style=flat-square&maxAge=600" alt="Coverage"></a>
|
||||||
|
<a href="https://marketplace.qt.io/products/qtpromise"><img src="https://img.shields.io/static/v1?style=flat-square&label=Qt&message=Marketplace&colorB=40cd52&maxAge=600" alt="Marketplace"></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
# QtPromise
|
|
||||||
[Promises/A+](https://promisesaplus.com/) implementation for [Qt/C++](https://www.qt.io/).
|
[Promises/A+](https://promisesaplus.com/) implementation for [Qt/C++](https://www.qt.io/).
|
||||||
|
|
||||||
Requires [Qt 5.6](https://www.qt.io/download/) (or later) with [C++11 support enabled](https://wiki.qt.io/How_to_use_C++11_in_your_Qt_Projects).
|
Requires [Qt 5.6](https://doc.qt.io/qt-5/) (or later) with [C++11 support enabled](https://wiki.qt.io/How_to_use_C++11_in_your_Qt_Projects) or [Qt 6](https://doc.qt.io/qt-6/).
|
||||||
|
|
||||||
|
## Documentation
|
||||||
|
|
||||||
|
- [Getting Started](/qtpromise/getting-started.md)
|
||||||
|
- [Qt Concurrent](/qtpromise/qtconcurrent.md)
|
||||||
|
- [Qt Signals](/qtpromise/qtsignals.md)
|
||||||
|
- [Thread-Safety](/qtpromise/thread-safety.md)
|
||||||
|
- [API Reference](/qtpromise/api-reference.md)
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
QtPromise is available under the [MIT license](https://github.com/simonbrunel/qtpromise/blob/master/LICENSE).
|
QtPromise is available under the [MIT license](https://github.com/simonbrunel/qtpromise/blob/master/LICENSE).
|
||||||
|
@ -2,48 +2,50 @@
|
|||||||
|
|
||||||
## Functions
|
## Functions
|
||||||
|
|
||||||
* [`QPromise<T>::QPromise`](qpromise/constructor.md)
|
- [`QPromise<T>::QPromise`](qpromise/constructor.md)
|
||||||
* [`QPromise<T>::delay`](qpromise/delay.md)
|
- [`QPromise<T>::convert`](qpromise/convert.md)
|
||||||
* [`QPromise<T>::each`](qpromise/each.md)
|
- [`QPromise<T>::delay`](qpromise/delay.md)
|
||||||
* [`QPromise<T>::fail`](qpromise/fail.md)
|
- [`QPromise<T>::each`](qpromise/each.md)
|
||||||
* [`QPromise<T>::filter`](qpromise/filter.md)
|
- [`QPromise<T>::fail`](qpromise/fail.md)
|
||||||
* [`QPromise<T>::finally`](qpromise/finally.md)
|
- [`QPromise<T>::filter`](qpromise/filter.md)
|
||||||
* [`QPromise<T>::isFulfilled`](qpromise/isfulfilled.md)
|
- [`QPromise<T>::finally`](qpromise/finally.md)
|
||||||
* [`QPromise<T>::isPending`](qpromise/ispending.md)
|
- [`QPromise<T>::isFulfilled`](qpromise/isfulfilled.md)
|
||||||
* [`QPromise<T>::isRejected`](qpromise/isrejected.md)
|
- [`QPromise<T>::isPending`](qpromise/ispending.md)
|
||||||
* [`QPromise<T>::map`](qpromise/map.md)
|
- [`QPromise<T>::isRejected`](qpromise/isrejected.md)
|
||||||
* [`QPromise<T>::reduce`](qpromise/reduce.md)
|
- [`QPromise<T>::map`](qpromise/map.md)
|
||||||
* [`QPromise<T>::tap`](qpromise/tap.md)
|
- [`QPromise<T>::reduce`](qpromise/reduce.md)
|
||||||
* [`QPromise<T>::tapFail`](qpromise/tapfail.md)
|
- [`QPromise<T>::tap`](qpromise/tap.md)
|
||||||
* [`QPromise<T>::then`](qpromise/then.md)
|
- [`QPromise<T>::tapFail`](qpromise/tapfail.md)
|
||||||
* [`QPromise<T>::timeout`](qpromise/timeout.md)
|
- [`QPromise<T>::then`](qpromise/then.md)
|
||||||
* [`QPromise<T>::wait`](qpromise/wait.md)
|
- [`QPromise<T>::timeout`](qpromise/timeout.md)
|
||||||
|
- [`QPromise<T>::wait`](qpromise/wait.md)
|
||||||
|
|
||||||
## Static Functions
|
## Static Functions
|
||||||
|
|
||||||
* [`[static] QPromise<T>::reject`](qpromise/reject.md)
|
- [`(static) QPromise<T>::reject`](qpromise/reject.md)
|
||||||
* [`[static] QPromise<T>::resolve`](qpromise/resolve.md)
|
- [`(static) QPromise<T>::resolve`](qpromise/resolve.md)
|
||||||
|
|
||||||
## Helpers
|
## Helpers
|
||||||
|
|
||||||
* [`QtPromise::all`](helpers/all.md)
|
- [`QtPromise::all`](helpers/all.md)
|
||||||
* [`QtPromise::attempt`](helpers/attempt.md)
|
- [`QtPromise::attempt`](helpers/attempt.md)
|
||||||
* [`QtPromise::connect`](helpers/connect.md)
|
- [`QtPromise::connect`](helpers/connect.md)
|
||||||
* [`QtPromise::each`](helpers/each.md)
|
- [`QtPromise::each`](helpers/each.md)
|
||||||
* [`QtPromise::filter`](helpers/filter.md)
|
- [`QtPromise::filter`](helpers/filter.md)
|
||||||
* [`QtPromise::map`](helpers/map.md)
|
- [`QtPromise::map`](helpers/map.md)
|
||||||
* [`QtPromise::reduce`](helpers/reduce.md)
|
- [`QtPromise::reduce`](helpers/reduce.md)
|
||||||
* [`QtPromise::resolve`](helpers/resolve.md)
|
- [`QtPromise::resolve`](helpers/resolve.md)
|
||||||
|
|
||||||
## Exceptions
|
## Exceptions
|
||||||
|
|
||||||
* [`QPromiseCanceledException`](exceptions/canceled.md)
|
- [`QPromiseCanceledException`](exceptions/canceled.md)
|
||||||
* [`QPromiseContextException`](exceptions/context.md)
|
- [`QPromiseContextException`](exceptions/context.md)
|
||||||
* [`QPromiseTimeoutException`](exceptions/timeout.md)
|
- [`QPromiseConversionException`](exceptions/conversion.md)
|
||||||
* [`QPromiseUndefinedException`](exceptions/undefined.md)
|
- [`QPromiseTimeoutException`](exceptions/timeout.md)
|
||||||
|
- [`QPromiseUndefinedException`](exceptions/undefined.md)
|
||||||
|
|
||||||
## Deprecations
|
## Deprecations
|
||||||
|
|
||||||
* `[static] QPromise<T>::all`: use [`QtPromise::all`](helpers/all.md) instead (since 0.5.0)
|
- `(static) QPromise<T>::all`: use [`QtPromise::all`](helpers/all.md) instead (since 0.5.0)
|
||||||
* `QtPromise::qPromise`: use [`QtPromise::resolve`](helpers/resolve.md) instead (since 0.5.0)
|
- `QtPromise::qPromise`: use [`QtPromise::resolve`](helpers/resolve.md) instead (since 0.5.0)
|
||||||
* `QtPromise::qPromiseAll`: use [`QtPromise::all`](helpers/all.md) instead (since 0.5.0)
|
- `QtPromise::qPromiseAll`: use [`QtPromise::all`](helpers/all.md) instead (since 0.5.0)
|
||||||
|
@ -1,16 +1,13 @@
|
|||||||
---
|
|
||||||
title: QPromiseCanceledException
|
|
||||||
---
|
|
||||||
|
|
||||||
# QPromiseCanceledException
|
# QPromiseCanceledException
|
||||||
|
|
||||||
*Since: 0.1.0*
|
*Since: 0.1.0*
|
||||||
|
|
||||||
This exception is thrown for promise created from a [`QFuture`](../qtconcurrent.md) which has been canceled (e.g. using [`QFuture::cancel()`](http://doc.qt.io/qt-5/qfuture.html#cancel)), for example:
|
This exception is thrown for promise created from a [`QFuture`](../qtconcurrent.md) which has been
|
||||||
|
canceled (e.g. using [`QFuture::cancel()`](http://doc.qt.io/qt-5/qfuture.html#cancel)), for example:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
auto output = QtPromise::resolve(future)
|
auto output = QtPromise::resolve(future)
|
||||||
.fail([](const QPromiseCanceledException&) {
|
.fail([](const QPromiseCanceledException& error) {
|
||||||
// `future` has been canceled!
|
// `future` has been canceled!
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
@ -1,12 +1,9 @@
|
|||||||
---
|
|
||||||
title: QPromiseContextException
|
|
||||||
---
|
|
||||||
|
|
||||||
# QPromiseContextException
|
# QPromiseContextException
|
||||||
|
|
||||||
*Since: 0.5.0*
|
*Since: 0.5.0*
|
||||||
|
|
||||||
When a promise is created using [`QtPromise::connect()`](../helpers/connect.md), this exception is thrown when the `sender` object is destroyed, for example:
|
When a promise is created using [`QtPromise::connect()`](../helpers/connect.md), this exception is
|
||||||
|
thrown when the `sender` object is destroyed, for example:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
auto promise = QtPromise::connect(sender, &Object::finished, &Object::error);
|
auto promise = QtPromise::connect(sender, &Object::finished, &Object::error);
|
||||||
|
13
docs/qtpromise/exceptions/conversion.md
Normal file
13
docs/qtpromise/exceptions/conversion.md
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# QPromiseConversionException
|
||||||
|
|
||||||
|
*Since: 0.7.0*
|
||||||
|
|
||||||
|
This exception is thrown whenever a promise result conversion fails, for example:
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<QVariant> input = {...};
|
||||||
|
auto output = input.convert<int>()
|
||||||
|
.fail([](const QPromiseconversionException& e) {
|
||||||
|
// conversion may fail because input could not be converted to number
|
||||||
|
});
|
||||||
|
```
|
@ -1,17 +1,14 @@
|
|||||||
---
|
|
||||||
title: QPromiseTimeoutException
|
|
||||||
---
|
|
||||||
|
|
||||||
# QPromiseTimeoutException
|
# QPromiseTimeoutException
|
||||||
|
|
||||||
*Since: 0.2.0*
|
*Since: 0.2.0*
|
||||||
|
|
||||||
This is the default exception thrown when reaching the time limit when using the [`QPromise::timeout()`](../qpromise/timeout.md) method, for example:
|
This is the default exception thrown when reaching the time limit when using the
|
||||||
|
[`QPromise::timeout()`](../qpromise/timeout.md) method, for example:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> input = {...}
|
QPromise<int> input = {...}
|
||||||
auto output = input.timeout(2000)
|
auto output = input.timeout(2000)
|
||||||
.fail([](const QPromiseTimeoutException& e) {
|
.fail([](const QPromiseTimeoutException& error) {
|
||||||
// operation timed out after 2s!
|
// operation timed out after 2s!
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
@ -1,7 +1,3 @@
|
|||||||
---
|
|
||||||
title: QPromiseUndefinedException
|
|
||||||
---
|
|
||||||
|
|
||||||
# QPromiseUndefinedException
|
# QPromiseUndefinedException
|
||||||
|
|
||||||
*Since: 0.5.0*
|
*Since: 0.5.0*
|
||||||
|
@ -2,42 +2,154 @@
|
|||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
QtPromise is a [header-only](https://en.wikipedia.org/wiki/Header-only) library, simply download the [latest release](https://github.com/simonbrunel/qtpromise/releases/latest) (or [`git submodule`](https://git-scm.com/docs/git-submodule)) and include `qtpromise.pri` from your project `.pro`.
|
### Using CMake
|
||||||
|
|
||||||
### qpm
|
*Since: 0.6.0*
|
||||||
|
|
||||||
Alternatively and **only** if your project relies on [qpm](https://www.qpm.io/), you can install QtPromise as follow:
|
If your project uses [CMake](https://cmake.org/) as build system, QtPromise can be installed using
|
||||||
|
the `FetchContent` module. Please refer to the [CMake (FetchContent)](#cmake-fetchcontent) section
|
||||||
|
for details of using this method.
|
||||||
|
|
||||||
```bash
|
### Using qpm
|
||||||
qpm install com.github.simonbrunel.qtpromise
|
|
||||||
|
If your project is configured to use [qpm](https://www.qpm.io/), QtPromise can be installed using
|
||||||
|
the following command:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
qpm install com.github.simonbrunel.qtpromise
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
See also: [com.github.simonbrunel.qtpromise](https://www.qpm.io/packages/com.github.simonbrunel.qtpromise/)
|
||||||
|
|
||||||
The recommended way to use QtPromise is to include the single module header:
|
### Using Git
|
||||||
|
|
||||||
|
If your project uses [Git](https://git-scm.com/) as version control system, QtPromise can be
|
||||||
|
installed either as a [`subtree`](#subtree) or a [`submodule`](#submodule). Read more about these
|
||||||
|
commands in ["Git: submodules vs subtrees"](https://nering.dev/2016/git-submodules-vs-subtrees/)
|
||||||
|
which provides a good comparison between these two workflows.
|
||||||
|
|
||||||
|
The following examples install QtPromise version 0.7.0 under the `3rdparty/qtpromise` subdirectory.
|
||||||
|
Note that the install directory is arbitrary and can be any empty directory under your repository.
|
||||||
|
Once installed, refer to the [CMake](#cmake) or [qmake](#qmake) sections for details of integrating
|
||||||
|
QtPromise into your project.
|
||||||
|
|
||||||
|
#### subtree
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cd <your/project/repository>
|
||||||
|
git remote add qtpromise https://github.com/simonbrunel/qtpromise.git
|
||||||
|
git subtree add -P 3rdparty/qtpromise qtpromise v0.7.0 --squash -m "Add QtPromise v0.7.0"
|
||||||
|
```
|
||||||
|
|
||||||
|
#### submodule
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cd <your/project/repository>
|
||||||
|
git submodule add https://github.com/simonbrunel/qtpromise.git 3rdparty/qtpromise
|
||||||
|
cd 3rdparty/qtpromise
|
||||||
|
git checkout v0.7.0
|
||||||
|
cd ../..
|
||||||
|
git add 3rdparty/qtpromise
|
||||||
|
git commit -m "Add QtPromise v0.7.0"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Download
|
||||||
|
|
||||||
|
QtPromise can be downloaded from the [GitHub release page](https://github.com/simonbrunel/qtpromise/releases)
|
||||||
|
as a `zip` or `tar.gz` archive. Under Linux, you can use the following commands:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cd <your/project/repository>
|
||||||
|
wget -q -O qtpromise.tar.gz https://github.com/simonbrunel/qtpromise/archive/v0.7.0.tar.gz
|
||||||
|
tar xzf qtpromise.tar.gz --strip 1 --one-top-level=3rdparty/qtpromise
|
||||||
|
rm qtpromise.tar.gz
|
||||||
|
```
|
||||||
|
|
||||||
|
## Integration
|
||||||
|
|
||||||
|
QtPromise is a [header-only](https://en.wikipedia.org/wiki/Header-only) library. Integrating it
|
||||||
|
within your project consists only in configuring the include path(s) to the library headers. To
|
||||||
|
simplify this step, [qmake](#qmake) and [CMake](#cmake) integrations are provided.
|
||||||
|
|
||||||
|
### CMake
|
||||||
|
|
||||||
|
*Since: 0.6.0*
|
||||||
|
|
||||||
|
After installing QtPromise using [Git](#using-git) or the [download](#download) method, you can use
|
||||||
|
the [`add_subdirectory`](https://cmake.org/cmake/help/latest/command/add_subdirectory.html) command
|
||||||
|
to make its targets available to your CMake project:
|
||||||
|
|
||||||
|
```cmake
|
||||||
|
add_subdirectory(<path/to/qtpromise>)
|
||||||
|
|
||||||
|
target_link_libraries(<target> qtpromise)
|
||||||
|
```
|
||||||
|
|
||||||
|
### CMake (FetchContent)
|
||||||
|
|
||||||
|
*Since: 0.6.0*
|
||||||
|
|
||||||
|
Alternatively, the [`FetchContent`](https://cmake.org/cmake/help/latest/module/FetchContent.html)
|
||||||
|
module (**CMake 3.11+**) allows to install QtPromise from your CMake project at configure time:
|
||||||
|
|
||||||
|
```cmake
|
||||||
|
include(FetchContent)
|
||||||
|
|
||||||
|
FetchContent_Declare(qtpromise
|
||||||
|
GIT_REPOSITORY https://github.com/simonbrunel/qtpromise.git
|
||||||
|
GIT_TAG v0.7.0
|
||||||
|
GIT_SHALLOW true
|
||||||
|
)
|
||||||
|
|
||||||
|
# CMake v3.14+
|
||||||
|
FetchContent_MakeAvailable(qtpromise)
|
||||||
|
|
||||||
|
target_link_libraries(<target> qtpromise)
|
||||||
|
```
|
||||||
|
|
||||||
|
If your CMake version **is prior to v3.14**, you need to explicitly define the population steps:
|
||||||
|
|
||||||
|
```cmake
|
||||||
|
# CMake v3.11+ (alternative for FetchContent_MakeAvailable)
|
||||||
|
FetchContent_GetProperties(qtpromise)
|
||||||
|
if(NOT qtpromise_POPULATED)
|
||||||
|
FetchContent_Populate(qtpromise)
|
||||||
|
add_subdirectory(
|
||||||
|
${qtpromise_SOURCE_DIR}
|
||||||
|
${qtpromise_BINARY_DIR}
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
```
|
||||||
|
|
||||||
|
### qmake
|
||||||
|
|
||||||
|
After installing QtPromise using [Git](#using-git) or the [download](#download) method, you can
|
||||||
|
[include](https://doc.qt.io/qt-5/qmake-test-function-reference.html#include-filename) `qtpromise.pri`
|
||||||
|
file from the install directory:
|
||||||
|
|
||||||
|
```cmake
|
||||||
|
include(<path/to/qtpromise>/qtpromise.pri)
|
||||||
|
```
|
||||||
|
|
||||||
|
## Example
|
||||||
|
|
||||||
|
To start using QtPromise in your code, you first need to include the single module header:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
#include <QtPromise>
|
#include <QtPromise>
|
||||||
```
|
```
|
||||||
|
|
||||||
## Example
|
The following `download` function creates a [promise from callbacks](qpromise/constructor.md) which
|
||||||
|
will be resolved when the network request is finished:
|
||||||
Let's first make the code more readable by using the library namespace:
|
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
using namespace QtPromise;
|
QtPromise::QPromise<QByteArray> download(const QUrl& url)
|
||||||
```
|
|
||||||
|
|
||||||
This `download` function creates a [promise from callbacks](qpromise/constructor.md) which will be resolved when the network request is finished:
|
|
||||||
|
|
||||||
```cpp
|
|
||||||
QPromise<QByteArray> download(const QUrl& url)
|
|
||||||
{
|
{
|
||||||
return QPromise<QByteArray>([&](
|
return QtPromise::QPromise<QByteArray>{[&](
|
||||||
const QPromiseResolve<QByteArray>& resolve,
|
const QtPromise::QPromiseResolve<QByteArray>& resolve,
|
||||||
const QPromiseReject<QByteArray>& reject) {
|
const QtPromise::QPromiseReject<QByteArray>& reject) {
|
||||||
|
|
||||||
QNetworkReply* reply = manager->get(QNetworkRequest(url));
|
QNetworkReply* reply = manager->get(QNetworkRequest{url});
|
||||||
QObject::connect(reply, &QNetworkReply::finished, [=]() {
|
QObject::connect(reply, &QNetworkReply::finished, [=]() {
|
||||||
if (reply->error() == QNetworkReply::NoError) {
|
if (reply->error() == QNetworkReply::NoError) {
|
||||||
resolve(reply->readAll());
|
resolve(reply->readAll());
|
||||||
@ -47,14 +159,14 @@ QPromise<QByteArray> download(const QUrl& url)
|
|||||||
|
|
||||||
reply->deleteLater();
|
reply->deleteLater();
|
||||||
});
|
});
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
The following method `uncompress` data in a separate thread and returns a [promise from QFuture](qtconcurrent.md):
|
The following method `uncompress` data in a separate thread and returns a [promise from QFuture](qtconcurrent.md):
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<Entries> uncompress(const QByteArray& data)
|
QtPromise::QPromise<Entries> uncompress(const QByteArray& data)
|
||||||
{
|
{
|
||||||
return QtPromise::resolve(QtConcurrent::run([](const QByteArray& data) {
|
return QtPromise::resolve(QtConcurrent::run([](const QByteArray& data) {
|
||||||
Entries entries;
|
Entries entries;
|
||||||
@ -62,7 +174,7 @@ QPromise<Entries> uncompress(const QByteArray& data)
|
|||||||
// {...} uncompress data and parse content.
|
// {...} uncompress data and parse content.
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
throw MalformedException();
|
throw MalformedException{};
|
||||||
}
|
}
|
||||||
|
|
||||||
return entries;
|
return entries;
|
||||||
@ -71,6 +183,7 @@ QPromise<Entries> uncompress(const QByteArray& data)
|
|||||||
```
|
```
|
||||||
|
|
||||||
It's then easy to chain the whole asynchronous process using promises:
|
It's then easy to chain the whole asynchronous process using promises:
|
||||||
|
|
||||||
- initiate the promise chain by downloading a specific URL,
|
- initiate the promise chain by downloading a specific URL,
|
||||||
- [`then`](qpromise/then.md) *and only if download succeeded*, uncompress received data,
|
- [`then`](qpromise/then.md) *and only if download succeeded*, uncompress received data,
|
||||||
- [`then`](qpromise/then.md) validate and process the uncompressed entries,
|
- [`then`](qpromise/then.md) validate and process the uncompressed entries,
|
||||||
@ -80,16 +193,19 @@ It's then easy to chain the whole asynchronous process using promises:
|
|||||||
```cpp
|
```cpp
|
||||||
download(url).then(&uncompress).then([](const Entries& entries) {
|
download(url).then(&uncompress).then([](const Entries& entries) {
|
||||||
if (entries.isEmpty()) {
|
if (entries.isEmpty()) {
|
||||||
throw UpdateException("No entries");
|
throw UpdateException{"No entries"};
|
||||||
}
|
}
|
||||||
// {...} process entries
|
// {...} process entries
|
||||||
}).finally([]() {
|
}).finally([]() {
|
||||||
// {...} cleanup
|
// {...} cleanup
|
||||||
}).fail([](QNetworkReply::NetworkError err) {
|
}).fail([](QNetworkReply::NetworkError error) {
|
||||||
// {...} handle network error
|
// {...} handle network error
|
||||||
}).fail([](const UpdateException& err) {
|
}).fail([](const UpdateException& error) {
|
||||||
// {...} handle update error
|
// {...} handle update error
|
||||||
}).fail([]() {
|
}).fail([]() {
|
||||||
// {...} catch all
|
// {...} catch all
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Note that `MalformedException` in the example above is thrown from a QtConcurrent thread and should
|
||||||
|
meet [specific conditions](qtconcurrent.md#error).
|
||||||
|
@ -6,14 +6,18 @@ title: all
|
|||||||
|
|
||||||
*Since: 0.5.0*
|
*Since: 0.5.0*
|
||||||
|
|
||||||
```
|
```cpp
|
||||||
QtPromise::all(Sequence<QPromise<T>> promises) -> QPromise<QVector<T>>
|
QtPromise::all(Sequence<QPromise<T>> promises) -> QPromise<QVector<T>>
|
||||||
QtPromise::all(Sequence<QPromise<void>> promises) -> QPromise<void>
|
QtPromise::all(Sequence<QPromise<void>> promises) -> QPromise<void>
|
||||||
```
|
```
|
||||||
|
|
||||||
Returns a `QPromise<QVector<T>>` (or `QPromise<void>`) that fulfills when **all** `promises` of (the same) type `T` have been fulfilled. The `output` value is a vector containing all the values of `promises`, in the same order, i.e., at the respective positions to the original sequence, regardless of completion order.
|
Returns a `QPromise<QVector<T>>` (or `QPromise<void>`) that fulfills when **all** `promises` of
|
||||||
|
(the same) type `T` have been fulfilled. The `output` value is a vector containing all the values
|
||||||
|
of `promises`, in the same order, i.e., at the respective positions to the original sequence,
|
||||||
|
regardless of completion order.
|
||||||
|
|
||||||
If any of the given `promises` fail, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
If any of the given `promises` fail, `output` immediately rejects with the error of the promise that
|
||||||
|
rejected, whether or not the other promises are resolved.
|
||||||
|
|
||||||
`Sequence` is any STL compatible container (eg. `QVector`, `QList`, `std::vector`, etc.)
|
`Sequence` is any STL compatible container (eg. `QVector`, `QList`, `std::vector`, etc.)
|
||||||
|
|
||||||
|
@ -13,9 +13,14 @@ QtPromise::attempt(Functor functor, Args...) -> QPromise<R>
|
|||||||
// - Functor: Function(Args...) -> R | QPromise<R>
|
// - Functor: Function(Args...) -> R | QPromise<R>
|
||||||
```
|
```
|
||||||
|
|
||||||
Calls `functor` immediately and returns a promise fulfilled with the value returned by `functor`. Any synchronous exceptions will be turned into rejections on the returned promise. This is a convenient method that can be used instead of handling both synchronous and asynchronous exception flows.
|
Calls `functor` immediately and returns a promise fulfilled with the value returned by `functor`.
|
||||||
|
Any synchronous exceptions will be turned into rejections on the returned promise. This is a
|
||||||
|
convenient method that can be used instead of handling both synchronous and asynchronous exception
|
||||||
|
flows.
|
||||||
|
|
||||||
The type `R` of the `output` promise depends on the type returned by the `functor` function. If `functor` returns a promise (or `QFuture`), the `output` promise is delayed and will be resolved by the returned promise.
|
The type `R` of the `output` promise depends on the type returned by the `functor` function. If
|
||||||
|
`functor` returns a promise (or `QFuture`), the `output` promise is delayed and will be resolved
|
||||||
|
by the returned promise.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<QByteArray> download(const QUrl& url);
|
QPromise<QByteArray> download(const QUrl& url);
|
||||||
@ -24,7 +29,7 @@ QPromise<QByteArray> process(const QUrl& url)
|
|||||||
{
|
{
|
||||||
return QtPromise::attempt([&]() {
|
return QtPromise::attempt([&]() {
|
||||||
if (!url.isValid()) {
|
if (!url.isValid()) {
|
||||||
throw InvalidUrlException();
|
throw InvalidUrlException{};
|
||||||
}
|
}
|
||||||
|
|
||||||
return download(url);
|
return download(url);
|
||||||
@ -36,7 +41,7 @@ auto output = process(url);
|
|||||||
// 'output' type: QPromise<QByteArray>
|
// 'output' type: QPromise<QByteArray>
|
||||||
output.then([](const QByteArray& res) {
|
output.then([](const QByteArray& res) {
|
||||||
// {...}
|
// {...}
|
||||||
}).fail([](const InvalidUrlException& err) {
|
}).fail([](const InvalidUrlException& error) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
@ -12,11 +12,16 @@ title: connect
|
|||||||
(3) QtPromise::connect(QObject* sender, Signal(T) resolver, QObject* sender2, Signal(R) rejecter) -> QPromise<T>
|
(3) QtPromise::connect(QObject* sender, Signal(T) resolver, QObject* sender2, Signal(R) rejecter) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
Creates a `QPromise<T>` that will be fulfilled with the `resolver` signal's first argument, or a `QPromise<void>` if `resolver` doesn't provide any argument.
|
Creates a `QPromise<T>` that will be fulfilled with the `resolver` signal's first argument, or a
|
||||||
|
`QPromise<void>` if `resolver` doesn't provide any argument.
|
||||||
|
|
||||||
The second `(2)` and third `(3)` variants of this method will reject the `output` promise when the `rejecter` signal is emitted. The rejection reason is the value of the `rejecter` signal's first argument or [`QPromiseUndefinedException`](../exceptions/undefined) if `rejected` doesn't provide any argument.
|
The second `(2)` and third `(3)` variants of this method will reject the `output` promise when the
|
||||||
|
`rejecter` signal is emitted. The rejection reason is the value of the `rejecter` signal's first
|
||||||
|
argument or [`QPromiseUndefinedException`](../exceptions/undefined.md) if `rejected` doesn't provide
|
||||||
|
any argument.
|
||||||
|
|
||||||
Additionally, the `output` promise will be automatically rejected with [`QPromiseContextException`](../exceptions/context.md) if `sender` is destroyed before the promise is resolved (that doesn't apply to `sender2`).
|
Additionally, the `output` promise will be automatically rejected with [`QPromiseContextException`](../exceptions/context.md)
|
||||||
|
if `sender` is destroyed before the promise is resolved (that doesn't apply to `sender2`).
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
class Sender : public QObject
|
class Sender : public QObject
|
||||||
@ -26,7 +31,7 @@ Q_SIGNALS:
|
|||||||
void error(ErrorCode);
|
void error(ErrorCode);
|
||||||
};
|
};
|
||||||
|
|
||||||
auto sender = new Sender();
|
auto sender = new Sender{};
|
||||||
auto output = QtPromise::connect(sender, &Sender::finished, &Sender::error);
|
auto output = QtPromise::connect(sender, &Sender::finished, &Sender::error);
|
||||||
|
|
||||||
// 'output' resolves as soon as one of the following events happens:
|
// 'output' resolves as soon as one of the following events happens:
|
||||||
@ -37,9 +42,9 @@ auto output = QtPromise::connect(sender, &Sender::finished, &Sender::error);
|
|||||||
// 'output' type: QPromise<QByteArray>
|
// 'output' type: QPromise<QByteArray>
|
||||||
output.then([](const QByteArray& res) {
|
output.then([](const QByteArray& res) {
|
||||||
// 'res' is the first argument of the 'finished' signal.
|
// 'res' is the first argument of the 'finished' signal.
|
||||||
}).fail([](ErrorCode err) {
|
}).fail([](ErrorCode error) {
|
||||||
// 'err' is the first argument of the 'error' signal.
|
// 'error' is the first argument of the 'error' signal.
|
||||||
}).fail([](const QPromiseContextException& err) {
|
}).fail([](const QPromiseContextException& error) {
|
||||||
// the 'sender' object has been destroyed before any of
|
// the 'sender' object has been destroyed before any of
|
||||||
// the 'finished' or 'error' signals have been emitted.
|
// the 'finished' or 'error' signals have been emitted.
|
||||||
});
|
});
|
||||||
|
@ -14,9 +14,12 @@ QtPromise::each(Sequence<T> values, Functor functor) -> QPromise<Sequence<T>>
|
|||||||
// - Functor: Function(T value, int index) -> void | QPromise<void>
|
// - Functor: Function(T value, int index) -> void | QPromise<void>
|
||||||
```
|
```
|
||||||
|
|
||||||
Calls the given `functor` on each element in `values` then resolves to the original sequence unmodified. If `functor` throws, `output` is rejected with the new exception.
|
Calls the given `functor` on each element in `values` then resolves to the original sequence
|
||||||
|
unmodified. If `functor` throws, `output` is rejected with the new exception.
|
||||||
|
|
||||||
If `functor` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises are resolved. If any of the promises fail, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
If `functor` returns a promise (or `QFuture`), the `output` promise is delayed until all the
|
||||||
|
promises are resolved. If any of the promises fail, `output` immediately rejects with the error
|
||||||
|
of the promise that rejected, whether or not the other promises are resolved.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
auto output = QtPromise::each(QVector<QUrl>{
|
auto output = QtPromise::each(QVector<QUrl>{
|
||||||
@ -24,9 +27,9 @@ auto output = QtPromise::each(QVector<QUrl>{
|
|||||||
QUrl("http://b..."),
|
QUrl("http://b..."),
|
||||||
QUrl("http://c...")
|
QUrl("http://c...")
|
||||||
}, [](const QUrl& url, ...) {
|
}, [](const QUrl& url, ...) {
|
||||||
return QPromise<void>([&](auto resolve, auto reject) {
|
return QPromise<void>{[&](auto resolve, auto reject) {
|
||||||
// process url asynchronously ...
|
// process url asynchronously ...
|
||||||
})
|
}};
|
||||||
});
|
});
|
||||||
|
|
||||||
// `output` resolves as soon as all promises returned by
|
// `output` resolves as soon as all promises returned by
|
||||||
|
@ -14,9 +14,14 @@ QtPromise::filter(Sequence<T> values, Filterer filterer) -> QPromise<Sequence<T>
|
|||||||
// - Filterer: Function(T value, int index) -> bool
|
// - Filterer: Function(T value, int index) -> bool
|
||||||
```
|
```
|
||||||
|
|
||||||
Iterates over `values` and [filters the sequence](https://en.wikipedia.org/wiki/Filter_%28higher-order_function%29) to another using the given `filterer` function. If `filterer` returns `true`, a copy of the item is put in the `output` sequence, otherwise, the item will not appear in `output`. If `filterer` throws, `output` is rejected with the new exception.
|
Iterates over `values` and [filters the sequence](https://en.wikipedia.org/wiki/Filter_%28higher-order_function%29)
|
||||||
|
to another using the given `filterer` function. If `filterer` returns `true`, a copy of the item
|
||||||
|
is put in the `output` sequence, otherwise, the item will not appear in `output`. If `filterer`
|
||||||
|
throws, `output` is rejected with the new exception.
|
||||||
|
|
||||||
If `filterer` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises are resolved. If any of the promises fail, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
If `filterer` returns a promise (or `QFuture`), the `output` promise is delayed until all the
|
||||||
|
promises are resolved. If any of the promises fail, `output` immediately rejects with the error
|
||||||
|
of the promise that rejected, whether or not the other promises are resolved.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
auto output = QtPromise::filter(QVector{
|
auto output = QtPromise::filter(QVector{
|
||||||
@ -24,10 +29,10 @@ auto output = QtPromise::filter(QVector{
|
|||||||
QUrl("http://b..."),
|
QUrl("http://b..."),
|
||||||
QUrl("http://c...")
|
QUrl("http://c...")
|
||||||
}, [](const QUrl& url, ...) {
|
}, [](const QUrl& url, ...) {
|
||||||
return QPromise<bool>([&](auto resolve, auto reject) {
|
return QPromise<bool>{[&](auto resolve, auto reject) {
|
||||||
// resolve(true) if 'url' is reachable, else resolve(false)
|
// resolve(true) if 'url' is reachable, else resolve(false)
|
||||||
// {...}
|
// {...}
|
||||||
});
|
}};
|
||||||
});
|
});
|
||||||
|
|
||||||
// 'output' resolves as soon as all promises returned by
|
// 'output' resolves as soon as all promises returned by
|
||||||
@ -40,7 +45,8 @@ output.then([](const QVector<QUrl>& res) {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
The order of the output sequence values is guarantee to be the same as the original sequence, regardless of completion order of the promises returned by `filterer`.
|
The order of the output sequence values is guarantee to be the same as the original sequence,
|
||||||
|
regardless of completion order of the promises returned by `filterer`.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
See also: [`QPromise<T>::filter`](../qpromise/filter.md)
|
See also: [`QPromise<T>::filter`](../qpromise/filter.md)
|
||||||
|
@ -14,9 +14,13 @@ QtPromise::map(Sequence<T> values, Mapper mapper) -> QPromise<QVector<R>>
|
|||||||
// - Mapper: Function(T value, int index) -> R | QPromise<R>
|
// - Mapper: Function(T value, int index) -> R | QPromise<R>
|
||||||
```
|
```
|
||||||
|
|
||||||
Iterates over `values` and [maps the sequence](https://en.wikipedia.org/wiki/Map_%28higher-order_function%29) to another using the given `mapper` function. The type returned by `mapper` determines the type of the `output` promise. If `mapper` throws, `output` is rejected with the new exception.
|
Iterates over `values` and [maps the sequence](https://en.wikipedia.org/wiki/Map_%28higher-order_function%29)
|
||||||
|
to another using the given `mapper` function. The type returned by `mapper` determines the type of
|
||||||
|
the `output` promise. If `mapper` throws, `output` is rejected with the new exception.
|
||||||
|
|
||||||
If `mapper` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises are resolved. If any of the promises fails, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
If `mapper` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises
|
||||||
|
are resolved. If any of the promises fails, `output` immediately rejects with the error of the
|
||||||
|
promise that rejected, whether or not the other promises are resolved.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
auto output = QtPromise::map(QVector{
|
auto output = QtPromise::map(QVector{
|
||||||
@ -24,10 +28,10 @@ auto output = QtPromise::map(QVector{
|
|||||||
QUrl("http://b..."),
|
QUrl("http://b..."),
|
||||||
QUrl("http://c...")
|
QUrl("http://c...")
|
||||||
}, [](const QUrl& url, ...) {
|
}, [](const QUrl& url, ...) {
|
||||||
return QPromise<QByteArray>([&](auto resolve, auto reject) {
|
return QPromise<QByteArray>{[&](auto resolve, auto reject) {
|
||||||
// download content at url and resolve
|
// download content at url and resolve
|
||||||
// {...}
|
// {...}
|
||||||
});
|
}};
|
||||||
});
|
});
|
||||||
|
|
||||||
// 'output' resolves as soon as all promises returned by
|
// 'output' resolves as soon as all promises returned by
|
||||||
@ -40,7 +44,8 @@ output.then([](const QVector<QByteArray>& res) {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
The order of the output sequence values is guarantee to be the same as the original sequence, regardless of completion order of the promises returned by `mapper`.
|
The order of the output sequence values is guarantee to be the same as the original sequence,
|
||||||
|
regardless of completion order of the promises returned by `mapper`.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
See also: [`QPromise<T>::map`](../qpromise/map.md)
|
See also: [`QPromise<T>::map`](../qpromise/map.md)
|
||||||
|
@ -15,9 +15,15 @@ QPromise::reduce(Sequence<T|QPromise<T>> values, Reducer reducer, R|QPromise<R>
|
|||||||
// - Reducer: Function(T|R accumulator, T item, int index) -> R|QPromise<R>
|
// - Reducer: Function(T|R accumulator, T item, int index) -> R|QPromise<R>
|
||||||
```
|
```
|
||||||
|
|
||||||
Iterates over `values` and [reduces the sequence to a single value](https://en.wikipedia.org/wiki/Fold_%28higher-order_function%29) using the given `reducer` function and an optional `initialValue`. The type returned by `reducer` determines the type of the `output` promise. If `reducer` throws, `output` is rejected with the new exception.
|
Iterates over `values` and [reduces the sequence to a single value](https://en.wikipedia.org/wiki/Fold_%28higher-order_function%29)
|
||||||
|
using the given `reducer` function and an optional `initialValue`. The type returned by `reducer`
|
||||||
|
determines the type of the `output` promise. If `reducer` throws, `output` is rejected with the
|
||||||
|
new exception.
|
||||||
|
|
||||||
If `reducer` returns a promise (or `QFuture`), then the result of the promise is awaited, before continuing with next iteration. If any promise in the `values` sequence is rejected or a promise returned by the `reducer` function is rejected, `output` immediately rejects with the error of the promise that rejected.
|
If `reducer` returns a promise (or `QFuture`), then the result of the promise is awaited, before
|
||||||
|
continuing with next iteration. If any promise in the `values` sequence is rejected or a promise
|
||||||
|
returned by the `reducer` function is rejected, `output` immediately rejects with the error of
|
||||||
|
the promise that rejected.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
// Concatenate the content of the given files, read asynchronously
|
// Concatenate the content of the given files, read asynchronously
|
||||||
@ -27,9 +33,9 @@ auto output = QtPromise::reduce(QList<QUrl>{
|
|||||||
"file:f2.txt" // contains "42"
|
"file:f2.txt" // contains "42"
|
||||||
}, [](const QString& acc, const QString& cur, int idx) {
|
}, [](const QString& acc, const QString& cur, int idx) {
|
||||||
return readAsync(cur).then([=](const QString& res) {
|
return readAsync(cur).then([=](const QString& res) {
|
||||||
return QString("%1;%2:%3").arg(acc).arg(idx).arg(res);
|
return QString{"%1;%2:%3"}.arg(acc).arg(idx).arg(res);
|
||||||
});
|
});
|
||||||
}, QString("index:text"));
|
}, QString{"index:text"});
|
||||||
|
|
||||||
// 'output' resolves as soon as all promises returned by
|
// 'output' resolves as soon as all promises returned by
|
||||||
// 'reducer' are fulfilled or at least one is rejected.
|
// 'reducer' are fulfilled or at least one is rejected.
|
||||||
@ -42,7 +48,8 @@ output.then([](const QString& res) {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: warning IMPORTANT
|
::: warning IMPORTANT
|
||||||
The first time `reducer` is called, if no `initialValue` is provided, `accumulator` will be equal to the first value in the sequence, and `currentValue` to the second one (thus index will be `1`).
|
The first time `reducer` is called, if no `initialValue` is provided, `accumulator` will be equal
|
||||||
|
to the first value in the sequence, and `currentValue` to the second one (thus index will be `1`).
|
||||||
:::
|
:::
|
||||||
|
|
||||||
See also: [`QPromise<T>::reduce`](../qpromise/reduce.md)
|
See also: [`QPromise<T>::reduce`](../qpromise/reduce.md)
|
||||||
|
@ -6,16 +6,18 @@ title: resolve
|
|||||||
|
|
||||||
*Since: 0.5.0*
|
*Since: 0.5.0*
|
||||||
|
|
||||||
```
|
```cpp
|
||||||
QtPromise::resolve(T value) -> QPromise<R>
|
QtPromise::resolve(T value) -> QPromise<R>
|
||||||
```
|
```
|
||||||
|
|
||||||
Similar to the [`QPromise<T>::resolve`](../qpromise/resolve.md) static method, creates a promise resolved from a given `value` but without the extra typing:
|
Similar to the [`QPromise<T>::resolve`](../qpromise/resolve.md) static method, creates a promise
|
||||||
|
resolved from a given `value` but without the extra typing:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
auto promise = QtPromise::resolve(); // QPromise<void>
|
auto promise = QtPromise::resolve(); // QPromise<void>
|
||||||
auto promise = QtPromise::resolve(42); // QPromise<int>
|
auto promise = QtPromise::resolve(42); // QPromise<int>
|
||||||
auto promise = QtPromise::resolve(QString("foo")); // QPromise<QString>
|
auto promise = QtPromise::resolve(QString{"foo"}); // QPromise<QString>
|
||||||
```
|
```
|
||||||
|
|
||||||
This method also allows to convert `QFuture<T>` to `QPromise<T>`, delayed until the `QFuture` is finished ([read more](../qtconcurrent.md#convert)).
|
This method also allows to convert `QFuture<T>` to `QPromise<T>`, delayed until the `QFuture` is
|
||||||
|
finished ([read more](../qtconcurrent.md#convert)).
|
||||||
|
@ -1,34 +0,0 @@
|
|||||||
---
|
|
||||||
title: ::all [static]
|
|
||||||
---
|
|
||||||
|
|
||||||
# QPromise::all [static]
|
|
||||||
|
|
||||||
*Since: 0.1.0*
|
|
||||||
|
|
||||||
```
|
|
||||||
[static] QPromise<T>::all(Sequence<QPromise<T>> promises) -> QPromise<QVector<T>>
|
|
||||||
```
|
|
||||||
|
|
||||||
Returns a `QPromise<QVector<T>>` that fulfills when **all** `promises` of (the same) type `T` have been fulfilled. The `output` value is a vector containing all the values of `promises`, in the same order, i.e., at the respective positions to the original sequence, regardless of completion order.
|
|
||||||
|
|
||||||
If any of the given `promises` fail, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
|
||||||
|
|
||||||
`Sequence` is any STL compatible container (eg. `QVector`, `QList`, `std::vector`, etc.)
|
|
||||||
|
|
||||||
```cpp
|
|
||||||
QVector<QPromise<QByteArray> > promises{
|
|
||||||
download(QUrl("http://a...")),
|
|
||||||
download(QUrl("http://b...")),
|
|
||||||
download(QUrl("http://c..."))
|
|
||||||
};
|
|
||||||
|
|
||||||
auto output = QPromise<QByteArray>::all(promises);
|
|
||||||
|
|
||||||
// output type: QPromise<QVector<QByteArray>>
|
|
||||||
output.then([](const QVector<QByteArray>& res) {
|
|
||||||
// {...}
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
See also: [`QtPromise::all`](../helpers/all.md)
|
|
@ -18,32 +18,35 @@ QPromise<int> promise([](const QPromiseResolve<int>& resolve, const QPromiseReje
|
|||||||
if (success) {
|
if (success) {
|
||||||
resolve(result);
|
resolve(result);
|
||||||
} else {
|
} else {
|
||||||
reject(customException());
|
reject(customException{});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
`QPromise<void>` is specialized to not contain any value, meaning that the `resolve` callback takes no argument.
|
`QPromise<void>` is specialized to not contain any value, meaning that the `resolve` callback takes
|
||||||
|
no argument.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
**C++14**
|
C++14 alternative:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> promise([](const auto& resolve, const auto& reject) {
|
QPromise<int> promise{[](const auto& resolve, const auto& reject) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
}};
|
||||||
```
|
```
|
||||||
|
|
||||||
**Undefined rejection reason**
|
## Undefined rejection reason
|
||||||
|
|
||||||
*Since: 0.5.0*
|
*Since: 0.5.0*
|
||||||
|
|
||||||
While not recommended because it makes tracking errors more difficult, it's also possible to reject a promise without explicit reason, in which case, a built-in [`QPromiseUndefinedException`](../exceptions/undefined.md) is thrown:
|
While not recommended because it makes tracking errors more difficult, it's also possible to reject
|
||||||
|
a promise without explicit reason, in which case, a built-in [`QPromiseUndefinedException`](../exceptions/undefined.md)
|
||||||
|
is thrown:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> promise([](const QPromiseResolve<int>& resolve, const QPromiseReject<int>& reject) {
|
QPromise<int> promise{[](const QPromiseResolve<int>& resolve, const QPromiseReject<int>& reject) {
|
||||||
async_method([=](bool success, int result) {
|
async_method([=](bool success, int result) {
|
||||||
if (success) {
|
if (success) {
|
||||||
resolve(result);
|
resolve(result);
|
||||||
@ -51,7 +54,7 @@ QPromise<int> promise([](const QPromiseResolve<int>& resolve, const QPromiseReje
|
|||||||
reject();
|
reject();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
}};
|
||||||
```
|
```
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
|
124
docs/qtpromise/qpromise/convert.md
Normal file
124
docs/qtpromise/qpromise/convert.md
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
---
|
||||||
|
title: .convert
|
||||||
|
---
|
||||||
|
|
||||||
|
# QPromise::convert
|
||||||
|
|
||||||
|
*Since: 0.7.0*
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<T>::convert<U>() -> QPromise<U>
|
||||||
|
```
|
||||||
|
|
||||||
|
This method converts the resolved value of `QPromise<T>` to the type `U`. Depending on types `T`
|
||||||
|
and `U`, it performs a [static cast](https://en.cppreference.com/w/cpp/language/static_cast),
|
||||||
|
calls a [converting constructor](https://en.cppreference.com/w/cpp/language/converting_constructor),
|
||||||
|
or tries to convert using [QVariant](https://doc.qt.io/qt-5/qvariant.html).
|
||||||
|
|
||||||
|
If `T` and `U` are [fundamental types](https://en.cppreference.com/w/cpp/language/types) or
|
||||||
|
[enumerations](https://en.cppreference.com/w/cpp/language/enum), the result of the conversion is
|
||||||
|
the same as calling `static_cast<U>` for type `T`:
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<int> input = {...}
|
||||||
|
|
||||||
|
// output type: QPromise<double>
|
||||||
|
auto output = input.convert<double>();
|
||||||
|
|
||||||
|
output.then([](double value) {
|
||||||
|
// the value has been converted using static_cast
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
If `U` has a [converting constructor](https://en.cppreference.com/w/cpp/language/converting_constructor)
|
||||||
|
from `T`, i.e., a non-explicit constructor with a single argument accepting `T`, it is used to
|
||||||
|
convert the value:
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<QByteArray> input = {...}
|
||||||
|
|
||||||
|
// output type: QPromise<QString>
|
||||||
|
auto output = input.convert<QString>();
|
||||||
|
|
||||||
|
output.then([](const QString& value) {
|
||||||
|
// the value has been converted using static_cast that effectively calls QString(QByteArray)
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
::: tip NOTE
|
||||||
|
When using this method to convert to your own classes, make sure that the constructor meeting the
|
||||||
|
converting constructor criteria actually performs conversion.
|
||||||
|
:::
|
||||||
|
|
||||||
|
::: tip NOTE
|
||||||
|
If `U` is `void`, the resolved value of `QPromise<T>` is dropped.
|
||||||
|
:::
|
||||||
|
|
||||||
|
Calling this method for `QPromise<QVariant>` tries to convert the resolved `QVariant` to type `U`
|
||||||
|
using the `QVariant` [conversion algorithm](https://doc.qt.io/qt-5/qvariant.html#using-canconvert-and-convert-consecutively).
|
||||||
|
For example, this allows to convert a string contained in `QVariant` to number. If such a
|
||||||
|
conversion fails, the promise is rejected with
|
||||||
|
[`QPromiseConversionException`](../exceptions/conversion.md).
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
// resolves to QVariant(int, 42) or QVariant(string, "foo")
|
||||||
|
QPromise<QVariant> input = {...};
|
||||||
|
|
||||||
|
auto output = input.convert<int>();
|
||||||
|
|
||||||
|
// output type: QPromise<int>
|
||||||
|
output.then([](int value) {
|
||||||
|
// input was QVariant(int, 42), value is 42
|
||||||
|
})
|
||||||
|
.fail(const QPromiseConversionException& e) {
|
||||||
|
// input was QVariant(string, "foo")
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
Conversion of `T` to `QVariant` using this method effectively calls `QVariant::fromValue<T>()`.
|
||||||
|
All custom types must be registered with
|
||||||
|
[`Q_DECLARE_METATYPE`](https://doc.qt.io/qt-5/qmetatype.html#Q_DECLARE_METATYPE) for this
|
||||||
|
conversion to work:
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
struct Foo {};
|
||||||
|
Q_DECLARE_METATYPE(Foo);
|
||||||
|
|
||||||
|
QPromise<Foo> input = {...}
|
||||||
|
|
||||||
|
auto output = input.convert<QVariant>();
|
||||||
|
|
||||||
|
// output type: QPromise<QVariant>
|
||||||
|
output.then([](const QVariant& value) {
|
||||||
|
// value contains an instance of Foo
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
All other combinations of `T` and `U` are converted via `QVariant`. All non-Qt types should provide
|
||||||
|
a [conversion function](https://doc.qt.io/qt-5/qmetatype.html#registerConverter), otherwise the
|
||||||
|
promise is rejected with [`QPromiseConversionException`](../exceptions/conversion.md):
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
struct Foo {};
|
||||||
|
Q_DECLARE_METATYPE(Foo);
|
||||||
|
|
||||||
|
QMetaType::registerConverter<Foo, QString>([](const Foo& foo) {
|
||||||
|
return QString{...};
|
||||||
|
});
|
||||||
|
|
||||||
|
QPromise<Foo> input = {...}
|
||||||
|
|
||||||
|
auto output = input.convert<QVariant>();
|
||||||
|
|
||||||
|
// output type: QPromise<QString>
|
||||||
|
output.then([](const QString& value) {
|
||||||
|
// value contains a result produced by the custom converter
|
||||||
|
})
|
||||||
|
.fail([](const QPromiseConversionException& e) {
|
||||||
|
// QVariant was unable to convert Foo to QString
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
::: warning IMPORTANT
|
||||||
|
Calling this method for `QPromise<void>` is not supported.
|
||||||
|
:::
|
@ -10,7 +10,9 @@ title: .delay
|
|||||||
QPromise<T>::delay(int msec) -> QPromise<T>
|
QPromise<T>::delay(int msec) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
This method returns a promise that will be fulfilled with the same value as the `input` promise and after at least `msec` milliseconds. If the `input` promise is rejected, the `output` promise is immediately rejected with the same reason.
|
This method returns a promise that will be fulfilled with the same value as the `input` promise
|
||||||
|
and after at least `msec` milliseconds. If the `input` promise is rejected, the `output` promise
|
||||||
|
is immediately rejected with the same reason.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> input = {...}
|
QPromise<int> input = {...}
|
||||||
@ -18,3 +20,31 @@ auto output = input.delay(2000).then([](int res) {
|
|||||||
// called 2 seconds after `input` is fulfilled
|
// called 2 seconds after `input` is fulfilled
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
*Since: 0.6.0*
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<T>::delay(std::chrono::milliseconds msec) -> QPromise<T>
|
||||||
|
```
|
||||||
|
|
||||||
|
This is a convenience overload accepting [durations from the C++ Standard Library](https://en.cppreference.com/w/cpp/chrono/duration).
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<int> input = {...}
|
||||||
|
auto output = input.delay(std::chrono::seconds{2}).then([](int res) {
|
||||||
|
// called 2 seconds after `input` is fulfilled
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
C++14 alternative:
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
using namespace std::chrono_literals;
|
||||||
|
|
||||||
|
QPromise<int> input = {...}
|
||||||
|
auto output = input.delay(2s).then([](int res) {
|
||||||
|
// called 2 seconds after `input` is fulfilled
|
||||||
|
});
|
||||||
|
```
|
||||||
|
@ -18,7 +18,9 @@ QPromise<Sequence<T>>::each(Functor functor) -> QPromise<Sequence<T>>
|
|||||||
This method only applies to promise with sequence value.
|
This method only applies to promise with sequence value.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
Calls the given `functor` on each element in the promise value (i.e. `Sequence<T>`), then resolves to the original sequence unmodified. If `functor` throws, `output` is rejected with the new exception.
|
Calls the given `functor` on each element in the promise value (i.e. `Sequence<T>`), then resolves
|
||||||
|
to the original sequence unmodified. If `functor` throws, `output` is rejected with the new
|
||||||
|
exception.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<QList<QByteArray>> input = {...}
|
QPromise<QList<QByteArray>> input = {...}
|
||||||
@ -33,15 +35,17 @@ output.then([](const QList<QByteArray>& res) {
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
If `functor` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises are resolved. If any of the promises fail, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
If `functor` returns a promise (or `QFuture`), the `output` promise is delayed until all the
|
||||||
|
promises are resolved. If any of the promises fail, `output` immediately rejects with the error
|
||||||
|
of the promise that rejected, whether or not the other promises are resolved.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<QList<QUrl>> input = {...}
|
QPromise<QList<QUrl>> input = {...}
|
||||||
|
|
||||||
auto output = input.each([](const QUrl& url, ...) {
|
auto output = input.each([](const QUrl& url, ...) {
|
||||||
return QPromise<void>([&](auto resolve, auto reject) {
|
return QPromise<void>{[&](auto resolve, auto reject) {
|
||||||
// process url asynchronously ...
|
// process url asynchronously ...
|
||||||
})
|
}};
|
||||||
});
|
});
|
||||||
|
|
||||||
// `output` resolves as soon as all promises returned by
|
// `output` resolves as soon as all promises returned by
|
||||||
|
@ -10,16 +10,19 @@ title: .fail
|
|||||||
QPromise<T>::fail(Function onRejected) -> QPromise<T>
|
QPromise<T>::fail(Function onRejected) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
Shorthand to `promise.then(nullptr, onRejected)`, similar to the [`catch` statement](http://en.cppreference.com/w/cpp/language/try_catch):
|
Shorthand to [`promise.then(nullptr, onRejected)`](then.md) for handling errors in promise chains,
|
||||||
|
similar to the native C++ [`catch` statement](http://en.cppreference.com/w/cpp/language/try_catch):
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
promise.fail([](const MyException&) {
|
promise.fail([](const MyException& error) {
|
||||||
// {...}
|
// {...}
|
||||||
}).fail(const QException&) {
|
}).fail([](const QException& error) {
|
||||||
// {...}
|
// {...}
|
||||||
}).fail(const std::exception&) {
|
}).fail([](const std::exception& error) {
|
||||||
// {...}
|
// {...}
|
||||||
}).fail() {
|
}).fail([]() {
|
||||||
// {...} catch-all
|
// {...} catch-all
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
|
See also: [`QPromise::then`](then.md)
|
||||||
|
@ -18,9 +18,14 @@ QPromise<Sequence<T>>::filter(Filter filterer) -> QPromise<Sequence<T>>
|
|||||||
This method only applies to promise with sequence value.
|
This method only applies to promise with sequence value.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
Iterates over all the promise values (i.e. `Sequence<T>`) and [filters the sequence](https://en.wikipedia.org/wiki/Filter_%28higher-order_function%29) to another using the given `filterer` function. If `filterer` returns `true`, a copy of the item is put in the `output` sequence, otherwise, the item will not appear in `output`. If `filterer` throws, `output` is rejected with the new exception.
|
Iterates over all the promise values (i.e. `Sequence<T>`) and [filters the sequence](https://en.wikipedia.org/wiki/Filter_%28higher-order_function%29)
|
||||||
|
to another using the given `filterer` function. If `filterer` returns `true`, a copy of the item
|
||||||
|
is put in the `output` sequence, otherwise, the item will not appear in `output`. If `filterer`
|
||||||
|
throws, `output` is rejected with the new exception.
|
||||||
|
|
||||||
If `filterer` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises are resolved. If any of the promises fail, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
If `filterer` returns a promise (or `QFuture`), the `output` promise is delayed until all the
|
||||||
|
promises are resolved. If any of the promises fail, `output` immediately rejects with the error
|
||||||
|
of the promise that rejected, whether or not the other promises are resolved.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<QList<QUrl>> input = {...}
|
QPromise<QList<QUrl>> input = {...}
|
||||||
@ -28,9 +33,9 @@ QPromise<QList<QUrl>> input = {...}
|
|||||||
auto output = input.filter([](const QUrl& url, ...) {
|
auto output = input.filter([](const QUrl& url, ...) {
|
||||||
return url.isValid(); // Keep only valid URLs
|
return url.isValid(); // Keep only valid URLs
|
||||||
}).filter([](const QUrl& url, ...) {
|
}).filter([](const QUrl& url, ...) {
|
||||||
return QPromise<bool>([&](auto resolve, auto reject) {
|
return QPromise<bool>{[&](auto resolve, auto reject) {
|
||||||
// resolve(true) if `url` is reachable, else resolve(false)
|
// resolve(true) if `url` is reachable, else resolve(false)
|
||||||
});
|
}};
|
||||||
});
|
});
|
||||||
|
|
||||||
// 'output' resolves as soon as all promises returned by
|
// 'output' resolves as soon as all promises returned by
|
||||||
@ -43,7 +48,8 @@ output.then([](const QList<QUrl>& res) {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
The order of the output sequence values is guarantee to be the same as the original sequence, regardless of completion order of the promises returned by `filterer`.
|
The order of the output sequence values is guarantee to be the same as the original sequence,
|
||||||
|
regardless of completion order of the promises returned by `filterer`.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
See also: [`QtPromise::filter`](../helpers/filter.md)
|
See also: [`QtPromise::filter`](../helpers/filter.md)
|
||||||
|
@ -10,7 +10,10 @@ title: .finally
|
|||||||
QPromise<T>::finally(Function handler) -> QPromise<T>
|
QPromise<T>::finally(Function handler) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
This `handler` is **always** called, without any argument and whatever the `input` promise state (fulfilled or rejected). The `output` promise has the same type as the `input` one but also the same value or error. The finally `handler` **can not modify the fulfilled value** (the returned value is ignored), however, if `handler` throws, `output` is rejected with the new exception.
|
This `handler` is **always** called, without any argument and whatever the `input` promise state
|
||||||
|
(fulfilled or rejected). The `output` promise has the same type as the `input` one but also the
|
||||||
|
same value or error. The finally `handler` **can not modify the fulfilled value** (the returned
|
||||||
|
value is ignored), however, if `handler` throws, `output` is rejected with the new exception.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
auto output = input.finally([]() {
|
auto output = input.finally([]() {
|
||||||
@ -18,4 +21,6 @@ auto output = input.finally([]() {
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
If `handler` returns a promise (or QFuture), the `output` promise is delayed until the returned promise is resolved and under the same conditions: the delayed value is ignored, the error transmitted to the `output` promise.
|
If `handler` returns a promise (or QFuture), the `output` promise is delayed until the returned
|
||||||
|
promise is resolved and under the same conditions: the delayed value is ignored, the error
|
||||||
|
transmitted to the `output` promise.
|
||||||
|
@ -11,4 +11,3 @@ QPromise<T>::isRejected() -> bool
|
|||||||
```
|
```
|
||||||
|
|
||||||
Returns `true` if the promise is rejected, otherwise returns `false`.
|
Returns `true` if the promise is rejected, otherwise returns `false`.
|
||||||
|
|
||||||
|
@ -18,22 +18,26 @@ QPromise<Sequence<T>>::map(Mapper mapper) -> QPromise<QVector<R>>
|
|||||||
This method only applies to promise with sequence value.
|
This method only applies to promise with sequence value.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
Iterates over all the promise values (i.e. `Sequence<T>`) and [maps the sequence](https://en.wikipedia.org/wiki/Map_%28higher-order_function%29) to another using the given `mapper` function. The type returned by `mapper` determines the type of the `output` promise. If `mapper` throws, `output` is rejected with the new exception.
|
Iterates over all the promise values (i.e. `Sequence<T>`) and [maps the sequence](https://en.wikipedia.org/wiki/Map_%28higher-order_function%29)
|
||||||
|
to another using the given `mapper` function. The type returned by `mapper` determines the type of
|
||||||
|
the `output` promise. If `mapper` throws, `output` is rejected with the new exception.
|
||||||
|
|
||||||
If `mapper` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises are resolved. If any of the promises fails, `output` immediately rejects with the error of the promise that rejected, whether or not the other promises are resolved.
|
If `mapper` returns a promise (or `QFuture`), the `output` promise is delayed until all the promises
|
||||||
|
are resolved. If any of the promises fails, `output` immediately rejects with the error of the
|
||||||
|
promise that rejected, whether or not the other promises are resolved.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<QList<QUrl>> input = {...}
|
QPromise<QList<QUrl>> input = {...}
|
||||||
|
|
||||||
auto output = input.map([](const QUrl& url, int index) {
|
auto output = input.map([](const QUrl& url, int index) {
|
||||||
return QPromise<QByteArray>([&](auto resolve, auto reject) {
|
return QPromise<QByteArray>{[&](auto resolve, auto reject) {
|
||||||
// download content at 'url' and resolve
|
// download content at 'url' and resolve
|
||||||
// {...}
|
// {...}
|
||||||
});
|
}};
|
||||||
}).map([](const QByteArray& value, ...) {
|
}).map([](const QByteArray& value, ...) {
|
||||||
// process the downloaded QByteArray
|
// process the downloaded QByteArray
|
||||||
// {...}
|
// {...}
|
||||||
return DownloadResult(value);
|
return DownloadResult{value};
|
||||||
});
|
});
|
||||||
|
|
||||||
// 'output' resolves as soon as all promises returned by
|
// 'output' resolves as soon as all promises returned by
|
||||||
@ -46,7 +50,8 @@ output.then([](const QVector<DownloadResult>& res) {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
The order of the output sequence values is guarantee to be the same as the original sequence, regardless of completion order of the promises returned by `mapper`.
|
The order of the output sequence values is guarantee to be the same as the original sequence,
|
||||||
|
regardless of completion order of the promises returned by `mapper`.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
This function is provided for convenience and is similar to:
|
This function is provided for convenience and is similar to:
|
||||||
|
@ -19,9 +19,12 @@ QPromise<Sequence<T>>::reduce(Reducer reducer, R|QPromise<R> initialValue) -> QP
|
|||||||
This method only applies to promise with sequence value.
|
This method only applies to promise with sequence value.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
Iterates over all the promise values (i.e. `Sequence<T>`) and [reduces the sequence to a single value](https://en.wikipedia.org/wiki/Fold_%28higher-order_function%29) using the given `reducer` function and an optional `initialValue`. The type returned by `reducer` determines the type of the `output` promise.
|
Iterates over all the promise values (i.e. `Sequence<T>`) and [reduces the sequence](https://en.wikipedia.org/wiki/Fold_%28higher-order_function%29)
|
||||||
|
to a single value using the given `reducer` function and an optional `initialValue`. The type
|
||||||
|
returned by `reducer` determines the type of the `output` promise.
|
||||||
|
|
||||||
See [`QtPromise::reduce`](../helpers/reduce.md) for details, this method is provided for convenience and is similar to:
|
See [`QtPromise::reduce`](../helpers/reduce.md) for details, this method is provided for convenience
|
||||||
|
and is similar to:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
promise.then([](const T& values) {
|
promise.then([](const T& values) {
|
||||||
@ -41,9 +44,9 @@ auto input = QtPromise::resolve(QList<QUrl>{
|
|||||||
// Concatenate the content of the given files, read asynchronously
|
// Concatenate the content of the given files, read asynchronously
|
||||||
auto output = input.reduce([](const QString& acc, const QString& cur, int idx) {
|
auto output = input.reduce([](const QString& acc, const QString& cur, int idx) {
|
||||||
return readAsync(cur).then([=](const QString& res) {
|
return readAsync(cur).then([=](const QString& res) {
|
||||||
return QString("%1;%2:%3").arg(acc).arg(idx).arg(res);
|
return QString{"%1;%2:%3"}.arg(acc).arg(idx).arg(res);
|
||||||
});
|
});
|
||||||
}, QString("index:text"));
|
}, QString{"index:text"});
|
||||||
|
|
||||||
// 'output' resolves as soon as all promises returned by
|
// 'output' resolves as soon as all promises returned by
|
||||||
// 'reducer' are fulfilled or at least one is rejected.
|
// 'reducer' are fulfilled or at least one is rejected.
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
---
|
---
|
||||||
title: ::reject [static]
|
title: ::reject (static)
|
||||||
---
|
---
|
||||||
|
|
||||||
# QPromise::reject [static]
|
# QPromise::reject (static)
|
||||||
|
|
||||||
*Since: 0.1.0*
|
*Since: 0.1.0*
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
[static] QPromise<T>::reject(any reason) -> QPromise<T>
|
(static) QPromise<T>::reject(any reason) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
Creates a `QPromise<T>` that is rejected with the given `reason` of *whatever type*:
|
Creates a `QPromise<T>` that is rejected with the given `reason` of *whatever type*:
|
||||||
@ -16,11 +16,11 @@ Creates a `QPromise<T>` that is rejected with the given `reason` of *whatever ty
|
|||||||
QPromise<int> compute(const QString& type)
|
QPromise<int> compute(const QString& type)
|
||||||
{
|
{
|
||||||
if (type == "foobar") {
|
if (type == "foobar") {
|
||||||
return QPromise<int>::reject(QString("Unknown type: %1").arg(type));
|
return QPromise<int>::reject(QString{"Unknown type: %1"}.arg(type));
|
||||||
}
|
}
|
||||||
|
|
||||||
return QPromise<int>([](const QPromiseResolve<int>& resolve) {
|
return QPromise<int>{[](const QPromiseResolve<int>& resolve) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
---
|
---
|
||||||
title: ::resolve [static]
|
title: ::resolve (static)
|
||||||
---
|
---
|
||||||
|
|
||||||
# QPromise::resolve [static]
|
# QPromise::resolve (static)
|
||||||
|
|
||||||
*Since: 0.1.0*
|
*Since: 0.1.0*
|
||||||
|
|
||||||
```
|
```cpp
|
||||||
[static] QPromise<T>::resolve(T value) -> QPromise<T>
|
(static) QPromise<T>::resolve(T value) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
Creates a `QPromise<T>` that is fulfilled with the given `value` of type `T`:
|
Creates a `QPromise<T>` that is fulfilled with the given `value` of type `T`:
|
||||||
@ -19,9 +19,9 @@ QPromise<int> compute(const QString& type)
|
|||||||
return QPromise<int>::resolve(42);
|
return QPromise<int>::resolve(42);
|
||||||
}
|
}
|
||||||
|
|
||||||
return QPromise<int>([](const QPromiseResolve<int>& resolve) {
|
return QPromise<int>{[](const QPromiseResolve<int>& resolve) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -10,7 +10,10 @@ title: .tap
|
|||||||
QPromise<T>::tap(Function handler) -> QPromise<T>
|
QPromise<T>::tap(Function handler) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
This `handler` allows to observe the value of the `input` promise, without changing the propagated value. The `output` promise will be resolved with the same value as the `input` promise (the `handler` returned value will be ignored). However, if `handler` throws, `output` is rejected with the new exception. Unlike [`finally`](finally.md), this handler is **not** called for rejections.
|
This `handler` allows to observe the value of the `input` promise, without changing the propagated
|
||||||
|
value. The `output` promise will be resolved with the same value as the `input` promise (the `handler`
|
||||||
|
returned value will be ignored). However, if `handler` throws, `output` is rejected with the new
|
||||||
|
exception. Unlike [`finally`](finally.md), this handler is **not** called for rejections.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> input = {...}
|
QPromise<int> input = {...}
|
||||||
@ -21,4 +24,6 @@ auto output = input.tap([](int res) {
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
If `handler` returns a promise (or QFuture), the `output` promise is delayed until the returned promise is resolved and under the same conditions: the delayed value is ignored, the error transmitted to the `output` promise.
|
If `handler` returns a promise (or QFuture), the `output` promise is delayed until the returned
|
||||||
|
promise is resolved and under the same conditions: the delayed value is ignored, the error
|
||||||
|
transmitted to the `output` promise.
|
||||||
|
@ -10,18 +10,23 @@ title: .tapFail
|
|||||||
QPromise<T>::tapFail(Function handler) -> QPromise<T>
|
QPromise<T>::tapFail(Function handler) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
This `handler` allows to observe errors of the `input` promise without handling them - similar to [`finally`](finally.md) but **only** called on rejections. The `output` promise has the same type as the `input` one but also the same value or error. However, if `handler` throws, `output` is rejected with the new exception.
|
This `handler` allows to observe errors of the `input` promise without handling them - similar to
|
||||||
|
[`finally`](finally.md) but **only** called on rejections. The `output` promise has the same type
|
||||||
|
as the `input` one but also the same value or error. However, if `handler` throws, `output` is
|
||||||
|
rejected with the new exception.
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> input = {...}
|
QPromise<int> input = {...}
|
||||||
auto output = input.tapFail([](Error err) {
|
auto output = input.tapFail([](const Error& error) {
|
||||||
log(err);
|
log(error);
|
||||||
}).then([](int res) {
|
}).then([](int res) {
|
||||||
return process(res);
|
return process(res);
|
||||||
}).fail([](Error err) {
|
}).fail([](const Error& error) {
|
||||||
handle(err);
|
handle(error);
|
||||||
return -1;
|
return -1;
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
If `handler` returns a promise (or QFuture), the `output` promise is delayed until the returned promise is resolved and under the same conditions: the delayed value is ignored, the error transmitted to the `output` promise.
|
If `handler` returns a promise (or QFuture), the `output` promise is delayed until the returned
|
||||||
|
promise is resolved and under the same conditions: the delayed value is ignored, the error
|
||||||
|
transmitted to the `output` promise.
|
||||||
|
@ -24,7 +24,8 @@ auto output = input.then([](int res) {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
`onRejected` handler is optional, in which case `output` will be rejected with the same reason as `input`. Also note that it's recommended to use the [`fail`](fail.md) shorthand to handle errors.
|
`onRejected` handler is optional, in which case `output` will be rejected with the same reason as
|
||||||
|
`input`. Also note that it's recommended to use the [`fail`](fail.md) shorthand to handle errors.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
The type `<R>` of the `output` promise depends on the return type of the `onFulfilled` handler:
|
The type `<R>` of the `output` promise depends on the return type of the `onFulfilled` handler:
|
||||||
@ -42,7 +43,9 @@ output.then([](const QString& res) {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
Only `onFulfilled` can change the promise type, `onRejected` **must** return the same type as `onFulfilled`. That also means if `onFulfilled` is `nullptr`, `onRejected` must return the same type as the `input` promise.
|
Only `onFulfilled` can change the promise type, `onRejected` **must** return the same type as
|
||||||
|
`onFulfilled`. That also means if `onFulfilled` is `nullptr`, `onRejected` must return the same
|
||||||
|
type as the `input` promise.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
@ -92,4 +95,5 @@ auto output = input.then([](int res) {
|
|||||||
// output.isRejected() is true
|
// output.isRejected() is true
|
||||||
```
|
```
|
||||||
|
|
||||||
If an handler returns a promise (or QFuture), the `output` promise is delayed and will be resolved by the returned promise.
|
If a handler returns a promise (or QFuture), the `output` promise is delayed and will be resolved
|
||||||
|
by the returned promise.
|
||||||
|
@ -10,7 +10,10 @@ title: .timeout
|
|||||||
QPromise<T>::timeout(int msec, any error = QPromiseTimeoutException) -> QPromise<T>
|
QPromise<T>::timeout(int msec, any error = QPromiseTimeoutException) -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
This method returns a promise that will be resolved with the `input` promise's fulfillment value or rejection reason. However, if the `input` promise is not fulfilled or rejected within `msec` milliseconds, the `output` promise is rejected with `error` as the reason ([`QPromiseTimeoutException`](../exceptions/timeout.md) by default).
|
This method returns a promise that will be resolved with the `input` promise's fulfillment value
|
||||||
|
or rejection reason. However, if the `input` promise is not fulfilled or rejected within `msec`
|
||||||
|
milliseconds, the `output` promise is rejected with `error` as the reason ([`QPromiseTimeoutException`](../exceptions/timeout.md)
|
||||||
|
by default).
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> input = {...}
|
QPromise<int> input = {...}
|
||||||
@ -18,7 +21,43 @@ auto output = input.timeout(2000)
|
|||||||
.then([](int res) {
|
.then([](int res) {
|
||||||
// operation succeeded within 2 seconds
|
// operation succeeded within 2 seconds
|
||||||
})
|
})
|
||||||
.fail([](const QPromiseTimeoutException& e) {
|
.fail([](const QPromiseTimeoutException& error) {
|
||||||
|
// operation timed out!
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
*Since: 0.6.0*
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<T>::timeout(std::chrono::milliseconds msec, any error = QPromiseTimeoutException) -> QPromise<T>
|
||||||
|
```
|
||||||
|
|
||||||
|
This is a convenience overload accepting [durations from the C++ Standard Library](https://en.cppreference.com/w/cpp/chrono/duration).
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
QPromise<int> input = {...}
|
||||||
|
auto output = input.timeout(std::chrono::seconds{2})
|
||||||
|
.then([](int res) {
|
||||||
|
// operation succeeded within 2 seconds
|
||||||
|
})
|
||||||
|
.fail([](const QPromiseTimeoutException& error) {
|
||||||
|
// operation timed out!
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
C++14 alternative:
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
using namespace std::chrono_literals;
|
||||||
|
|
||||||
|
QPromise<int> input = {...}
|
||||||
|
auto output = input.timeout(2s)
|
||||||
|
.then([](int res) {
|
||||||
|
// operation succeeded within 2 seconds
|
||||||
|
})
|
||||||
|
.fail([](const QPromiseTimeoutException& error) {
|
||||||
// operation timed out!
|
// operation timed out!
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
@ -10,7 +10,8 @@ title: .wait
|
|||||||
QPromise<T>::wait() -> QPromise<T>
|
QPromise<T>::wait() -> QPromise<T>
|
||||||
```
|
```
|
||||||
|
|
||||||
This method holds the execution of the remaining code until the `input` promise is resolved (either fulfilled or rejected), **without** blocking the event loop of the current thread:
|
This method holds the execution of the remaining code until the `input` promise is resolved (either
|
||||||
|
fulfilled or rejected), **without** blocking the event loop of the current thread:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
int result = -1;
|
int result = -1;
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
# Qt Concurrent
|
# Qt Concurrent
|
||||||
|
|
||||||
QtPromise integrates with [QtConcurrent](https://doc.qt.io/qt-5/qtconcurrent-index.html) to make easy chaining QFuture with QPromise.
|
QtPromise integrates with [QtConcurrent](https://doc.qt.io/qt-5/qtconcurrent-index.html) to simplify
|
||||||
|
chaining QFuture with QPromise.
|
||||||
|
|
||||||
## <a name="qtconcurrent-convert"></a> Convert
|
## Convert
|
||||||
|
|
||||||
Converting `QFuture<T>` to `QPromise<T>` is done using the [`QtPromise::resolve`](helpers/resolve.md) helper:
|
Converting `QFuture<T>` to `QPromise<T>` is done using the [`QtPromise::resolve`](helpers/resolve.md)
|
||||||
|
helper:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QFuture<int> future = QtConcurrent::run([]() {
|
QFuture<int> future = QtConcurrent::run([]() {
|
||||||
@ -25,14 +27,15 @@ auto promise = QtPromise::resolve(QtConcurrent::run([]() {
|
|||||||
|
|
||||||
## Chain
|
## Chain
|
||||||
|
|
||||||
Returning a `QFuture<T>` in [`then`](qpromise/then.md) or [`fail`](qpromise/fail.md) automatically translate to `QPromise<T>`:
|
Returning a `QFuture<T>` in [`then`](qpromise/then.md) or [`fail`](qpromise/fail.md) automatically
|
||||||
|
translate to `QPromise<T>`:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
QPromise<int> input = ...
|
QPromise<int> input = ...
|
||||||
auto output = input.then([](int res) {
|
auto output = input.then([](int res) {
|
||||||
return QtConcurrent::run([]() {
|
return QtConcurrent::run([]() {
|
||||||
// {...}
|
// {...}
|
||||||
return QString("42");
|
return QString{"42"};
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -46,21 +49,40 @@ The `output` promise is resolved when the `QFuture` is [finished](https://doc.qt
|
|||||||
|
|
||||||
## Error
|
## Error
|
||||||
|
|
||||||
Exceptions thrown from a QtConcurrent thread reject the associated promise with the exception as the reason. Note that if you throw an exception that is not a subclass of `QException`, the promise will be rejected with [`QUnhandledException`](https://doc.qt.io/qt-5/qunhandledexception.html#details) (this restriction only applies to exceptions thrown from a QtConcurrent thread, [read more](https://doc.qt.io/qt-5/qexception.html#details)).
|
Exceptions thrown from a QtConcurrent thread reject the associated promise with the exception as the
|
||||||
|
reason. For this to work, the exception should be a [`QException`](https://doc.qt.io/qt-5/qexception.html)
|
||||||
|
or its subclass. Correct subclassing of [`QException`](https://doc.qt.io/qt-5/qexception.html)
|
||||||
|
includes overriding its methods [`clone()`](https://doc.qt.io/qt-5/qexception.html#clone) and
|
||||||
|
[`raise()`](https://doc.qt.io/qt-5/qexception.html#raise). Without these overrides the promise will
|
||||||
|
be rejected with [`QException`](https://doc.qt.io/qt-5/qexception.html).
|
||||||
|
|
||||||
|
Note that if you throw an exception that is not a subclass of `QException`, the promise will
|
||||||
|
be rejected with [`QUnhandledException`](https://doc.qt.io/qt-5/qunhandledexception.html#details)
|
||||||
|
(this restriction only applies to exceptions thrown from a QtConcurrent thread,
|
||||||
|
[read more](https://doc.qt.io/qt-5/qexception.html#details)).
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
|
class CustomException : public QException
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void raise() const override { throw *this; }
|
||||||
|
CustomException* clone() const override { return new CustomException{*this}; }
|
||||||
|
};
|
||||||
|
|
||||||
|
// {...}
|
||||||
|
|
||||||
QPromise<int> promise = ...
|
QPromise<int> promise = ...
|
||||||
promise.then([](int res) {
|
promise.then([](int res) {
|
||||||
return QtConcurrent::run([]() {
|
return QtConcurrent::run([]() {
|
||||||
// {...}
|
// {...}
|
||||||
|
|
||||||
if (!success) {
|
if (!success) {
|
||||||
throw CustomException();
|
throw CustomException{};
|
||||||
}
|
}
|
||||||
|
|
||||||
return QString("42");
|
return QString{"42"};
|
||||||
});
|
});
|
||||||
}).fail(const CustomException& err) {
|
}).fail([](const CustomException& error) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
@ -3,12 +3,15 @@
|
|||||||
QtPromise supports creating promises that are resolved or rejected by regular [Qt signals](https://doc.qt.io/qt-5/signalsandslots.html).
|
QtPromise supports creating promises that are resolved or rejected by regular [Qt signals](https://doc.qt.io/qt-5/signalsandslots.html).
|
||||||
|
|
||||||
::: warning IMPORTANT
|
::: warning IMPORTANT
|
||||||
A promise connected to a signal will be resolved (fulfilled or rejected) **only one time**, no matter if the signals are emitted multiple times. Internally, the promise is disconnected from all signals as soon as one signal is emitted.
|
A promise connected to a signal will be resolved (fulfilled or rejected) **only one time**, no
|
||||||
|
matter if the signals are emitted multiple times. Internally, the promise is disconnected from
|
||||||
|
all signals as soon as one signal is emitted.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
## Resolve Signal
|
## Resolve Signal
|
||||||
|
|
||||||
The [`QtPromise::connect()`](helpers/connect.md) helper allows to create a promise resolved from a single signal:
|
The [`QtPromise::connect()`](helpers/connect.md) helper allows to create a promise resolved from
|
||||||
|
a single signal:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
// [signal] Object::finished(const QByteArray&)
|
// [signal] Object::finished(const QByteArray&)
|
||||||
@ -33,12 +36,14 @@ output.then([]() {
|
|||||||
```
|
```
|
||||||
|
|
||||||
::: tip NOTE
|
::: tip NOTE
|
||||||
QtPromise currently only supports single argument signals, which means that only the first argument is used to fulfill or reject the connected promise, other arguments being ignored.
|
QtPromise currently only supports single argument signals, which means that only the first argument
|
||||||
|
is used to fulfill or reject the connected promise, other arguments being ignored.
|
||||||
:::
|
:::
|
||||||
|
|
||||||
## Reject Signal
|
## Reject Signal
|
||||||
|
|
||||||
The [`QtPromise::connect()`](helpers/connect.md) helper also allows to reject the promise from another signal:
|
The [`QtPromise::connect()`](helpers/connect.md) helper also allows to reject the promise from
|
||||||
|
another signal:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
// [signal] Object::finished(const QByteArray& data)
|
// [signal] Object::finished(const QByteArray& data)
|
||||||
@ -48,13 +53,13 @@ auto output = QtPromise::connect(obj, &Object::finished, &Object::error);
|
|||||||
// output type: QPromise<QByteArray>
|
// output type: QPromise<QByteArray>
|
||||||
output.then([](const QByteArray& data) {
|
output.then([](const QByteArray& data) {
|
||||||
// {...}
|
// {...}
|
||||||
}).fail(const ObjectError& error) {
|
}).fail([](const ObjectError& error) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
If the rejection signal doesn't provide any argument, the promise will be rejected
|
If the rejection signal doesn't provide any argument, the promise will be rejected with
|
||||||
with [`QPromiseUndefinedException`](../exceptions/undefined), for example:
|
[`QPromiseUndefinedException`](exceptions/undefined.md), for example:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
// [signal] Object::finished()
|
// [signal] Object::finished()
|
||||||
@ -64,7 +69,7 @@ auto output = QtPromise::connect(obj, &Object::finished, &Object::error);
|
|||||||
// output type: QPromise<QByteArray>
|
// output type: QPromise<QByteArray>
|
||||||
output.then([]() {
|
output.then([]() {
|
||||||
// {...}
|
// {...}
|
||||||
}).fail(const QPromiseUndefinedException& error) {
|
}).fail([](const QPromiseUndefinedException& error) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
@ -79,11 +84,13 @@ auto output = QtPromise::connect(objA, &ObjectA::finished, objB, &ObjectB::error
|
|||||||
// output type: QPromise<QByteArray>
|
// output type: QPromise<QByteArray>
|
||||||
output.then([](const QByteArray& data) {
|
output.then([](const QByteArray& data) {
|
||||||
// {...}
|
// {...}
|
||||||
}).fail(const ObjectBError& error) {
|
}).fail([](const ObjectBError& error) {
|
||||||
// {...}
|
// {...}
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
Additionally to the rejection signal, promises created using [`QtPromise::connect()`](helpers/connect.md) are automatically rejected with [`QPromiseContextException`](exceptions/context.md) if the sender is destroyed before fulfilling the promise.
|
Additionally to the rejection signal, promises created using [`QtPromise::connect()`](helpers/connect.md)
|
||||||
|
are automatically rejected with [`QPromiseContextException`](exceptions/context.md) if the sender is
|
||||||
|
destroyed before fulfilling the promise.
|
||||||
|
|
||||||
See [`QtPromise::connect()`](helpers/connect.md) for more details.
|
See [`QtPromise::connect()`](helpers/connect.md) for more details.
|
||||||
|
@ -1,7 +1,12 @@
|
|||||||
# Thread-Safety
|
# Thread-Safety
|
||||||
|
|
||||||
QPromise is thread-safe and can be copied and accessed across different threads. QPromise relies on [explicitly data sharing](https://doc.qt.io/qt-5/qexplicitlyshareddatapointer.html#details) and thus `auto p2 = p1` represents the same promise: when `p1` resolves, handlers registered on `p1` and `p2` are called, the fulfilled value being shared between both instances.
|
QPromise is thread-safe and can be copied and accessed across different threads. QPromise relies on
|
||||||
|
[explicitly data sharing](https://doc.qt.io/qt-5/qexplicitlyshareddatapointer.html#details) and thus
|
||||||
|
`auto p2 = p1` represents the same promise: when `p1` resolves, handlers registered on `p1` and `p2`
|
||||||
|
are called, the fulfilled value being shared between both instances.
|
||||||
|
|
||||||
::: warning IMPORTANT
|
::: warning IMPORTANT
|
||||||
While it's safe to access the resolved value from different threads using [`then`](qpromise/then.md), QPromise provides no guarantee about the object being pointed to. Thread-safety and reentrancy rules for that object still apply.
|
While it's safe to access the resolved value from different threads using [`then`](qpromise/then.md),
|
||||||
|
QPromise provides no guarantee about the object being pointed to. Thread-safety and reentrancy rules
|
||||||
|
for that object still apply.
|
||||||
:::
|
:::
|
||||||
|
@ -1,3 +1,10 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_MODULE_H
|
#ifndef QTPROMISE_MODULE_H
|
||||||
#define QTPROMISE_MODULE_H
|
#define QTPROMISE_MODULE_H
|
||||||
|
|
||||||
|
34975
package-lock.json
generated
Normal file
34975
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
18
package.json
Normal file
18
package.json
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
{
|
||||||
|
"description": "qtpromise",
|
||||||
|
"private": true,
|
||||||
|
"scripts": {
|
||||||
|
"docs:build": "vuepress build docs",
|
||||||
|
"docs:dev": "vuepress dev docs",
|
||||||
|
"docs:lint": "remark --quiet --frail ."
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@vuepress/plugin-google-analytics": "^1.9.9",
|
||||||
|
"remark-cli": "^11.0.0",
|
||||||
|
"remark-frontmatter": "^4.0.1",
|
||||||
|
"remark-preset-lint-markdown-style-guide": "^5.1.2",
|
||||||
|
"remark-preset-lint-recommended": "^6.1.2",
|
||||||
|
"remark-validate-links": "^12.1.0",
|
||||||
|
"vuepress": "^1.9.9"
|
||||||
|
}
|
||||||
|
}
|
@ -1 +0,0 @@
|
|||||||
include(../../qtpromise.pri)
|
|
2
qpm.json
2
qpm.json
@ -10,7 +10,7 @@
|
|||||||
"url": "https://github.com/simonbrunel/qtpromise.git"
|
"url": "https://github.com/simonbrunel/qtpromise.git"
|
||||||
},
|
},
|
||||||
"version": {
|
"version": {
|
||||||
"label": "0.5.0"
|
"label": "0.7.0"
|
||||||
},
|
},
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"pri_filename": "qtpromise.pri",
|
"pri_filename": "qtpromise.pri",
|
||||||
|
@ -1,12 +0,0 @@
|
|||||||
TEMPLATE = subdirs
|
|
||||||
SUBDIRS = \
|
|
||||||
tests
|
|
||||||
|
|
||||||
_qt_creator_ {
|
|
||||||
SUBDIRS += src
|
|
||||||
}
|
|
||||||
|
|
||||||
OTHER_FILES = \
|
|
||||||
package/features/*.prf \
|
|
||||||
include/* \
|
|
||||||
qtpromise.pri
|
|
@ -1,37 +1,54 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISE_H
|
#ifndef QTPROMISE_QPROMISE_H
|
||||||
#define QTPROMISE_QPROMISE_H
|
#define QTPROMISE_QPROMISE_H
|
||||||
|
|
||||||
#include "qpromise_p.h"
|
|
||||||
#include "qpromiseexceptions.h"
|
#include "qpromiseexceptions.h"
|
||||||
|
#include "qpromise_p.h"
|
||||||
#include "qpromiseglobal.h"
|
#include "qpromiseglobal.h"
|
||||||
#include "qpromiseresolver.h"
|
#include "qpromiseresolver.h"
|
||||||
|
|
||||||
// Qt
|
#include <QtCore/QExplicitlySharedDataPointer>
|
||||||
#include <QExplicitlySharedDataPointer>
|
|
||||||
|
#include <chrono>
|
||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class QPromiseBase
|
class QPromiseBase
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
using Type = T;
|
using Type = T;
|
||||||
|
|
||||||
template <typename F, typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count == 1, int>::type = 0>
|
template<typename F,
|
||||||
|
typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count == 1, int>::type = 0>
|
||||||
inline QPromiseBase(F resolver);
|
inline QPromiseBase(F resolver);
|
||||||
|
|
||||||
template <typename F, typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count != 1, int>::type = 0>
|
template<typename F,
|
||||||
|
typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count != 1, int>::type = 0>
|
||||||
inline QPromiseBase(F resolver);
|
inline QPromiseBase(F resolver);
|
||||||
|
|
||||||
QPromiseBase(const QPromiseBase<T>& other): m_d(other.m_d) {}
|
QPromiseBase(const QPromiseBase<T>& other) : m_d{other.m_d} { }
|
||||||
QPromiseBase(const QPromise<T>& other): m_d(other.m_d) {}
|
QPromiseBase(const QPromise<T>& other) : m_d{other.m_d} { }
|
||||||
QPromiseBase(QPromiseBase<T>&& other) Q_DECL_NOEXCEPT { swap(other); }
|
QPromiseBase(QPromiseBase<T>&& other) Q_DECL_NOEXCEPT { swap(other); }
|
||||||
|
|
||||||
virtual ~QPromiseBase() { }
|
virtual ~QPromiseBase() { }
|
||||||
|
|
||||||
QPromiseBase<T>& operator=(const QPromiseBase<T>& other) { m_d = other.m_d; return *this;}
|
QPromiseBase<T>& operator=(const QPromiseBase<T>& other)
|
||||||
|
{
|
||||||
|
m_d = other.m_d;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
QPromiseBase<T>& operator=(QPromiseBase<T>&& other) Q_DECL_NOEXCEPT
|
QPromiseBase<T>& operator=(QPromiseBase<T>&& other) Q_DECL_NOEXCEPT
|
||||||
{ QPromiseBase<T>(std::move(other)).swap(*this); return *this; }
|
{
|
||||||
|
QPromiseBase<T>(std::move(other)).swap(*this);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
bool operator==(const QPromiseBase<T>& other) const { return (m_d == other.m_d); }
|
bool operator==(const QPromiseBase<T>& other) const { return (m_d == other.m_d); }
|
||||||
bool operator!=(const QPromiseBase<T>& other) const { return (m_d != other.m_d); }
|
bool operator!=(const QPromiseBase<T>& other) const { return (m_d != other.m_d); }
|
||||||
@ -42,35 +59,40 @@ public:
|
|||||||
bool isRejected() const { return m_d->isRejected(); }
|
bool isRejected() const { return m_d->isRejected(); }
|
||||||
bool isPending() const { return m_d->isPending(); }
|
bool isPending() const { return m_d->isPending(); }
|
||||||
|
|
||||||
template <typename TFulfilled, typename TRejected>
|
template<typename TFulfilled, typename TRejected>
|
||||||
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
||||||
then(const TFulfilled& fulfilled, const TRejected& rejected) const;
|
then(const TFulfilled& fulfilled, const TRejected& rejected) const;
|
||||||
|
|
||||||
template <typename TFulfilled>
|
template<typename TFulfilled>
|
||||||
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
||||||
then(TFulfilled&& fulfilled) const;
|
then(TFulfilled&& fulfilled) const;
|
||||||
|
|
||||||
template <typename TRejected>
|
template<typename TRejected>
|
||||||
inline typename QtPromisePrivate::PromiseHandler<T, std::nullptr_t>::Promise
|
inline typename QtPromisePrivate::PromiseHandler<T, std::nullptr_t>::Promise
|
||||||
fail(TRejected&& rejected) const;
|
fail(TRejected&& rejected) const;
|
||||||
|
|
||||||
template <typename THandler>
|
template<typename THandler>
|
||||||
inline QPromise<T> finally(THandler handler) const;
|
inline QPromise<T> finally(THandler handler) const;
|
||||||
|
|
||||||
template <typename THandler>
|
template<typename THandler>
|
||||||
inline QPromise<T> tap(THandler handler) const;
|
inline QPromise<T> tap(THandler handler) const;
|
||||||
|
|
||||||
template <typename THandler>
|
template<typename THandler>
|
||||||
inline QPromise<T> tapFail(THandler handler) const;
|
inline QPromise<T> tapFail(THandler handler) const;
|
||||||
|
|
||||||
template <typename E = QPromiseTimeoutException>
|
template<typename E = QPromiseTimeoutException>
|
||||||
inline QPromise<T> timeout(int msec, E&& error = E()) const;
|
inline QPromise<T> timeout(int msec, E&& error = E{}) const;
|
||||||
|
|
||||||
|
template<typename E = QPromiseTimeoutException>
|
||||||
|
inline QPromise<T> timeout(std::chrono::milliseconds msec, E&& error = E{}) const;
|
||||||
|
|
||||||
inline QPromise<T> delay(int msec) const;
|
inline QPromise<T> delay(int msec) const;
|
||||||
|
inline QPromise<T> delay(std::chrono::milliseconds msec) const;
|
||||||
|
|
||||||
inline QPromise<T> wait() const;
|
inline QPromise<T> wait() const;
|
||||||
|
|
||||||
public: // STATIC
|
public: // STATIC
|
||||||
template <typename E>
|
template<typename E>
|
||||||
inline static QPromise<T> reject(E&& error);
|
inline static QPromise<T> reject(E&& error);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@ -81,39 +103,38 @@ protected:
|
|||||||
QExplicitlySharedDataPointer<QtPromisePrivate::PromiseData<T>> m_d;
|
QExplicitlySharedDataPointer<QtPromisePrivate::PromiseData<T>> m_d;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class QPromise : public QPromiseBase<T>
|
class QPromise : public QPromiseBase<T>
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
template <typename F>
|
template<typename F>
|
||||||
QPromise(F&& resolver): QPromiseBase<T>(std::forward<F>(resolver)) { }
|
QPromise(F&& resolver) : QPromiseBase<T>(std::forward<F>(resolver))
|
||||||
|
{ }
|
||||||
|
|
||||||
template <typename Functor>
|
template<typename U>
|
||||||
inline QPromise<T>
|
inline QPromise<U> convert() const;
|
||||||
each(Functor fn);
|
|
||||||
|
|
||||||
template <typename Functor>
|
template<typename Functor>
|
||||||
inline QPromise<T>
|
inline QPromise<T> each(Functor fn);
|
||||||
filter(Functor fn);
|
|
||||||
|
|
||||||
template <typename Functor>
|
template<typename Functor>
|
||||||
inline typename QtPromisePrivate::PromiseMapper<T, Functor>::PromiseType
|
inline QPromise<T> filter(Functor fn);
|
||||||
map(Functor fn);
|
|
||||||
|
|
||||||
template <typename Functor, typename Input>
|
template<typename Functor>
|
||||||
inline typename QtPromisePrivate::PromiseDeduce<Input>::Type
|
inline typename QtPromisePrivate::PromiseMapper<T, Functor>::PromiseType map(Functor fn);
|
||||||
reduce(Functor fn, Input initial);
|
|
||||||
|
|
||||||
template <typename Functor, typename U = T>
|
template<typename Functor, typename Input>
|
||||||
|
inline typename QtPromisePrivate::PromiseDeduce<Input>::Type reduce(Functor fn, Input initial);
|
||||||
|
|
||||||
|
template<typename Functor, typename U = T>
|
||||||
inline typename QtPromisePrivate::PromiseDeduce<typename U::value_type>::Type
|
inline typename QtPromisePrivate::PromiseDeduce<typename U::value_type>::Type
|
||||||
reduce(Functor fn);
|
reduce(Functor fn);
|
||||||
|
|
||||||
public: // STATIC
|
public: // STATIC
|
||||||
|
|
||||||
// DEPRECATED (remove at version 1)
|
// DEPRECATED (remove at version 1)
|
||||||
template <template <typename, typename...> class Sequence = QVector, typename ...Args>
|
template<template<typename, typename...> class Sequence = QVector, typename... Args>
|
||||||
Q_DECL_DEPRECATED_X("Use QtPromise::all instead") static inline QPromise<QVector<T>>
|
Q_DECL_DEPRECATED_X("Use QtPromise::all instead")
|
||||||
all(const Sequence<QPromise<T>, Args...>& promises);
|
static inline QPromise<QVector<T>> all(const Sequence<QPromise<T>, Args...>& promises);
|
||||||
|
|
||||||
inline static QPromise<T> resolve(const T& value);
|
inline static QPromise<T> resolve(const T& value);
|
||||||
inline static QPromise<T> resolve(T&& value);
|
inline static QPromise<T> resolve(T&& value);
|
||||||
@ -122,19 +143,19 @@ private:
|
|||||||
friend class QPromiseBase<T>;
|
friend class QPromiseBase<T>;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
class QPromise<void> : public QPromiseBase<void>
|
class QPromise<void> : public QPromiseBase<void>
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
template <typename F>
|
template<typename F>
|
||||||
QPromise(F&& resolver): QPromiseBase<void>(std::forward<F>(resolver)) { }
|
QPromise(F&& resolver) : QPromiseBase<void>(std::forward<F>(resolver))
|
||||||
|
{ }
|
||||||
|
|
||||||
public: // STATIC
|
public: // STATIC
|
||||||
|
|
||||||
// DEPRECATED (remove at version 1)
|
// DEPRECATED (remove at version 1)
|
||||||
template <template <typename, typename...> class Sequence = QVector, typename ...Args>
|
template<template<typename, typename...> class Sequence = QVector, typename... Args>
|
||||||
Q_DECL_DEPRECATED_X("Use QtPromise::all instead") static inline QPromise<void>
|
Q_DECL_DEPRECATED_X("Use QtPromise::all instead")
|
||||||
all(const Sequence<QPromise<void>, Args...>& promises);
|
static inline QPromise<void> all(const Sequence<QPromise<void>, Args...>& promises);
|
||||||
|
|
||||||
inline static QPromise<void> resolve();
|
inline static QPromise<void> resolve();
|
||||||
|
|
||||||
|
@ -1,19 +1,24 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#include "qpromise.h"
|
#include "qpromise.h"
|
||||||
#include "qpromisehelpers.h"
|
#include "qpromisehelpers.h"
|
||||||
|
|
||||||
// Qt
|
#include <QtCore/QCoreApplication>
|
||||||
#include <QCoreApplication>
|
#include <QtCore/QSharedPointer>
|
||||||
#include <QSharedPointer>
|
#include <QtCore/QTimer>
|
||||||
#include <QTimer>
|
|
||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename F, typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count == 1, int>::type>
|
template<typename F, typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count == 1, int>::type>
|
||||||
inline QPromiseBase<T>::QPromiseBase(F callback)
|
inline QPromiseBase<T>::QPromiseBase(F callback) : m_d{new QtPromisePrivate::PromiseData<T>{}}
|
||||||
: m_d(new QtPromisePrivate::PromiseData<T>())
|
|
||||||
{
|
{
|
||||||
QtPromisePrivate::PromiseResolver<T> resolver(*this);
|
QtPromisePrivate::PromiseResolver<T> resolver{*this};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
callback(QPromiseResolve<T>(resolver));
|
callback(QPromiseResolve<T>(resolver));
|
||||||
@ -22,12 +27,18 @@ inline QPromiseBase<T>::QPromiseBase(F callback)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename F, typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count != 1, int>::type>
|
template<typename F, typename std::enable_if<QtPromisePrivate::ArgsOf<F>::count != 1, int>::type>
|
||||||
inline QPromiseBase<T>::QPromiseBase(F callback)
|
inline QPromiseBase<T>::QPromiseBase(F callback) : m_d{new QtPromisePrivate::PromiseData<T>{}}
|
||||||
: m_d(new QtPromisePrivate::PromiseData<T>())
|
|
||||||
{
|
{
|
||||||
QtPromisePrivate::PromiseResolver<T> resolver(*this);
|
// To prevent infinite recursion at runtime when resolving the QPromise template
|
||||||
|
// constructor, we don't explicitly check for ArgsOf<F>::count == 2 so that this
|
||||||
|
// method is called for ALL callbacks other than the ones with a single typed
|
||||||
|
// argument. This includes valid callbacks such as with two args, variadic or
|
||||||
|
// auto args (c++14) but also invalid callbacks which are not functions or with
|
||||||
|
// 0 or more than 2 arguments, in which case this method MUST fail to compile.
|
||||||
|
|
||||||
|
QtPromisePrivate::PromiseResolver<T> resolver{*this};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
callback(QPromiseResolve<T>(resolver), QPromiseReject<T>(resolver));
|
callback(QPromiseResolve<T>(resolver), QPromiseReject<T>(resolver));
|
||||||
@ -36,17 +47,16 @@ inline QPromiseBase<T>::QPromiseBase(F callback)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename TFulfilled, typename TRejected>
|
template<typename TFulfilled, typename TRejected>
|
||||||
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
||||||
QPromiseBase<T>::then(const TFulfilled& fulfilled, const TRejected& rejected) const
|
QPromiseBase<T>::then(const TFulfilled& fulfilled, const TRejected& rejected) const
|
||||||
{
|
{
|
||||||
using namespace QtPromisePrivate;
|
using namespace QtPromisePrivate;
|
||||||
using PromiseType = typename PromiseHandler<T, TFulfilled>::Promise;
|
using PromiseType = typename PromiseHandler<T, TFulfilled>::Promise;
|
||||||
|
|
||||||
PromiseType next([&](
|
PromiseType next([&](const QPromiseResolve<typename PromiseType::Type>& resolve,
|
||||||
const QPromiseResolve<typename PromiseType::Type>& resolve,
|
const QPromiseReject<typename PromiseType::Type>& reject) {
|
||||||
const QPromiseReject<typename PromiseType::Type>& reject) {
|
|
||||||
m_d->addHandler(PromiseHandler<T, TFulfilled>::create(fulfilled, resolve, reject));
|
m_d->addHandler(PromiseHandler<T, TFulfilled>::create(fulfilled, resolve, reject));
|
||||||
m_d->addCatcher(PromiseCatcher<T, TRejected>::create(rejected, resolve, reject));
|
m_d->addCatcher(PromiseCatcher<T, TRejected>::create(rejected, resolve, reject));
|
||||||
});
|
});
|
||||||
@ -58,24 +68,24 @@ QPromiseBase<T>::then(const TFulfilled& fulfilled, const TRejected& rejected) co
|
|||||||
return next;
|
return next;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename TFulfilled>
|
template<typename TFulfilled>
|
||||||
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
inline typename QtPromisePrivate::PromiseHandler<T, TFulfilled>::Promise
|
||||||
QPromiseBase<T>::then(TFulfilled&& fulfilled) const
|
QPromiseBase<T>::then(TFulfilled&& fulfilled) const
|
||||||
{
|
{
|
||||||
return then(std::forward<TFulfilled>(fulfilled), nullptr);
|
return then(std::forward<TFulfilled>(fulfilled), nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename TRejected>
|
template<typename TRejected>
|
||||||
inline typename QtPromisePrivate::PromiseHandler<T, std::nullptr_t>::Promise
|
inline typename QtPromisePrivate::PromiseHandler<T, std::nullptr_t>::Promise
|
||||||
QPromiseBase<T>::fail(TRejected&& rejected) const
|
QPromiseBase<T>::fail(TRejected&& rejected) const
|
||||||
{
|
{
|
||||||
return then(nullptr, std::forward<TRejected>(rejected));
|
return then(nullptr, std::forward<TRejected>(rejected));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename THandler>
|
template<typename THandler>
|
||||||
inline QPromise<T> QPromiseBase<T>::finally(THandler handler) const
|
inline QPromise<T> QPromiseBase<T>::finally(THandler handler) const
|
||||||
{
|
{
|
||||||
QPromise<T> p = *this;
|
QPromise<T> p = *this;
|
||||||
@ -84,8 +94,8 @@ inline QPromise<T> QPromiseBase<T>::finally(THandler handler) const
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename THandler>
|
template<typename THandler>
|
||||||
inline QPromise<T> QPromiseBase<T>::tap(THandler handler) const
|
inline QPromise<T> QPromiseBase<T>::tap(THandler handler) const
|
||||||
{
|
{
|
||||||
QPromise<T> p = *this;
|
QPromise<T> p = *this;
|
||||||
@ -94,25 +104,22 @@ inline QPromise<T> QPromiseBase<T>::tap(THandler handler) const
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename THandler>
|
template<typename THandler>
|
||||||
inline QPromise<T> QPromiseBase<T>::tapFail(THandler handler) const
|
inline QPromise<T> QPromiseBase<T>::tapFail(THandler handler) const
|
||||||
{
|
{
|
||||||
QPromise<T> p = *this;
|
QPromise<T> p = *this;
|
||||||
return p.then([](){}, handler).then([=]() {
|
return p.then([]() {}, handler).then([=]() {
|
||||||
return p;
|
return p;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename E>
|
template<typename E>
|
||||||
inline QPromise<T> QPromiseBase<T>::timeout(int msec, E&& error) const
|
inline QPromise<T> QPromiseBase<T>::timeout(int msec, E&& error) const
|
||||||
{
|
{
|
||||||
QPromise<T> p = *this;
|
QPromise<T> p = *this;
|
||||||
return QPromise<T>([&](
|
return QPromise<T>{[&](const QPromiseResolve<T>& resolve, const QPromiseReject<T>& reject) {
|
||||||
const QPromiseResolve<T>& resolve,
|
|
||||||
const QPromiseReject<T>& reject) {
|
|
||||||
|
|
||||||
QTimer::singleShot(msec, [=]() {
|
QTimer::singleShot(msec, [=]() {
|
||||||
// we don't need to verify the current promise state, reject()
|
// we don't need to verify the current promise state, reject()
|
||||||
// takes care of checking if the promise is already resolved,
|
// takes care of checking if the promise is already resolved,
|
||||||
@ -121,20 +128,33 @@ inline QPromise<T> QPromiseBase<T>::timeout(int msec, E&& error) const
|
|||||||
});
|
});
|
||||||
|
|
||||||
QtPromisePrivate::PromiseFulfill<QPromise<T>>::call(p, resolve, reject);
|
QtPromisePrivate::PromiseFulfill<QPromise<T>>::call(p, resolve, reject);
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
|
template<typename E>
|
||||||
|
inline QPromise<T> QPromiseBase<T>::timeout(std::chrono::milliseconds msec, E&& error) const
|
||||||
|
{
|
||||||
|
return timeout(static_cast<int>(msec.count()), std::forward<E>(error));
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
inline QPromise<T> QPromiseBase<T>::delay(int msec) const
|
inline QPromise<T> QPromiseBase<T>::delay(int msec) const
|
||||||
{
|
{
|
||||||
return tap([=]() {
|
return tap([=]() {
|
||||||
return QPromise<void>([&](const QPromiseResolve<void>& resolve) {
|
return QPromise<void>{[&](const QPromiseResolve<void>& resolve) {
|
||||||
QTimer::singleShot(msec, resolve);
|
QTimer::singleShot(msec, resolve);
|
||||||
});
|
}};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
|
inline QPromise<T> QPromiseBase<T>::delay(std::chrono::milliseconds msec) const
|
||||||
|
{
|
||||||
|
return delay(static_cast<int>(msec.count()));
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
inline QPromise<T> QPromiseBase<T>::wait() const
|
inline QPromise<T> QPromiseBase<T>::wait() const
|
||||||
{
|
{
|
||||||
// @TODO wait timeout + global timeout
|
// @TODO wait timeout + global timeout
|
||||||
@ -146,17 +166,17 @@ inline QPromise<T> QPromiseBase<T>::wait() const
|
|||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename E>
|
template<typename E>
|
||||||
inline QPromise<T> QPromiseBase<T>::reject(E&& error)
|
inline QPromise<T> QPromiseBase<T>::reject(E&& error)
|
||||||
{
|
{
|
||||||
return QPromise<T>([&](const QPromiseResolve<T>&, const QPromiseReject<T>& reject) {
|
return QPromise<T>{[&](const QPromiseResolve<T>&, const QPromiseReject<T>& reject) {
|
||||||
reject(std::forward<E>(error));
|
reject(std::forward<E>(error));
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename Functor>
|
template<typename Functor>
|
||||||
inline QPromise<T> QPromise<T>::each(Functor fn)
|
inline QPromise<T> QPromise<T>::each(Functor fn)
|
||||||
{
|
{
|
||||||
return this->tap([=](const T& values) {
|
return this->tap([=](const T& values) {
|
||||||
@ -164,12 +184,10 @@ inline QPromise<T> QPromise<T>::each(Functor fn)
|
|||||||
|
|
||||||
std::vector<QPromise<void>> promises;
|
std::vector<QPromise<void>> promises;
|
||||||
for (const auto& v : values) {
|
for (const auto& v : values) {
|
||||||
promises.push_back(
|
promises.push_back(QtPromise::attempt(fn, v, i).then([]() {
|
||||||
QtPromise::attempt(fn, v, i)
|
// Cast to void in case fn returns a non promise value.
|
||||||
.then([]() {
|
// TODO remove when implicit cast is implemented.
|
||||||
// Cast to void in case fn returns a non promise value.
|
}));
|
||||||
// TODO remove when implicit cast is implemented.
|
|
||||||
}));
|
|
||||||
|
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
@ -178,8 +196,8 @@ inline QPromise<T> QPromise<T>::each(Functor fn)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename Functor>
|
template<typename Functor>
|
||||||
inline QPromise<T> QPromise<T>::filter(Functor fn)
|
inline QPromise<T> QPromise<T>::filter(Functor fn)
|
||||||
{
|
{
|
||||||
return this->then([=](const T& values) {
|
return this->then([=](const T& values) {
|
||||||
@ -187,8 +205,8 @@ inline QPromise<T> QPromise<T>::filter(Functor fn)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename Functor>
|
template<typename Functor>
|
||||||
inline typename QtPromisePrivate::PromiseMapper<T, Functor>::PromiseType
|
inline typename QtPromisePrivate::PromiseMapper<T, Functor>::PromiseType
|
||||||
QPromise<T>::map(Functor fn)
|
QPromise<T>::map(Functor fn)
|
||||||
{
|
{
|
||||||
@ -197,8 +215,8 @@ QPromise<T>::map(Functor fn)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename Functor, typename Input>
|
template<typename Functor, typename Input>
|
||||||
inline typename QtPromisePrivate::PromiseDeduce<Input>::Type
|
inline typename QtPromisePrivate::PromiseDeduce<Input>::Type
|
||||||
QPromise<T>::reduce(Functor fn, Input initial)
|
QPromise<T>::reduce(Functor fn, Input initial)
|
||||||
{
|
{
|
||||||
@ -207,8 +225,8 @@ QPromise<T>::reduce(Functor fn, Input initial)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <typename Functor, typename U>
|
template<typename Functor, typename U>
|
||||||
inline typename QtPromisePrivate::PromiseDeduce<typename U::value_type>::Type
|
inline typename QtPromisePrivate::PromiseDeduce<typename U::value_type>::Type
|
||||||
QPromise<T>::reduce(Functor fn)
|
QPromise<T>::reduce(Functor fn)
|
||||||
{
|
{
|
||||||
@ -217,30 +235,37 @@ QPromise<T>::reduce(Functor fn)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
template <template <typename, typename...> class Sequence, typename ...Args>
|
template<template<typename, typename...> class Sequence, typename... Args>
|
||||||
inline QPromise<QVector<T>> QPromise<T>::all(const Sequence<QPromise<T>, Args...>& promises)
|
inline QPromise<QVector<T>> QPromise<T>::all(const Sequence<QPromise<T>, Args...>& promises)
|
||||||
{
|
{
|
||||||
return QtPromise::all(promises);
|
return QtPromise::all(promises);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
|
template<typename U>
|
||||||
|
inline QPromise<U> QPromise<T>::convert() const
|
||||||
|
{
|
||||||
|
return QPromiseBase<T>::then(QtPromisePrivate::PromiseConverter<T, U>::create());
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
inline QPromise<T> QPromise<T>::resolve(const T& value)
|
inline QPromise<T> QPromise<T>::resolve(const T& value)
|
||||||
{
|
{
|
||||||
return QPromise<T>([&](const QPromiseResolve<T>& resolve) {
|
return QPromise<T>{[&](const QPromiseResolve<T>& resolve) {
|
||||||
resolve(value);
|
resolve(value);
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
inline QPromise<T> QPromise<T>::resolve(T&& value)
|
inline QPromise<T> QPromise<T>::resolve(T&& value)
|
||||||
{
|
{
|
||||||
return QPromise<T>([&](const QPromiseResolve<T>& resolve) {
|
return QPromise<T>{[&](const QPromiseResolve<T>& resolve) {
|
||||||
resolve(std::forward<T>(value));
|
resolve(std::forward<T>(value));
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <template <typename, typename...> class Sequence, typename ...Args>
|
template<template<typename, typename...> class Sequence, typename... Args>
|
||||||
inline QPromise<void> QPromise<void>::all(const Sequence<QPromise<void>, Args...>& promises)
|
inline QPromise<void> QPromise<void>::all(const Sequence<QPromise<void>, Args...>& promises)
|
||||||
{
|
{
|
||||||
return QtPromise::all(promises);
|
return QtPromise::all(promises);
|
||||||
|
@ -1,44 +1,63 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISE_P_H
|
#ifndef QTPROMISE_QPROMISE_P_H
|
||||||
#define QTPROMISE_QPROMISE_P_H
|
#define QTPROMISE_QPROMISE_P_H
|
||||||
|
|
||||||
#include "qpromiseglobal.h"
|
#include "qpromiseglobal.h"
|
||||||
|
|
||||||
// Qt
|
#include <QtCore/QAbstractEventDispatcher>
|
||||||
#include <QCoreApplication>
|
#include <QtCore/QCoreApplication>
|
||||||
#include <QAbstractEventDispatcher>
|
#include <QtCore/QPointer>
|
||||||
#include <QThread>
|
#include <QtCore/QReadWriteLock>
|
||||||
#include <QVector>
|
#include <QtCore/QSharedData>
|
||||||
#include <QReadWriteLock>
|
#include <QtCore/QSharedPointer>
|
||||||
#include <QSharedPointer>
|
#include <QtCore/QThread>
|
||||||
#include <QSharedData>
|
#include <QtCore/QVariant>
|
||||||
#include <QPointer>
|
#include <QtCore/QVector>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class QPromise;
|
class QPromise;
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class QPromiseResolve;
|
class QPromiseResolve;
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class QPromiseReject;
|
class QPromiseReject;
|
||||||
|
|
||||||
|
class QPromiseConversionException;
|
||||||
|
|
||||||
} // namespace QtPromise
|
} // namespace QtPromise
|
||||||
|
|
||||||
namespace QtPromisePrivate {
|
namespace QtPromisePrivate {
|
||||||
|
|
||||||
|
// Use std::invoke_result for C++17 and beyond
|
||||||
|
#if (__cplusplus >= 201703L) || defined(_MSVC_LANG) && (_MSVC_LANG >= 201703L)
|
||||||
|
using std::invoke_result;
|
||||||
|
#else
|
||||||
|
template<class F, class... ArgTypes>
|
||||||
|
using invoke_result = std::result_of<F(ArgTypes...)>;
|
||||||
|
#endif
|
||||||
|
|
||||||
// https://stackoverflow.com/a/21653558
|
// https://stackoverflow.com/a/21653558
|
||||||
template <typename F>
|
template<typename F>
|
||||||
static void qtpromise_defer(F&& f, const QPointer<QThread>& thread)
|
static void qtpromise_defer(F&& f, const QPointer<QThread>& thread)
|
||||||
{
|
{
|
||||||
using FType = typename std::decay<F>::type;
|
using FType = typename std::decay<F>::type;
|
||||||
|
|
||||||
struct Event : public QEvent
|
struct Event : public QEvent
|
||||||
{
|
{
|
||||||
Event(FType&& f) : QEvent(QEvent::None), m_f(std::move(f)) { }
|
Event(FType&& f) : QEvent{QEvent::None}, m_f{std::move(f)} { }
|
||||||
Event(const FType& f) : QEvent(QEvent::None), m_f(f) { }
|
Event(const FType& f) : QEvent{QEvent::None}, m_f{f} { }
|
||||||
~Event() { m_f(); }
|
~Event() override { m_f(); }
|
||||||
FType m_f;
|
FType m_f;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -60,34 +79,34 @@ static void qtpromise_defer(F&& f, const QPointer<QThread>& thread)
|
|||||||
}
|
}
|
||||||
|
|
||||||
Q_ASSERT_X(target, "postMetaCall", "Target thread must have an event loop");
|
Q_ASSERT_X(target, "postMetaCall", "Target thread must have an event loop");
|
||||||
QCoreApplication::postEvent(target, new Event(std::forward<F>(f)));
|
QCoreApplication::postEvent(target, new Event{std::forward<F>(f)});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename F>
|
template<typename F>
|
||||||
static void qtpromise_defer(F&& f)
|
static void qtpromise_defer(F&& f)
|
||||||
{
|
{
|
||||||
Q_ASSERT(QThread::currentThread());
|
Q_ASSERT(QThread::currentThread());
|
||||||
qtpromise_defer(std::forward<F>(f), QThread::currentThread());
|
qtpromise_defer(std::forward<F>(f), QThread::currentThread());
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class PromiseValue
|
class PromiseValue
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
PromiseValue() { }
|
PromiseValue() { }
|
||||||
PromiseValue(const T& data) : m_data(new T(data)) { }
|
PromiseValue(const T& data) : m_data(std::make_shared<T>(data)) { }
|
||||||
PromiseValue(T&& data) : m_data(new T(std::move(data))) { }
|
PromiseValue(T&& data) : m_data(std::make_shared<T>(std::forward<T>(data))) { }
|
||||||
bool isNull() const { return m_data.isNull(); }
|
bool isNull() const { return m_data == nullptr; }
|
||||||
const T& data() const { return *m_data; }
|
const T& data() const { return *m_data; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QSharedPointer<T> m_data;
|
std::shared_ptr<T> m_data;
|
||||||
};
|
};
|
||||||
|
|
||||||
class PromiseError
|
class PromiseError
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
template <typename T>
|
template<typename T>
|
||||||
PromiseError(const T& value)
|
PromiseError(const T& value)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
@ -98,8 +117,8 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
PromiseError() { }
|
PromiseError() { }
|
||||||
PromiseError(const std::exception_ptr& exception) : m_data(exception) { }
|
PromiseError(const std::exception_ptr& exception) : m_data{exception} { }
|
||||||
void rethrow() const { std::rethrow_exception(m_data); }
|
Q_NORETURN void rethrow() const { std::rethrow_exception(m_data); }
|
||||||
bool isNull() const { return m_data == nullptr; }
|
bool isNull() const { return m_data == nullptr; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
@ -107,116 +126,110 @@ private:
|
|||||||
std::exception_ptr m_data;
|
std::exception_ptr m_data;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseDeduce
|
struct PromiseDeduce
|
||||||
{
|
{
|
||||||
using Type = QtPromise::QPromise<T>;
|
using Type = QtPromise::QPromise<T>;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseDeduce<T&>
|
struct PromiseDeduce<T&> : public PromiseDeduce<T>
|
||||||
: public PromiseDeduce<T>
|
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseDeduce<const T>
|
struct PromiseDeduce<const T> : public PromiseDeduce<T>
|
||||||
: public PromiseDeduce<T>
|
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseDeduce<const volatile T>
|
struct PromiseDeduce<const volatile T> : public PromiseDeduce<T>
|
||||||
: public PromiseDeduce<T>
|
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseDeduce<QtPromise::QPromise<T>>
|
struct PromiseDeduce<QtPromise::QPromise<T>> : public PromiseDeduce<T>
|
||||||
: public PromiseDeduce<T>
|
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename Functor, typename... Args>
|
template<typename Functor, typename... Args>
|
||||||
struct PromiseFunctor
|
struct PromiseFunctor
|
||||||
{
|
{
|
||||||
using ResultType = typename std::result_of<Functor(Args...)>::type;
|
using ResultType = typename invoke_result<Functor, Args...>::type;
|
||||||
using PromiseType = typename PromiseDeduce<ResultType>::Type;
|
using PromiseType = typename PromiseDeduce<ResultType>::Type;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseFulfill
|
struct PromiseFulfill
|
||||||
{
|
{
|
||||||
template <typename V, typename TResolve, typename TReject>
|
template<typename V, typename TResolve, typename TReject>
|
||||||
static void call(V&& value, const TResolve& resolve, const TReject&)
|
static void call(V&& value, const TResolve& resolve, const TReject&)
|
||||||
{
|
{
|
||||||
resolve(std::forward<V>(value));
|
resolve(std::forward<V>(value));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseFulfill<QtPromise::QPromise<T>>
|
struct PromiseFulfill<QtPromise::QPromise<T>>
|
||||||
{
|
{
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static void call(
|
static void
|
||||||
const QtPromise::QPromise<T>& promise,
|
call(const QtPromise::QPromise<T>& promise, const TResolve& resolve, const TReject& reject)
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
if (promise.isFulfilled()) {
|
if (promise.isFulfilled()) {
|
||||||
resolve(promise.m_d->value());
|
resolve(promise.m_d->value());
|
||||||
} else if (promise.isRejected()) {
|
} else if (promise.isRejected()) {
|
||||||
reject(promise.m_d->error());
|
reject(promise.m_d->error());
|
||||||
} else {
|
} else {
|
||||||
promise.then([=]() {
|
promise.then(
|
||||||
resolve(promise.m_d->value());
|
[=]() {
|
||||||
}, [=]() { // catch all
|
resolve(promise.m_d->value());
|
||||||
reject(promise.m_d->error());
|
},
|
||||||
});
|
[=]() { // catch all
|
||||||
|
reject(promise.m_d->error());
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
struct PromiseFulfill<QtPromise::QPromise<void>>
|
struct PromiseFulfill<QtPromise::QPromise<void>>
|
||||||
{
|
{
|
||||||
template <typename TPromise, typename TResolve, typename TReject>
|
template<typename TPromise, typename TResolve, typename TReject>
|
||||||
static void call(
|
static void call(const TPromise& promise, const TResolve& resolve, const TReject& reject)
|
||||||
const TPromise& promise,
|
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
if (promise.isFulfilled()) {
|
if (promise.isFulfilled()) {
|
||||||
resolve();
|
resolve();
|
||||||
} else if (promise.isRejected()) {
|
} else if (promise.isRejected()) {
|
||||||
reject(promise.m_d->error());
|
reject(promise.m_d->error());
|
||||||
} else {
|
} else {
|
||||||
promise.then([=]() {
|
promise.then(
|
||||||
resolve();
|
[=]() {
|
||||||
}, [=]() { // catch all
|
resolve();
|
||||||
reject(promise.m_d->error());
|
},
|
||||||
});
|
[=]() { // catch all
|
||||||
|
reject(promise.m_d->error());
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Result>
|
template<typename Result>
|
||||||
struct PromiseDispatch
|
struct PromiseDispatch
|
||||||
{
|
{
|
||||||
template <typename Resolve, typename Reject, typename Functor, typename... Args>
|
template<typename Resolve, typename Reject, typename Functor, typename... Args>
|
||||||
static void call(const Resolve& resolve, const Reject& reject, Functor fn, Args&&... args)
|
static void call(const Resolve& resolve, const Reject& reject, Functor fn, Args&&... args)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
PromiseFulfill<Unqualified<Result>>::call(
|
PromiseFulfill<Unqualified<Result>>::call(fn(std::forward<Args>(args)...),
|
||||||
fn(std::forward<Args>(args)...),
|
resolve,
|
||||||
resolve,
|
reject);
|
||||||
reject);
|
|
||||||
} catch (...) {
|
} catch (...) {
|
||||||
reject(std::current_exception());
|
reject(std::current_exception());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
struct PromiseDispatch<void>
|
struct PromiseDispatch<void>
|
||||||
{
|
{
|
||||||
template <typename Resolve, typename Reject, typename Functor, typename... Args>
|
template<typename Resolve, typename Reject, typename Functor, typename... Args>
|
||||||
static void call(const Resolve& resolve, const Reject& reject, Functor fn, Args&&... args)
|
static void call(const Resolve& resolve, const Reject& reject, Functor fn, Args&&... args)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
@ -228,17 +241,15 @@ struct PromiseDispatch<void>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename THandler, typename TArg = typename ArgsOf<THandler>::first>
|
template<typename T, typename THandler, typename TArg = typename ArgsOf<THandler>::first>
|
||||||
struct PromiseHandler
|
struct PromiseHandler
|
||||||
{
|
{
|
||||||
using ResType = typename std::result_of<THandler(T)>::type;
|
using ResType = typename invoke_result<THandler, T>::type;
|
||||||
using Promise = typename PromiseDeduce<ResType>::Type;
|
using Promise = typename PromiseDeduce<ResType>::Type;
|
||||||
|
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void(const T&)> create(
|
static std::function<void(const T&)>
|
||||||
const THandler& handler,
|
create(const THandler& handler, const TResolve& resolve, const TReject& reject)
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
return [=](const T& value) {
|
return [=](const T& value) {
|
||||||
PromiseDispatch<ResType>::call(resolve, reject, handler, value);
|
PromiseDispatch<ResType>::call(resolve, reject, handler, value);
|
||||||
@ -246,17 +257,15 @@ struct PromiseHandler
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename THandler>
|
template<typename T, typename THandler>
|
||||||
struct PromiseHandler<T, THandler, void>
|
struct PromiseHandler<T, THandler, void>
|
||||||
{
|
{
|
||||||
using ResType = typename std::result_of<THandler()>::type;
|
using ResType = typename invoke_result<THandler>::type;
|
||||||
using Promise = typename PromiseDeduce<ResType>::Type;
|
using Promise = typename PromiseDeduce<ResType>::Type;
|
||||||
|
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void(const T&)> create(
|
static std::function<void(const T&)>
|
||||||
const THandler& handler,
|
create(const THandler& handler, const TResolve& resolve, const TReject& reject)
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
return [=](const T&) {
|
return [=](const T&) {
|
||||||
PromiseDispatch<ResType>::call(resolve, reject, handler);
|
PromiseDispatch<ResType>::call(resolve, reject, handler);
|
||||||
@ -264,17 +273,15 @@ struct PromiseHandler<T, THandler, void>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename THandler>
|
template<typename THandler>
|
||||||
struct PromiseHandler<void, THandler, void>
|
struct PromiseHandler<void, THandler, void>
|
||||||
{
|
{
|
||||||
using ResType = typename std::result_of<THandler()>::type;
|
using ResType = typename invoke_result<THandler>::type;
|
||||||
using Promise = typename PromiseDeduce<ResType>::Type;
|
using Promise = typename PromiseDeduce<ResType>::Type;
|
||||||
|
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void()> create(
|
static std::function<void()>
|
||||||
const THandler& handler,
|
create(const THandler& handler, const TResolve& resolve, const TReject& reject)
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
return [=]() {
|
return [=]() {
|
||||||
PromiseDispatch<ResType>::call(resolve, reject, handler);
|
PromiseDispatch<ResType>::call(resolve, reject, handler);
|
||||||
@ -282,16 +289,14 @@ struct PromiseHandler<void, THandler, void>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseHandler<T, std::nullptr_t, void>
|
struct PromiseHandler<T, std::nullptr_t, void>
|
||||||
{
|
{
|
||||||
using Promise = QtPromise::QPromise<T>;
|
using Promise = QtPromise::QPromise<T>;
|
||||||
|
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void(const T&)> create(
|
static std::function<void(const T&)>
|
||||||
std::nullptr_t,
|
create(std::nullptr_t, const TResolve& resolve, const TReject& reject)
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
return [=](const T& value) {
|
return [=](const T& value) {
|
||||||
// 2.2.7.3. If onFulfilled is not a function and promise1 is fulfilled,
|
// 2.2.7.3. If onFulfilled is not a function and promise1 is fulfilled,
|
||||||
@ -301,16 +306,13 @@ struct PromiseHandler<T, std::nullptr_t, void>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
struct PromiseHandler<void, std::nullptr_t, void>
|
struct PromiseHandler<void, std::nullptr_t, void>
|
||||||
{
|
{
|
||||||
using Promise = QtPromise::QPromise<void>;
|
using Promise = QtPromise::QPromise<void>;
|
||||||
|
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void()> create(
|
static std::function<void()> create(std::nullptr_t, const TResolve& resolve, const TReject&)
|
||||||
std::nullptr_t,
|
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject&)
|
|
||||||
{
|
{
|
||||||
return [=]() {
|
return [=]() {
|
||||||
// 2.2.7.3. If onFulfilled is not a function and promise1 is fulfilled,
|
// 2.2.7.3. If onFulfilled is not a function and promise1 is fulfilled,
|
||||||
@ -320,22 +322,20 @@ struct PromiseHandler<void, std::nullptr_t, void>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename THandler, typename TArg = typename ArgsOf<THandler>::first>
|
template<typename T, typename THandler, typename TArg = typename ArgsOf<THandler>::first>
|
||||||
struct PromiseCatcher
|
struct PromiseCatcher
|
||||||
{
|
{
|
||||||
using ResType = typename std::result_of<THandler(TArg)>::type;
|
using ResType = typename invoke_result<THandler, TArg>::type;
|
||||||
|
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void(const PromiseError&)> create(
|
static std::function<void(const PromiseError&)>
|
||||||
const THandler& handler,
|
create(const THandler& handler, const TResolve& resolve, const TReject& reject)
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
return [=](const PromiseError& error) {
|
return [=](const PromiseError& error) {
|
||||||
try {
|
try {
|
||||||
error.rethrow();
|
error.rethrow();
|
||||||
} catch (const TArg& error) {
|
} catch (const TArg& argError) {
|
||||||
PromiseDispatch<ResType>::call(resolve, reject, handler, error);
|
PromiseDispatch<ResType>::call(resolve, reject, handler, argError);
|
||||||
} catch (...) {
|
} catch (...) {
|
||||||
reject(std::current_exception());
|
reject(std::current_exception());
|
||||||
}
|
}
|
||||||
@ -343,16 +343,14 @@ struct PromiseCatcher
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename THandler>
|
template<typename T, typename THandler>
|
||||||
struct PromiseCatcher<T, THandler, void>
|
struct PromiseCatcher<T, THandler, void>
|
||||||
{
|
{
|
||||||
using ResType = typename std::result_of<THandler()>::type;
|
using ResType = typename invoke_result<THandler>::type;
|
||||||
|
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void(const PromiseError&)> create(
|
static std::function<void(const PromiseError&)>
|
||||||
const THandler& handler,
|
create(const THandler& handler, const TResolve& resolve, const TReject& reject)
|
||||||
const TResolve& resolve,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
return [=](const PromiseError& error) {
|
return [=](const PromiseError& error) {
|
||||||
try {
|
try {
|
||||||
@ -364,14 +362,12 @@ struct PromiseCatcher<T, THandler, void>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseCatcher<T, std::nullptr_t, void>
|
struct PromiseCatcher<T, std::nullptr_t, void>
|
||||||
{
|
{
|
||||||
template <typename TResolve, typename TReject>
|
template<typename TResolve, typename TReject>
|
||||||
static std::function<void(const PromiseError&)> create(
|
static std::function<void(const PromiseError&)>
|
||||||
std::nullptr_t,
|
create(std::nullptr_t, const TResolve&, const TReject& reject)
|
||||||
const TResolve&,
|
|
||||||
const TReject& reject)
|
|
||||||
{
|
{
|
||||||
return [=](const PromiseError& error) {
|
return [=](const PromiseError& error) {
|
||||||
// 2.2.7.4. If onRejected is not a function and promise1 is rejected,
|
// 2.2.7.4. If onRejected is not a function and promise1 is rejected,
|
||||||
@ -381,63 +377,57 @@ struct PromiseCatcher<T, std::nullptr_t, void>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename F>
|
template<typename T, typename F>
|
||||||
struct PromiseMapper
|
struct PromiseMapper
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T, typename F, template <typename, typename...> class Sequence, typename ...Args>
|
template<typename T, typename F, template<typename, typename...> class Sequence, typename... Args>
|
||||||
struct PromiseMapper<Sequence<T, Args...>, F>
|
struct PromiseMapper<Sequence<T, Args...>, F>
|
||||||
{
|
{
|
||||||
using ReturnType = typename std::result_of<F(T, int)>::type;
|
using ReturnType = typename invoke_result<F, T, int>::type;
|
||||||
using ResultType = QVector<typename PromiseDeduce<ReturnType>::Type::Type>;
|
using ResultType = QVector<typename PromiseDeduce<ReturnType>::Type::Type>;
|
||||||
using PromiseType = QtPromise::QPromise<ResultType>;
|
using PromiseType = QtPromise::QPromise<ResultType>;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T> class PromiseData;
|
template<typename T>
|
||||||
|
class PromiseData;
|
||||||
|
|
||||||
template <typename T, typename F>
|
template<typename T, typename F>
|
||||||
class PromiseDataBase : public QSharedData
|
class PromiseDataBase : public QSharedData
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
using Handler = std::pair<QPointer<QThread>, std::function<F>>;
|
using Handler = std::pair<QPointer<QThread>, std::function<F>>;
|
||||||
using Catcher = std::pair<QPointer<QThread>, std::function<void(const PromiseError&)>>;
|
using Catcher = std::pair<QPointer<QThread>, std::function<void(const PromiseError&)>>;
|
||||||
|
|
||||||
virtual ~PromiseDataBase() {}
|
virtual ~PromiseDataBase() { }
|
||||||
|
|
||||||
bool isFulfilled() const
|
bool isFulfilled() const { return !isPending() && m_error.isNull(); }
|
||||||
{
|
bool isRejected() const { return !isPending() && !m_error.isNull(); }
|
||||||
return !isPending() && m_error.isNull();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isRejected() const
|
|
||||||
{
|
|
||||||
return !isPending() && !m_error.isNull();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isPending() const
|
bool isPending() const
|
||||||
{
|
{
|
||||||
QReadLocker lock(&m_lock);
|
QReadLocker lock{&m_lock};
|
||||||
return !m_settled;
|
return !m_settled;
|
||||||
}
|
}
|
||||||
|
|
||||||
void addHandler(std::function<F> handler)
|
void addHandler(std::function<F> handler)
|
||||||
{
|
{
|
||||||
QWriteLocker lock(&m_lock);
|
QWriteLocker lock{&m_lock};
|
||||||
m_handlers.append({QThread::currentThread(), std::move(handler)});
|
m_handlers.append({QThread::currentThread(), std::move(handler)});
|
||||||
}
|
}
|
||||||
|
|
||||||
void addCatcher(std::function<void(const PromiseError&)> catcher)
|
void addCatcher(std::function<void(const PromiseError&)> catcher)
|
||||||
{
|
{
|
||||||
QWriteLocker lock(&m_lock);
|
QWriteLocker lock{&m_lock};
|
||||||
m_catchers.append({QThread::currentThread(), std::move(catcher)});
|
m_catchers.append({QThread::currentThread(), std::move(catcher)});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename E>
|
template<typename E>
|
||||||
void reject(E&& error)
|
void reject(E&& error)
|
||||||
{
|
{
|
||||||
Q_ASSERT(isPending());
|
Q_ASSERT(isPending());
|
||||||
Q_ASSERT(m_error.isNull());
|
Q_ASSERT(m_error.isNull());
|
||||||
m_error = PromiseError(std::forward<E>(error));
|
m_error = PromiseError{std::forward<E>(error)};
|
||||||
setSettled();
|
setSettled();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -461,8 +451,8 @@ public:
|
|||||||
// captured in the handler and/or catcher lambdas.
|
// captured in the handler and/or catcher lambdas.
|
||||||
|
|
||||||
m_lock.lockForWrite();
|
m_lock.lockForWrite();
|
||||||
QVector<Handler> handlers(std::move(m_handlers));
|
QVector<Handler> handlers = std::move(m_handlers);
|
||||||
QVector<Catcher> catchers(std::move(m_catchers));
|
QVector<Catcher> catchers = std::move(m_catchers);
|
||||||
m_lock.unlock();
|
m_lock.unlock();
|
||||||
|
|
||||||
if (m_error.isNull()) {
|
if (m_error.isNull()) {
|
||||||
@ -470,14 +460,16 @@ public:
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
PromiseError error(m_error);
|
PromiseError error = m_error;
|
||||||
Q_ASSERT(!error.isNull());
|
Q_ASSERT(!error.isNull());
|
||||||
|
|
||||||
for (const auto& catcher: catchers) {
|
for (const auto& catcher : catchers) {
|
||||||
const auto& fn = catcher.second;
|
const auto& fn = catcher.second;
|
||||||
qtpromise_defer([=]() {
|
qtpromise_defer(
|
||||||
fn(error);
|
[=]() {
|
||||||
}, catcher.first);
|
fn(error);
|
||||||
|
},
|
||||||
|
catcher.first);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -486,7 +478,7 @@ protected:
|
|||||||
|
|
||||||
void setSettled()
|
void setSettled()
|
||||||
{
|
{
|
||||||
QWriteLocker lock(&m_lock);
|
QWriteLocker lock{&m_lock};
|
||||||
Q_ASSERT(!m_settled);
|
Q_ASSERT(!m_settled);
|
||||||
m_settled = true;
|
m_settled = true;
|
||||||
}
|
}
|
||||||
@ -500,18 +492,18 @@ private:
|
|||||||
PromiseError m_error;
|
PromiseError m_error;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class PromiseData : public PromiseDataBase<T, void(const T&)>
|
class PromiseData : public PromiseDataBase<T, void(const T&)>
|
||||||
{
|
{
|
||||||
using Handler = typename PromiseDataBase<T, void(const T&)>::Handler;
|
using Handler = typename PromiseDataBase<T, void(const T&)>::Handler;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
template <typename V>
|
template<typename V>
|
||||||
void resolve(V&& value)
|
void resolve(V&& value)
|
||||||
{
|
{
|
||||||
Q_ASSERT(this->isPending());
|
Q_ASSERT(this->isPending());
|
||||||
Q_ASSERT(m_value.isNull());
|
Q_ASSERT(m_value.isNull());
|
||||||
m_value = PromiseValue<T>(std::forward<V>(value));
|
m_value = PromiseValue<T>{std::forward<V>(value)};
|
||||||
this->setSettled();
|
this->setSettled();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -523,14 +515,16 @@ public:
|
|||||||
|
|
||||||
void notify(const QVector<Handler>& handlers) Q_DECL_OVERRIDE
|
void notify(const QVector<Handler>& handlers) Q_DECL_OVERRIDE
|
||||||
{
|
{
|
||||||
PromiseValue<T> value(m_value);
|
PromiseValue<T> value = m_value;
|
||||||
Q_ASSERT(!value.isNull());
|
Q_ASSERT(!value.isNull());
|
||||||
|
|
||||||
for (const auto& handler: handlers) {
|
for (const auto& handler : handlers) {
|
||||||
const auto& fn = handler.second;
|
const auto& fn = handler.second;
|
||||||
qtpromise_defer([=]() {
|
qtpromise_defer(
|
||||||
fn(value.data());
|
[=]() {
|
||||||
}, handler.first);
|
fn(value.data());
|
||||||
|
},
|
||||||
|
handler.first);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -538,21 +532,18 @@ private:
|
|||||||
PromiseValue<T> m_value;
|
PromiseValue<T> m_value;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
class PromiseData<void> : public PromiseDataBase<void, void()>
|
class PromiseData<void> : public PromiseDataBase<void, void()>
|
||||||
{
|
{
|
||||||
using Handler = PromiseDataBase<void, void()>::Handler;
|
using Handler = PromiseDataBase<void, void()>::Handler;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void resolve()
|
void resolve() { setSettled(); }
|
||||||
{
|
|
||||||
setSettled();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void notify(const QVector<Handler>& handlers) Q_DECL_OVERRIDE
|
void notify(const QVector<Handler>& handlers) Q_DECL_OVERRIDE
|
||||||
{
|
{
|
||||||
for (const auto& handler: handlers) {
|
for (const auto& handler : handlers) {
|
||||||
qtpromise_defer(handler.second, handler.first);
|
qtpromise_defer(handler.second, handler.first);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -560,13 +551,77 @@ protected:
|
|||||||
|
|
||||||
struct PromiseInspect
|
struct PromiseInspect
|
||||||
{
|
{
|
||||||
template <typename T>
|
template<typename T>
|
||||||
static inline PromiseData<T>* get(const QtPromise::QPromise<T>& p)
|
static inline PromiseData<T>* get(const QtPromise::QPromise<T>& p)
|
||||||
{
|
{
|
||||||
return p.m_d.data();
|
return p.m_d.data();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<typename T, typename U, bool IsConvertibleViaStaticCast>
|
||||||
|
struct PromiseConverterBase;
|
||||||
|
|
||||||
|
template<typename T, typename U>
|
||||||
|
struct PromiseConverterBase<T, U, true>
|
||||||
|
{
|
||||||
|
static std::function<U(const T&)> create()
|
||||||
|
{
|
||||||
|
return [](const T& value) {
|
||||||
|
return static_cast<U>(value);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename T, typename U>
|
||||||
|
struct PromiseConverterBase<T, U, false>
|
||||||
|
{
|
||||||
|
static std::function<U(const T&)> create()
|
||||||
|
{
|
||||||
|
return [](const T& value) {
|
||||||
|
auto tmp = QVariant::fromValue(value);
|
||||||
|
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
|
// https://doc.qt.io/qt-6/qvariant.html#using-canconvert-and-convert-consecutively
|
||||||
|
if (tmp.canConvert(QMetaType{qMetaTypeId<U>()})
|
||||||
|
&& tmp.convert(QMetaType{qMetaTypeId<U>()})) {
|
||||||
|
return qvariant_cast<U>(tmp);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
// https://doc.qt.io/qt-5/qvariant.html#using-canconvert-and-convert-consecutively
|
||||||
|
if (tmp.canConvert(qMetaTypeId<U>()) && tmp.convert(qMetaTypeId<U>())) {
|
||||||
|
return qvariant_cast<U>(tmp);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
throw QtPromise::QPromiseConversionException{};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
struct PromiseConverterBase<T, QVariant, false>
|
||||||
|
{
|
||||||
|
static std::function<QVariant(const T&)> create()
|
||||||
|
{
|
||||||
|
return [](const T& value) {
|
||||||
|
return QVariant::fromValue(value);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename T, typename U>
|
||||||
|
struct PromiseConverter
|
||||||
|
: PromiseConverterBase<T,
|
||||||
|
U,
|
||||||
|
// Fundamental types and converting constructors.
|
||||||
|
std::is_convertible<T, U>::value ||
|
||||||
|
// Conversion to void.
|
||||||
|
std::is_same<U, void>::value ||
|
||||||
|
// Conversion between enums and arithmetic types.
|
||||||
|
((std::is_enum<T>::value && std::is_arithmetic<U>::value)
|
||||||
|
|| (std::is_arithmetic<T>::value && std::is_enum<U>::value)
|
||||||
|
|| (std::is_enum<T>::value && std::is_enum<U>::value))>
|
||||||
|
{ };
|
||||||
|
|
||||||
} // namespace QtPromisePrivate
|
} // namespace QtPromisePrivate
|
||||||
|
|
||||||
#endif // QTPROMISE_QPROMISE_H
|
#endif // QTPROMISE_QPROMISE_H
|
||||||
|
@ -1,17 +1,25 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISECONNECTIONS_H
|
#ifndef QTPROMISE_QPROMISECONNECTIONS_H
|
||||||
#define QTPROMISE_QPROMISECONNECTIONS_H
|
#define QTPROMISE_QPROMISECONNECTIONS_H
|
||||||
|
|
||||||
// Qt
|
#include <QtCore/QObject>
|
||||||
#include <QSharedPointer>
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
class QPromiseConnections
|
class QPromiseConnections
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
QPromiseConnections() : m_d(new Data()) { }
|
QPromiseConnections() : m_d(std::make_shared<Data>()) { }
|
||||||
|
|
||||||
int count() const { return m_d->connections.count(); }
|
int count() const { return static_cast<int>(m_d->connections.count()); }
|
||||||
|
|
||||||
void disconnect() const { m_d->disconnect(); }
|
void disconnect() const { m_d->disconnect(); }
|
||||||
|
|
||||||
@ -25,22 +33,24 @@ private:
|
|||||||
{
|
{
|
||||||
QVector<QMetaObject::Connection> connections;
|
QVector<QMetaObject::Connection> connections;
|
||||||
|
|
||||||
~Data() {
|
~Data()
|
||||||
|
{
|
||||||
if (!connections.empty()) {
|
if (!connections.empty()) {
|
||||||
qWarning("QPromiseConnections: destroyed with unhandled connections.");
|
qWarning("QPromiseConnections: destroyed with unhandled connections.");
|
||||||
disconnect();
|
disconnect();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void disconnect() {
|
void disconnect()
|
||||||
for (const auto& connection: connections) {
|
{
|
||||||
|
for (const auto& connection : connections) {
|
||||||
QObject::disconnect(connection);
|
QObject::disconnect(connection);
|
||||||
}
|
}
|
||||||
connections.clear();
|
connections.clear();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
QSharedPointer<Data> m_d;
|
std::shared_ptr<Data> m_d;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace QtPromise
|
} // namespace QtPromise
|
||||||
|
@ -1,11 +1,16 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISEEXCEPTIONS_H
|
#ifndef QTPROMISE_QPROMISEEXCEPTIONS_H
|
||||||
#define QTPROMISE_QPROMISEEXCEPTIONS_H
|
#define QTPROMISE_QPROMISEEXCEPTIONS_H
|
||||||
|
|
||||||
#include "qpromise_p.h"
|
|
||||||
#include "qpromiseglobal.h"
|
#include "qpromiseglobal.h"
|
||||||
|
|
||||||
// Qt
|
#include <QtCore/QException>
|
||||||
#include <QException>
|
|
||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
@ -15,7 +20,7 @@ public:
|
|||||||
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
||||||
QPromiseCanceledException* clone() const Q_DECL_OVERRIDE
|
QPromiseCanceledException* clone() const Q_DECL_OVERRIDE
|
||||||
{
|
{
|
||||||
return new QPromiseCanceledException(*this);
|
return new QPromiseCanceledException{*this};
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -25,7 +30,17 @@ public:
|
|||||||
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
||||||
QPromiseContextException* clone() const Q_DECL_OVERRIDE
|
QPromiseContextException* clone() const Q_DECL_OVERRIDE
|
||||||
{
|
{
|
||||||
return new QPromiseContextException(*this);
|
return new QPromiseContextException{*this};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class QPromiseConversionException : public QException
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
||||||
|
QPromiseConversionException* clone() const Q_DECL_OVERRIDE
|
||||||
|
{
|
||||||
|
return new QPromiseConversionException{*this};
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -35,7 +50,7 @@ public:
|
|||||||
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
||||||
QPromiseTimeoutException* clone() const Q_DECL_OVERRIDE
|
QPromiseTimeoutException* clone() const Q_DECL_OVERRIDE
|
||||||
{
|
{
|
||||||
return new QPromiseTimeoutException(*this);
|
return new QPromiseTimeoutException{*this};
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -45,16 +60,10 @@ public:
|
|||||||
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
void raise() const Q_DECL_OVERRIDE { throw *this; }
|
||||||
QPromiseUndefinedException* clone() const Q_DECL_OVERRIDE
|
QPromiseUndefinedException* clone() const Q_DECL_OVERRIDE
|
||||||
{
|
{
|
||||||
return new QPromiseUndefinedException(*this);
|
return new QPromiseUndefinedException{*this};
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// QPromiseError is provided for backward compatibility and will be
|
|
||||||
// removed in the next major version: it wasn't intended to be used
|
|
||||||
// directly and thus should not be part of the public API.
|
|
||||||
// TODO Remove QPromiseError at version 1.0
|
|
||||||
using QPromiseError = QtPromisePrivate::PromiseError;
|
|
||||||
|
|
||||||
} // namespace QtPromise
|
} // namespace QtPromise
|
||||||
|
|
||||||
#endif // QTPROMISE_QPROMISEEXCEPTIONS_H
|
#endif // QTPROMISE_QPROMISEEXCEPTIONS_H
|
||||||
|
@ -1,30 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISEFUTURE_P_H
|
#ifndef QTPROMISE_QPROMISEFUTURE_P_H
|
||||||
#define QTPROMISE_QPROMISEFUTURE_P_H
|
#define QTPROMISE_QPROMISEFUTURE_P_H
|
||||||
|
|
||||||
#include "qpromiseexceptions.h"
|
#include "qpromiseexceptions.h"
|
||||||
|
|
||||||
// Qt
|
#include <QtCore/QFuture>
|
||||||
#include <QFutureWatcher>
|
#include <QtCore/QFutureWatcher>
|
||||||
#include <QFuture>
|
|
||||||
|
|
||||||
namespace QtPromisePrivate {
|
namespace QtPromisePrivate {
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseDeduce<QFuture<T>>
|
struct PromiseDeduce<QFuture<T>> : public PromiseDeduce<T>
|
||||||
: public PromiseDeduce<T>
|
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct PromiseFulfill<QFuture<T>>
|
struct PromiseFulfill<QFuture<T>>
|
||||||
{
|
{
|
||||||
static void call(
|
static void call(const QFuture<T>& future,
|
||||||
const QFuture<T>& future,
|
const QtPromise::QPromiseResolve<T>& resolve,
|
||||||
const QtPromise::QPromiseResolve<T>& resolve,
|
const QtPromise::QPromiseReject<T>& reject)
|
||||||
const QtPromise::QPromiseReject<T>& reject)
|
|
||||||
{
|
{
|
||||||
using Watcher = QFutureWatcher<T>;
|
using Watcher = QFutureWatcher<T>;
|
||||||
|
|
||||||
Watcher* watcher = new Watcher();
|
Watcher* watcher = new Watcher{};
|
||||||
QObject::connect(watcher, &Watcher::finished, [=]() mutable {
|
QObject::connect(watcher, &Watcher::finished, [=]() mutable {
|
||||||
try {
|
try {
|
||||||
if (watcher->isCanceled()) {
|
if (watcher->isCanceled()) {
|
||||||
@ -34,7 +38,7 @@ struct PromiseFulfill<QFuture<T>>
|
|||||||
// rethrown potential exceptions using waitForFinished() and thus detect
|
// rethrown potential exceptions using waitForFinished() and thus detect
|
||||||
// if the future has been canceled by the user or an exception.
|
// if the future has been canceled by the user or an exception.
|
||||||
watcher->waitForFinished();
|
watcher->waitForFinished();
|
||||||
reject(QtPromise::QPromiseCanceledException());
|
reject(QtPromise::QPromiseCanceledException{});
|
||||||
} else {
|
} else {
|
||||||
PromiseFulfill<T>::call(watcher->result(), resolve, reject);
|
PromiseFulfill<T>::call(watcher->result(), resolve, reject);
|
||||||
}
|
}
|
||||||
@ -49,23 +53,22 @@ struct PromiseFulfill<QFuture<T>>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
struct PromiseFulfill<QFuture<void>>
|
struct PromiseFulfill<QFuture<void>>
|
||||||
{
|
{
|
||||||
static void call(
|
static void call(const QFuture<void>& future,
|
||||||
const QFuture<void>& future,
|
const QtPromise::QPromiseResolve<void>& resolve,
|
||||||
const QtPromise::QPromiseResolve<void>& resolve,
|
const QtPromise::QPromiseReject<void>& reject)
|
||||||
const QtPromise::QPromiseReject<void>& reject)
|
|
||||||
{
|
{
|
||||||
using Watcher = QFutureWatcher<void>;
|
using Watcher = QFutureWatcher<void>;
|
||||||
|
|
||||||
Watcher* watcher = new Watcher();
|
Watcher* watcher = new Watcher{};
|
||||||
QObject::connect(watcher, &Watcher::finished, [=]() mutable {
|
QObject::connect(watcher, &Watcher::finished, [=]() mutable {
|
||||||
try {
|
try {
|
||||||
if (watcher->isCanceled()) {
|
if (watcher->isCanceled()) {
|
||||||
// let's rethrown potential exception
|
// let's rethrown potential exception
|
||||||
watcher->waitForFinished();
|
watcher->waitForFinished();
|
||||||
reject(QtPromise::QPromiseCanceledException());
|
reject(QtPromise::QPromiseCanceledException{});
|
||||||
} else {
|
} else {
|
||||||
resolve();
|
resolve();
|
||||||
}
|
}
|
||||||
|
@ -1,30 +1,35 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISEGLOBAL_H
|
#ifndef QTPROMISE_QPROMISEGLOBAL_H
|
||||||
#define QTPROMISE_QPROMISEGLOBAL_H
|
#define QTPROMISE_QPROMISEGLOBAL_H
|
||||||
|
|
||||||
// QtCore
|
|
||||||
#include <QtGlobal>
|
#include <QtGlobal>
|
||||||
|
|
||||||
// STL
|
|
||||||
#include <functional>
|
|
||||||
#include <array>
|
#include <array>
|
||||||
|
#include <functional>
|
||||||
|
|
||||||
|
namespace QtPromisePrivate {
|
||||||
|
|
||||||
namespace QtPromisePrivate
|
|
||||||
{
|
|
||||||
// https://rmf.io/cxx11/even-more-traits#unqualified_types
|
// https://rmf.io/cxx11/even-more-traits#unqualified_types
|
||||||
template <typename T>
|
template<typename T>
|
||||||
using Unqualified = typename std::remove_cv<typename std::remove_reference<T>::type>::type;
|
using Unqualified = typename std::remove_cv<typename std::remove_reference<T>::type>::type;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \struct HasCallOperator
|
* \struct HasCallOperator
|
||||||
* http://stackoverflow.com/a/5117641
|
* http://stackoverflow.com/a/5117641
|
||||||
*/
|
*/
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct HasCallOperator
|
struct HasCallOperator
|
||||||
{
|
{
|
||||||
template <typename U>
|
template<typename U>
|
||||||
static char check(decltype(&U::operator(), char(0)));
|
static char check(decltype(&U::operator(), char(0)));
|
||||||
|
|
||||||
template <typename U>
|
template<typename U>
|
||||||
static char (&check(...))[2];
|
static char (&check(...))[2];
|
||||||
|
|
||||||
static const bool value = (sizeof(check<T>(0)) == 1);
|
static const bool value = (sizeof(check<T>(0)) == 1);
|
||||||
@ -35,7 +40,7 @@ struct HasCallOperator
|
|||||||
* http://stackoverflow.com/a/7943765
|
* http://stackoverflow.com/a/7943765
|
||||||
* http://stackoverflow.com/a/27885283
|
* http://stackoverflow.com/a/27885283
|
||||||
*/
|
*/
|
||||||
template <typename... Args>
|
template<typename... Args>
|
||||||
struct ArgsTraits
|
struct ArgsTraits
|
||||||
{
|
{
|
||||||
using types = std::tuple<Args...>;
|
using types = std::tuple<Args...>;
|
||||||
@ -43,7 +48,7 @@ struct ArgsTraits
|
|||||||
static const size_t count = std::tuple_size<types>::value;
|
static const size_t count = std::tuple_size<types>::value;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
struct ArgsTraits<>
|
struct ArgsTraits<>
|
||||||
{
|
{
|
||||||
using types = std::tuple<>;
|
using types = std::tuple<>;
|
||||||
@ -51,73 +56,59 @@ struct ArgsTraits<>
|
|||||||
static const size_t count = 0;
|
static const size_t count = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename Enable = void>
|
// Fallback implementation, including types (T) which are not functions but
|
||||||
|
// also lambda with `auto` arguments, which are not covered but still valid
|
||||||
|
// callbacks (see the QPromiseBase<T> template constructor).
|
||||||
|
template<typename T, typename Enabled = void>
|
||||||
struct ArgsOf : public ArgsTraits<>
|
struct ArgsOf : public ArgsTraits<>
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
// Partial specialization for null function.
|
||||||
|
template<>
|
||||||
|
struct ArgsOf<std::nullptr_t> : public ArgsTraits<>
|
||||||
|
{ };
|
||||||
|
|
||||||
|
// Partial specialization for type with a non-overloaded operator().
|
||||||
|
// This applies to lambda, std::function but not to std::bind result.
|
||||||
|
template<typename T>
|
||||||
struct ArgsOf<T, typename std::enable_if<HasCallOperator<T>::value>::type>
|
struct ArgsOf<T, typename std::enable_if<HasCallOperator<T>::value>::type>
|
||||||
: public ArgsOf<decltype(&T::operator())>
|
: public ArgsOf<decltype(&T::operator())>
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename TReturn, typename... Args>
|
// Partial specialization to remove reference and rvalue (e.g. lambda, std::function, etc.).
|
||||||
struct ArgsOf<TReturn(Args...)> : public ArgsTraits<Args...>
|
template<typename T>
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename TReturn, typename... Args>
|
|
||||||
struct ArgsOf<TReturn(*)(Args...)> : public ArgsTraits<Args...>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T, typename TReturn, typename... Args>
|
|
||||||
struct ArgsOf<TReturn(T::*)(Args...)> : public ArgsTraits<Args...>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T, typename TReturn, typename... Args>
|
|
||||||
struct ArgsOf<TReturn(T::*)(Args...) const> : public ArgsTraits<Args...>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T, typename TReturn, typename... Args>
|
|
||||||
struct ArgsOf<TReturn(T::*)(Args...) volatile> : public ArgsTraits<Args...>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T, typename TReturn, typename... Args>
|
|
||||||
struct ArgsOf<TReturn(T::*)(Args...) const volatile> : public ArgsTraits<Args...>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct ArgsOf<std::function<T>> : public ArgsOf<T>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct ArgsOf<T&> : public ArgsOf<T>
|
struct ArgsOf<T&> : public ArgsOf<T>
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
struct ArgsOf<const T&> : public ArgsOf<T>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct ArgsOf<volatile T&> : public ArgsOf<T>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct ArgsOf<const volatile T&> : public ArgsOf<T>
|
|
||||||
{ };
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct ArgsOf<T&&> : public ArgsOf<T>
|
struct ArgsOf<T&&> : public ArgsOf<T>
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
// Partial specialization for function type.
|
||||||
struct ArgsOf<const T&&> : public ArgsOf<T>
|
template<typename R, typename... Args>
|
||||||
|
struct ArgsOf<R(Args...)> : public ArgsTraits<Args...>
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
// Partial specialization for function pointer.
|
||||||
struct ArgsOf<volatile T&&> : public ArgsOf<T>
|
template<typename R, typename... Args>
|
||||||
|
struct ArgsOf<R (*)(Args...)> : public ArgsTraits<Args...>
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
template <typename T>
|
// Partial specialization for pointer-to-member-function (i.e. operator()'s).
|
||||||
struct ArgsOf<const volatile T&&> : public ArgsOf<T>
|
template<typename R, typename T, typename... Args>
|
||||||
|
struct ArgsOf<R (T::*)(Args...)> : public ArgsTraits<Args...>
|
||||||
|
{ };
|
||||||
|
|
||||||
|
template<typename R, typename T, typename... Args>
|
||||||
|
struct ArgsOf<R (T::*)(Args...) const> : public ArgsTraits<Args...>
|
||||||
|
{ };
|
||||||
|
|
||||||
|
template<typename R, typename T, typename... Args>
|
||||||
|
struct ArgsOf<R (T::*)(Args...) volatile> : public ArgsTraits<Args...>
|
||||||
|
{ };
|
||||||
|
|
||||||
|
template<typename R, typename T, typename... Args>
|
||||||
|
struct ArgsOf<R (T::*)(Args...) const volatile> : public ArgsTraits<Args...>
|
||||||
{ };
|
{ };
|
||||||
|
|
||||||
} // namespace QtPromisePrivate
|
} // namespace QtPromisePrivate
|
||||||
|
@ -1,3 +1,10 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISEHELPERS_H
|
#ifndef QTPROMISE_QPROMISEHELPERS_H
|
||||||
#define QTPROMISE_QPROMISEHELPERS_H
|
#define QTPROMISE_QPROMISEHELPERS_H
|
||||||
|
|
||||||
@ -6,9 +13,8 @@
|
|||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
static inline typename QtPromisePrivate::PromiseDeduce<T>::Type
|
static inline typename QtPromisePrivate::PromiseDeduce<T>::Type resolve(T&& value)
|
||||||
resolve(T&& value)
|
|
||||||
{
|
{
|
||||||
using namespace QtPromisePrivate;
|
using namespace QtPromisePrivate;
|
||||||
using PromiseType = typename PromiseDeduce<T>::Type;
|
using PromiseType = typename PromiseDeduce<T>::Type;
|
||||||
@ -16,95 +22,90 @@ resolve(T&& value)
|
|||||||
using ResolveType = QPromiseResolve<ValueType>;
|
using ResolveType = QPromiseResolve<ValueType>;
|
||||||
using RejectType = QPromiseReject<ValueType>;
|
using RejectType = QPromiseReject<ValueType>;
|
||||||
|
|
||||||
return PromiseType([&](ResolveType&& resolve, RejectType&& reject) {
|
return PromiseType{[&](ResolveType&& resolve, RejectType&& reject) {
|
||||||
PromiseFulfill<Unqualified<T>>::call(
|
PromiseFulfill<Unqualified<T>>::call(std::forward<T>(value),
|
||||||
std::forward<T>(value),
|
std::forward<ResolveType>(resolve),
|
||||||
std::forward<ResolveType>(resolve),
|
std::forward<RejectType>(reject));
|
||||||
std::forward<RejectType>(reject));
|
}};
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
static inline QPromise<T>
|
static inline QPromise<T> resolve(QPromise<T> value)
|
||||||
resolve(QPromise<T> value)
|
|
||||||
{
|
{
|
||||||
return std::move(value);
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline QPromise<void>
|
static inline QPromise<void> resolve()
|
||||||
resolve()
|
|
||||||
{
|
{
|
||||||
return QPromise<void>([](const QPromiseResolve<void>& resolve) {
|
return QPromise<void>{[](const QPromiseResolve<void>& resolve) {
|
||||||
resolve();
|
resolve();
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, template <typename, typename...> class Sequence = QVector, typename ...Args>
|
template<typename T, template<typename, typename...> class Sequence = QVector, typename... Args>
|
||||||
static inline QPromise<QVector<T>>
|
static inline QPromise<QVector<T>> all(const Sequence<QPromise<T>, Args...>& promises)
|
||||||
all(const Sequence<QPromise<T>, Args...>& promises)
|
|
||||||
{
|
{
|
||||||
const int count = static_cast<int>(promises.size());
|
const int count = static_cast<int>(promises.size());
|
||||||
if (count == 0) {
|
if (count == 0) {
|
||||||
return QtPromise::resolve(QVector<T>{});
|
return QtPromise::resolve(QVector<T>{});
|
||||||
}
|
}
|
||||||
|
|
||||||
return QPromise<QVector<T>>([=](
|
return QPromise<QVector<T>>{
|
||||||
const QPromiseResolve<QVector<T>>& resolve,
|
[=](const QPromiseResolve<QVector<T>>& resolve, const QPromiseReject<QVector<T>>& reject) {
|
||||||
const QPromiseReject<QVector<T>>& reject) {
|
auto remaining = QSharedPointer<int>::create(count);
|
||||||
|
auto results = QSharedPointer<QVector<T>>::create(count);
|
||||||
|
|
||||||
QSharedPointer<int> remaining(new int(count));
|
int i = 0;
|
||||||
QSharedPointer<QVector<T>> results(new QVector<T>(count));
|
for (const auto& promise : promises) {
|
||||||
|
promise.then(
|
||||||
|
[=](const T& res) mutable {
|
||||||
|
(*results)[i] = res;
|
||||||
|
if (--(*remaining) == 0) {
|
||||||
|
resolve(*results);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[=]() mutable {
|
||||||
|
if (*remaining != -1) {
|
||||||
|
*remaining = -1;
|
||||||
|
reject(std::current_exception());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
int i = 0;
|
i++;
|
||||||
for (const auto& promise: promises) {
|
}
|
||||||
promise.then([=](const T& res) mutable {
|
}};
|
||||||
(*results)[i] = res;
|
|
||||||
if (--(*remaining) == 0) {
|
|
||||||
resolve(*results);
|
|
||||||
}
|
|
||||||
}, [=]() mutable {
|
|
||||||
if (*remaining != -1) {
|
|
||||||
*remaining = -1;
|
|
||||||
reject(std::current_exception());
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <template <typename, typename...> class Sequence = QVector, typename ...Args>
|
template<template<typename, typename...> class Sequence = QVector, typename... Args>
|
||||||
static inline QPromise<void>
|
static inline QPromise<void> all(const Sequence<QPromise<void>, Args...>& promises)
|
||||||
all(const Sequence<QPromise<void>, Args...>& promises)
|
|
||||||
{
|
{
|
||||||
const int count = static_cast<int>(promises.size());
|
const int count = static_cast<int>(promises.size());
|
||||||
if (count == 0) {
|
if (count == 0) {
|
||||||
return QtPromise::resolve();
|
return QtPromise::resolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
return QPromise<void>([=](
|
return QPromise<void>{
|
||||||
const QPromiseResolve<void>& resolve,
|
[=](const QPromiseResolve<void>& resolve, const QPromiseReject<void>& reject) {
|
||||||
const QPromiseReject<void>& reject) {
|
auto remaining = QSharedPointer<int>::create(count);
|
||||||
|
|
||||||
QSharedPointer<int> remaining(new int(count));
|
for (const auto& promise : promises) {
|
||||||
|
promise.then(
|
||||||
for (const auto& promise: promises) {
|
[=]() {
|
||||||
promise.then([=]() {
|
if (--(*remaining) == 0) {
|
||||||
if (--(*remaining) == 0) {
|
resolve();
|
||||||
resolve();
|
}
|
||||||
}
|
},
|
||||||
}, [=]() {
|
[=]() {
|
||||||
if (*remaining != -1) {
|
if (*remaining != -1) {
|
||||||
*remaining = -1;
|
*remaining = -1;
|
||||||
reject(std::current_exception());
|
reject(std::current_exception());
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Functor, typename... Args>
|
template<typename Functor, typename... Args>
|
||||||
static inline typename QtPromisePrivate::PromiseFunctor<Functor, Args...>::PromiseType
|
static inline typename QtPromisePrivate::PromiseFunctor<Functor, Args...>::PromiseType
|
||||||
attempt(Functor&& fn, Args&&... args)
|
attempt(Functor&& fn, Args&&... args)
|
||||||
{
|
{
|
||||||
@ -120,62 +121,57 @@ attempt(Functor&& fn, Args&&... args)
|
|||||||
using ResolveType = QPromiseResolve<ValueType>;
|
using ResolveType = QPromiseResolve<ValueType>;
|
||||||
using RejectType = QPromiseReject<ValueType>;
|
using RejectType = QPromiseReject<ValueType>;
|
||||||
|
|
||||||
return PromiseType(
|
return PromiseType{[&](ResolveType&& resolve, RejectType&& reject) {
|
||||||
[&](ResolveType&& resolve, RejectType&& reject) {
|
PromiseDispatch<typename FunctorType::ResultType>::call(std::forward<ResolveType>(resolve),
|
||||||
PromiseDispatch<typename FunctorType::ResultType>::call(
|
std::forward<RejectType>(reject),
|
||||||
std::forward<ResolveType>(resolve),
|
std::forward<Functor>(fn),
|
||||||
std::forward<RejectType>(reject),
|
std::forward<Args>(args)...);
|
||||||
std::forward<Functor>(fn),
|
}};
|
||||||
std::forward<Args>(args)...);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Sender, typename Signal>
|
template<typename Sender, typename Signal>
|
||||||
static inline typename QtPromisePrivate::PromiseFromSignal<Signal>
|
static inline typename QtPromisePrivate::PromiseFromSignal<Signal>
|
||||||
connect(const Sender* sender, Signal signal)
|
connect(const Sender* sender, Signal signal)
|
||||||
{
|
{
|
||||||
using namespace QtPromisePrivate;
|
using namespace QtPromisePrivate;
|
||||||
using T = typename PromiseFromSignal<Signal>::Type;
|
using T = typename PromiseFromSignal<Signal>::Type;
|
||||||
|
|
||||||
return QPromise<T>(
|
return QPromise<T>{[&](const QPromiseResolve<T>& resolve, const QPromiseReject<T>& reject) {
|
||||||
[&](const QPromiseResolve<T>& resolve, const QPromiseReject<T>& reject) {
|
QPromiseConnections connections;
|
||||||
QPromiseConnections connections;
|
connectSignalToResolver(connections, resolve, sender, signal);
|
||||||
connectSignalToResolver(connections, resolve, sender, signal);
|
connectDestroyedToReject(connections, reject, sender);
|
||||||
connectDestroyedToReject(connections, reject, sender);
|
}};
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FSender, typename FSignal, typename RSender, typename RSignal>
|
template<typename FSender, typename FSignal, typename RSender, typename RSignal>
|
||||||
static inline typename QtPromisePrivate::PromiseFromSignal<FSignal>
|
static inline typename QtPromisePrivate::PromiseFromSignal<FSignal>
|
||||||
connect(const FSender* fsender, FSignal fsignal, const RSender* rsender, RSignal rsignal)
|
connect(const FSender* fsender, FSignal fsignal, const RSender* rsender, RSignal rsignal)
|
||||||
{
|
{
|
||||||
using namespace QtPromisePrivate;
|
using namespace QtPromisePrivate;
|
||||||
using T = typename PromiseFromSignal<FSignal>::Type;
|
using T = typename PromiseFromSignal<FSignal>::Type;
|
||||||
|
|
||||||
return QPromise<T>(
|
return QPromise<T>{[&](const QPromiseResolve<T>& resolve, const QPromiseReject<T>& reject) {
|
||||||
[&](const QPromiseResolve<T>& resolve, const QPromiseReject<T>& reject) {
|
QPromiseConnections connections;
|
||||||
QPromiseConnections connections;
|
connectSignalToResolver(connections, resolve, fsender, fsignal);
|
||||||
connectSignalToResolver(connections, resolve, fsender, fsignal);
|
connectSignalToResolver(connections, reject, rsender, rsignal);
|
||||||
connectSignalToResolver(connections, reject, rsender, rsignal);
|
connectDestroyedToReject(connections, reject, fsender);
|
||||||
connectDestroyedToReject(connections, reject, fsender);
|
}};
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Sender, typename FSignal, typename RSignal>
|
template<typename Sender, typename FSignal, typename RSignal>
|
||||||
static inline typename QtPromisePrivate::PromiseFromSignal<FSignal>
|
static inline typename QtPromisePrivate::PromiseFromSignal<FSignal>
|
||||||
connect(const Sender* sender, FSignal fsignal, RSignal rsignal)
|
connect(const Sender* sender, FSignal fsignal, RSignal rsignal)
|
||||||
{
|
{
|
||||||
return connect(sender, fsignal, sender, rsignal);
|
return connect(sender, fsignal, sender, rsignal);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Sequence, typename Functor>
|
template<typename Sequence, typename Functor>
|
||||||
static inline QPromise<Sequence>
|
static inline QPromise<Sequence> each(const Sequence& values, Functor&& fn)
|
||||||
each(const Sequence& values, Functor&& fn)
|
|
||||||
{
|
{
|
||||||
return QPromise<Sequence>::resolve(values).each(std::forward<Functor>(fn));
|
return QPromise<Sequence>::resolve(values).each(std::forward<Functor>(fn));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Sequence, typename Functor>
|
template<typename Sequence, typename Functor>
|
||||||
static inline typename QtPromisePrivate::PromiseMapper<Sequence, Functor>::PromiseType
|
static inline typename QtPromisePrivate::PromiseMapper<Sequence, Functor>::PromiseType
|
||||||
map(const Sequence& values, Functor fn)
|
map(const Sequence& values, Functor fn)
|
||||||
{
|
{
|
||||||
@ -188,11 +184,10 @@ map(const Sequence& values, Functor fn)
|
|||||||
|
|
||||||
std::vector<QPromise<ResType>> promises;
|
std::vector<QPromise<ResType>> promises;
|
||||||
for (const auto& v : values) {
|
for (const auto& v : values) {
|
||||||
promises.push_back(QPromise<ResType>([&](
|
promises.push_back(QPromise<ResType>{
|
||||||
const QPromiseResolve<ResType>& resolve,
|
[&](const QPromiseResolve<ResType>& resolve, const QPromiseReject<ResType>& reject) {
|
||||||
const QPromiseReject<ResType>& reject) {
|
|
||||||
PromiseFulfill<RetType>::call(fn(v, i), resolve, reject);
|
PromiseFulfill<RetType>::call(fn(v, i), resolve, reject);
|
||||||
}));
|
}});
|
||||||
|
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
@ -200,28 +195,30 @@ map(const Sequence& values, Functor fn)
|
|||||||
return QtPromise::all(promises);
|
return QtPromise::all(promises);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Sequence, typename Functor>
|
template<typename Sequence, typename Functor>
|
||||||
static inline QPromise<Sequence>
|
static inline QPromise<Sequence> filter(const Sequence& values, Functor fn)
|
||||||
filter(const Sequence& values, Functor fn)
|
|
||||||
{
|
{
|
||||||
return QtPromise::map(values, fn)
|
return QtPromise::map(values, fn).then([=](const QVector<bool>& filters) {
|
||||||
.then([=](const QVector<bool>& filters) {
|
Sequence filtered;
|
||||||
Sequence filtered;
|
|
||||||
|
|
||||||
auto filter = filters.begin();
|
auto filter = filters.begin();
|
||||||
for (auto& value : values) {
|
for (auto& value : values) {
|
||||||
if (*filter) {
|
if (*filter) {
|
||||||
filtered.push_back(std::move(value));
|
filtered.push_back(std::move(value));
|
||||||
}
|
|
||||||
|
|
||||||
filter++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return filtered;
|
filter++;
|
||||||
});
|
}
|
||||||
|
|
||||||
|
return filtered;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, template <typename...> class Sequence = QVector, typename Reducer, typename Input, typename ...Args>
|
template<typename T,
|
||||||
|
template<typename...> class Sequence = QVector,
|
||||||
|
typename Reducer,
|
||||||
|
typename Input,
|
||||||
|
typename... Args>
|
||||||
static inline typename QtPromisePrivate::PromiseDeduce<Input>::Type
|
static inline typename QtPromisePrivate::PromiseDeduce<Input>::Type
|
||||||
reduce(const Sequence<T, Args...>& values, Reducer fn, Input initial)
|
reduce(const Sequence<T, Args...>& values, Reducer fn, Input initial)
|
||||||
{
|
{
|
||||||
@ -246,7 +243,10 @@ reduce(const Sequence<T, Args...>& values, Reducer fn, Input initial)
|
|||||||
return promise;
|
return promise;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, template <typename...> class Sequence = QVector, typename Reducer, typename ...Args>
|
template<typename T,
|
||||||
|
template<typename...> class Sequence = QVector,
|
||||||
|
typename Reducer,
|
||||||
|
typename... Args>
|
||||||
static inline typename QtPromisePrivate::PromiseDeduce<T>::Type
|
static inline typename QtPromisePrivate::PromiseDeduce<T>::Type
|
||||||
reduce(const Sequence<T, Args...>& values, Reducer fn)
|
reduce(const Sequence<T, Args...>& values, Reducer fn)
|
||||||
{
|
{
|
||||||
@ -276,19 +276,17 @@ reduce(const Sequence<T, Args...>& values, Reducer fn)
|
|||||||
|
|
||||||
// DEPRECATIONS (remove at version 1)
|
// DEPRECATIONS (remove at version 1)
|
||||||
|
|
||||||
template <typename... Args>
|
template<typename... Args>
|
||||||
Q_DECL_DEPRECATED_X("Use QtPromise::resolve instead")
|
Q_DECL_DEPRECATED_X("Use QtPromise::resolve instead")
|
||||||
static inline auto
|
static inline auto qPromise(Args&&... args)
|
||||||
qPromise(Args&&... args)
|
|
||||||
-> decltype(QtPromise::resolve(std::forward<Args>(args)...))
|
-> decltype(QtPromise::resolve(std::forward<Args>(args)...))
|
||||||
{
|
{
|
||||||
return QtPromise::resolve(std::forward<Args>(args)...);
|
return QtPromise::resolve(std::forward<Args>(args)...);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename... Args>
|
template<typename... Args>
|
||||||
Q_DECL_DEPRECATED_X("Use QtPromise::all instead")
|
Q_DECL_DEPRECATED_X("Use QtPromise::all instead")
|
||||||
static inline auto
|
static inline auto qPromiseAll(Args&&... args)
|
||||||
qPromiseAll(Args&&... args)
|
|
||||||
-> decltype(QtPromise::all(std::forward<Args>(args)...))
|
-> decltype(QtPromise::all(std::forward<Args>(args)...))
|
||||||
{
|
{
|
||||||
return QtPromise::all(std::forward<Args>(args)...);
|
return QtPromise::all(std::forward<Args>(args)...);
|
||||||
|
@ -1,3 +1,10 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISEHELPERS_P_H
|
#ifndef QTPROMISE_QPROMISEHELPERS_P_H
|
||||||
#define QTPROMISE_QPROMISEHELPERS_P_H
|
#define QTPROMISE_QPROMISEHELPERS_P_H
|
||||||
|
|
||||||
@ -9,17 +16,16 @@ namespace QtPromisePrivate {
|
|||||||
// TODO: Suppress QPrivateSignal trailing private signal args
|
// TODO: Suppress QPrivateSignal trailing private signal args
|
||||||
// TODO: Support deducing tuple from args (might require MSVC2017)
|
// TODO: Support deducing tuple from args (might require MSVC2017)
|
||||||
|
|
||||||
template <typename Signal>
|
template<typename Signal>
|
||||||
using PromiseFromSignal = typename QtPromise::QPromise<Unqualified<typename ArgsOf<Signal>::first>>;
|
using PromiseFromSignal = typename QtPromise::QPromise<Unqualified<typename ArgsOf<Signal>::first>>;
|
||||||
|
|
||||||
// Connect signal() to QPromiseResolve
|
// Connect signal() to QPromiseResolve
|
||||||
template <typename Sender, typename Signal>
|
template<typename Sender, typename Signal>
|
||||||
typename std::enable_if<(ArgsOf<Signal>::count == 0)>::type
|
typename std::enable_if<(ArgsOf<Signal>::count == 0)>::type
|
||||||
connectSignalToResolver(
|
connectSignalToResolver(const QtPromise::QPromiseConnections& connections,
|
||||||
const QtPromise::QPromiseConnections& connections,
|
const QtPromise::QPromiseResolve<void>& resolve,
|
||||||
const QtPromise::QPromiseResolve<void>& resolve,
|
const Sender* sender,
|
||||||
const Sender* sender,
|
Signal signal)
|
||||||
Signal signal)
|
|
||||||
{
|
{
|
||||||
connections << QObject::connect(sender, signal, [=]() {
|
connections << QObject::connect(sender, signal, [=]() {
|
||||||
connections.disconnect();
|
connections.disconnect();
|
||||||
@ -28,28 +34,26 @@ connectSignalToResolver(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Connect signal() to QPromiseReject
|
// Connect signal() to QPromiseReject
|
||||||
template <typename T, typename Sender, typename Signal>
|
template<typename T, typename Sender, typename Signal>
|
||||||
typename std::enable_if<(ArgsOf<Signal>::count == 0)>::type
|
typename std::enable_if<(ArgsOf<Signal>::count == 0)>::type
|
||||||
connectSignalToResolver(
|
connectSignalToResolver(const QtPromise::QPromiseConnections& connections,
|
||||||
const QtPromise::QPromiseConnections& connections,
|
const QtPromise::QPromiseReject<T>& reject,
|
||||||
const QtPromise::QPromiseReject<T>& reject,
|
const Sender* sender,
|
||||||
const Sender* sender,
|
Signal signal)
|
||||||
Signal signal)
|
|
||||||
{
|
{
|
||||||
connections << QObject::connect(sender, signal, [=]() {
|
connections << QObject::connect(sender, signal, [=]() {
|
||||||
connections.disconnect();
|
connections.disconnect();
|
||||||
reject(QtPromise::QPromiseUndefinedException());
|
reject(QtPromise::QPromiseUndefinedException{});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Connect signal(args...) to QPromiseResolve
|
// Connect signal(args...) to QPromiseResolve
|
||||||
template <typename T, typename Sender, typename Signal>
|
template<typename T, typename Sender, typename Signal>
|
||||||
typename std::enable_if<(ArgsOf<Signal>::count >= 1)>::type
|
typename std::enable_if<(ArgsOf<Signal>::count >= 1)>::type
|
||||||
connectSignalToResolver(
|
connectSignalToResolver(const QtPromise::QPromiseConnections& connections,
|
||||||
const QtPromise::QPromiseConnections& connections,
|
const QtPromise::QPromiseResolve<T>& resolve,
|
||||||
const QtPromise::QPromiseResolve<T>& resolve,
|
const Sender* sender,
|
||||||
const Sender* sender,
|
Signal signal)
|
||||||
Signal signal)
|
|
||||||
{
|
{
|
||||||
connections << QObject::connect(sender, signal, [=](const T& value) {
|
connections << QObject::connect(sender, signal, [=](const T& value) {
|
||||||
connections.disconnect();
|
connections.disconnect();
|
||||||
@ -58,13 +62,12 @@ connectSignalToResolver(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Connect signal(args...) to QPromiseReject
|
// Connect signal(args...) to QPromiseReject
|
||||||
template <typename T, typename Sender, typename Signal>
|
template<typename T, typename Sender, typename Signal>
|
||||||
typename std::enable_if<(ArgsOf<Signal>::count >= 1)>::type
|
typename std::enable_if<(ArgsOf<Signal>::count >= 1)>::type
|
||||||
connectSignalToResolver(
|
connectSignalToResolver(const QtPromise::QPromiseConnections& connections,
|
||||||
const QtPromise::QPromiseConnections& connections,
|
const QtPromise::QPromiseReject<T>& reject,
|
||||||
const QtPromise::QPromiseReject<T>& reject,
|
const Sender* sender,
|
||||||
const Sender* sender,
|
Signal signal)
|
||||||
Signal signal)
|
|
||||||
{
|
{
|
||||||
using V = Unqualified<typename ArgsOf<Signal>::first>;
|
using V = Unqualified<typename ArgsOf<Signal>::first>;
|
||||||
connections << QObject::connect(sender, signal, [=](const V& value) {
|
connections << QObject::connect(sender, signal, [=](const V& value) {
|
||||||
@ -74,15 +77,14 @@ connectSignalToResolver(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Connect QObject::destroyed signal to QPromiseReject
|
// Connect QObject::destroyed signal to QPromiseReject
|
||||||
template <typename T, typename Sender>
|
template<typename T, typename Sender>
|
||||||
void connectDestroyedToReject(
|
void connectDestroyedToReject(const QtPromise::QPromiseConnections& connections,
|
||||||
const QtPromise::QPromiseConnections& connections,
|
const QtPromise::QPromiseReject<T>& reject,
|
||||||
const QtPromise::QPromiseReject<T>& reject,
|
const Sender* sender)
|
||||||
const Sender* sender)
|
|
||||||
{
|
{
|
||||||
connections << QObject::connect(sender, &QObject::destroyed, [=]() {
|
connections << QObject::connect(sender, &QObject::destroyed, [=]() {
|
||||||
connections.disconnect();
|
connections.disconnect();
|
||||||
reject(QtPromise::QPromiseContextException());
|
reject(QtPromise::QPromiseContextException{});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,30 +1,36 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef QTPROMISE_QPROMISERESOLVER_H
|
#ifndef QTPROMISE_QPROMISERESOLVER_H
|
||||||
#define QTPROMISE_QPROMISERESOLVER_H
|
#define QTPROMISE_QPROMISERESOLVER_H
|
||||||
|
|
||||||
#include "qpromiseexceptions.h"
|
#include "qpromiseexceptions.h"
|
||||||
|
|
||||||
// Qt
|
#include <QtCore/QExplicitlySharedDataPointer>
|
||||||
#include <QExplicitlySharedDataPointer>
|
|
||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
template <typename T> class QPromise;
|
template<typename T>
|
||||||
|
class QPromise;
|
||||||
|
|
||||||
} // namespace QtPromise
|
} // namespace QtPromise
|
||||||
|
|
||||||
namespace QtPromisePrivate {
|
namespace QtPromisePrivate {
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
class PromiseResolver
|
class PromiseResolver
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
PromiseResolver(QtPromise::QPromise<T> promise)
|
PromiseResolver(QtPromise::QPromise<T> promise) : m_d{new Data{}}
|
||||||
: m_d(new Data())
|
|
||||||
{
|
{
|
||||||
m_d->promise = new QtPromise::QPromise<T>(std::move(promise));
|
m_d->promise = new QtPromise::QPromise<T>{std::move(promise)};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename E>
|
template<typename E>
|
||||||
void reject(E&& error)
|
void reject(E&& error)
|
||||||
{
|
{
|
||||||
auto promise = m_d->promise;
|
auto promise = m_d->promise;
|
||||||
@ -41,13 +47,13 @@ public:
|
|||||||
auto promise = m_d->promise;
|
auto promise = m_d->promise;
|
||||||
if (promise) {
|
if (promise) {
|
||||||
Q_ASSERT(promise->isPending());
|
Q_ASSERT(promise->isPending());
|
||||||
promise->m_d->reject(QtPromise::QPromiseUndefinedException());
|
promise->m_d->reject(QtPromise::QPromiseUndefinedException{});
|
||||||
promise->m_d->dispatch();
|
promise->m_d->dispatch();
|
||||||
release();
|
release();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename V>
|
template<typename V>
|
||||||
void resolve(V&& value)
|
void resolve(V&& value)
|
||||||
{
|
{
|
||||||
auto promise = m_d->promise;
|
auto promise = m_d->promise;
|
||||||
@ -87,51 +93,43 @@ private:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} // QtPromisePrivate
|
} // namespace QtPromisePrivate
|
||||||
|
|
||||||
namespace QtPromise {
|
namespace QtPromise {
|
||||||
|
|
||||||
template <class T>
|
template<class T>
|
||||||
class QPromiseResolve
|
class QPromiseResolve
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
QPromiseResolve(QtPromisePrivate::PromiseResolver<T> resolver)
|
QPromiseResolve(QtPromisePrivate::PromiseResolver<T> resolver) : m_resolver{std::move(resolver)}
|
||||||
: m_resolver(std::move(resolver))
|
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
template <typename V>
|
template<typename V>
|
||||||
void operator()(V&& value) const
|
void operator()(V&& value) const
|
||||||
{
|
{
|
||||||
m_resolver.resolve(std::forward<V>(value));
|
m_resolver.resolve(std::forward<V>(value));
|
||||||
}
|
}
|
||||||
|
|
||||||
void operator()() const
|
void operator()() const { m_resolver.resolve(); }
|
||||||
{
|
|
||||||
m_resolver.resolve();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
mutable QtPromisePrivate::PromiseResolver<T> m_resolver;
|
mutable QtPromisePrivate::PromiseResolver<T> m_resolver;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <class T>
|
template<class T>
|
||||||
class QPromiseReject
|
class QPromiseReject
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
QPromiseReject(QtPromisePrivate::PromiseResolver<T> resolver)
|
QPromiseReject(QtPromisePrivate::PromiseResolver<T> resolver) : m_resolver{std::move(resolver)}
|
||||||
: m_resolver(std::move(resolver))
|
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
template <typename E>
|
template<typename E>
|
||||||
void operator()(E&& error) const
|
void operator()(E&& error) const
|
||||||
{
|
{
|
||||||
m_resolver.reject(std::forward<E>(error));
|
m_resolver.reject(std::forward<E>(error));
|
||||||
}
|
}
|
||||||
|
|
||||||
void operator()() const
|
void operator()() const { m_resolver.reject(); }
|
||||||
{
|
|
||||||
m_resolver.reject();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
mutable QtPromisePrivate::PromiseResolver<T> m_resolver;
|
mutable QtPromisePrivate::PromiseResolver<T> m_resolver;
|
||||||
|
@ -1,11 +0,0 @@
|
|||||||
HEADERS += \
|
|
||||||
$$PWD/qpromise.h \
|
|
||||||
$$PWD/qpromise.inl \
|
|
||||||
$$PWD/qpromise_p.h \
|
|
||||||
$$PWD/qpromiseconnections.h \
|
|
||||||
$$PWD/qpromiseexceptions.h \
|
|
||||||
$$PWD/qpromisefuture.h \
|
|
||||||
$$PWD/qpromiseglobal.h \
|
|
||||||
$$PWD/qpromisehelpers.h \
|
|
||||||
$$PWD/qpromisehelpers_p.h \
|
|
||||||
$$PWD/qpromiseresolver.h
|
|
@ -1,5 +0,0 @@
|
|||||||
TEMPLATE = lib
|
|
||||||
CONFIG += c++11 qt thread warn_on
|
|
||||||
DEFINES += QT_DEPRECATED_WARNINGS
|
|
||||||
|
|
||||||
include(qtpromise.pri)
|
|
@ -1,2 +0,0 @@
|
|||||||
TEMPLATE = subdirs
|
|
||||||
SUBDIRS = qtpromise
|
|
9
tests/CMakeLists.txt
Normal file
9
tests/CMakeLists.txt
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
find_package(Qt${QT_VERSION_MAJOR} REQUIRED
|
||||||
|
COMPONENTS
|
||||||
|
Concurrent
|
||||||
|
Test
|
||||||
|
)
|
||||||
|
|
||||||
|
include(QtPromiseAddTest)
|
||||||
|
|
||||||
|
add_subdirectory(auto)
|
1
tests/auto/CMakeLists.txt
Normal file
1
tests/auto/CMakeLists.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
add_subdirectory(qtpromise)
|
@ -1,2 +0,0 @@
|
|||||||
TEMPLATE = subdirs
|
|
||||||
SUBDIRS += qtpromise
|
|
13
tests/auto/qtpromise/CMakeLists.txt
Normal file
13
tests/auto/qtpromise/CMakeLists.txt
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
add_subdirectory(shared)
|
||||||
|
|
||||||
|
add_subdirectory(benchmark)
|
||||||
|
add_subdirectory(cpp14)
|
||||||
|
add_subdirectory(deprecations)
|
||||||
|
add_subdirectory(exceptions)
|
||||||
|
add_subdirectory(future)
|
||||||
|
add_subdirectory(helpers)
|
||||||
|
add_subdirectory(internals)
|
||||||
|
add_subdirectory(qpromise)
|
||||||
|
add_subdirectory(qpromiseconnections)
|
||||||
|
add_subdirectory(requirements)
|
||||||
|
add_subdirectory(thread)
|
4
tests/auto/qtpromise/benchmark/CMakeLists.txt
Normal file
4
tests/auto/qtpromise/benchmark/CMakeLists.txt
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
qtpromise_add_test(benchmark
|
||||||
|
SOURCES
|
||||||
|
tst_benchmark.cpp
|
||||||
|
)
|
@ -1,4 +0,0 @@
|
|||||||
TARGET = tst_benchmark
|
|
||||||
SOURCES += $$PWD/tst_benchmark.cpp
|
|
||||||
|
|
||||||
include(../qtpromise.pri)
|
|
@ -1,21 +1,23 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#include "../shared/data.h"
|
#include "../shared/data.h"
|
||||||
|
|
||||||
// QtPromise
|
|
||||||
#include <QtPromise>
|
#include <QtPromise>
|
||||||
|
|
||||||
// Qt
|
|
||||||
#include <QtTest>
|
#include <QtTest>
|
||||||
|
|
||||||
#ifdef Q_CC_MSVC
|
#ifdef Q_CC_MSVC
|
||||||
// MSVC calls the copy constructor on std::current_exception AND std::rethrow_exception
|
// MSVC calls the copy constructor on std::current_exception AND std::rethrow_exception
|
||||||
// https://stackoverflow.com/a/31820854
|
// https://stackoverflow.com/a/31820854
|
||||||
#define EXCEPT_CALL_COPY_CTOR 1
|
# define EXCEPT_CALL_COPY_CTOR 1
|
||||||
#else
|
#else
|
||||||
#define EXCEPT_CALL_COPY_CTOR 0
|
# define EXCEPT_CALL_COPY_CTOR 0
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
using namespace QtPromise;
|
|
||||||
|
|
||||||
class tst_benchmark : public QObject
|
class tst_benchmark : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@ -37,26 +39,26 @@ QTEST_MAIN(tst_benchmark)
|
|||||||
|
|
||||||
void tst_benchmark::valueResolve()
|
void tst_benchmark::valueResolve()
|
||||||
{
|
{
|
||||||
{ // should move the value when resolved by rvalue
|
{ // should move the value when resolved by rvalue
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>([&](const QPromiseResolve<Data>& resolve) {
|
QtPromise::QPromise<Data>{[&](const QtPromise::QPromiseResolve<Data>& resolve) {
|
||||||
resolve(Data(42));
|
resolve(Data{42});
|
||||||
}).wait();
|
}}.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
QCOMPARE(Data::logs().move, 1); // move value to the promise data
|
QCOMPARE(Data::logs().move, 1); // move value to the promise data
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
}
|
}
|
||||||
{ // should create one copy of the value when resolved by lvalue
|
{ // should create one copy of the value when resolved by lvalue
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>([&](const QPromiseResolve<Data>& resolve) {
|
QtPromise::QPromise<Data>{[&](const QtPromise::QPromiseResolve<Data>& resolve) {
|
||||||
Data value(42);
|
Data value{42};
|
||||||
resolve(value);
|
resolve(value);
|
||||||
}).wait();
|
}}.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 1); // copy value to the promise data
|
QCOMPARE(Data::logs().copy, 1); // copy value to the promise data
|
||||||
QCOMPARE(Data::logs().move, 0);
|
QCOMPARE(Data::logs().move, 0);
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
}
|
}
|
||||||
@ -64,11 +66,12 @@ void tst_benchmark::valueResolve()
|
|||||||
|
|
||||||
void tst_benchmark::valueReject()
|
void tst_benchmark::valueReject()
|
||||||
{
|
{
|
||||||
{ // should not create any data if rejected
|
{ // should not create any data if rejected
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>([&](const QPromiseResolve<Data>&, const QPromiseReject<Data>& reject) {
|
QtPromise::QPromise<Data>{[&](const QtPromise::QPromiseResolve<Data>&,
|
||||||
reject(QString("foo"));
|
const QtPromise::QPromiseReject<Data>& reject) {
|
||||||
}).wait();
|
reject(QString{"foo"});
|
||||||
|
}}.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 0);
|
QCOMPARE(Data::logs().ctor, 0);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
@ -79,57 +82,68 @@ void tst_benchmark::valueReject()
|
|||||||
|
|
||||||
void tst_benchmark::valueThen()
|
void tst_benchmark::valueThen()
|
||||||
{
|
{
|
||||||
{ // should not copy value on continutation if fulfilled
|
{ // should not copy value on continuation if fulfilled
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>::resolve(Data(42)).then([&](const Data& res) {
|
QtPromise::QPromise<Data>::resolve(Data{42})
|
||||||
value = res.value();
|
.then([&](const Data& res) {
|
||||||
}).wait();
|
value = res.value();
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
QCOMPARE(Data::logs().move, 1); // move value to the promise data
|
QCOMPARE(Data::logs().move, 1); // move value to the promise data
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(value, 42);
|
QCOMPARE(value, 42);
|
||||||
}
|
}
|
||||||
{ // should not create value on continutation if rejected
|
{ // should not create value on continuation if rejected
|
||||||
int value = -1;
|
int value = -1;
|
||||||
QString error;
|
QString error;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>::reject(QString("foo")).then([&](const Data& res) {
|
QtPromise::QPromise<Data>::reject(QString{"foo"})
|
||||||
value = res.value();
|
.then(
|
||||||
}, [&](const QString& err) {
|
[&](const Data& res) {
|
||||||
error = err;
|
value = res.value();
|
||||||
}).wait();
|
},
|
||||||
|
[&](const QString& err) {
|
||||||
|
error = err;
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 0);
|
QCOMPARE(Data::logs().ctor, 0);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
QCOMPARE(Data::logs().move, 0);
|
QCOMPARE(Data::logs().move, 0);
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(error, QString("foo"));
|
QCOMPARE(error, QString{"foo"});
|
||||||
QCOMPARE(value, -1);
|
QCOMPARE(value, -1);
|
||||||
}
|
}
|
||||||
{ // should move the returned value when fulfilled
|
{ // should move the returned value when fulfilled
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<int>::resolve(42).then([&](int res) {
|
QtPromise::QPromise<int>::resolve(42)
|
||||||
return Data(res+2);
|
.then([&](int res) {
|
||||||
}).then([&](const Data& res) {
|
return Data{res + 2};
|
||||||
value = res.value();
|
})
|
||||||
}).wait();
|
.then([&](const Data& res) {
|
||||||
|
value = res.value();
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
QCOMPARE(Data::logs().move, 1); // move values to the next promise data
|
QCOMPARE(Data::logs().move, 1); // move values to the next promise data
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(value, 44);
|
QCOMPARE(value, 44);
|
||||||
}
|
}
|
||||||
{ // should not create any data if handler throws
|
{ // should not create any data if handler throws
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<int>::resolve(42).then([&](int res) {
|
QtPromise::QPromise<int>::resolve(42)
|
||||||
throw QString("foo");
|
.then([&](int res) {
|
||||||
return Data(res+2);
|
throw QString{"foo"};
|
||||||
}).wait();
|
return Data{res + 2};
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 0);
|
QCOMPARE(Data::logs().ctor, 0);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
@ -140,26 +154,31 @@ void tst_benchmark::valueThen()
|
|||||||
|
|
||||||
void tst_benchmark::valueDelayed()
|
void tst_benchmark::valueDelayed()
|
||||||
{
|
{
|
||||||
{ // should not copy the value on continutation if fulfilled
|
{ // should not copy the value on continuation if fulfilled
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<int>::resolve(42).then([&](int res) {
|
QtPromise::QPromise<int>::resolve(42)
|
||||||
return QPromise<Data>::resolve(Data(res + 1));
|
.then([&](int res) {
|
||||||
}).then([&](const Data& res) {
|
return QtPromise::QPromise<Data>::resolve(Data{res + 1});
|
||||||
value = res.value();
|
})
|
||||||
}).wait();
|
.then([&](const Data& res) {
|
||||||
|
value = res.value();
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
QCOMPARE(Data::logs().move, 1); // move value to the input promise data
|
QCOMPARE(Data::logs().move, 1); // move value to the input promise data
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(value, 43);
|
QCOMPARE(value, 43);
|
||||||
}
|
}
|
||||||
{ // should not create value on continutation if rejected
|
{ // should not create value on continuation if rejected
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<int>::resolve(42).then([&]() {
|
QtPromise::QPromise<int>::resolve(42)
|
||||||
return QPromise<Data>::reject(QString("foo"));
|
.then([&]() {
|
||||||
}).wait();
|
return QtPromise::QPromise<Data>::reject(QString{"foo"});
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 0);
|
QCOMPARE(Data::logs().ctor, 0);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
@ -170,25 +189,29 @@ void tst_benchmark::valueDelayed()
|
|||||||
|
|
||||||
void tst_benchmark::valueFinally()
|
void tst_benchmark::valueFinally()
|
||||||
{
|
{
|
||||||
{ // should not copy the value on continutation if fulfilled
|
{ // should not copy the value on continuation if fulfilled
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>::resolve(Data(42)).finally([&]() {
|
QtPromise::QPromise<Data>::resolve(Data{42})
|
||||||
value = 42;
|
.finally([&]() {
|
||||||
}).wait();
|
value = 42;
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
QCOMPARE(Data::logs().move, 1); // move value to the input and output promise data
|
QCOMPARE(Data::logs().move, 1); // move value to the input and output promise data
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(value, 42);
|
QCOMPARE(value, 42);
|
||||||
}
|
}
|
||||||
{ // should not create value on continutation if rejected
|
{ // should not create value on continuation if rejected
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>::reject(QString("foo")).finally([&]() {
|
QtPromise::QPromise<Data>::reject(QString{"foo"})
|
||||||
value = 42;
|
.finally([&]() {
|
||||||
}).wait();
|
value = 42;
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 0);
|
QCOMPARE(Data::logs().ctor, 0);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
@ -200,25 +223,29 @@ void tst_benchmark::valueFinally()
|
|||||||
|
|
||||||
void tst_benchmark::valueTap()
|
void tst_benchmark::valueTap()
|
||||||
{
|
{
|
||||||
{ // should not copy the value on continutation if fulfilled
|
{ // should not copy the value on continuation if fulfilled
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>::resolve(Data(42)).tap([&](const Data& res) {
|
QtPromise::QPromise<Data>::resolve(Data{42})
|
||||||
value = res.value();
|
.tap([&](const Data& res) {
|
||||||
}).wait();
|
value = res.value();
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
QCOMPARE(Data::logs().move, 1); // move value to the input and output promise data
|
QCOMPARE(Data::logs().move, 1); // move value to the input and output promise data
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(value, 42);
|
QCOMPARE(value, 42);
|
||||||
}
|
}
|
||||||
{ // should not create value on continutation if rejected
|
{ // should not create value on continuation if rejected
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<Data>::reject(QString("foo")).tap([&](const Data& res) {
|
QtPromise::QPromise<Data>::reject(QString{"foo"})
|
||||||
value = res.value();
|
.tap([&](const Data& res) {
|
||||||
}).wait();
|
value = res.value();
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 0);
|
QCOMPARE(Data::logs().ctor, 0);
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
@ -230,26 +257,28 @@ void tst_benchmark::valueTap()
|
|||||||
|
|
||||||
void tst_benchmark::errorReject()
|
void tst_benchmark::errorReject()
|
||||||
{
|
{
|
||||||
{ // should create one copy of the error when rejected by rvalue
|
{ // should create one copy of the error when rejected by rvalue
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<int>([&](const QPromiseResolve<int>&, const QPromiseReject<int>& reject) {
|
QtPromise::QPromise<int>{[&](const QtPromise::QPromiseResolve<int>&,
|
||||||
reject(Data(42));
|
const QtPromise::QPromiseReject<int>& reject) {
|
||||||
}).wait();
|
reject(Data{42});
|
||||||
|
}}.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 1 + EXCEPT_CALL_COPY_CTOR); // copy value in std::exception_ptr
|
QCOMPARE(Data::logs().copy, 1 + EXCEPT_CALL_COPY_CTOR); // copy value in std::exception_ptr
|
||||||
QCOMPARE(Data::logs().move, 0);
|
QCOMPARE(Data::logs().move, 0);
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
}
|
}
|
||||||
{ // should create one copy of the error when rejected by lvalue (no extra copy)
|
{ // should create one copy of the error when rejected by lvalue (no extra copy)
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<int>([&](const QPromiseResolve<int>&, const QPromiseReject<int>& reject) {
|
QtPromise::QPromise<int>{[&](const QtPromise::QPromiseResolve<int>&,
|
||||||
Data error(42);
|
const QtPromise::QPromiseReject<int>& reject) {
|
||||||
|
Data error{42};
|
||||||
reject(error);
|
reject(error);
|
||||||
}).wait();
|
}}.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 1 + EXCEPT_CALL_COPY_CTOR); // copy value to the promise data
|
QCOMPARE(Data::logs().copy, 1 + EXCEPT_CALL_COPY_CTOR); // copy value to the promise data
|
||||||
QCOMPARE(Data::logs().move, 0);
|
QCOMPARE(Data::logs().move, 0);
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
}
|
}
|
||||||
@ -257,30 +286,37 @@ void tst_benchmark::errorReject()
|
|||||||
|
|
||||||
void tst_benchmark::errorThen()
|
void tst_benchmark::errorThen()
|
||||||
{
|
{
|
||||||
{ // should not copy error on continutation if rejected
|
{ // should not copy error on continuation if rejected
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<void>::reject(Data(42)).fail([&](const Data& res) {
|
QtPromise::QPromise<void>::reject(Data{42})
|
||||||
value = res.value();
|
.fail([&](const Data& res) {
|
||||||
}).wait();
|
value = res.value();
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 1 + 2 * EXCEPT_CALL_COPY_CTOR); // (initial) copy value in std::exception_ptr
|
QCOMPARE(Data::logs().copy,
|
||||||
|
1 + 2 * EXCEPT_CALL_COPY_CTOR); // (initial) copy value in std::exception_ptr
|
||||||
QCOMPARE(Data::logs().move, 0);
|
QCOMPARE(Data::logs().move, 0);
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(value, 42);
|
QCOMPARE(value, 42);
|
||||||
}
|
}
|
||||||
{ // should not copy error on continutation if rethrown
|
{ // should not copy error on continuation if rethrown
|
||||||
int value = -1;
|
int value = -1;
|
||||||
Data::logs().reset();
|
Data::logs().reset();
|
||||||
QPromise<void>::reject(Data(42)).fail([](const Data&) {
|
QtPromise::QPromise<void>::reject(Data{42})
|
||||||
throw;
|
.fail([](const Data&) {
|
||||||
}).fail([&](const Data& res) {
|
throw;
|
||||||
value = res.value();
|
})
|
||||||
}).wait();
|
.fail([&](const Data& res) {
|
||||||
|
value = res.value();
|
||||||
|
})
|
||||||
|
.wait();
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
QCOMPARE(Data::logs().copy, 1 + 4 * EXCEPT_CALL_COPY_CTOR); // (initial) copy value in std::exception_ptr
|
QCOMPARE(Data::logs().copy,
|
||||||
|
1 + 4 * EXCEPT_CALL_COPY_CTOR); // (initial) copy value in std::exception_ptr
|
||||||
QCOMPARE(Data::logs().move, 0);
|
QCOMPARE(Data::logs().move, 0);
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
QCOMPARE(value, 42);
|
QCOMPARE(value, 42);
|
||||||
|
10
tests/auto/qtpromise/cpp14/CMakeLists.txt
Normal file
10
tests/auto/qtpromise/cpp14/CMakeLists.txt
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
# https://cmake.org/cmake/help/latest/prop_gbl/CMAKE_CXX_KNOWN_FEATURES.html
|
||||||
|
# https://gcc.gnu.org/projects/cxx-status.html#cxx14
|
||||||
|
if ("cxx_generic_lambdas" IN_LIST CMAKE_CXX_COMPILE_FEATURES)
|
||||||
|
set(CMAKE_CXX_STANDARD 14)
|
||||||
|
qtpromise_add_tests(cpp14
|
||||||
|
SOURCES
|
||||||
|
tst_argsof_lambda_auto.cpp
|
||||||
|
tst_resolver_lambda_auto.cpp
|
||||||
|
)
|
||||||
|
endif()
|
33
tests/auto/qtpromise/cpp14/tst_argsof_lambda_auto.cpp
Normal file
33
tests/auto/qtpromise/cpp14/tst_argsof_lambda_auto.cpp
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <QtPromise>
|
||||||
|
#include <QtTest>
|
||||||
|
|
||||||
|
using namespace QtPromisePrivate;
|
||||||
|
|
||||||
|
class tst_cpp14_argsof_lambda_auto : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void lambdaAutoArgs();
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_MAIN(tst_cpp14_argsof_lambda_auto)
|
||||||
|
#include "tst_argsof_lambda_auto.moc"
|
||||||
|
|
||||||
|
void tst_cpp14_argsof_lambda_auto::lambdaAutoArgs()
|
||||||
|
{
|
||||||
|
auto lOneArg = [](auto) {};
|
||||||
|
auto lManyArgs = [](const auto&, auto, auto) {};
|
||||||
|
auto lMutable = [](const auto&, auto) mutable {};
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((ArgsOf<decltype(lOneArg)>::count == 0));
|
||||||
|
Q_STATIC_ASSERT((ArgsOf<decltype(lManyArgs)>::count == 0));
|
||||||
|
Q_STATIC_ASSERT((ArgsOf<decltype(lMutable)>::count == 0));
|
||||||
|
}
|
77
tests/auto/qtpromise/cpp14/tst_resolver_lambda_auto.cpp
Normal file
77
tests/auto/qtpromise/cpp14/tst_resolver_lambda_auto.cpp
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "../shared/utils.h"
|
||||||
|
|
||||||
|
#include <QtPromise>
|
||||||
|
#include <QtTest>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
class tst_cpp14_resolver_lambda_auto : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void resolverTwoAutoArgs();
|
||||||
|
void resolverTwoAutoArgs_void();
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_MAIN(tst_cpp14_resolver_lambda_auto)
|
||||||
|
#include "tst_resolver_lambda_auto.moc"
|
||||||
|
|
||||||
|
void tst_cpp14_resolver_lambda_auto::resolverTwoAutoArgs()
|
||||||
|
{
|
||||||
|
QtPromise::QPromise<int> p0{[](auto resolve, auto reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve(42);
|
||||||
|
}};
|
||||||
|
QtPromise::QPromise<int> p1{[](auto resolve, const auto& reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve(42);
|
||||||
|
}};
|
||||||
|
QtPromise::QPromise<int> p2{[](const auto& resolve, auto reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve(42);
|
||||||
|
}};
|
||||||
|
QtPromise::QPromise<int> p3{[](const auto& resolve, const auto& reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve(42);
|
||||||
|
}};
|
||||||
|
|
||||||
|
for (const auto& p : {p0, p1, p2, p3}) {
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
QCOMPARE(waitForError(p, -1), -1);
|
||||||
|
QCOMPARE(waitForValue(p, -1), 42);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_cpp14_resolver_lambda_auto::resolverTwoAutoArgs_void()
|
||||||
|
{
|
||||||
|
QtPromise::QPromise<void> p0{[](auto resolve, auto reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve();
|
||||||
|
}};
|
||||||
|
QtPromise::QPromise<void> p1{[](auto resolve, const auto& reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve();
|
||||||
|
}};
|
||||||
|
QtPromise::QPromise<void> p2{[](const auto& resolve, auto reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve();
|
||||||
|
}};
|
||||||
|
QtPromise::QPromise<void> p3{[](const auto& resolve, const auto& reject) {
|
||||||
|
Q_UNUSED(reject)
|
||||||
|
resolve();
|
||||||
|
}};
|
||||||
|
|
||||||
|
for (const auto& p : {p0, p1, p2, p3}) {
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
QCOMPARE(waitForError(p, -1), -1);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
}
|
||||||
|
}
|
13
tests/auto/qtpromise/deprecations/CMakeLists.txt
Normal file
13
tests/auto/qtpromise/deprecations/CMakeLists.txt
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Disable deprecation warnings since we are testing deprecated features.
|
||||||
|
if(CMAKE_CXX_COMPILER_ID MATCHES "GNU")
|
||||||
|
add_compile_options(-Wno-deprecated-declarations)
|
||||||
|
elseif(CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
|
||||||
|
add_compile_options(/wd4996)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
qtpromise_add_tests(deprecation
|
||||||
|
SOURCES
|
||||||
|
tst_helpers_qpromise.cpp
|
||||||
|
tst_helpers_qpromiseall.cpp
|
||||||
|
tst_qpromise_all.cpp
|
||||||
|
)
|
@ -1,5 +0,0 @@
|
|||||||
include(../qtpromise.pri)
|
|
||||||
|
|
||||||
DEFINES -= QT_DEPRECATED_WARNINGS
|
|
||||||
gcc:QMAKE_CXXFLAGS += -Wno-deprecated-declarations
|
|
||||||
msvc:QMAKE_CXXFLAGS -= -wd4996
|
|
@ -1,3 +0,0 @@
|
|||||||
TEMPLATE = subdirs
|
|
||||||
SUBDIRS += \
|
|
||||||
helpers
|
|
@ -1,4 +0,0 @@
|
|||||||
TEMPLATE = subdirs
|
|
||||||
SUBDIRS += \
|
|
||||||
qpromise \
|
|
||||||
qpromiseall
|
|
@ -1,5 +0,0 @@
|
|||||||
QT += concurrent
|
|
||||||
TARGET = tst_deprecations_helpers_qpromise
|
|
||||||
SOURCES += $$PWD/tst_qpromise.cpp
|
|
||||||
|
|
||||||
include(../../deprecations.pri)
|
|
@ -1,268 +0,0 @@
|
|||||||
#include "../../../shared/data.h"
|
|
||||||
#include "../../../shared/utils.h"
|
|
||||||
|
|
||||||
// QtPromise
|
|
||||||
#include <QtPromise>
|
|
||||||
|
|
||||||
// Qt
|
|
||||||
#include <QtConcurrent>
|
|
||||||
#include <QtTest>
|
|
||||||
|
|
||||||
// STL
|
|
||||||
#include <memory>
|
|
||||||
|
|
||||||
using namespace QtPromise;
|
|
||||||
|
|
||||||
class tst_deprecations_helpers_qpromise : public QObject
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
private Q_SLOTS:
|
|
||||||
void value();
|
|
||||||
void noValue();
|
|
||||||
void moveRValue();
|
|
||||||
void copyLValue();
|
|
||||||
void qtSharedPtr();
|
|
||||||
void stdSharedPtr();
|
|
||||||
void typedPromise();
|
|
||||||
void voidPromise();
|
|
||||||
void typedFuture();
|
|
||||||
void voidFuture();
|
|
||||||
};
|
|
||||||
|
|
||||||
QTEST_MAIN(tst_deprecations_helpers_qpromise)
|
|
||||||
#include "tst_qpromise.moc"
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::value()
|
|
||||||
{
|
|
||||||
int v0 = 42;
|
|
||||||
const int v1 = 42;
|
|
||||||
|
|
||||||
auto p0 = qPromise(42);
|
|
||||||
auto p1 = qPromise(v0);
|
|
||||||
auto p2 = qPromise(v1);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p0), QPromise<int>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p1), QPromise<int>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p2), QPromise<int>>::value));
|
|
||||||
|
|
||||||
for (const auto& p : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(p.isFulfilled(), true);
|
|
||||||
}
|
|
||||||
for (const auto& p : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(waitForValue(p, -1), 42);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::noValue()
|
|
||||||
{
|
|
||||||
auto p = qPromise();
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p.isFulfilled(), true);
|
|
||||||
QCOMPARE(waitForValue(p, -1, 42), 42);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::moveRValue()
|
|
||||||
{
|
|
||||||
Data::logs().reset();
|
|
||||||
|
|
||||||
{
|
|
||||||
auto p = qPromise(Data(42)).wait();
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<Data>>::value));
|
|
||||||
}
|
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
|
||||||
QCOMPARE(Data::logs().move, 1);
|
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::copyLValue()
|
|
||||||
{
|
|
||||||
Data::logs().reset();
|
|
||||||
|
|
||||||
{
|
|
||||||
Data value(42);
|
|
||||||
auto p = qPromise(value).wait();
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<Data>>::value));
|
|
||||||
}
|
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 1);
|
|
||||||
QCOMPARE(Data::logs().copy, 1);
|
|
||||||
QCOMPARE(Data::logs().move, 0);
|
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://github.com/simonbrunel/qtpromise/issues/6
|
|
||||||
void tst_deprecations_helpers_qpromise::qtSharedPtr()
|
|
||||||
{
|
|
||||||
Data::logs().reset();
|
|
||||||
|
|
||||||
QWeakPointer<Data> wptr;
|
|
||||||
|
|
||||||
{
|
|
||||||
QSharedPointer<Data> sptr0(new Data(42));
|
|
||||||
const QSharedPointer<Data> sptr1 = sptr0;
|
|
||||||
|
|
||||||
auto p0 = qPromise(QSharedPointer<Data>(new Data(42)));
|
|
||||||
auto p1 = qPromise(sptr0);
|
|
||||||
auto p2 = qPromise(sptr1);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p0), QPromise<QSharedPointer<Data>>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p1), QPromise<QSharedPointer<Data>>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p2), QPromise<QSharedPointer<Data>>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(waitForValue(p1, QSharedPointer<Data>()), sptr0);
|
|
||||||
QCOMPARE(waitForValue(p2, QSharedPointer<Data>()), sptr1);
|
|
||||||
|
|
||||||
wptr = sptr0;
|
|
||||||
|
|
||||||
QCOMPARE(wptr.isNull(), false);
|
|
||||||
QCOMPARE(Data::logs().refs, 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
QCOMPARE(wptr.isNull(), true);
|
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 2);
|
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
|
||||||
QCOMPARE(Data::logs().move, 0);
|
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://github.com/simonbrunel/qtpromise/issues/6
|
|
||||||
void tst_deprecations_helpers_qpromise::stdSharedPtr()
|
|
||||||
{
|
|
||||||
Data::logs().reset();
|
|
||||||
|
|
||||||
std::weak_ptr<Data> wptr;
|
|
||||||
|
|
||||||
{
|
|
||||||
std::shared_ptr<Data> sptr0(new Data(42));
|
|
||||||
const std::shared_ptr<Data> sptr1 = sptr0;
|
|
||||||
|
|
||||||
auto p0 = qPromise(std::shared_ptr<Data>(new Data(42)));
|
|
||||||
auto p1 = qPromise(sptr0);
|
|
||||||
auto p2 = qPromise(sptr1);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p0), QPromise<std::shared_ptr<Data>>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p1), QPromise<std::shared_ptr<Data>>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p2), QPromise<std::shared_ptr<Data>>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(waitForValue(p1, std::shared_ptr<Data>()), sptr0);
|
|
||||||
QCOMPARE(waitForValue(p2, std::shared_ptr<Data>()), sptr1);
|
|
||||||
|
|
||||||
wptr = sptr0;
|
|
||||||
|
|
||||||
QCOMPARE(wptr.use_count(), 4l);
|
|
||||||
QCOMPARE(Data::logs().refs, 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
QCOMPARE(wptr.use_count(), 0l);
|
|
||||||
|
|
||||||
QCOMPARE(Data::logs().ctor, 2);
|
|
||||||
QCOMPARE(Data::logs().copy, 0);
|
|
||||||
QCOMPARE(Data::logs().move, 0);
|
|
||||||
QCOMPARE(Data::logs().refs, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::typedPromise()
|
|
||||||
{
|
|
||||||
auto resolver = [](const QPromiseResolve<int>& resolve) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
resolve(42);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
QPromise<int> v0(resolver);
|
|
||||||
const QPromise<int> v1 = v0;
|
|
||||||
|
|
||||||
auto p0 = qPromise(QPromise<int>(resolver));
|
|
||||||
auto p1 = qPromise(v0);
|
|
||||||
auto p2 = qPromise(v1);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p0), QPromise<int>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p1), QPromise<int>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p2), QPromise<int>>::value));
|
|
||||||
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(promise.isPending(), true);
|
|
||||||
}
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(waitForValue(promise, -1), 42);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::voidPromise()
|
|
||||||
{
|
|
||||||
auto resolver = [](const QPromiseResolve<void>& resolve) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
resolve();
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
QPromise<void> v0(resolver);
|
|
||||||
const QPromise<void> v1 = v0;
|
|
||||||
|
|
||||||
auto p0 = qPromise(QPromise<void>(resolver));
|
|
||||||
auto p1 = qPromise(v0);
|
|
||||||
auto p2 = qPromise(v1);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p0), QPromise<void>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p1), QPromise<void>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p2), QPromise<void>>::value));
|
|
||||||
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(promise.isPending(), true);
|
|
||||||
}
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(waitForValue(promise, -1, 42), 42);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::typedFuture()
|
|
||||||
{
|
|
||||||
auto fn = [](){ return 42; };
|
|
||||||
QFuture<int> v0 = QtConcurrent::run(fn);
|
|
||||||
const QFuture<int> v1 = v0;
|
|
||||||
|
|
||||||
auto p0 = qPromise(QtConcurrent::run(fn));
|
|
||||||
auto p1 = qPromise(v0);
|
|
||||||
auto p2 = qPromise(v1);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p0), QPromise<int>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p1), QPromise<int>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p2), QPromise<int>>::value));
|
|
||||||
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(promise.isPending(), true);
|
|
||||||
}
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(waitForValue(promise, -1), 42);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromise::voidFuture()
|
|
||||||
{
|
|
||||||
auto fn = [](){ };
|
|
||||||
QFuture<void> v0 = QtConcurrent::run(fn);
|
|
||||||
const QFuture<void> v1 = v0;
|
|
||||||
|
|
||||||
auto p0 = qPromise(QtConcurrent::run(fn));
|
|
||||||
auto p1 = qPromise(v0);
|
|
||||||
auto p2 = qPromise(v1);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p0), QPromise<void>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p1), QPromise<void>>::value));
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p2), QPromise<void>>::value));
|
|
||||||
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(promise.isPending(), true);
|
|
||||||
}
|
|
||||||
for (const auto& promise : {p0, p1, p2}) {
|
|
||||||
QCOMPARE(waitForValue(promise, -1, 42), 42);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,4 +0,0 @@
|
|||||||
TARGET = tst_deprecations_helpers_qpromiseall
|
|
||||||
SOURCES += $$PWD/tst_qpromiseall.cpp
|
|
||||||
|
|
||||||
include(../../deprecations.pri)
|
|
@ -1,220 +0,0 @@
|
|||||||
#include "../../../shared/utils.h"
|
|
||||||
|
|
||||||
// QtPromise
|
|
||||||
#include <QtPromise>
|
|
||||||
|
|
||||||
// Qt
|
|
||||||
#include <QtTest>
|
|
||||||
|
|
||||||
using namespace QtPromise;
|
|
||||||
|
|
||||||
class tst_deprecations_helpers_qpromiseall : public QObject
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
private Q_SLOTS:
|
|
||||||
void emptySequence();
|
|
||||||
void emptySequence_void();
|
|
||||||
void allPromisesSucceed();
|
|
||||||
void allPromisesSucceed_void();
|
|
||||||
void atLeastOnePromiseReject();
|
|
||||||
void atLeastOnePromiseReject_void();
|
|
||||||
void preserveOrder();
|
|
||||||
void sequenceTypes();
|
|
||||||
void sequenceTypes_void();
|
|
||||||
};
|
|
||||||
|
|
||||||
QTEST_MAIN(tst_deprecations_helpers_qpromiseall)
|
|
||||||
#include "tst_qpromiseall.moc"
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
template <class Sequence>
|
|
||||||
struct SequenceTester
|
|
||||||
{
|
|
||||||
Q_STATIC_ASSERT((std::is_same<typename Sequence::value_type, QPromise<int>>::value));
|
|
||||||
|
|
||||||
static void exec()
|
|
||||||
{
|
|
||||||
Sequence promises{
|
|
||||||
QtPromise::resolve(42),
|
|
||||||
QtPromise::resolve(43),
|
|
||||||
QtPromise::resolve(44)
|
|
||||||
};
|
|
||||||
|
|
||||||
promises.push_back(QtPromise::resolve(45));
|
|
||||||
promises.insert(++promises.begin(), QtPromise::resolve(46));
|
|
||||||
promises.pop_back();
|
|
||||||
|
|
||||||
auto p = qPromiseAll(promises);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>({42, 46, 43, 44}));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <template <typename, typename...> class Sequence, typename ...Args>
|
|
||||||
struct SequenceTester<Sequence<QPromise<void>, Args...>>
|
|
||||||
{
|
|
||||||
static void exec()
|
|
||||||
{
|
|
||||||
Sequence<QPromise<void>, Args...> promises{
|
|
||||||
QtPromise::resolve(),
|
|
||||||
QtPromise::resolve(),
|
|
||||||
QtPromise::resolve()
|
|
||||||
};
|
|
||||||
|
|
||||||
promises.push_back(QtPromise::resolve());
|
|
||||||
promises.insert(++promises.begin(), QtPromise::resolve());
|
|
||||||
promises.pop_back();
|
|
||||||
|
|
||||||
auto p = qPromiseAll(promises);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, -1, 42), 42);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // anonymous namespace
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::emptySequence()
|
|
||||||
{
|
|
||||||
auto p = qPromiseAll(QVector<QPromise<int>>());
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
QCOMPARE(p.isFulfilled(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>({}));
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::emptySequence_void()
|
|
||||||
{
|
|
||||||
auto p = qPromiseAll(QVector<QPromise<void>>());
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
QCOMPARE(p.isFulfilled(), true);
|
|
||||||
QCOMPARE(waitForValue(p, -1, 42), 42);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::allPromisesSucceed()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve(42);
|
|
||||||
auto p1 = QtPromise::resolve(44);
|
|
||||||
auto p2 = QPromise<int>([](const QPromiseResolve<int>& resolve) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
resolve(43);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = qPromiseAll(QVector<QPromise<int>>{p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>({42, 43, 44}));
|
|
||||||
QCOMPARE(p2.isFulfilled(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::allPromisesSucceed_void()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve();
|
|
||||||
auto p1 = QtPromise::resolve();
|
|
||||||
auto p2 = QPromise<void>([](const QPromiseResolve<void>& resolve) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
resolve();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = qPromiseAll(QVector<QPromise<void>>{p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, -1, 42), 42);
|
|
||||||
QCOMPARE(p2.isFulfilled(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::atLeastOnePromiseReject()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve(42);
|
|
||||||
auto p1 = QtPromise::resolve(44);
|
|
||||||
auto p2 = QPromise<int>([](const QPromiseResolve<int>&, const QPromiseReject<int>& reject) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
reject(QString("foo"));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = qPromiseAll(QVector<QPromise<int>>{p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForError(p, QString()), QString("foo"));
|
|
||||||
QCOMPARE(p2.isRejected(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::atLeastOnePromiseReject_void()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve();
|
|
||||||
auto p1 = QtPromise::resolve();
|
|
||||||
auto p2 = QPromise<void>([](const QPromiseResolve<void>&, const QPromiseReject<void>& reject) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
reject(QString("foo"));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = qPromiseAll(QVector<QPromise<void>>{p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForError(p, QString()), QString("foo"));
|
|
||||||
QCOMPARE(p2.isRejected(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::preserveOrder()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve(42).delay(500);
|
|
||||||
auto p1 = QtPromise::resolve(43).delay(100);
|
|
||||||
auto p2 = QtPromise::resolve(44).delay(250);
|
|
||||||
|
|
||||||
auto p = qPromiseAll(QVector<QPromise<int>>{p0, p1, p2});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
QCOMPARE(p0.isPending(), true);
|
|
||||||
QCOMPARE(p1.isPending(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>({42, 43, 44}));
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isFulfilled(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
// QVector::push_back/append isn't supported since it requires a default
|
|
||||||
// constructor (see https://github.com/simonbrunel/qtpromise/issues/3)
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::sequenceTypes()
|
|
||||||
{
|
|
||||||
SequenceTester<QList<QPromise<int>>>::exec();
|
|
||||||
//SequenceTester<QVector<QPromise<int>>>::exec();
|
|
||||||
SequenceTester<std::list<QPromise<int>>>::exec();
|
|
||||||
SequenceTester<std::vector<QPromise<int>>>::exec();
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_helpers_qpromiseall::sequenceTypes_void()
|
|
||||||
{
|
|
||||||
SequenceTester<QList<QPromise<void>>>::exec();
|
|
||||||
//SequenceTester<QVector<QPromise<void>>>::exec();
|
|
||||||
SequenceTester<std::list<QPromise<void>>>::exec();
|
|
||||||
SequenceTester<std::vector<QPromise<void>>>::exec();
|
|
||||||
}
|
|
@ -1,4 +0,0 @@
|
|||||||
TARGET = tst_deprecations_qpromise_all
|
|
||||||
SOURCES += $$PWD/tst_all.cpp
|
|
||||||
|
|
||||||
include(../../deprecations.pri)
|
|
@ -1,229 +0,0 @@
|
|||||||
#include "../../../shared/utils.h"
|
|
||||||
|
|
||||||
// QtPromise
|
|
||||||
#include <QtPromise>
|
|
||||||
|
|
||||||
// Qt
|
|
||||||
#include <QtTest>
|
|
||||||
|
|
||||||
using namespace QtPromise;
|
|
||||||
|
|
||||||
class tst_deprecations_qpromise_all : public QObject
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
private Q_SLOTS:
|
|
||||||
void emptySequence();
|
|
||||||
void emptySequence_void();
|
|
||||||
void allPromisesSucceed();
|
|
||||||
void allPromisesSucceed_void();
|
|
||||||
void atLeastOnePromiseReject();
|
|
||||||
void atLeastOnePromiseReject_void();
|
|
||||||
void preserveOrder();
|
|
||||||
void sequenceTypes();
|
|
||||||
void sequenceTypes_void();
|
|
||||||
};
|
|
||||||
|
|
||||||
QTEST_MAIN(tst_deprecations_qpromise_all)
|
|
||||||
#include "tst_all.moc"
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
template <class Sequence>
|
|
||||||
struct SequenceTester
|
|
||||||
{
|
|
||||||
Q_STATIC_ASSERT((std::is_same<typename Sequence::value_type, QPromise<int>>::value));
|
|
||||||
|
|
||||||
static void exec()
|
|
||||||
{
|
|
||||||
Sequence promises{
|
|
||||||
QtPromise::resolve(42),
|
|
||||||
QtPromise::resolve(43),
|
|
||||||
QtPromise::resolve(44)
|
|
||||||
};
|
|
||||||
|
|
||||||
promises.push_back(QtPromise::resolve(45));
|
|
||||||
promises.insert(++promises.begin(), QtPromise::resolve(46));
|
|
||||||
promises.pop_back();
|
|
||||||
|
|
||||||
auto p = QPromise<int>::all(promises);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>({42, 46, 43, 44}));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <template <typename, typename...> class Sequence, typename ...Args>
|
|
||||||
struct SequenceTester<Sequence<QPromise<void>, Args...>>
|
|
||||||
{
|
|
||||||
static void exec()
|
|
||||||
{
|
|
||||||
Sequence<QPromise<void>, Args...> promises{
|
|
||||||
QtPromise::resolve(),
|
|
||||||
QtPromise::resolve(),
|
|
||||||
QtPromise::resolve()
|
|
||||||
};
|
|
||||||
|
|
||||||
promises.push_back(QtPromise::resolve());
|
|
||||||
promises.insert(++promises.begin(), QtPromise::resolve());
|
|
||||||
promises.pop_back();
|
|
||||||
|
|
||||||
auto p = QPromise<void>::all(promises);
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, -1, 42), 42);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // anonymous namespace
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::emptySequence()
|
|
||||||
{
|
|
||||||
auto p = QPromise<int>::all({});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p.isFulfilled(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>{});
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::emptySequence_void()
|
|
||||||
{
|
|
||||||
auto p = QPromise<void>::all({});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p.isFulfilled(), true);
|
|
||||||
QCOMPARE(waitForValue(p, -1, 42), 42);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::allPromisesSucceed()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve(42);
|
|
||||||
auto p1 = QtPromise::resolve(44);
|
|
||||||
auto p2 = QPromise<int>([](const QPromiseResolve<int>& resolve) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
resolve(43);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = QPromise<int>::all({p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>({42, 43, 44}));
|
|
||||||
QCOMPARE(p2.isFulfilled(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::allPromisesSucceed_void()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve();
|
|
||||||
auto p1 = QtPromise::resolve();
|
|
||||||
auto p2 = QPromise<void>([](const QPromiseResolve<void>& resolve) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
resolve();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = QPromise<void>::all({p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, -1, 42), 42);
|
|
||||||
QCOMPARE(p2.isFulfilled(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::atLeastOnePromiseReject()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve(42);
|
|
||||||
auto p1 = QtPromise::resolve(44);
|
|
||||||
auto p2 = QPromise<int>([](const QPromiseResolve<int>&, const QPromiseReject<int>& reject) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
reject(QString("foo"));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = QPromise<int>::all({p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForError(p, QString()), QString("foo"));
|
|
||||||
QCOMPARE(p2.isRejected(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::atLeastOnePromiseReject_void()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve();
|
|
||||||
auto p1 = QtPromise::resolve();
|
|
||||||
auto p2 = QPromise<void>([](const QPromiseResolve<void>&, const QPromiseReject<void>& reject) {
|
|
||||||
QtPromisePrivate::qtpromise_defer([=](){
|
|
||||||
reject(QString("foo"));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
auto p = QPromise<void>::all({p0, p2, p1});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<void>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForError(p, QString()), QString("foo"));
|
|
||||||
QCOMPARE(p2.isRejected(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::preserveOrder()
|
|
||||||
{
|
|
||||||
auto p0 = QtPromise::resolve(42).delay(500);
|
|
||||||
auto p1 = QtPromise::resolve(43).delay(100);
|
|
||||||
auto p2 = QtPromise::resolve(44).delay(250);
|
|
||||||
|
|
||||||
auto p = QPromise<int>::all({p0, p1, p2});
|
|
||||||
|
|
||||||
Q_STATIC_ASSERT((std::is_same<decltype(p), QPromise<QVector<int>>>::value));
|
|
||||||
|
|
||||||
QCOMPARE(p0.isPending(), true);
|
|
||||||
QCOMPARE(p1.isPending(), true);
|
|
||||||
QCOMPARE(p2.isPending(), true);
|
|
||||||
QCOMPARE(p.isPending(), true);
|
|
||||||
QCOMPARE(waitForValue(p, QVector<int>()), QVector<int>({42, 43, 44}));
|
|
||||||
QCOMPARE(p0.isFulfilled(), true);
|
|
||||||
QCOMPARE(p1.isFulfilled(), true);
|
|
||||||
QCOMPARE(p2.isFulfilled(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
// QVector::push_back/append isn't supported since it requires a default
|
|
||||||
// constructor (see https://github.com/simonbrunel/qtpromise/issues/3)
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::sequenceTypes()
|
|
||||||
{
|
|
||||||
SequenceTester<QList<QPromise<int>>>::exec();
|
|
||||||
//SequenceTester<QVector<QPromise<int>>>::exec();
|
|
||||||
SequenceTester<std::list<QPromise<int>>>::exec();
|
|
||||||
SequenceTester<std::vector<QPromise<int>>>::exec();
|
|
||||||
}
|
|
||||||
|
|
||||||
void tst_deprecations_qpromise_all::sequenceTypes_void()
|
|
||||||
{
|
|
||||||
SequenceTester<QList<QPromise<void>>>::exec();
|
|
||||||
//SequenceTester<QVector<QPromise<void>>>::exec();
|
|
||||||
SequenceTester<std::list<QPromise<void>>>::exec();
|
|
||||||
SequenceTester<std::vector<QPromise<void>>>::exec();
|
|
||||||
}
|
|
@ -1,3 +0,0 @@
|
|||||||
TEMPLATE = subdirs
|
|
||||||
SUBDIRS += \
|
|
||||||
all
|
|
275
tests/auto/qtpromise/deprecations/tst_helpers_qpromise.cpp
Normal file
275
tests/auto/qtpromise/deprecations/tst_helpers_qpromise.cpp
Normal file
@ -0,0 +1,275 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "../shared/data.h"
|
||||||
|
#include "../shared/utils.h"
|
||||||
|
|
||||||
|
#include <QtConcurrent>
|
||||||
|
#include <QtPromise>
|
||||||
|
#include <QtTest>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
class tst_deprecations_helpers_qpromise : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void value();
|
||||||
|
void noValue();
|
||||||
|
void moveRValue();
|
||||||
|
void copyLValue();
|
||||||
|
void qtSharedPtr();
|
||||||
|
void stdSharedPtr();
|
||||||
|
void typedPromise();
|
||||||
|
void voidPromise();
|
||||||
|
void typedFuture();
|
||||||
|
void voidFuture();
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_MAIN(tst_deprecations_helpers_qpromise)
|
||||||
|
#include "tst_helpers_qpromise.moc"
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::value()
|
||||||
|
{
|
||||||
|
int v0 = 42;
|
||||||
|
const int v1 = 42;
|
||||||
|
|
||||||
|
auto p0 = QtPromise::qPromise(42);
|
||||||
|
auto p1 = QtPromise::qPromise(v0);
|
||||||
|
auto p2 = QtPromise::qPromise(v1);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p0), QtPromise::QPromise<int>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p1), QtPromise::QPromise<int>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p2), QtPromise::QPromise<int>>::value));
|
||||||
|
|
||||||
|
for (const auto& p : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
}
|
||||||
|
for (const auto& p : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(waitForValue(p, -1), 42);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::noValue()
|
||||||
|
{
|
||||||
|
auto p = QtPromise::qPromise();
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::moveRValue()
|
||||||
|
{
|
||||||
|
Data::logs().reset();
|
||||||
|
|
||||||
|
{
|
||||||
|
auto p = QtPromise::qPromise(Data{42}).wait();
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<Data>>::value));
|
||||||
|
}
|
||||||
|
|
||||||
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
|
QCOMPARE(Data::logs().move, 1);
|
||||||
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::copyLValue()
|
||||||
|
{
|
||||||
|
Data::logs().reset();
|
||||||
|
|
||||||
|
{
|
||||||
|
Data value{42};
|
||||||
|
auto p = QtPromise::qPromise(value).wait();
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<Data>>::value));
|
||||||
|
}
|
||||||
|
|
||||||
|
QCOMPARE(Data::logs().ctor, 1);
|
||||||
|
QCOMPARE(Data::logs().copy, 1);
|
||||||
|
QCOMPARE(Data::logs().move, 0);
|
||||||
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://github.com/simonbrunel/qtpromise/issues/6
|
||||||
|
void tst_deprecations_helpers_qpromise::qtSharedPtr()
|
||||||
|
{
|
||||||
|
using DataSPtr = QSharedPointer<Data>;
|
||||||
|
|
||||||
|
Data::logs().reset();
|
||||||
|
|
||||||
|
QWeakPointer<Data> wptr;
|
||||||
|
|
||||||
|
{
|
||||||
|
auto sptr0 = DataSPtr::create(42);
|
||||||
|
const DataSPtr sptr1 = sptr0;
|
||||||
|
|
||||||
|
auto p0 = QtPromise::qPromise(DataSPtr::create(42));
|
||||||
|
auto p1 = QtPromise::qPromise(sptr0);
|
||||||
|
auto p2 = QtPromise::qPromise(sptr1);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p0), QtPromise::QPromise<DataSPtr>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p1), QtPromise::QPromise<DataSPtr>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p2), QtPromise::QPromise<DataSPtr>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(waitForValue(p1, DataSPtr{}), sptr0);
|
||||||
|
QCOMPARE(waitForValue(p2, DataSPtr{}), sptr1);
|
||||||
|
|
||||||
|
wptr = sptr0;
|
||||||
|
|
||||||
|
QCOMPARE(wptr.isNull(), false);
|
||||||
|
QCOMPARE(Data::logs().refs, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
QCOMPARE(wptr.isNull(), true);
|
||||||
|
|
||||||
|
QCOMPARE(Data::logs().ctor, 2);
|
||||||
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
|
QCOMPARE(Data::logs().move, 0);
|
||||||
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://github.com/simonbrunel/qtpromise/issues/6
|
||||||
|
void tst_deprecations_helpers_qpromise::stdSharedPtr()
|
||||||
|
{
|
||||||
|
using DataSPtr = std::shared_ptr<Data>;
|
||||||
|
|
||||||
|
Data::logs().reset();
|
||||||
|
|
||||||
|
std::weak_ptr<Data> wptr;
|
||||||
|
|
||||||
|
{
|
||||||
|
auto sptr0 = std::make_shared<Data>(42);
|
||||||
|
const DataSPtr sptr1 = sptr0;
|
||||||
|
|
||||||
|
auto p0 = QtPromise::qPromise(std::make_shared<Data>(42));
|
||||||
|
auto p1 = QtPromise::qPromise(sptr0);
|
||||||
|
auto p2 = QtPromise::qPromise(sptr1);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p0), QtPromise::QPromise<DataSPtr>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p1), QtPromise::QPromise<DataSPtr>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p2), QtPromise::QPromise<DataSPtr>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(waitForValue(p1, DataSPtr{}), sptr0);
|
||||||
|
QCOMPARE(waitForValue(p2, DataSPtr{}), sptr1);
|
||||||
|
|
||||||
|
wptr = sptr0;
|
||||||
|
|
||||||
|
QCOMPARE(wptr.use_count(), 4l);
|
||||||
|
QCOMPARE(Data::logs().refs, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
QCOMPARE(wptr.use_count(), 0l);
|
||||||
|
|
||||||
|
QCOMPARE(Data::logs().ctor, 2);
|
||||||
|
QCOMPARE(Data::logs().copy, 0);
|
||||||
|
QCOMPARE(Data::logs().move, 0);
|
||||||
|
QCOMPARE(Data::logs().refs, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::typedPromise()
|
||||||
|
{
|
||||||
|
auto resolver = [](const QtPromise::QPromiseResolve<int>& resolve) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
resolve(42);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
QtPromise::QPromise<int> v0{resolver};
|
||||||
|
const QtPromise::QPromise<int> v1 = v0;
|
||||||
|
|
||||||
|
auto p0 = QtPromise::qPromise(QtPromise::QPromise<int>{resolver});
|
||||||
|
auto p1 = QtPromise::qPromise(v0);
|
||||||
|
auto p2 = QtPromise::qPromise(v1);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p0), QtPromise::QPromise<int>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p1), QtPromise::QPromise<int>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p2), QtPromise::QPromise<int>>::value));
|
||||||
|
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(promise.isPending(), true);
|
||||||
|
}
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(waitForValue(promise, -1), 42);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::voidPromise()
|
||||||
|
{
|
||||||
|
auto resolver = [](const QtPromise::QPromiseResolve<void>& resolve) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
QtPromise::QPromise<void> v0{resolver};
|
||||||
|
const QtPromise::QPromise<void> v1 = v0;
|
||||||
|
|
||||||
|
auto p0 = QtPromise::qPromise(QtPromise::QPromise<void>{resolver});
|
||||||
|
auto p1 = QtPromise::qPromise(v0);
|
||||||
|
auto p2 = QtPromise::qPromise(v1);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p0), QtPromise::QPromise<void>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p1), QtPromise::QPromise<void>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p2), QtPromise::QPromise<void>>::value));
|
||||||
|
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(promise.isPending(), true);
|
||||||
|
}
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(waitForValue(promise, -1, 42), 42);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::typedFuture()
|
||||||
|
{
|
||||||
|
auto fn = []() {
|
||||||
|
return 42;
|
||||||
|
};
|
||||||
|
QFuture<int> v0 = QtConcurrent::run(fn);
|
||||||
|
const QFuture<int> v1 = v0;
|
||||||
|
|
||||||
|
auto p0 = QtPromise::qPromise(QtConcurrent::run(fn));
|
||||||
|
auto p1 = QtPromise::qPromise(v0);
|
||||||
|
auto p2 = QtPromise::qPromise(v1);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p0), QtPromise::QPromise<int>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p1), QtPromise::QPromise<int>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p2), QtPromise::QPromise<int>>::value));
|
||||||
|
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(promise.isPending(), true);
|
||||||
|
}
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(waitForValue(promise, -1), 42);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromise::voidFuture()
|
||||||
|
{
|
||||||
|
auto fn = []() {};
|
||||||
|
QFuture<void> v0 = QtConcurrent::run(fn);
|
||||||
|
const QFuture<void> v1 = v0;
|
||||||
|
|
||||||
|
auto p0 = QtPromise::qPromise(QtConcurrent::run(fn));
|
||||||
|
auto p1 = QtPromise::qPromise(v0);
|
||||||
|
auto p2 = QtPromise::qPromise(v1);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p0), QtPromise::QPromise<void>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p1), QtPromise::QPromise<void>>::value));
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p2), QtPromise::QPromise<void>>::value));
|
||||||
|
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(promise.isPending(), true);
|
||||||
|
}
|
||||||
|
for (const auto& promise : {p0, p1, p2}) {
|
||||||
|
QCOMPARE(waitForValue(promise, -1, 42), 42);
|
||||||
|
}
|
||||||
|
}
|
218
tests/auto/qtpromise/deprecations/tst_helpers_qpromiseall.cpp
Normal file
218
tests/auto/qtpromise/deprecations/tst_helpers_qpromiseall.cpp
Normal file
@ -0,0 +1,218 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "../shared/utils.h"
|
||||||
|
|
||||||
|
#include <QtPromise>
|
||||||
|
#include <QtTest>
|
||||||
|
|
||||||
|
class tst_deprecations_helpers_qpromiseall : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void emptySequence();
|
||||||
|
void emptySequence_void();
|
||||||
|
void allPromisesSucceed();
|
||||||
|
void allPromisesSucceed_void();
|
||||||
|
void atLeastOnePromiseReject();
|
||||||
|
void atLeastOnePromiseReject_void();
|
||||||
|
void preserveOrder();
|
||||||
|
void sequenceTypes();
|
||||||
|
void sequenceTypes_void();
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_MAIN(tst_deprecations_helpers_qpromiseall)
|
||||||
|
#include "tst_helpers_qpromiseall.moc"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
template<class Sequence>
|
||||||
|
struct SequenceTester
|
||||||
|
{
|
||||||
|
Q_STATIC_ASSERT((std::is_same<typename Sequence::value_type, QtPromise::QPromise<int>>::value));
|
||||||
|
|
||||||
|
static void exec()
|
||||||
|
{
|
||||||
|
Sequence promises{QtPromise::resolve(42), QtPromise::resolve(43), QtPromise::resolve(44)};
|
||||||
|
|
||||||
|
promises.push_back(QtPromise::resolve(45));
|
||||||
|
promises.insert(++promises.begin(), QtPromise::resolve(46));
|
||||||
|
promises.pop_back();
|
||||||
|
|
||||||
|
auto p = qPromiseAll(promises);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), (QVector<int>{42, 46, 43, 44}));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<template<typename, typename...> class Sequence, typename... Args>
|
||||||
|
struct SequenceTester<Sequence<QtPromise::QPromise<void>, Args...>>
|
||||||
|
{
|
||||||
|
static void exec()
|
||||||
|
{
|
||||||
|
Sequence<QtPromise::QPromise<void>, Args...> promises{QtPromise::resolve(),
|
||||||
|
QtPromise::resolve(),
|
||||||
|
QtPromise::resolve()};
|
||||||
|
|
||||||
|
promises.push_back(QtPromise::resolve());
|
||||||
|
promises.insert(++promises.begin(), QtPromise::resolve());
|
||||||
|
promises.pop_back();
|
||||||
|
|
||||||
|
auto p = qPromiseAll(promises);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // anonymous namespace
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::emptySequence()
|
||||||
|
{
|
||||||
|
auto p = qPromiseAll(QVector<QtPromise::QPromise<int>>());
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), QVector<int>{});
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::emptySequence_void()
|
||||||
|
{
|
||||||
|
auto p = qPromiseAll(QVector<QtPromise::QPromise<void>>());
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::allPromisesSucceed()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve(42);
|
||||||
|
auto p1 = QtPromise::resolve(44);
|
||||||
|
auto p2 = QtPromise::QPromise<int>{[](const QtPromise::QPromiseResolve<int>& resolve) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
resolve(43);
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::qPromiseAll(QVector<QtPromise::QPromise<int>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), (QVector<int>{42, 43, 44}));
|
||||||
|
QCOMPARE(p2.isFulfilled(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::allPromisesSucceed_void()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve();
|
||||||
|
auto p1 = QtPromise::resolve();
|
||||||
|
auto p2 = QtPromise::QPromise<void>{[](const QtPromise::QPromiseResolve<void>& resolve) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::qPromiseAll(QVector<QtPromise::QPromise<void>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
QCOMPARE(p2.isFulfilled(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::atLeastOnePromiseReject()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve(42);
|
||||||
|
auto p1 = QtPromise::resolve(44);
|
||||||
|
auto p2 = QtPromise::QPromise<int>{
|
||||||
|
[](const QtPromise::QPromiseResolve<int>&, const QtPromise::QPromiseReject<int>& reject) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
reject(QString{"foo"});
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::qPromiseAll(QVector<QtPromise::QPromise<int>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForError(p, QString{}), QString{"foo"});
|
||||||
|
QCOMPARE(p2.isRejected(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::atLeastOnePromiseReject_void()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve();
|
||||||
|
auto p1 = QtPromise::resolve();
|
||||||
|
auto p2 = QtPromise::QPromise<void>{
|
||||||
|
[](const QtPromise::QPromiseResolve<void>&, const QtPromise::QPromiseReject<void>& reject) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
reject(QString{"foo"});
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::qPromiseAll(QVector<QtPromise::QPromise<void>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForError(p, QString{}), QString{"foo"});
|
||||||
|
QCOMPARE(p2.isRejected(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::preserveOrder()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve(42).delay(500);
|
||||||
|
auto p1 = QtPromise::resolve(43).delay(100);
|
||||||
|
auto p2 = QtPromise::resolve(44).delay(250);
|
||||||
|
|
||||||
|
auto p = QtPromise::qPromiseAll(QVector<QtPromise::QPromise<int>>{p0, p1, p2});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
QCOMPARE(p0.isPending(), true);
|
||||||
|
QCOMPARE(p1.isPending(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), (QVector<int>{42, 43, 44}));
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isFulfilled(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
// QVector::push_back/append isn't supported since it requires a default
|
||||||
|
// constructor (see https://github.com/simonbrunel/qtpromise/issues/3)
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::sequenceTypes()
|
||||||
|
{
|
||||||
|
SequenceTester<QList<QtPromise::QPromise<int>>>::exec();
|
||||||
|
// SequenceTester<QVector<QtPromise::QPromise<int>>>::exec();
|
||||||
|
SequenceTester<std::list<QtPromise::QPromise<int>>>::exec();
|
||||||
|
SequenceTester<std::vector<QtPromise::QPromise<int>>>::exec();
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_helpers_qpromiseall::sequenceTypes_void()
|
||||||
|
{
|
||||||
|
SequenceTester<QList<QtPromise::QPromise<void>>>::exec();
|
||||||
|
// SequenceTester<QVector<QtPromise::QPromise<void>>>::exec();
|
||||||
|
SequenceTester<std::list<QtPromise::QPromise<void>>>::exec();
|
||||||
|
SequenceTester<std::vector<QtPromise::QPromise<void>>>::exec();
|
||||||
|
}
|
227
tests/auto/qtpromise/deprecations/tst_qpromise_all.cpp
Normal file
227
tests/auto/qtpromise/deprecations/tst_qpromise_all.cpp
Normal file
@ -0,0 +1,227 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "../shared/utils.h"
|
||||||
|
|
||||||
|
#include <QtPromise>
|
||||||
|
#include <QtTest>
|
||||||
|
|
||||||
|
class tst_deprecations_qpromise_all : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void emptySequence();
|
||||||
|
void emptySequence_void();
|
||||||
|
void allPromisesSucceed();
|
||||||
|
void allPromisesSucceed_void();
|
||||||
|
void atLeastOnePromiseReject();
|
||||||
|
void atLeastOnePromiseReject_void();
|
||||||
|
void preserveOrder();
|
||||||
|
void sequenceTypes();
|
||||||
|
void sequenceTypes_void();
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_MAIN(tst_deprecations_qpromise_all)
|
||||||
|
#include "tst_qpromise_all.moc"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
template<class Sequence>
|
||||||
|
struct SequenceTester
|
||||||
|
{
|
||||||
|
Q_STATIC_ASSERT((std::is_same<typename Sequence::value_type, QtPromise::QPromise<int>>::value));
|
||||||
|
|
||||||
|
static void exec()
|
||||||
|
{
|
||||||
|
Sequence promises{QtPromise::resolve(42), QtPromise::resolve(43), QtPromise::resolve(44)};
|
||||||
|
|
||||||
|
promises.push_back(QtPromise::resolve(45));
|
||||||
|
promises.insert(++promises.begin(), QtPromise::resolve(46));
|
||||||
|
promises.pop_back();
|
||||||
|
|
||||||
|
auto p = QtPromise::QPromise<int>::all(promises);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), (QVector<int>{42, 46, 43, 44}));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<template<typename, typename...> class Sequence, typename... Args>
|
||||||
|
struct SequenceTester<Sequence<QtPromise::QPromise<void>, Args...>>
|
||||||
|
{
|
||||||
|
static void exec()
|
||||||
|
{
|
||||||
|
Sequence<QtPromise::QPromise<void>, Args...> promises{QtPromise::resolve(),
|
||||||
|
QtPromise::resolve(),
|
||||||
|
QtPromise::resolve()};
|
||||||
|
|
||||||
|
promises.push_back(QtPromise::resolve());
|
||||||
|
promises.insert(++promises.begin(), QtPromise::resolve());
|
||||||
|
promises.pop_back();
|
||||||
|
|
||||||
|
auto p = QtPromise::QPromise<void>::all(promises);
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // anonymous namespace
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::emptySequence()
|
||||||
|
{
|
||||||
|
auto p = QtPromise::QPromise<int>::all(QVector<QtPromise::QPromise<int>>{});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), QVector<int>{});
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::emptySequence_void()
|
||||||
|
{
|
||||||
|
auto p = QtPromise::QPromise<void>::all(QVector<QtPromise::QPromise<void>>{});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p.isFulfilled(), true);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::allPromisesSucceed()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve(42);
|
||||||
|
auto p1 = QtPromise::resolve(44);
|
||||||
|
auto p2 = QtPromise::QPromise<int>{[](const QtPromise::QPromiseResolve<int>& resolve) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
resolve(43);
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::QPromise<int>::all(QVector<QtPromise::QPromise<int>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), (QVector<int>{42, 43, 44}));
|
||||||
|
QCOMPARE(p2.isFulfilled(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::allPromisesSucceed_void()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve();
|
||||||
|
auto p1 = QtPromise::resolve();
|
||||||
|
auto p2 = QtPromise::QPromise<void>{[](const QtPromise::QPromiseResolve<void>& resolve) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::QPromise<void>::all(QVector<QtPromise::QPromise<void>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, -1, 42), 42);
|
||||||
|
QCOMPARE(p2.isFulfilled(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::atLeastOnePromiseReject()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve(42);
|
||||||
|
auto p1 = QtPromise::resolve(44);
|
||||||
|
auto p2 = QtPromise::QPromise<int>{
|
||||||
|
[](const QtPromise::QPromiseResolve<int>&, const QtPromise::QPromiseReject<int>& reject) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
reject(QString{"foo"});
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::QPromise<int>::all(QVector<QtPromise::QPromise<int>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForError(p, QString{}), QString{"foo"});
|
||||||
|
QCOMPARE(p2.isRejected(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::atLeastOnePromiseReject_void()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve();
|
||||||
|
auto p1 = QtPromise::resolve();
|
||||||
|
auto p2 = QtPromise::QPromise<void>{
|
||||||
|
[](const QtPromise::QPromiseResolve<void>&, const QtPromise::QPromiseReject<void>& reject) {
|
||||||
|
QtPromisePrivate::qtpromise_defer([=]() {
|
||||||
|
reject(QString{"foo"});
|
||||||
|
});
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto p = QtPromise::QPromise<void>::all(QVector<QtPromise::QPromise<void>>{p0, p2, p1});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<void>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForError(p, QString{}), QString{"foo"});
|
||||||
|
QCOMPARE(p2.isRejected(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::preserveOrder()
|
||||||
|
{
|
||||||
|
auto p0 = QtPromise::resolve(42).delay(500);
|
||||||
|
auto p1 = QtPromise::resolve(43).delay(100);
|
||||||
|
auto p2 = QtPromise::resolve(44).delay(250);
|
||||||
|
|
||||||
|
auto p = QtPromise::QPromise<int>::all(QVector<QtPromise::QPromise<int>>{p0, p1, p2});
|
||||||
|
|
||||||
|
Q_STATIC_ASSERT((std::is_same<decltype(p), QtPromise::QPromise<QVector<int>>>::value));
|
||||||
|
|
||||||
|
QCOMPARE(p0.isPending(), true);
|
||||||
|
QCOMPARE(p1.isPending(), true);
|
||||||
|
QCOMPARE(p2.isPending(), true);
|
||||||
|
QCOMPARE(p.isPending(), true);
|
||||||
|
QCOMPARE(waitForValue(p, QVector<int>{}), (QVector<int>{42, 43, 44}));
|
||||||
|
QCOMPARE(p0.isFulfilled(), true);
|
||||||
|
QCOMPARE(p1.isFulfilled(), true);
|
||||||
|
QCOMPARE(p2.isFulfilled(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
// QVector::push_back/append isn't supported since it requires a default
|
||||||
|
// constructor (see https://github.com/simonbrunel/qtpromise/issues/3)
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::sequenceTypes()
|
||||||
|
{
|
||||||
|
SequenceTester<QList<QtPromise::QPromise<int>>>::exec();
|
||||||
|
// SequenceTester<QVector<QtPromise::QPromise<int>>>::exec();
|
||||||
|
SequenceTester<std::list<QtPromise::QPromise<int>>>::exec();
|
||||||
|
SequenceTester<std::vector<QtPromise::QPromise<int>>>::exec();
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_deprecations_qpromise_all::sequenceTypes_void()
|
||||||
|
{
|
||||||
|
SequenceTester<QList<QtPromise::QPromise<void>>>::exec();
|
||||||
|
// SequenceTester<QVector<QtPromise::QPromise<void>>>::exec();
|
||||||
|
SequenceTester<std::list<QtPromise::QPromise<void>>>::exec();
|
||||||
|
SequenceTester<std::vector<QtPromise::QPromise<void>>>::exec();
|
||||||
|
}
|
4
tests/auto/qtpromise/exceptions/CMakeLists.txt
Normal file
4
tests/auto/qtpromise/exceptions/CMakeLists.txt
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
qtpromise_add_test(exceptions
|
||||||
|
SOURCES
|
||||||
|
tst_exceptions.cpp
|
||||||
|
)
|
@ -1,5 +0,0 @@
|
|||||||
QT += concurrent
|
|
||||||
TARGET = tst_exceptions
|
|
||||||
SOURCES += $$PWD/tst_exceptions.cpp
|
|
||||||
|
|
||||||
include(../qtpromise.pri)
|
|
@ -1,14 +1,16 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) Simon Brunel, https://github.com/simonbrunel
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in
|
||||||
|
* the LICENSE file in the root directory of this source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
#include "../shared/utils.h"
|
#include "../shared/utils.h"
|
||||||
|
|
||||||
// QtPromise
|
|
||||||
#include <QtPromise>
|
|
||||||
|
|
||||||
// Qt
|
|
||||||
#include <QtConcurrent>
|
#include <QtConcurrent>
|
||||||
|
#include <QtPromise>
|
||||||
#include <QtTest>
|
#include <QtTest>
|
||||||
|
|
||||||
using namespace QtPromise;
|
|
||||||
|
|
||||||
class tst_exceptions : public QObject
|
class tst_exceptions : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@ -16,6 +18,7 @@ class tst_exceptions : public QObject
|
|||||||
private Q_SLOTS:
|
private Q_SLOTS:
|
||||||
void canceled();
|
void canceled();
|
||||||
void context();
|
void context();
|
||||||
|
void conversion();
|
||||||
void timeout();
|
void timeout();
|
||||||
void undefined();
|
void undefined();
|
||||||
|
|
||||||
@ -26,10 +29,12 @@ QTEST_MAIN(tst_exceptions)
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
template <class E>
|
template<class E>
|
||||||
void verify()
|
void verify()
|
||||||
{
|
{
|
||||||
auto p = QtPromise::resolve(QtConcurrent::run([]() { throw E(); }));
|
auto p = QtPromise::resolve(QtConcurrent::run([]() {
|
||||||
|
throw E();
|
||||||
|
}));
|
||||||
QCOMPARE(p.isPending(), true);
|
QCOMPARE(p.isPending(), true);
|
||||||
QCOMPARE(waitForRejected<E>(p), true);
|
QCOMPARE(waitForRejected<E>(p), true);
|
||||||
QCOMPARE(p.isRejected(), true);
|
QCOMPARE(p.isRejected(), true);
|
||||||
@ -39,20 +44,25 @@ void verify()
|
|||||||
|
|
||||||
void tst_exceptions::canceled()
|
void tst_exceptions::canceled()
|
||||||
{
|
{
|
||||||
verify<QPromiseCanceledException>();
|
verify<QtPromise::QPromiseCanceledException>();
|
||||||
}
|
}
|
||||||
|
|
||||||
void tst_exceptions::context()
|
void tst_exceptions::context()
|
||||||
{
|
{
|
||||||
verify<QPromiseContextException>();
|
verify<QtPromise::QPromiseContextException>();
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_exceptions::conversion()
|
||||||
|
{
|
||||||
|
verify<QtPromise::QPromiseConversionException>();
|
||||||
}
|
}
|
||||||
|
|
||||||
void tst_exceptions::timeout()
|
void tst_exceptions::timeout()
|
||||||
{
|
{
|
||||||
verify<QPromiseTimeoutException>();
|
verify<QtPromise::QPromiseTimeoutException>();
|
||||||
}
|
}
|
||||||
|
|
||||||
void tst_exceptions::undefined()
|
void tst_exceptions::undefined()
|
||||||
{
|
{
|
||||||
verify<QPromiseUndefinedException>();
|
verify<QtPromise::QPromiseUndefinedException>();
|
||||||
}
|
}
|
||||||
|
4
tests/auto/qtpromise/future/CMakeLists.txt
Normal file
4
tests/auto/qtpromise/future/CMakeLists.txt
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
qtpromise_add_test(future
|
||||||
|
SOURCES
|
||||||
|
tst_future.cpp
|
||||||
|
)
|
@ -1,5 +0,0 @@
|
|||||||
QT += concurrent
|
|
||||||
TARGET = tst_future
|
|
||||||
SOURCES += $$PWD/tst_future.cpp
|
|
||||||
|
|
||||||
include(../qtpromise.pri)
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user