mirror of https://github.com/Cysharp/UniTask
Compare commits
No commits in common. "master" and "2.0.3-preview4" have entirely different histories.
master
...
2.0.3-prev
|
@ -0,0 +1,114 @@
|
||||||
|
version: 2.1
|
||||||
|
executors:
|
||||||
|
unity:
|
||||||
|
# https://hub.docker.com/r/gableroux/unity3d/tags
|
||||||
|
parameters:
|
||||||
|
version: {type: string}
|
||||||
|
docker:
|
||||||
|
- image: gableroux/unity3d:<< parameters.version >>
|
||||||
|
go:
|
||||||
|
docker:
|
||||||
|
- image: circleci/golang
|
||||||
|
commands:
|
||||||
|
unity_activate:
|
||||||
|
parameters:
|
||||||
|
unity_version: {type: string}
|
||||||
|
unity_license: {type: string}
|
||||||
|
steps:
|
||||||
|
# get activation file, if fail to activate unity, use this key and activate from https://license.unity3d.com/manual
|
||||||
|
- run: apt update && apt install libunwind8 -y
|
||||||
|
- run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -logFile -createManualActivationFile || exit 0
|
||||||
|
- run: cat Unity_v<< parameters.unity_version >>.alf
|
||||||
|
# get from UNITY_LICENSE envvar(base64 encoded(cat foo.ulf | base64 )), this file is generated from above manual activation
|
||||||
|
- run: echo << parameters.unity_license >> | base64 -di >> .circleci/Unity.ulf
|
||||||
|
- run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -manualLicenseFile .circleci/Unity.ulf || exit 0
|
||||||
|
jobs:
|
||||||
|
build-and-test:
|
||||||
|
parameters:
|
||||||
|
unity_version: {type: string}
|
||||||
|
unity_license: {type: string}
|
||||||
|
executor:
|
||||||
|
name: unity
|
||||||
|
version: << parameters.unity_version >>
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- unity_activate:
|
||||||
|
unity_version: << parameters.unity_version >>
|
||||||
|
unity_license: << parameters.unity_license >>
|
||||||
|
- run:
|
||||||
|
name: Build Linux(Mono)
|
||||||
|
command: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod UnitTestBuilder.BuildUnitTest /headless /ScriptBackend Mono2x /BuildTarget StandaloneLinux64
|
||||||
|
working_directory: .
|
||||||
|
# TODO:check unity version and packages...
|
||||||
|
# - run: ./bin/UnitTest/StandaloneLinux64_Mono2x/test
|
||||||
|
build-and-create-package:
|
||||||
|
parameters:
|
||||||
|
unity_version: {type: string}
|
||||||
|
unity_license: {type: string}
|
||||||
|
executor:
|
||||||
|
name: unity
|
||||||
|
version: << parameters.unity_version >>
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- unity_activate:
|
||||||
|
unity_version: << parameters.unity_version >>
|
||||||
|
unity_license: << parameters.unity_license >>
|
||||||
|
- run:
|
||||||
|
name: Export unitypackage
|
||||||
|
command: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod PackageExporter.Export
|
||||||
|
working_directory: .
|
||||||
|
- store_artifacts:
|
||||||
|
path: ./UniRx.Async.unitypackage
|
||||||
|
destination: /UniRx.Async.unitypackage
|
||||||
|
# upload to github by ghr
|
||||||
|
upload-github:
|
||||||
|
executor: go
|
||||||
|
steps:
|
||||||
|
- attach_workspace:
|
||||||
|
at: .
|
||||||
|
- run: go get github.com/tcnksm/ghr
|
||||||
|
- run: ghr -t ${GITHUB_TOKEN} -u ${CIRCLE_PROJECT_USERNAME} -r ${CIRCLE_PROJECT_REPONAME} ${CIRCLE_TAG} .
|
||||||
|
- store_artifacts:
|
||||||
|
path: UniRx.Async.unitypackage
|
||||||
|
destination: UniRx.Async.unitypackage
|
||||||
|
workflows:
|
||||||
|
version: 2
|
||||||
|
build-unity:
|
||||||
|
jobs:
|
||||||
|
# does not exists yet.
|
||||||
|
# - build-and-test:
|
||||||
|
# unity_version: 2019.3.0a2
|
||||||
|
# unity_license: ${UNITY_LICENSE_2019_3}
|
||||||
|
# - build-and-test:
|
||||||
|
# unity_version: 2019.2.0b2
|
||||||
|
# unity_license: ${UNITY_LICENSE_2019_2}
|
||||||
|
- build-and-test:
|
||||||
|
unity_version: 2019.1.2f1
|
||||||
|
unity_license: ${UNITY_LICENSE_2019_1}
|
||||||
|
filters:
|
||||||
|
tags:
|
||||||
|
only: /.*/
|
||||||
|
# test asmdef will not found.
|
||||||
|
# - build-and-test:
|
||||||
|
# unity_version: 2018.4.0f1
|
||||||
|
# unity_license: ${UNITY_LICENSE_2018_4}
|
||||||
|
# # UniTask minimum support version is 2018.3(C# 7.x)
|
||||||
|
# - build-and-test:
|
||||||
|
# unity_version: 2018.3.12f1
|
||||||
|
# unity_license: ${UNITY_LICENSE_2018_3}
|
||||||
|
- build-and-create-package:
|
||||||
|
unity_version: 2019.1.2f1
|
||||||
|
unity_license: ${UNITY_LICENSE_2019_1}
|
||||||
|
filters:
|
||||||
|
tags:
|
||||||
|
only: /^\d\.\d\.\d.*/
|
||||||
|
branches:
|
||||||
|
ignore: /.*/
|
||||||
|
- upload-github:
|
||||||
|
requires:
|
||||||
|
- build-and-create-package
|
||||||
|
filters:
|
||||||
|
tags:
|
||||||
|
only: /^\d\.\d\.\d.*/
|
||||||
|
branches:
|
||||||
|
ignore: /.*/
|
|
@ -1,41 +0,0 @@
|
||||||
# top-most EditorConfig file
|
|
||||||
root = true
|
|
||||||
|
|
||||||
[*]
|
|
||||||
charset = utf-8
|
|
||||||
end_of_line = lf
|
|
||||||
indent_style = space
|
|
||||||
indent_size = 2
|
|
||||||
insert_final_newline = true
|
|
||||||
trim_trailing_whitespace = true
|
|
||||||
|
|
||||||
# Visual Studio Spell checker configs (https://learn.microsoft.com/en-us/visualstudio/ide/text-spell-checker?view=vs-2022#how-to-customize-the-spell-checker)
|
|
||||||
spelling_exclusion_path = ./exclusion.dic
|
|
||||||
|
|
||||||
[*.cs]
|
|
||||||
indent_size = 4
|
|
||||||
charset = utf-8-bom
|
|
||||||
end_of_line = unset
|
|
||||||
|
|
||||||
# Solution files
|
|
||||||
[*.{sln,slnx}]
|
|
||||||
end_of_line = unset
|
|
||||||
|
|
||||||
# MSBuild project files
|
|
||||||
[*.{csproj,props,targets}]
|
|
||||||
end_of_line = unset
|
|
||||||
|
|
||||||
# Xml config files
|
|
||||||
[*.{ruleset,config,nuspec,resx,runsettings,DotSettings}]
|
|
||||||
end_of_line = unset
|
|
||||||
|
|
||||||
[*{_AssemblyInfo.cs,.notsupported.cs}]
|
|
||||||
generated_code = true
|
|
||||||
|
|
||||||
# C# code style settings
|
|
||||||
[*.{cs}]
|
|
||||||
dotnet_diagnostic.IDE0044.severity = none # IDE0044: Make field readonly
|
|
||||||
|
|
||||||
# https://stackoverflow.com/questions/79195382/how-to-disable-fading-unused-methods-in-visual-studio-2022-17-12-0
|
|
||||||
dotnet_diagnostic.IDE0051.severity = none # IDE0051: Remove unused private member
|
|
||||||
dotnet_diagnostic.IDE0130.severity = none # IDE0130: Namespace does not match folder structure
|
|
|
@ -1 +0,0 @@
|
||||||
github: [neuecc]
|
|
|
@ -1,12 +0,0 @@
|
||||||
# ref: https://docs.github.com/en/code-security/dependabot/working-with-dependabot/keeping-your-actions-up-to-date-with-dependabot
|
|
||||||
version: 2
|
|
||||||
updates:
|
|
||||||
- package-ecosystem: "github-actions"
|
|
||||||
directory: "/"
|
|
||||||
schedule:
|
|
||||||
interval: "weekly" # Check for updates to GitHub Actions every week
|
|
||||||
ignore:
|
|
||||||
# I just want update action when major/minor version is updated. patch updates are too noisy.
|
|
||||||
- dependency-name: '*'
|
|
||||||
update-types:
|
|
||||||
- version-update:semver-patch
|
|
|
@ -1,92 +0,0 @@
|
||||||
name: Build-Debug
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- "master"
|
|
||||||
pull_request:
|
|
||||||
branches:
|
|
||||||
- "master"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build-dotnet:
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
timeout-minutes: 10
|
|
||||||
steps:
|
|
||||||
- uses: Cysharp/Actions/.github/actions/checkout@main
|
|
||||||
- uses: Cysharp/Actions/.github/actions/setup-dotnet@main
|
|
||||||
- run: dotnet build -c Debug
|
|
||||||
- run: dotnet test -c Debug
|
|
||||||
|
|
||||||
build-unity:
|
|
||||||
if: ${{ ((github.event_name == 'push' && github.repository_owner == 'Cysharp') || startsWith(github.event.pull_request.head.label, 'Cysharp:')) && github.triggering_actor != 'dependabot[bot]' }}
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
max-parallel: 2
|
|
||||||
matrix:
|
|
||||||
unity: ["2022.3.39f1", "6000.0.12f1"] # Test with LTS
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
timeout-minutes: 30 # Unity build takes more than 20min.
|
|
||||||
steps:
|
|
||||||
- name: Load secrets
|
|
||||||
id: op-load-secret
|
|
||||||
uses: 1password/load-secrets-action@581a835fb51b8e7ec56b71cf2ffddd7e68bb25e0 # v2.0.0
|
|
||||||
with:
|
|
||||||
export-env: false
|
|
||||||
env:
|
|
||||||
OP_SERVICE_ACCOUNT_TOKEN: ${{ secrets.OP_SERVICE_ACCOUNT_TOKEN_PUBLIC }}
|
|
||||||
UNITY_EMAIL: "op://${{ vars.OP_VAULT_ACTIONS_PUBLIC }}/UNITY_LICENSE/username"
|
|
||||||
UNITY_PASSWORD: "op://${{ vars.OP_VAULT_ACTIONS_PUBLIC }}/UNITY_LICENSE/credential"
|
|
||||||
UNITY_SERIAL: "op://${{ vars.OP_VAULT_ACTIONS_PUBLIC }}/UNITY_LICENSE/serial"
|
|
||||||
|
|
||||||
- uses: Cysharp/Actions/.github/actions/checkout@main
|
|
||||||
|
|
||||||
# Execute scripts: Export Package
|
|
||||||
# /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod PackageExporter.Export
|
|
||||||
- name: Build Unity (.unitypacakge)
|
|
||||||
if: ${{ startsWith(matrix.unity, '2022') }} # only execute once
|
|
||||||
uses: Cysharp/Actions/.github/actions/unity-builder@main
|
|
||||||
env:
|
|
||||||
UNITY_EMAIL: ${{ steps.op-load-secret.outputs.UNITY_EMAIL }}
|
|
||||||
UNITY_PASSWORD: ${{ steps.op-load-secret.outputs.UNITY_PASSWORD }}
|
|
||||||
UNITY_SERIAL: ${{ steps.op-load-secret.outputs.UNITY_SERIAL }}
|
|
||||||
with:
|
|
||||||
projectPath: src/UniTask
|
|
||||||
unityVersion: ${{ matrix.unity }}
|
|
||||||
targetPlatform: StandaloneLinux64
|
|
||||||
buildMethod: PackageExporter.Export
|
|
||||||
|
|
||||||
# Execute UnitTest
|
|
||||||
# /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod UnitTestBuilder.BuildUnitTest /headless /ScriptBackend IL2CPP /BuildTarget StandaloneLinux64
|
|
||||||
- name: Build UnitTest (IL2CPP)
|
|
||||||
uses: Cysharp/Actions/.github/actions/unity-builder@main
|
|
||||||
env:
|
|
||||||
UNITY_EMAIL: ${{ steps.op-load-secret.outputs.UNITY_EMAIL }}
|
|
||||||
UNITY_PASSWORD: ${{ steps.op-load-secret.outputs.UNITY_PASSWORD }}
|
|
||||||
UNITY_SERIAL: ${{ steps.op-load-secret.outputs.UNITY_SERIAL }}
|
|
||||||
with:
|
|
||||||
projectPath: src/UniTask
|
|
||||||
unityVersion: ${{ matrix.unity }}
|
|
||||||
targetPlatform: StandaloneLinux64
|
|
||||||
buildMethod: UnitTestBuilder.BuildUnitTest
|
|
||||||
customParameters: "/headless /ScriptBackend IL2CPP"
|
|
||||||
- name: Check UnitTest file is generated
|
|
||||||
run: ls -lR ./src/UniTask/bin/UnitTest
|
|
||||||
- name: Execute UnitTest
|
|
||||||
run: ./src/UniTask/bin/UnitTest/StandaloneLinux64_IL2CPP/test
|
|
||||||
|
|
||||||
- uses: Cysharp/Actions/.github/actions/check-metas@main # check meta files
|
|
||||||
with:
|
|
||||||
directory: src/UniTask
|
|
||||||
|
|
||||||
# Store artifacts.
|
|
||||||
- uses: Cysharp/Actions/.github/actions/upload-artifact@main
|
|
||||||
if: ${{ startsWith(matrix.unity, '2021') }} # only execute 2021
|
|
||||||
with:
|
|
||||||
name: UniTask.unitypackage-${{ matrix.unity }}.zip
|
|
||||||
path: ./src/UniTask/*.unitypackage
|
|
||||||
retention-days: 1
|
|
|
@ -1,34 +0,0 @@
|
||||||
name: build-docs
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
- feature/docs
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
run-docfx:
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
pages: write
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
timeout-minutes: 10
|
|
||||||
steps:
|
|
||||||
- uses: Cysharp/Actions/.github/actions/checkout@main
|
|
||||||
- uses: Cysharp/Actions/.github/actions/checkout@main
|
|
||||||
with:
|
|
||||||
repository: Cysharp/DocfxTemplate
|
|
||||||
path: docs/_DocfxTemplate
|
|
||||||
- uses: Kirbyrawr/docfx-action@db9a22c8fe1e8693a2a21be54cb0b87dfaa72cc4
|
|
||||||
name: Docfx metadata
|
|
||||||
with:
|
|
||||||
args: metadata docs/docfx.json
|
|
||||||
- uses: Kirbyrawr/docfx-action@db9a22c8fe1e8693a2a21be54cb0b87dfaa72cc4
|
|
||||||
name: Docfx build
|
|
||||||
with:
|
|
||||||
args: build docs/docfx.json
|
|
||||||
- name: Publish to GitHub Pages
|
|
||||||
uses: peaceiris/actions-gh-pages@4f9cc6602d3f66b9c108549d475ec49e8ef4d45e # v4.0.0
|
|
||||||
with:
|
|
||||||
github_token: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
publish_dir: docs/_site
|
|
|
@ -1,121 +0,0 @@
|
||||||
name: build-release
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
inputs:
|
|
||||||
tag:
|
|
||||||
description: "tag: git tag you want create. (sample 1.0.0)"
|
|
||||||
required: true
|
|
||||||
dry-run:
|
|
||||||
description: "dry-run: true will never create relase/nuget."
|
|
||||||
required: true
|
|
||||||
default: false
|
|
||||||
type: boolean
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
update-packagejson:
|
|
||||||
permissions:
|
|
||||||
actions: read
|
|
||||||
contents: write
|
|
||||||
uses: Cysharp/Actions/.github/workflows/update-packagejson.yaml@main
|
|
||||||
with:
|
|
||||||
file-path: ./src/UniTask/Assets/Plugins/UniTask/package.json
|
|
||||||
tag: ${{ inputs.tag }}
|
|
||||||
dry-run: ${{ inputs.dry-run }}
|
|
||||||
|
|
||||||
build-dotnet:
|
|
||||||
needs: [update-packagejson]
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
timeout-minutes: 10
|
|
||||||
steps:
|
|
||||||
- run: echo ${{ needs.update-packagejson.outputs.sha }}
|
|
||||||
- uses: Cysharp/Actions/.github/actions/checkout@main
|
|
||||||
with:
|
|
||||||
ref: ${{ needs.update-packagejson.outputs.sha }}
|
|
||||||
- uses: Cysharp/Actions/.github/actions/setup-dotnet@main
|
|
||||||
# build and pack
|
|
||||||
- run: dotnet build -c Release -p:Version=${{ inputs.tag }}
|
|
||||||
- run: dotnet test -c Release --no-build
|
|
||||||
- run: dotnet pack ./src/UniTask.NetCore/UniTask.NetCore.csproj -c Release --no-build -p:Version=${{ inputs.tag }} -o ./publish
|
|
||||||
# Store artifacts.
|
|
||||||
- uses: Cysharp/Actions/.github/actions/upload-artifact@main
|
|
||||||
with:
|
|
||||||
name: nuget
|
|
||||||
path: ./publish/
|
|
||||||
retention-days: 1
|
|
||||||
|
|
||||||
build-unity:
|
|
||||||
needs: [update-packagejson]
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
unity: ["2022.3.39f1"]
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
timeout-minutes: 15
|
|
||||||
steps:
|
|
||||||
- name: Load secrets
|
|
||||||
id: op-load-secret
|
|
||||||
uses: 1password/load-secrets-action@581a835fb51b8e7ec56b71cf2ffddd7e68bb25e0 # v2.0.0
|
|
||||||
with:
|
|
||||||
export-env: false
|
|
||||||
env:
|
|
||||||
OP_SERVICE_ACCOUNT_TOKEN: ${{ secrets.OP_SERVICE_ACCOUNT_TOKEN_PUBLIC }}
|
|
||||||
UNITY_EMAIL: "op://${{ vars.OP_VAULT_ACTIONS_PUBLIC }}/UNITY_LICENSE/username"
|
|
||||||
UNITY_PASSWORD: "op://${{ vars.OP_VAULT_ACTIONS_PUBLIC }}/UNITY_LICENSE/credential"
|
|
||||||
UNITY_SERIAL: "op://${{ vars.OP_VAULT_ACTIONS_PUBLIC }}/UNITY_LICENSE/serial"
|
|
||||||
|
|
||||||
- run: echo ${{ needs.update-packagejson.outputs.sha }}
|
|
||||||
- uses: Cysharp/Actions/.github/actions/checkout@main
|
|
||||||
with:
|
|
||||||
ref: ${{ needs.update-packagejson.outputs.sha }}
|
|
||||||
# Execute scripts: Export Package
|
|
||||||
# /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod PackageExporter.Export
|
|
||||||
- name: Build Unity (.unitypacakge)
|
|
||||||
uses: Cysharp/Actions/.github/actions/unity-builder@main
|
|
||||||
env:
|
|
||||||
UNITY_EMAIL: ${{ steps.op-load-secret.outputs.UNITY_EMAIL }}
|
|
||||||
UNITY_PASSWORD: ${{ steps.op-load-secret.outputs.UNITY_PASSWORD }}
|
|
||||||
UNITY_SERIAL: ${{ steps.op-load-secret.outputs.UNITY_SERIAL }}
|
|
||||||
with:
|
|
||||||
projectPath: src/UniTask
|
|
||||||
unityVersion: ${{ matrix.unity }}
|
|
||||||
targetPlatform: StandaloneLinux64
|
|
||||||
buildMethod: PackageExporter.Export
|
|
||||||
|
|
||||||
- uses: Cysharp/Actions/.github/actions/check-metas@main # check meta files
|
|
||||||
with:
|
|
||||||
directory: src/UniTask
|
|
||||||
|
|
||||||
# Store artifacts.
|
|
||||||
- uses: Cysharp/Actions/.github/actions/upload-artifact@main
|
|
||||||
with:
|
|
||||||
name: UniTask.${{ inputs.tag }}.unitypackage
|
|
||||||
path: ./src/UniTask/UniTask.${{ inputs.tag }}.unitypackage
|
|
||||||
retention-days: 1
|
|
||||||
|
|
||||||
# release
|
|
||||||
create-release:
|
|
||||||
needs: [update-packagejson, build-dotnet, build-unity]
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
uses: Cysharp/Actions/.github/workflows/create-release.yaml@main
|
|
||||||
with:
|
|
||||||
commit-id: ${{ needs.update-packagejson.outputs.sha }}
|
|
||||||
dry-run: ${{ inputs.dry-run }}
|
|
||||||
tag: ${{ inputs.tag }}
|
|
||||||
nuget-push: true
|
|
||||||
release-upload: true
|
|
||||||
release-asset-path: ./UniTask.${{ inputs.tag }}.unitypackage/UniTask.${{ inputs.tag }}.unitypackage
|
|
||||||
secrets: inherit
|
|
||||||
|
|
||||||
cleanup:
|
|
||||||
if: ${{ needs.update-packagejson.outputs.is-branch-created == 'true' }}
|
|
||||||
needs: [update-packagejson, build-dotnet, build-unity]
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
uses: Cysharp/Actions/.github/workflows/clean-packagejson-branch.yaml@main
|
|
||||||
with:
|
|
||||||
branch: ${{ needs.update-packagejson.outputs.branch-name }}
|
|
|
@ -1,12 +0,0 @@
|
||||||
name: Prevent github change
|
|
||||||
on:
|
|
||||||
pull_request:
|
|
||||||
paths:
|
|
||||||
- ".github/**/*.yaml"
|
|
||||||
- ".github/**/*.yml"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
detect:
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
uses: Cysharp/Actions/.github/workflows/prevent-github-change.yaml@main
|
|
|
@ -1,14 +0,0 @@
|
||||||
name: "Close stale issues"
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
schedule:
|
|
||||||
- cron: "0 0 * * *"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
stale:
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
pull-requests: write
|
|
||||||
issues: write
|
|
||||||
uses: Cysharp/Actions/.github/workflows/stale-issue.yaml@main
|
|
|
@ -1,15 +0,0 @@
|
||||||
name: TOC Generator
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
paths:
|
|
||||||
- 'README.md'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
toc:
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
uses: Cysharp/Actions/.github/workflows/toc-generator.yaml@main
|
|
||||||
with:
|
|
||||||
TOC_TITLE: "## Table of Contents"
|
|
||||||
secrets: inherit
|
|
|
@ -80,7 +80,7 @@ _ReSharper*
|
||||||
*.ncrunch*
|
*.ncrunch*
|
||||||
.*crunch*.local.xml
|
.*crunch*.local.xml
|
||||||
|
|
||||||
# Installshield output folder
|
# Installshield output folder
|
||||||
[Ee]xpress
|
[Ee]xpress
|
||||||
|
|
||||||
# DocProject is a documentation generator add-in
|
# DocProject is a documentation generator add-in
|
||||||
|
@ -130,15 +130,30 @@ UpgradeLog*.XML
|
||||||
Assets/WSATestCertificate.pfx
|
Assets/WSATestCertificate.pfx
|
||||||
.vs/
|
.vs/
|
||||||
|
|
||||||
# Unity
|
Assembly-CSharp\.csproj
|
||||||
|
|
||||||
# Unity
|
UniRx\.Async\.csproj
|
||||||
.vsconfig
|
|
||||||
src/UniTask/Library/*
|
UniRx\.Async\.Editor\.csproj
|
||||||
src/UniTask/Temp/*
|
|
||||||
src/UniTask/Logs/*
|
UniRx\.Async\.Tests\.csproj
|
||||||
src/UniTask/[Uu]ser[Ss]ettings/
|
|
||||||
src/UniTask/*.sln
|
UniTask\.sln
|
||||||
src/UniTask/*.csproj
|
|
||||||
src/UniTask/*.unitypackage
|
RuntimeUnitTestToolkit\.csproj
|
||||||
!src/UniTask/Packages/
|
|
||||||
|
Assembly-CSharp-Editor\.csproj
|
||||||
|
|
||||||
|
UniRx\.Async\.unitypackage
|
||||||
|
|
||||||
|
UniRx.Async.Tests.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/UniTask.csproj
|
||||||
|
|
||||||
|
src/UniTask/UniTask.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/UniTask.Tests.csproj
|
||||||
|
|
||||||
|
src/UniTask/UniTask.Tests.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/UniTask.*.unitypackage
|
||||||
|
|
1170
README_CN.md
1170
README_CN.md
File diff suppressed because it is too large
Load Diff
|
@ -1,15 +1,13 @@
|
||||||
|
|
||||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
# Visual Studio Version 17
|
# Visual Studio Version 16
|
||||||
VisualStudioVersion = 17.0.31606.5
|
VisualStudioVersion = 16.0.29613.14
|
||||||
MinimumVisualStudioVersion = 10.0.40219.1
|
MinimumVisualStudioVersion = 10.0.40219.1
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "UniTask.NetCoreTests", "src\UniTask.NetCoreTests\UniTask.NetCoreTests.csproj", "{B3E311A4-70D8-4131-9965-C073A99D201A}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UniTask.NetCoreTests", "src\UniTask.NetCoreTests\UniTask.NetCoreTests.csproj", "{B3E311A4-70D8-4131-9965-C073A99D201A}"
|
||||||
EndProject
|
EndProject
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "UniTask.NetCore", "src\UniTask.NetCore\UniTask.NetCore.csproj", "{16EE20D0-7FB1-483A-8467-A5EEDBF1F5BF}"
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "UniTask.NetCore", "src\UniTask.NetCore\UniTask.NetCore.csproj", "{16EE20D0-7FB1-483A-8467-A5EEDBF1F5BF}"
|
||||||
EndProject
|
EndProject
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "UniTask.NetCoreSandbox", "src\UniTask.NetCoreSandbox\UniTask.NetCoreSandbox.csproj", "{3915E72E-33E0-4A14-A6D8-872702200E58}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UniTask.NetCoreSandbox", "src\UniTask.NetCoreSandbox\UniTask.NetCoreSandbox.csproj", "{3915E72E-33E0-4A14-A6D8-872702200E58}"
|
||||||
EndProject
|
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UniTask.Analyzer", "src\UniTask.Analyzer\UniTask.Analyzer.csproj", "{0AC6F052-A255-4EE3-9E05-1C02D49AB1C2}"
|
|
||||||
EndProject
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
|
@ -29,10 +27,6 @@ Global
|
||||||
{3915E72E-33E0-4A14-A6D8-872702200E58}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{3915E72E-33E0-4A14-A6D8-872702200E58}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{3915E72E-33E0-4A14-A6D8-872702200E58}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{3915E72E-33E0-4A14-A6D8-872702200E58}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{3915E72E-33E0-4A14-A6D8-872702200E58}.Release|Any CPU.Build.0 = Release|Any CPU
|
{3915E72E-33E0-4A14-A6D8-872702200E58}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
{0AC6F052-A255-4EE3-9E05-1C02D49AB1C2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
|
||||||
{0AC6F052-A255-4EE3-9E05-1C02D49AB1C2}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
|
||||||
{0AC6F052-A255-4EE3-9E05-1C02D49AB1C2}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
|
||||||
{0AC6F052-A255-4EE3-9E05-1C02D49AB1C2}.Release|Any CPU.Build.0 = Release|Any CPU
|
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
HideSolutionNode = FALSE
|
HideSolutionNode = FALSE
|
||||||
|
|
|
@ -1,10 +0,0 @@
|
||||||
###############
|
|
||||||
# folder #
|
|
||||||
###############
|
|
||||||
/**/DROP/
|
|
||||||
/**/TEMP/
|
|
||||||
/**/packages/
|
|
||||||
/**/bin/
|
|
||||||
/**/obj/
|
|
||||||
_site
|
|
||||||
_DocfxTemplate
|
|
|
@ -1,5 +0,0 @@
|
||||||
###############
|
|
||||||
# temp file #
|
|
||||||
###############
|
|
||||||
*.yml
|
|
||||||
.manifest
|
|
|
@ -1,70 +0,0 @@
|
||||||
{
|
|
||||||
"metadata": [
|
|
||||||
{
|
|
||||||
"src": [
|
|
||||||
{
|
|
||||||
"files": [
|
|
||||||
"UniTask/Assets/Plugins/UniTask/Runtime/**/*.cs"
|
|
||||||
],
|
|
||||||
"src": "../src"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"dest": "api",
|
|
||||||
"disableGitFeatures": false,
|
|
||||||
"disableDefaultFilter": false
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"build": {
|
|
||||||
"globalMetadata": {
|
|
||||||
"_disableContribution": true,
|
|
||||||
"_appTitle": "UniTask"
|
|
||||||
},
|
|
||||||
"content": [
|
|
||||||
{
|
|
||||||
"files": [
|
|
||||||
"api/**.yml",
|
|
||||||
"api/index.md"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"files": [
|
|
||||||
"articles/**.md",
|
|
||||||
"articles/**/toc.yml",
|
|
||||||
"toc.yml",
|
|
||||||
"*.md"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"resource": [
|
|
||||||
{
|
|
||||||
"files": [
|
|
||||||
"images/**"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"overwrite": [
|
|
||||||
{
|
|
||||||
"files": [
|
|
||||||
"apidoc/**.md"
|
|
||||||
],
|
|
||||||
"exclude": [
|
|
||||||
"obj/**",
|
|
||||||
"_site/**"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"dest": "_site",
|
|
||||||
|
|
||||||
"globalMetadataFiles": [],
|
|
||||||
"fileMetadataFiles": [],
|
|
||||||
"template": [
|
|
||||||
"_DocfxTemplate/templates/default-v2.5.2",
|
|
||||||
"_DocfxTemplate/templates/cysharp"
|
|
||||||
],
|
|
||||||
"postProcessors": [],
|
|
||||||
"markdownEngineName": "markdig",
|
|
||||||
"noLangKeyword": false,
|
|
||||||
"keepFileLink": false,
|
|
||||||
"cleanupCacheHistory": false
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,8 +0,0 @@
|
||||||
---
|
|
||||||
title: Home
|
|
||||||
---
|
|
||||||
# UniTask
|
|
||||||
|
|
||||||
Provides an efficient async/await integration to Unity.
|
|
||||||
|
|
||||||
https://github.com/Cysharp/UniTask
|
|
11
docs/toc.yml
11
docs/toc.yml
|
@ -1,11 +0,0 @@
|
||||||
- name: API Documentation
|
|
||||||
href: api/
|
|
||||||
homepage: api/Cysharp.Threading.Tasks.html
|
|
||||||
|
|
||||||
- name: Repository
|
|
||||||
href: https://github.com/Cysharp/UniTask
|
|
||||||
homepage: https://github.com/Cysharp/UniTask
|
|
||||||
|
|
||||||
- name: Releases
|
|
||||||
href: https://github.com/Cysharp/UniTask/releases
|
|
||||||
homepage: https://github.com/Cysharp/UniTask/releases
|
|
|
@ -1,8 +0,0 @@
|
||||||
{
|
|
||||||
"profiles": {
|
|
||||||
"UniTask.Analyzer": {
|
|
||||||
"commandName": "DebugRoslynComponent",
|
|
||||||
"targetProject": "..\\UniTask.NetCoreSandbox\\UniTask.NetCoreSandbox.csproj"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,29 +0,0 @@
|
||||||
<Project Sdk="Microsoft.NET.Sdk">
|
|
||||||
<PropertyGroup>
|
|
||||||
<OutputType>library</OutputType>
|
|
||||||
<TargetFramework>netstandard2.0</TargetFramework>
|
|
||||||
<LangVersion>latest</LangVersion>
|
|
||||||
<Nullable>enable</Nullable>
|
|
||||||
<IsRoslynComponent>true</IsRoslynComponent>
|
|
||||||
<TargetsForTfmSpecificContentInPackage>$(TargetsForTfmSpecificContentInPackage);PackBuildOutputs</TargetsForTfmSpecificContentInPackage>
|
|
||||||
<IncludeBuildOutput>false</IncludeBuildOutput>
|
|
||||||
<IncludeSymbols>false</IncludeSymbols>
|
|
||||||
<SuppressDependenciesWhenPacking>true</SuppressDependenciesWhenPacking>
|
|
||||||
<DevelopmentDependency>true</DevelopmentDependency>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.2">
|
|
||||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets>
|
|
||||||
<PrivateAssets>all</PrivateAssets>
|
|
||||||
</PackageReference>
|
|
||||||
<PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="3.8.0" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<Target Name="PackBuildOutputs" DependsOnTargets="SatelliteDllsProjectOutputGroup;DebugSymbolsProjectOutputGroup">
|
|
||||||
<ItemGroup>
|
|
||||||
<TfmSpecificPackageFile Include="$(TargetDir)\*.dll" PackagePath="analyzers\dotnet\cs" />
|
|
||||||
<TfmSpecificPackageFile Include="@(SatelliteDllsProjectOutputGroupOutput->'%(FinalOutputPath)')" PackagePath="analyzers\dotnet\cs\%(SatelliteDllsProjectOutputGroupOutput.Culture)\" />
|
|
||||||
</ItemGroup>
|
|
||||||
</Target>
|
|
||||||
</Project>
|
|
|
@ -1,54 +0,0 @@
|
||||||
#pragma warning disable RS2008
|
|
||||||
|
|
||||||
using Microsoft.CodeAnalysis;
|
|
||||||
using Microsoft.CodeAnalysis.Diagnostics;
|
|
||||||
using Microsoft.CodeAnalysis.Operations;
|
|
||||||
using System.Collections.Immutable;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace UniTask.Analyzer
|
|
||||||
{
|
|
||||||
[DiagnosticAnalyzer(LanguageNames.CSharp)]
|
|
||||||
public class UniTaskAnalyzer : DiagnosticAnalyzer
|
|
||||||
{
|
|
||||||
private static readonly DiagnosticDescriptor Rule = new DiagnosticDescriptor(
|
|
||||||
id: "UNITASK001",
|
|
||||||
title: "UniTaskAnalyzer001: Must pass CancellationToken",
|
|
||||||
messageFormat: "Must pass CancellationToken",
|
|
||||||
category: "Usage",
|
|
||||||
defaultSeverity: DiagnosticSeverity.Error,
|
|
||||||
isEnabledByDefault: true,
|
|
||||||
description: "Pass CancellationToken or CancellationToken.None.");
|
|
||||||
|
|
||||||
public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics { get { return ImmutableArray.Create(Rule); } }
|
|
||||||
|
|
||||||
public override void Initialize(AnalysisContext context)
|
|
||||||
{
|
|
||||||
context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None);
|
|
||||||
context.EnableConcurrentExecution();
|
|
||||||
|
|
||||||
context.RegisterOperationAction(AnalyzeOperation, OperationKind.Invocation);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void AnalyzeOperation(OperationAnalysisContext context)
|
|
||||||
{
|
|
||||||
var token = context.Compilation.GetTypeByMetadataName(typeof(CancellationToken).FullName);
|
|
||||||
if (token == null) return;
|
|
||||||
|
|
||||||
if (context.Operation is IInvocationOperation invocation)
|
|
||||||
{
|
|
||||||
foreach (var arg in invocation.Arguments)
|
|
||||||
{
|
|
||||||
if (arg.ArgumentKind == ArgumentKind.DefaultValue)
|
|
||||||
{
|
|
||||||
if (SymbolEqualityComparer.Default.Equals(arg.Parameter.Type, token))
|
|
||||||
{
|
|
||||||
var diagnostic = Diagnostic.Create(Rule, arg.Syntax.GetLocation());
|
|
||||||
context.ReportDiagnostic(diagnostic);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Binary file not shown.
Before Width: | Height: | Size: 3.1 KiB |
|
@ -1,101 +0,0 @@
|
||||||
#if !NETSTANDARD2_0
|
|
||||||
|
|
||||||
#pragma warning disable 0649
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using System.Threading.Tasks.Sources;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public static class AsyncEnumerableExtensions
|
|
||||||
{
|
|
||||||
public static IUniTaskAsyncEnumerable<T> AsUniTaskAsyncEnumerable<T>(this IAsyncEnumerable<T> source)
|
|
||||||
{
|
|
||||||
return new AsyncEnumerableToUniTaskAsyncEnumerable<T>(source);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncEnumerable<T> AsAsyncEnumerable<T>(this IUniTaskAsyncEnumerable<T> source)
|
|
||||||
{
|
|
||||||
return new UniTaskAsyncEnumerableToAsyncEnumerable<T>(source);
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class AsyncEnumerableToUniTaskAsyncEnumerable<T> : IUniTaskAsyncEnumerable<T>
|
|
||||||
{
|
|
||||||
readonly IAsyncEnumerable<T> source;
|
|
||||||
|
|
||||||
public AsyncEnumerableToUniTaskAsyncEnumerable(IAsyncEnumerable<T> source)
|
|
||||||
{
|
|
||||||
this.source = source;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
return new Enumerator(source.GetAsyncEnumerator(cancellationToken));
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class Enumerator : IUniTaskAsyncEnumerator<T>
|
|
||||||
{
|
|
||||||
readonly IAsyncEnumerator<T> enumerator;
|
|
||||||
|
|
||||||
public Enumerator(IAsyncEnumerator<T> enumerator)
|
|
||||||
{
|
|
||||||
this.enumerator = enumerator;
|
|
||||||
}
|
|
||||||
|
|
||||||
public T Current => enumerator.Current;
|
|
||||||
|
|
||||||
public async UniTask DisposeAsync()
|
|
||||||
{
|
|
||||||
await enumerator.DisposeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async UniTask<bool> MoveNextAsync()
|
|
||||||
{
|
|
||||||
return await enumerator.MoveNextAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class UniTaskAsyncEnumerableToAsyncEnumerable<T> : IAsyncEnumerable<T>
|
|
||||||
{
|
|
||||||
readonly IUniTaskAsyncEnumerable<T> source;
|
|
||||||
|
|
||||||
public UniTaskAsyncEnumerableToAsyncEnumerable(IUniTaskAsyncEnumerable<T> source)
|
|
||||||
{
|
|
||||||
this.source = source;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
return new Enumerator(source.GetAsyncEnumerator(cancellationToken));
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class Enumerator : IAsyncEnumerator<T>
|
|
||||||
{
|
|
||||||
readonly IUniTaskAsyncEnumerator<T> enumerator;
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerator<T> enumerator)
|
|
||||||
{
|
|
||||||
this.enumerator = enumerator;
|
|
||||||
}
|
|
||||||
|
|
||||||
public T Current => enumerator.Current;
|
|
||||||
|
|
||||||
public ValueTask DisposeAsync()
|
|
||||||
{
|
|
||||||
return enumerator.DisposeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public ValueTask<bool> MoveNextAsync()
|
|
||||||
{
|
|
||||||
return enumerator.MoveNextAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -0,0 +1,123 @@
|
||||||
|
#pragma warning disable 0649
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Threading.Tasks.Sources;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks
|
||||||
|
{
|
||||||
|
public static class UniTaskValueTaskExtensions
|
||||||
|
{
|
||||||
|
public static ValueTask AsValueTask(this UniTask task)
|
||||||
|
{
|
||||||
|
ref var core = ref Unsafe.As<UniTask, UniTaskToValueTask>(ref task);
|
||||||
|
if (core.source == null)
|
||||||
|
{
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ValueTask(new UniTaskValueTaskSource(core.source), core.token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ValueTask<T> AsValueTask<T>(this UniTask<T> task)
|
||||||
|
{
|
||||||
|
ref var core = ref Unsafe.As<UniTask<T>, UniTaskToValueTask<T>>(ref task);
|
||||||
|
if (core.source == null)
|
||||||
|
{
|
||||||
|
return new ValueTask<T>(core.result);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ValueTask<T>(new UniTaskValueTaskSource<T>(core.source), core.token);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct UniTaskToValueTask
|
||||||
|
{
|
||||||
|
public IUniTaskSource source;
|
||||||
|
public short token;
|
||||||
|
}
|
||||||
|
|
||||||
|
class UniTaskValueTaskSource : IValueTaskSource
|
||||||
|
{
|
||||||
|
readonly IUniTaskSource source;
|
||||||
|
|
||||||
|
public UniTaskValueTaskSource(IUniTaskSource source)
|
||||||
|
{
|
||||||
|
this.source = source;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void GetResult(short token)
|
||||||
|
{
|
||||||
|
source.GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ValueTaskSourceStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
var status = source.GetStatus(token);
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
return ValueTaskSourceStatus.Pending;
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
return ValueTaskSourceStatus.Succeeded;
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
return ValueTaskSourceStatus.Faulted;
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
return ValueTaskSourceStatus.Canceled;
|
||||||
|
default:
|
||||||
|
return (ValueTaskSourceStatus)status;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token, ValueTaskSourceOnCompletedFlags flags)
|
||||||
|
{
|
||||||
|
source.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct UniTaskToValueTask<T>
|
||||||
|
{
|
||||||
|
public IUniTaskSource<T> source;
|
||||||
|
public T result;
|
||||||
|
public short token;
|
||||||
|
}
|
||||||
|
|
||||||
|
class UniTaskValueTaskSource<T> : IValueTaskSource<T>
|
||||||
|
{
|
||||||
|
readonly IUniTaskSource<T> source;
|
||||||
|
|
||||||
|
public UniTaskValueTaskSource(IUniTaskSource<T> source)
|
||||||
|
{
|
||||||
|
this.source = source;
|
||||||
|
}
|
||||||
|
|
||||||
|
public T GetResult(short token)
|
||||||
|
{
|
||||||
|
return source.GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ValueTaskSourceStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
var status = source.GetStatus(token);
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
return ValueTaskSourceStatus.Pending;
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
return ValueTaskSourceStatus.Succeeded;
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
return ValueTaskSourceStatus.Faulted;
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
return ValueTaskSourceStatus.Canceled;
|
||||||
|
default:
|
||||||
|
return (ValueTaskSourceStatus)status;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token, ValueTaskSourceOnCompletedFlags flags)
|
||||||
|
{
|
||||||
|
source.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,39 +0,0 @@
|
||||||
//using Cysharp.Threading.Tasks.Internal;
|
|
||||||
//using System;
|
|
||||||
//using System.Collections.Concurrent;
|
|
||||||
//using System.Runtime.CompilerServices;
|
|
||||||
//using System.Threading;
|
|
||||||
|
|
||||||
//namespace Cysharp.Threading.Tasks
|
|
||||||
//{
|
|
||||||
// public partial struct UniTask
|
|
||||||
// {
|
|
||||||
// public static UniTask Delay()
|
|
||||||
// {
|
|
||||||
// return default;
|
|
||||||
// }
|
|
||||||
|
|
||||||
// sealed class DelayPromise : IUniTaskSource
|
|
||||||
// {
|
|
||||||
// public void GetResult(short token)
|
|
||||||
// {
|
|
||||||
// throw new NotImplementedException();
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public UniTaskStatus GetStatus(short token)
|
|
||||||
// {
|
|
||||||
// throw new NotImplementedException();
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
// {
|
|
||||||
// throw new NotImplementedException();
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public UniTaskStatus UnsafeGetStatus()
|
|
||||||
// {
|
|
||||||
// throw new NotImplementedException();
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
//}
|
|
|
@ -1,6 +1,4 @@
|
||||||
using Cysharp.Threading.Tasks.Internal;
|
using System;
|
||||||
using System;
|
|
||||||
using System.Collections.Concurrent;
|
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
|
||||||
|
@ -43,11 +41,7 @@ namespace Cysharp.Threading.Tasks
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
#if NETCOREAPP3_1
|
|
||||||
ThreadPool.UnsafeQueueUserWorkItem(ThreadPoolWorkItem.Create(continuation), false);
|
|
||||||
#else
|
|
||||||
ThreadPool.UnsafeQueueUserWorkItem(WaitCallbackDelegate, continuation);
|
ThreadPool.UnsafeQueueUserWorkItem(WaitCallbackDelegate, continuation);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -56,48 +50,6 @@ namespace Cysharp.Threading.Tasks
|
||||||
((Action)state).Invoke();
|
((Action)state).Invoke();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if NETCOREAPP3_1
|
|
||||||
|
|
||||||
sealed class ThreadPoolWorkItem : IThreadPoolWorkItem, ITaskPoolNode<ThreadPoolWorkItem>
|
|
||||||
{
|
|
||||||
static TaskPool<ThreadPoolWorkItem> pool;
|
|
||||||
ThreadPoolWorkItem nextNode;
|
|
||||||
public ref ThreadPoolWorkItem NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static ThreadPoolWorkItem()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(ThreadPoolWorkItem), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
Action continuation;
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public static ThreadPoolWorkItem Create(Action continuation)
|
|
||||||
{
|
|
||||||
if (!pool.TryPop(out var item))
|
|
||||||
{
|
|
||||||
item = new ThreadPoolWorkItem();
|
|
||||||
}
|
|
||||||
|
|
||||||
item.continuation = continuation;
|
|
||||||
return item;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void Execute()
|
|
||||||
{
|
|
||||||
var call = continuation;
|
|
||||||
continuation = null;
|
|
||||||
if (call != null)
|
|
||||||
{
|
|
||||||
pool.TryPush(this);
|
|
||||||
call.Invoke();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,43 +1,37 @@
|
||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFrameworks>net6.0;net7.0;netstandard2.1;netstandard2.0</TargetFrameworks>
|
<TargetFramework>netstandard2.0</TargetFramework>
|
||||||
<AssemblyName>UniTask</AssemblyName>
|
<AssemblyName>UniTask</AssemblyName>
|
||||||
<LangVersion>8.0</LangVersion>
|
<RootNamespace>Cysharp.Threading.Tasks</RootNamespace>
|
||||||
<RootNamespace>Cysharp.Threading.Tasks</RootNamespace>
|
</PropertyGroup>
|
||||||
<DefineConstants>UNITASK_NETCORE</DefineConstants>
|
|
||||||
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
|
||||||
<NoWarn>$(NoWarn);CS1591</NoWarn>
|
|
||||||
|
|
||||||
<!-- NuGet Packaging -->
|
<ItemGroup>
|
||||||
<Id>UniTask</Id>
|
<Compile Include="..\UniTask\Assets\Plugins\UniTask\Runtime\**\*.cs"
|
||||||
<PackageVersion>$(Version)</PackageVersion>
|
Exclude="
|
||||||
<Company>Cysharp</Company>
|
..\UniTask\Assets\Plugins\UniTask\Editor\*.cs;
|
||||||
<Authors>Cysharp</Authors>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Triggers\*.cs;
|
||||||
<Copyright>© Cysharp, Inc.</Copyright>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Linq\UnityExtensions\*.cs;
|
||||||
<PackageTags>task;async</PackageTags>
|
|
||||||
<Description>Provides an efficient async/await integration to Unity and .NET Core.</Description>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\UnityEqualityComparer.cs;
|
||||||
<PackageProjectUrl>https://github.com/Cysharp/UniTask</PackageProjectUrl>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\DiagnosticsExtensions.cs;
|
||||||
<RepositoryUrl>$(PackageProjectUrl)</RepositoryUrl>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\PlayerLoopRunner.cs;
|
||||||
<RepositoryType>git</RepositoryType>
|
|
||||||
<PackageLicenseExpression>MIT</PackageLicenseExpression>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\CancellationTokenSourceExtensions.cs;
|
||||||
<PackageIcon>Icon.png</PackageIcon>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\EnumeratorAsyncExtensions.cs;
|
||||||
<SignAssembly>true</SignAssembly>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\PlayerLoopHelper.cs;
|
||||||
<AssemblyOriginatorKeyFile>opensource.snk</AssemblyOriginatorKeyFile>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.Delay.cs;
|
||||||
<IsPackable>true</IsPackable>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.Run.cs;
|
||||||
</PropertyGroup>
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.Bridge.cs;
|
||||||
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.WaitUntil.cs;
|
||||||
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.cs;
|
||||||
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.uGUI.cs;
|
||||||
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.MonoBehaviour.cs;
|
||||||
|
" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<None Include="Icon.png" Pack="true" PackagePath="/" />
|
<PackageReference Include="System.Threading.Tasks.Extensions" Version="4.5.4" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<Compile Include="..\UniTask\Assets\Plugins\UniTask\Runtime\**\*.cs" Exclude="
..\UniTask\Assets\Plugins\UniTask\Editor\*.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\Triggers\*.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\Linq\UnityExtensions\*.cs;
 
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\UnityEqualityComparer.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\DiagnosticsExtensions.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\PlayerLoopRunner.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\ContinuationQueue.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\UnityWebRequestExtensions.cs;
 
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTaskSynchronizationContext.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\CancellationTokenSourceExtensions.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\EnumeratorAsyncExtensions.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\TimeoutController.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\PlayerLoopHelper.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\PlayerLoopTimer.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.Delay.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.Run.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.Bridge.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\UniTask.WaitUntil.cs;
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.*;
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityBindingExtensions.cs;
" />
|
|
||||||
<Compile Remove="..\UniTask\Assets\Plugins\UniTask\Runtime\_InternalVisibleTo.cs" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="System.Threading.Tasks.Extensions" Version="4.5.4" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|
Binary file not shown.
|
@ -1,6 +1,4 @@
|
||||||
#pragma warning disable CS1998
|
using Cysharp.Threading.Tasks;
|
||||||
|
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System;
|
using System;
|
||||||
|
@ -17,34 +15,229 @@ using System.Reactive.Concurrency;
|
||||||
|
|
||||||
namespace NetCoreSandbox
|
namespace NetCoreSandbox
|
||||||
{
|
{
|
||||||
public class Program
|
class Program
|
||||||
{
|
{
|
||||||
|
static string FlattenGenArgs(Type type)
|
||||||
|
{
|
||||||
|
if (type.IsGenericType)
|
||||||
|
{
|
||||||
|
var t = string.Join(", ", type.GetGenericArguments().Select(x => FlattenGenArgs(x)));
|
||||||
|
return Regex.Replace(type.Name, "`.+", "") + "<" + t + ">";
|
||||||
|
}
|
||||||
|
//x.ReturnType.GetGenericArguments()
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return type.Name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static async IAsyncEnumerable<int> FooAsync([EnumeratorCancellation]CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
yield return 1;
|
||||||
|
await Task.Delay(10, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
static async Task Main(string[] args)
|
static async Task Main(string[] args)
|
||||||
{
|
{
|
||||||
var cts = new CancellationTokenSource();
|
await foreach (var item in UniTaskAsyncEnumerable.Range(1, 10)
|
||||||
|
.SelectAwait(x => UniTask.Run(() => x))
|
||||||
|
.TakeLast(6)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
Console.WriteLine(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
// AsyncEnumerable.Range(1,10).FirstAsync(
|
||||||
|
// AsyncEnumerable.Range(1, 10).GroupBy(x=>x).Select(x=>x.first
|
||||||
|
|
||||||
|
|
||||||
// OK.
|
// AsyncEnumerable.Range(1,10).WithCancellation(CancellationToken.None).WithCancellation
|
||||||
await FooAsync(10, cts.Token);
|
|
||||||
|
|
||||||
// NG(Compiler Error)
|
|
||||||
// await FooAsync(10);
|
//Enumerable.Range(1,10).ToHashSet(
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void Foo()
|
||||||
|
{
|
||||||
|
|
||||||
|
// AsyncEnumerable.Range(1,10).Do(
|
||||||
|
|
||||||
|
// AsyncEnumerable.t
|
||||||
|
|
||||||
|
var sb = new StringBuilder();
|
||||||
|
sb.AppendLine(@"using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.Linq
|
||||||
|
{
|
||||||
|
");
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
var chako = typeof(AsyncEnumerable).GetMethods()
|
||||||
|
.OrderBy(x => x.Name)
|
||||||
|
.Select(x =>
|
||||||
|
{
|
||||||
|
var ret = FlattenGenArgs(x.ReturnType);
|
||||||
|
|
||||||
|
|
||||||
|
var generics = string.Join(", ", x.GetGenericArguments().Select(x => x.Name));
|
||||||
|
|
||||||
|
if (x.GetParameters().Length == 0) return "";
|
||||||
|
|
||||||
|
var self = x.GetParameters().First();
|
||||||
|
if (x.GetCustomAttributes(typeof(ExtensionAttribute), true).Length == 0)
|
||||||
|
{
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
var arg1Type = FlattenGenArgs(x.GetParameters().First().ParameterType);
|
||||||
|
|
||||||
|
var others = string.Join(", ", x.GetParameters().Skip(1).Select(y => FlattenGenArgs(y.ParameterType) + " " + y.Name));
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(others))
|
||||||
|
{
|
||||||
|
others = ", " + others;
|
||||||
|
}
|
||||||
|
|
||||||
|
var template = $"public static {ret} {x.Name}<{generics}>(this {arg1Type} {self.Name}{others})";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
return template.Replace("ValueTask", "UniTask").Replace("IAsyncEnumerable", "IUniTaskAsyncEnumerable").Replace("<>", "");
|
||||||
|
})
|
||||||
|
.Where(x => x != "")
|
||||||
|
.Select(x => x + "\r\n{\r\n throw new NotImplementedException();\r\n}")
|
||||||
|
.ToArray();
|
||||||
|
|
||||||
|
var huga = string.Join("\r\n\r\n", chako);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
foreach (var item in typeof(AsyncEnumerable).GetMethods().Select(x => x.Name).Distinct())
|
||||||
|
{
|
||||||
|
if (item.EndsWith("AwaitAsync") || item.EndsWith("AwaitWithCancellationAsync") || item.EndsWith("WithCancellation"))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var item2 = item.Replace("Async", "");
|
||||||
|
item2 = item2.Replace("Await", "");
|
||||||
|
|
||||||
|
var format = @"
|
||||||
|
internal sealed class {0}
|
||||||
|
{{
|
||||||
|
}}
|
||||||
|
";
|
||||||
|
|
||||||
|
sb.Append(string.Format(format, item2));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
sb.Append("}");
|
||||||
|
|
||||||
|
|
||||||
|
Console.WriteLine(sb.ToString());
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static async UniTask FooAsync(int x, CancellationToken cancellationToken = default)
|
|
||||||
|
public static async IAsyncEnumerable<int> AsyncGen()
|
||||||
{
|
{
|
||||||
await UniTask.Yield();
|
await UniTask.SwitchToThreadPool();
|
||||||
|
yield return 10;
|
||||||
|
await UniTask.SwitchToThreadPool();
|
||||||
|
yield return 100;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class MyEnumerable : IEnumerable<int>
|
||||||
|
{
|
||||||
|
public IEnumerator<int> GetEnumerator()
|
||||||
|
{
|
||||||
|
return new MyEnumerator();
|
||||||
|
}
|
||||||
|
|
||||||
|
IEnumerator IEnumerable.GetEnumerator()
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MyEnumerator : IEnumerator<int>
|
||||||
|
{
|
||||||
|
public int Current => throw new NotImplementedException();
|
||||||
|
|
||||||
|
object IEnumerator.Current => throw new NotImplementedException();
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
Console.WriteLine("Called Dispose");
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool MoveNext()
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Reset()
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public class MyClass<T>
|
||||||
|
{
|
||||||
|
public CustomAsyncEnumerator<T> GetAsyncEnumerator()
|
||||||
|
{
|
||||||
|
//IAsyncEnumerable
|
||||||
|
return new CustomAsyncEnumerator<T>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public struct CustomAsyncEnumerator<T>
|
||||||
|
{
|
||||||
|
int count;
|
||||||
|
|
||||||
|
public T Current
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
|
{
|
||||||
|
if (count++ == 3)
|
||||||
|
{
|
||||||
|
return UniTask.FromResult(false);
|
||||||
|
//return false;
|
||||||
|
}
|
||||||
|
return UniTask.FromResult(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,26 +2,17 @@
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<OutputType>Exe</OutputType>
|
<OutputType>Exe</OutputType>
|
||||||
<TargetFramework>net7.0</TargetFramework>
|
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||||
<RootNamespace>NetCoreSandbox</RootNamespace>
|
<RootNamespace>NetCoreSandbox</RootNamespace>
|
||||||
<IsPackable>false</IsPackable>
|
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="BenchmarkDotNet" Version="0.12.1" />
|
|
||||||
<PackageReference Include="PooledAwait" Version="1.0.49" />
|
|
||||||
<PackageReference Include="System.Interactive.Async" Version="4.1.1" />
|
<PackageReference Include="System.Interactive.Async" Version="4.1.1" />
|
||||||
<PackageReference Include="System.Reactive" Version="4.4.1" />
|
<PackageReference Include="System.Reactive" Version="4.4.1" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\UniTask.NetCore\UniTask.NetCore.csproj" />
|
<ProjectReference Include="..\UniTask.NetCore\UniTask.NetCore.csproj" />
|
||||||
|
|
||||||
|
|
||||||
<ProjectReference Include="..\UniTask.Analyzer\UniTask.Analyzer.csproj">
|
|
||||||
<ReferenceOutputAssembly>false</ReferenceOutputAssembly>
|
|
||||||
<OutputItemType>Analyzer</OutputItemType>
|
|
||||||
</ProjectReference>
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|
|
@ -1,167 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class AsyncLazyTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task LazyLazy()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => After());
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
await a;
|
|
||||||
await b;
|
|
||||||
await c;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => AfterException());
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task LazyImmediate()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => UniTask.FromResult(1).AsUniTask());
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
await a;
|
|
||||||
await b;
|
|
||||||
await c;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => UniTask.FromException(new TaskTestException()));
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static async UniTask AwaitAwait(UniTask t)
|
|
||||||
{
|
|
||||||
await t;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
async UniTask After()
|
|
||||||
{
|
|
||||||
await UniTask.Yield();
|
|
||||||
Thread.Sleep(TimeSpan.FromSeconds(1));
|
|
||||||
await UniTask.Yield();
|
|
||||||
await UniTask.Yield();
|
|
||||||
}
|
|
||||||
|
|
||||||
async UniTask AfterException()
|
|
||||||
{
|
|
||||||
await UniTask.Yield();
|
|
||||||
Thread.Sleep(TimeSpan.FromSeconds(1));
|
|
||||||
await UniTask.Yield();
|
|
||||||
throw new TaskTestException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class AsyncLazyTest2
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task LazyLazy()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => After());
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
var a2 = await a;
|
|
||||||
var b2 = await b;
|
|
||||||
var c2 = await c;
|
|
||||||
(a2, b2, c2).Should().Be((10, 10, 10));
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => AfterException());
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task LazyImmediate()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => UniTask.FromResult(1));
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
var a2 = await a;
|
|
||||||
var b2 = await b;
|
|
||||||
var c2 = await c;
|
|
||||||
(a2, b2, c2).Should().Be((1, 1, 1));
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var l = UniTask.Lazy(() => UniTask.FromException<int>(new TaskTestException()));
|
|
||||||
var a = AwaitAwait(l.Task);
|
|
||||||
var b = AwaitAwait(l.Task);
|
|
||||||
var c = AwaitAwait(l.Task);
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static async UniTask<int> AwaitAwait(UniTask<int> t)
|
|
||||||
{
|
|
||||||
return await t;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
async UniTask<int> After()
|
|
||||||
{
|
|
||||||
await UniTask.Yield();
|
|
||||||
Thread.Sleep(TimeSpan.FromSeconds(1));
|
|
||||||
await UniTask.Yield();
|
|
||||||
await UniTask.Yield();
|
|
||||||
return 10;
|
|
||||||
}
|
|
||||||
|
|
||||||
async UniTask<int> AfterException()
|
|
||||||
{
|
|
||||||
await UniTask.Yield();
|
|
||||||
Thread.Sleep(TimeSpan.FromSeconds(1));
|
|
||||||
await UniTask.Yield();
|
|
||||||
throw new TaskTestException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,197 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class AsyncReactivePropertyTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Iteration()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(99);
|
|
||||||
|
|
||||||
var f = await rp.FirstAsync();
|
|
||||||
f.Should().Be(99);
|
|
||||||
|
|
||||||
var array = rp.Take(5).ToArrayAsync();
|
|
||||||
|
|
||||||
rp.Value = 100;
|
|
||||||
rp.Value = 100;
|
|
||||||
rp.Value = 100;
|
|
||||||
rp.Value = 131;
|
|
||||||
|
|
||||||
var ar = await array;
|
|
||||||
|
|
||||||
ar.Should().Equal(new[] { 99, 100, 100, 100, 131 });
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task WithoutCurrent()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(99);
|
|
||||||
|
|
||||||
var array = rp.WithoutCurrent().Take(5).ToArrayAsync();
|
|
||||||
|
|
||||||
rp.Value = 100;
|
|
||||||
rp.Value = 100;
|
|
||||||
rp.Value = 100;
|
|
||||||
rp.Value = 131;
|
|
||||||
rp.Value = 191;
|
|
||||||
|
|
||||||
var ar = await array;
|
|
||||||
|
|
||||||
ar.Should().Equal(new[] { 100, 100, 100, 131, 191 });
|
|
||||||
}
|
|
||||||
|
|
||||||
//[Fact]
|
|
||||||
//public async Task StateIteration()
|
|
||||||
//{
|
|
||||||
// var rp = new ReadOnlyAsyncReactiveProperty<int>(99);
|
|
||||||
// var setter = rp.GetSetter();
|
|
||||||
|
|
||||||
// var f = await rp.FirstAsync();
|
|
||||||
// f.Should().Be(99);
|
|
||||||
|
|
||||||
// var array = rp.Take(5).ToArrayAsync();
|
|
||||||
|
|
||||||
// setter(100);
|
|
||||||
// setter(100);
|
|
||||||
// setter(100);
|
|
||||||
// setter(131);
|
|
||||||
|
|
||||||
// var ar = await array;
|
|
||||||
|
|
||||||
// ar.Should().Equal(new[] { 99, 100, 100, 100, 131 });
|
|
||||||
//}
|
|
||||||
|
|
||||||
//[Fact]
|
|
||||||
//public async Task StateWithoutCurrent()
|
|
||||||
//{
|
|
||||||
// var rp = new ReadOnlyAsyncReactiveProperty<int>(99);
|
|
||||||
// var setter = rp.GetSetter();
|
|
||||||
|
|
||||||
// var array = rp.WithoutCurrent().Take(5).ToArrayAsync();
|
|
||||||
// setter(100);
|
|
||||||
// setter(100);
|
|
||||||
// setter(100);
|
|
||||||
// setter(131);
|
|
||||||
// setter(191);
|
|
||||||
|
|
||||||
// var ar = await array;
|
|
||||||
|
|
||||||
// ar.Should().Equal(new[] { 100, 100, 100, 131, 191 });
|
|
||||||
//}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void StateFromEnumeration()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(10);
|
|
||||||
|
|
||||||
var state = rp.ToReadOnlyAsyncReactiveProperty(CancellationToken.None);
|
|
||||||
|
|
||||||
rp.Value = 10;
|
|
||||||
state.Value.Should().Be(10);
|
|
||||||
|
|
||||||
rp.Value = 20;
|
|
||||||
state.Value.Should().Be(20);
|
|
||||||
|
|
||||||
state.Dispose();
|
|
||||||
|
|
||||||
rp.Value = 30;
|
|
||||||
state.Value.Should().Be(20);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task WaitAsyncTest()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(128);
|
|
||||||
|
|
||||||
var f = await rp.FirstAsync();
|
|
||||||
f.Should().Be(128);
|
|
||||||
|
|
||||||
{
|
|
||||||
var t = rp.WaitAsync();
|
|
||||||
rp.Value = 99;
|
|
||||||
rp.Value = 100;
|
|
||||||
var v = await t;
|
|
||||||
|
|
||||||
v.Should().Be(99);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var t = rp.WaitAsync();
|
|
||||||
rp.Value = 99;
|
|
||||||
rp.Value = 100;
|
|
||||||
var v = await t;
|
|
||||||
|
|
||||||
v.Should().Be(99);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task WaitAsyncCancellationTest()
|
|
||||||
{
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var rp = new AsyncReactiveProperty<int>(128);
|
|
||||||
|
|
||||||
var t = rp.WaitAsync(cts.Token);
|
|
||||||
|
|
||||||
cts.Cancel();
|
|
||||||
|
|
||||||
rp.Value = 99;
|
|
||||||
rp.Value = 100;
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => { await t; });
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task ReadOnlyWaitAsyncTest()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(128);
|
|
||||||
var rrp = rp.ToReadOnlyAsyncReactiveProperty(CancellationToken.None);
|
|
||||||
|
|
||||||
var t = rrp.WaitAsync();
|
|
||||||
rp.Value = 99;
|
|
||||||
rp.Value = 100;
|
|
||||||
var v = await t;
|
|
||||||
|
|
||||||
v.Should().Be(99);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task ReadOnlyWaitAsyncCancellationTest()
|
|
||||||
{
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var rp = new AsyncReactiveProperty<int>(128);
|
|
||||||
var rrp = rp.ToReadOnlyAsyncReactiveProperty(CancellationToken.None);
|
|
||||||
|
|
||||||
var t = rrp.WaitAsync(cts.Token);
|
|
||||||
|
|
||||||
cts.Cancel();
|
|
||||||
|
|
||||||
rp.Value = 99;
|
|
||||||
rp.Value = 100;
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => { await t; });
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,51 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class CancellationTokenTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task WaitUntilCanceled()
|
|
||||||
{
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
cts.CancelAfter(TimeSpan.FromSeconds(1.5));
|
|
||||||
|
|
||||||
var now = DateTime.UtcNow;
|
|
||||||
|
|
||||||
await cts.Token.WaitUntilCanceled();
|
|
||||||
|
|
||||||
var elapsed = DateTime.UtcNow - now;
|
|
||||||
|
|
||||||
elapsed.Should().BeGreaterThan(TimeSpan.FromSeconds(1));
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void AlreadyCanceled()
|
|
||||||
{
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
cts.Cancel();
|
|
||||||
|
|
||||||
cts.Token.WaitUntilCanceled().GetAwaiter().IsCompleted.Should().BeTrue();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void None()
|
|
||||||
{
|
|
||||||
CancellationToken.None.WaitUntilCanceled().GetAwaiter().IsCompleted.Should().BeTrue();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,370 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class ChannelTest
|
|
||||||
{
|
|
||||||
(System.Threading.Channels.Channel<int>, Cysharp.Threading.Tasks.Channel<int>) CreateChannel()
|
|
||||||
{
|
|
||||||
var reference = System.Threading.Channels.Channel.CreateUnbounded<int>(new UnboundedChannelOptions
|
|
||||||
{
|
|
||||||
AllowSynchronousContinuations = true,
|
|
||||||
SingleReader = true,
|
|
||||||
SingleWriter = false
|
|
||||||
});
|
|
||||||
|
|
||||||
var channel = Cysharp.Threading.Tasks.Channel.CreateSingleConsumerUnbounded<int>();
|
|
||||||
|
|
||||||
return (reference, channel);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SingleWriteSingleRead()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
var t1 = reference.Reader.WaitToReadAsync();
|
|
||||||
var t2 = channel.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
t1.IsCompleted.Should().BeFalse();
|
|
||||||
t2.Status.IsCompleted().Should().BeFalse();
|
|
||||||
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
|
|
||||||
(await t1).Should().BeTrue();
|
|
||||||
(await t2).Should().BeTrue();
|
|
||||||
|
|
||||||
reference.Reader.TryRead(out var refitem).Should().BeTrue();
|
|
||||||
channel.Reader.TryRead(out var chanitem).Should().BeTrue();
|
|
||||||
refitem.Should().Be(item);
|
|
||||||
chanitem.Should().Be(item);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task MultiWrite()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
var t1 = reference.Reader.WaitToReadAsync();
|
|
||||||
var t2 = channel.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
t1.IsCompleted.Should().BeFalse();
|
|
||||||
t2.Status.IsCompleted().Should().BeFalse();
|
|
||||||
|
|
||||||
foreach (var i in Enumerable.Range(1, 3))
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item * i);
|
|
||||||
channel.Writer.TryWrite(item * i);
|
|
||||||
}
|
|
||||||
|
|
||||||
(await t1).Should().BeTrue();
|
|
||||||
(await t2).Should().BeTrue();
|
|
||||||
|
|
||||||
foreach (var i in Enumerable.Range(1, 3))
|
|
||||||
{
|
|
||||||
(await reference.Reader.WaitToReadAsync()).Should().BeTrue();
|
|
||||||
(await channel.Reader.WaitToReadAsync()).Should().BeTrue();
|
|
||||||
|
|
||||||
reference.Reader.TryRead(out var refitem).Should().BeTrue();
|
|
||||||
channel.Reader.TryRead(out var chanitem).Should().BeTrue();
|
|
||||||
refitem.Should().Be(item * i);
|
|
||||||
chanitem.Should().Be(item * i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task CompleteOnEmpty()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
reference.Reader.TryRead(out var refitem);
|
|
||||||
channel.Reader.TryRead(out var chanitem);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Empty.
|
|
||||||
|
|
||||||
var completion1 = reference.Reader.Completion;
|
|
||||||
var wait1 = reference.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
var completion2 = channel.Reader.Completion;
|
|
||||||
var wait2 = channel.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
reference.Writer.TryComplete();
|
|
||||||
channel.Writer.TryComplete();
|
|
||||||
|
|
||||||
completion1.Status.Should().Be(TaskStatus.RanToCompletion);
|
|
||||||
completion2.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
|
|
||||||
(await wait1).Should().BeFalse();
|
|
||||||
(await wait2).Should().BeFalse();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task CompleteErrorOnEmpty()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
reference.Reader.TryRead(out var refitem);
|
|
||||||
channel.Reader.TryRead(out var chanitem);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Empty.
|
|
||||||
|
|
||||||
var completion1 = reference.Reader.Completion;
|
|
||||||
var wait1 = reference.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
var completion2 = channel.Reader.Completion;
|
|
||||||
var wait2 = channel.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
var ex = new Exception();
|
|
||||||
reference.Writer.TryComplete(ex);
|
|
||||||
channel.Writer.TryComplete(ex);
|
|
||||||
|
|
||||||
completion1.Status.Should().Be(TaskStatus.Faulted);
|
|
||||||
completion2.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<Exception>(async () => await wait1)).Should().Be(ex);
|
|
||||||
(await Assert.ThrowsAsync<Exception>(async () => await wait2)).Should().Be(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task CompleteWithRest()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Three Item2.
|
|
||||||
|
|
||||||
var completion1 = reference.Reader.Completion;
|
|
||||||
var wait1 = reference.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
var completion2 = channel.Reader.Completion;
|
|
||||||
var wait2 = channel.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
reference.Writer.TryComplete();
|
|
||||||
channel.Writer.TryComplete();
|
|
||||||
|
|
||||||
// completion1.Status.Should().Be(TaskStatus.WaitingForActivation);
|
|
||||||
completion2.Status.Should().Be(UniTaskStatus.Pending);
|
|
||||||
|
|
||||||
(await wait1).Should().BeTrue();
|
|
||||||
(await wait2).Should().BeTrue();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Reader.TryRead(out var i1).Should().BeTrue();
|
|
||||||
channel.Reader.TryRead(out var i2).Should().BeTrue();
|
|
||||||
i1.Should().Be(item);
|
|
||||||
i2.Should().Be(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
(await reference.Reader.WaitToReadAsync()).Should().BeFalse();
|
|
||||||
(await channel.Reader.WaitToReadAsync()).Should().BeFalse();
|
|
||||||
|
|
||||||
completion1.Status.Should().Be(TaskStatus.RanToCompletion);
|
|
||||||
completion2.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task CompleteErrorWithRest()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Three Item2.
|
|
||||||
|
|
||||||
var completion1 = reference.Reader.Completion;
|
|
||||||
var wait1 = reference.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
var completion2 = channel.Reader.Completion;
|
|
||||||
var wait2 = channel.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
var ex = new Exception();
|
|
||||||
reference.Writer.TryComplete(ex);
|
|
||||||
channel.Writer.TryComplete(ex);
|
|
||||||
|
|
||||||
// completion1.Status.Should().Be(TaskStatus.WaitingForActivation);
|
|
||||||
completion2.Status.Should().Be(UniTaskStatus.Pending);
|
|
||||||
|
|
||||||
(await wait1).Should().BeTrue();
|
|
||||||
(await wait2).Should().BeTrue();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Reader.TryRead(out var i1).Should().BeTrue();
|
|
||||||
channel.Reader.TryRead(out var i2).Should().BeTrue();
|
|
||||||
i1.Should().Be(item);
|
|
||||||
i2.Should().Be(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
wait1 = reference.Reader.WaitToReadAsync();
|
|
||||||
wait2 = channel.Reader.WaitToReadAsync();
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<Exception>(async () => await wait1)).Should().Be(ex);
|
|
||||||
(await Assert.ThrowsAsync<Exception>(async () => await wait2)).Should().Be(ex);
|
|
||||||
|
|
||||||
completion1.Status.Should().Be(TaskStatus.Faulted);
|
|
||||||
completion2.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Cancellation()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var wait1 = reference.Reader.WaitToReadAsync(cts.Token);
|
|
||||||
var wait2 = channel.Reader.WaitToReadAsync(cts.Token);
|
|
||||||
|
|
||||||
cts.Cancel();
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await wait1)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await wait2)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AsyncEnumerator()
|
|
||||||
{
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
var ta1 = reference.Reader.ReadAllAsync().ToArrayAsync();
|
|
||||||
var ta2 = channel.Reader.ReadAllAsync().ToArrayAsync();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
reference.Writer.TryComplete();
|
|
||||||
channel.Writer.TryComplete();
|
|
||||||
|
|
||||||
(await ta1).Should().Equal(new[] { 10, 20, 30 });
|
|
||||||
(await ta2).Should().Equal(new[] { 10, 20, 30 });
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AsyncEnumeratorCancellation()
|
|
||||||
{
|
|
||||||
// Token1, Token2 and Cancel1
|
|
||||||
{
|
|
||||||
var cts1 = new CancellationTokenSource();
|
|
||||||
var cts2 = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
var ta1 = reference.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
|
||||||
var ta2 = channel.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
cts1.Cancel();
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts1.Token);
|
|
||||||
}
|
|
||||||
// Token1, Token2 and Cancel2
|
|
||||||
{
|
|
||||||
var cts1 = new CancellationTokenSource();
|
|
||||||
var cts2 = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
var ta1 = reference.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
|
||||||
var ta2 = channel.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
cts2.Cancel();
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts2.Token);
|
|
||||||
}
|
|
||||||
// Token1 and Cancel1
|
|
||||||
{
|
|
||||||
var cts1 = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
var ta1 = reference.Reader.ReadAllAsync(cts1.Token).ToArrayAsync();
|
|
||||||
var ta2 = channel.Reader.ReadAllAsync(cts1.Token).ToArrayAsync();
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
cts1.Cancel();
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts1.Token);
|
|
||||||
}
|
|
||||||
// Token2 and Cancel2
|
|
||||||
{
|
|
||||||
var cts2 = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var (reference, channel) = CreateChannel();
|
|
||||||
|
|
||||||
var ta1 = reference.Reader.ReadAllAsync().ToArrayAsync(cts2.Token);
|
|
||||||
var ta2 = channel.Reader.ReadAllAsync().ToArrayAsync(cts2.Token);
|
|
||||||
|
|
||||||
foreach (var item in new[] { 10, 20, 30 })
|
|
||||||
{
|
|
||||||
reference.Writer.TryWrite(item);
|
|
||||||
channel.Writer.TryWrite(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
cts2.Cancel();
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts2.Token);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,590 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class CompletionSourceTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task SetFirst()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
tcs.TrySetResult();
|
|
||||||
await tcs.Task; // ok.
|
|
||||||
await tcs.Task; // ok.
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SingleOnFirst()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetResult();
|
|
||||||
await a;
|
|
||||||
await tcs.Task; // ok.
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task MultiOne()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
tcs.TrySetResult();
|
|
||||||
await a;
|
|
||||||
await b;
|
|
||||||
await tcs.Task; // ok.
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task MultiTwo()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
tcs.TrySetResult();
|
|
||||||
await a;
|
|
||||||
await b;
|
|
||||||
await c;
|
|
||||||
await tcs.Task; // ok.
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await c);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await c)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource();
|
|
||||||
|
|
||||||
async UniTask Await()
|
|
||||||
{
|
|
||||||
await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await c)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class TestException : Exception
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class CompletionSourceTest2
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task SetFirst()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
tcs.TrySetResult(10);
|
|
||||||
var a = await tcs.Task; // ok.
|
|
||||||
var b = await tcs.Task; // ok.
|
|
||||||
a.Should().Be(10);
|
|
||||||
b.Should().Be(10);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SingleOnFirst()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetResult(10);
|
|
||||||
var r1 = await a;
|
|
||||||
var r2 = await tcs.Task; // ok.
|
|
||||||
r1.Should().Be(10);
|
|
||||||
r2.Should().Be(10);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task MultiOne()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
tcs.TrySetResult(10);
|
|
||||||
var r1 = await a;
|
|
||||||
var r2 = await b;
|
|
||||||
var r3 = await tcs.Task; // ok.
|
|
||||||
(r1, r2, r3).Should().Be((10, 10, 10));
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task MultiTwo()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
tcs.TrySetResult(10);
|
|
||||||
var r1 = await a;
|
|
||||||
var r2 = await b;
|
|
||||||
var r3 = await c;
|
|
||||||
var r4 = await tcs.Task; // ok.
|
|
||||||
(r1, r2, r3, r4).Should().Be((10, 10, 10, 10));
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new TestException());
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await b);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await c);
|
|
||||||
await Assert.ThrowsAsync<TestException>(async () => await tcs.Task);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
|
|
||||||
tcs.TrySetException(new OperationCanceledException(cts.Token));
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await c)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var tcs = new UniTaskCompletionSource<int>();
|
|
||||||
|
|
||||||
async UniTask<int> Await()
|
|
||||||
{
|
|
||||||
return await tcs.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
var a = Await();
|
|
||||||
var b = Await();
|
|
||||||
var c = Await();
|
|
||||||
|
|
||||||
tcs.TrySetCanceled(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await a)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await b)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await c)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await tcs.Task)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
tcs.Task.Status.Should().Be(UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class TestException : Exception
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,47 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class DeferTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task D()
|
|
||||||
{
|
|
||||||
var created = false;
|
|
||||||
var v = UniTask.Defer(() => { created = true; return UniTask.Run(() => 10); });
|
|
||||||
|
|
||||||
created.Should().BeFalse();
|
|
||||||
|
|
||||||
var t = await v;
|
|
||||||
|
|
||||||
created.Should().BeTrue();
|
|
||||||
|
|
||||||
t.Should().Be(10);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task D2()
|
|
||||||
{
|
|
||||||
var created = false;
|
|
||||||
var v = UniTask.Defer(() => { created = true; return UniTask.Run(() => 10).AsUniTask(); });
|
|
||||||
|
|
||||||
created.Should().BeFalse();
|
|
||||||
|
|
||||||
await v;
|
|
||||||
|
|
||||||
created.Should().BeTrue();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
|
@ -35,7 +35,7 @@ namespace NetCoreTests.Linq
|
||||||
xs.Should().Be(ys);
|
xs.Should().Be(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(start, count).SumAwaitWithCancellationAsync((x, _) => UniTask.Run(() => x));
|
var xs = await UniTaskAsyncEnumerable.Range(start, count).SumAwaitCancellationAsync((x, _) => UniTask.Run(() => x));
|
||||||
var ys = Enumerable.Range(start, count).Sum(x => x);
|
var ys = Enumerable.Range(start, count).Sum(x => x);
|
||||||
xs.Should().Be(ys);
|
xs.Should().Be(ys);
|
||||||
}
|
}
|
||||||
|
@ -481,7 +481,7 @@ namespace NetCoreTests.Linq
|
||||||
list.Add(x);
|
list.Add(x);
|
||||||
});
|
});
|
||||||
|
|
||||||
list.Should().Equal(Enumerable.Range(1, 10));
|
list.Should().BeEquivalentTo(Enumerable.Range(1, 10));
|
||||||
|
|
||||||
var list2 = new List<(int, int)>();
|
var list2 = new List<(int, int)>();
|
||||||
await Enumerable.Range(5, 10).ToUniTaskAsyncEnumerable().ForEachAsync((index, x) =>
|
await Enumerable.Range(5, 10).ToUniTaskAsyncEnumerable().ForEachAsync((index, x) =>
|
||||||
|
@ -490,7 +490,7 @@ namespace NetCoreTests.Linq
|
||||||
});
|
});
|
||||||
|
|
||||||
var list3 = Enumerable.Range(5, 10).Select((index, x) => (index, x)).ToArray();
|
var list3 = Enumerable.Range(5, 10).Select((index, x) => (index, x)).ToArray();
|
||||||
list2.Should().Equal(list3);
|
list2.Should().BeEquivalentTo(list3);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,7 +24,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await Enumerable.Range(start, count).ToUniTaskAsyncEnumerable().Append(99).ToArrayAsync();
|
var xs = await Enumerable.Range(start, count).ToUniTaskAsyncEnumerable().Append(99).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(start, count).Append(99).ToArray();
|
var ys = Enumerable.Range(start, count).Append(99).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -50,7 +50,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await Enumerable.Range(start, count).ToUniTaskAsyncEnumerable().Prepend(99).ToArrayAsync();
|
var xs = await Enumerable.Range(start, count).ToUniTaskAsyncEnumerable().Prepend(99).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(start, count).Prepend(99).ToArray();
|
var ys = Enumerable.Range(start, count).Prepend(99).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -85,7 +85,7 @@ namespace NetCoreTests.Linq
|
||||||
|
|
||||||
var xs = await l.ToUniTaskAsyncEnumerable().Concat(r.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
var xs = await l.ToUniTaskAsyncEnumerable().Concat(r.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
||||||
var ys = l.Concat(r).ToArray();
|
var ys = l.Concat(r).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -119,17 +119,17 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().DefaultIfEmpty(99).ToArrayAsync();
|
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().DefaultIfEmpty(99).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, 0).DefaultIfEmpty(99).ToArray();
|
var ys = Enumerable.Range(1, 0).DefaultIfEmpty(99).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 1).ToUniTaskAsyncEnumerable().DefaultIfEmpty(99).ToArrayAsync();
|
var xs = await Enumerable.Range(1, 1).ToUniTaskAsyncEnumerable().DefaultIfEmpty(99).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, 1).DefaultIfEmpty(99).ToArray();
|
var ys = Enumerable.Range(1, 1).DefaultIfEmpty(99).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 10).ToUniTaskAsyncEnumerable().DefaultIfEmpty(99).ToArrayAsync();
|
var xs = await Enumerable.Range(1, 10).ToUniTaskAsyncEnumerable().DefaultIfEmpty(99).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, 10).DefaultIfEmpty(99).ToArray();
|
var ys = Enumerable.Range(1, 10).DefaultIfEmpty(99).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
// Throw
|
// Throw
|
||||||
{
|
{
|
||||||
|
|
|
@ -34,11 +34,11 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, 10).ToObservable().ToArray();
|
var xs = await UniTaskAsyncEnumerable.Range(1, 10).ToObservable().ToArray();
|
||||||
xs.Should().Equal(Enumerable.Range(1, 10));
|
xs.Should().BeEquivalentTo(Enumerable.Range(1, 10));
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, 0).ToObservable().ToArray();
|
var xs = await UniTaskAsyncEnumerable.Range(1, 0).ToObservable().ToArray();
|
||||||
xs.Should().Equal(Enumerable.Range(1, 0));
|
xs.Should().BeEquivalentTo(Enumerable.Range(1, 0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -70,21 +70,21 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await Observable.Range(1, 100).ToUniTaskAsyncEnumerable().ToArrayAsync();
|
var xs = await Observable.Range(1, 100).ToUniTaskAsyncEnumerable().ToArrayAsync();
|
||||||
var ys = await Observable.Range(1, 100).ToArray();
|
var ys = await Observable.Range(1, 100).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
var xs = await Observable.Range(1, 100, ThreadPoolScheduler.Instance).ToUniTaskAsyncEnumerable().ToArrayAsync();
|
var xs = await Observable.Range(1, 100, ThreadPoolScheduler.Instance).ToUniTaskAsyncEnumerable().ToArrayAsync();
|
||||||
var ys = await Observable.Range(1, 100, ThreadPoolScheduler.Instance).ToArray();
|
var ys = await Observable.Range(1, 100, ThreadPoolScheduler.Instance).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
var xs = await Observable.Empty<int>(ThreadPoolScheduler.Instance).ToUniTaskAsyncEnumerable().ToArrayAsync();
|
var xs = await Observable.Empty<int>(ThreadPoolScheduler.Instance).ToUniTaskAsyncEnumerable().ToArrayAsync();
|
||||||
var ys = await Observable.Empty<int>(ThreadPoolScheduler.Instance).ToArray();
|
var ys = await Observable.Empty<int>(ThreadPoolScheduler.Instance).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -95,25 +95,25 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await Enumerable.Range(1, 100).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x);
|
var xs = await Enumerable.Range(1, 100).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x);
|
||||||
var ys = Enumerable.Range(1, 100).ToDictionary(x => x);
|
var ys = Enumerable.Range(1, 100).ToDictionary(x => x);
|
||||||
|
|
||||||
xs.OrderBy(x => x.Key).Should().Equal(ys.OrderBy(x => x.Key));
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x);
|
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x);
|
||||||
var ys = Enumerable.Range(1, 0).ToDictionary(x => x);
|
var ys = Enumerable.Range(1, 0).ToDictionary(x => x);
|
||||||
|
|
||||||
xs.OrderBy(x => x.Key).Should().Equal(ys.OrderBy(x => x.Key));
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 100).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x, x => x * 2);
|
var xs = await Enumerable.Range(1, 100).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x, x => x * 2);
|
||||||
var ys = Enumerable.Range(1, 100).ToDictionary(x => x, x => x * 2);
|
var ys = Enumerable.Range(1, 100).ToDictionary(x => x, x => x * 2);
|
||||||
|
|
||||||
xs.OrderBy(x => x.Key).Should().Equal(ys.OrderBy(x => x.Key));
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x, x => x * 2);
|
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToDictionaryAsync(x => x, x => x * 2);
|
||||||
var ys = Enumerable.Range(1, 0).ToDictionary(x => x, x => x * 2);
|
var ys = Enumerable.Range(1, 0).ToDictionary(x => x, x => x * 2);
|
||||||
|
|
||||||
xs.OrderBy(x => x.Key).Should().Equal(ys.OrderBy(x => x.Key));
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -126,34 +126,34 @@ namespace NetCoreTests.Linq
|
||||||
var ys = arr.ToLookup(x => x);
|
var ys = arr.ToLookup(x => x);
|
||||||
|
|
||||||
xs.Count.Should().Be(ys.Count);
|
xs.Count.Should().Be(ys.Count);
|
||||||
xs.Should().BeEquivalentTo(ys);
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
foreach (var key in xs.Select(x => x.Key))
|
foreach (var key in xs.Select(x => x.Key))
|
||||||
{
|
{
|
||||||
xs[key].Should().Equal(ys[key]);
|
xs[key].Should().BeEquivalentTo(ys[key]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToLookupAsync(x => x);
|
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToLookupAsync(x => x);
|
||||||
var ys = Enumerable.Range(1, 0).ToLookup(x => x);
|
var ys = Enumerable.Range(1, 0).ToLookup(x => x);
|
||||||
|
|
||||||
xs.Should().BeEquivalentTo(ys);
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await arr.ToUniTaskAsyncEnumerable().ToLookupAsync(x => x, x => x * 2);
|
var xs = await arr.ToUniTaskAsyncEnumerable().ToLookupAsync(x => x, x => x * 2);
|
||||||
var ys = arr.ToLookup(x => x, x => x * 2);
|
var ys = arr.ToLookup(x => x, x => x * 2);
|
||||||
|
|
||||||
xs.Count.Should().Be(ys.Count);
|
xs.Count.Should().Be(ys.Count);
|
||||||
xs.Should().BeEquivalentTo(ys);
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
foreach (var key in xs.Select(x => x.Key))
|
foreach (var key in xs.Select(x => x.Key))
|
||||||
{
|
{
|
||||||
xs[key].Should().Equal(ys[key]);
|
xs[key].Should().BeEquivalentTo(ys[key]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToLookupAsync(x => x, x => x * 2);
|
var xs = await Enumerable.Range(1, 0).ToUniTaskAsyncEnumerable().ToLookupAsync(x => x, x => x * 2);
|
||||||
var ys = Enumerable.Range(1, 0).ToLookup(x => x, x => x * 2);
|
var ys = Enumerable.Range(1, 0).ToLookup(x => x, x => x * 2);
|
||||||
|
|
||||||
xs.Should().BeEquivalentTo(ys);
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -164,13 +164,13 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await Enumerable.Range(1, 100).ToUniTaskAsyncEnumerable().ToListAsync();
|
var xs = await Enumerable.Range(1, 100).ToUniTaskAsyncEnumerable().ToListAsync();
|
||||||
var ys = Enumerable.Range(1, 100).ToList();
|
var ys = Enumerable.Range(1, 100).ToList();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Empty<int>().ToUniTaskAsyncEnumerable().ToListAsync();
|
var xs = await Enumerable.Empty<int>().ToUniTaskAsyncEnumerable().ToListAsync();
|
||||||
var ys = Enumerable.Empty<int>().ToList();
|
var ys = Enumerable.Empty<int>().ToList();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -181,13 +181,13 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await new[] { 1, 20, 4, 5, 20, 4, 6 }.ToUniTaskAsyncEnumerable().ToHashSetAsync();
|
var xs = await new[] { 1, 20, 4, 5, 20, 4, 6 }.ToUniTaskAsyncEnumerable().ToHashSetAsync();
|
||||||
var ys = new[] { 1, 20, 4, 5, 20, 4, 6 }.ToHashSet();
|
var ys = new[] { 1, 20, 4, 5, 20, 4, 6 }.ToHashSet();
|
||||||
|
|
||||||
xs.OrderBy(x => x).Should().Equal(ys.OrderBy(x => x));
|
xs.OrderBy(x => x).Should().BeEquivalentTo(ys.OrderBy(x => x));
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await Enumerable.Empty<int>().ToUniTaskAsyncEnumerable().ToHashSetAsync();
|
var xs = await Enumerable.Empty<int>().ToUniTaskAsyncEnumerable().ToHashSetAsync();
|
||||||
var ys = Enumerable.Empty<int>().ToHashSet();
|
var ys = Enumerable.Empty<int>().ToHashSet();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,194 +0,0 @@
|
||||||
#pragma warning disable CS1998
|
|
||||||
#pragma warning disable CS0162
|
|
||||||
|
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests.Linq
|
|
||||||
{
|
|
||||||
public class CreateTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task SyncCreation()
|
|
||||||
{
|
|
||||||
var from = 10;
|
|
||||||
var count = 100;
|
|
||||||
|
|
||||||
var xs = await UniTaskAsyncEnumerable.Create<int>(async (writer, token) =>
|
|
||||||
{
|
|
||||||
for (int i = 0; i < count; i++)
|
|
||||||
{
|
|
||||||
await writer.YieldAsync(from + i);
|
|
||||||
}
|
|
||||||
}).ToArrayAsync();
|
|
||||||
|
|
||||||
var ys = await Range(from, count).AsUniTaskAsyncEnumerable().ToArrayAsync();
|
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SyncManually()
|
|
||||||
{
|
|
||||||
var list = new List<int>();
|
|
||||||
var xs = UniTaskAsyncEnumerable.Create<int>(async (writer, token) =>
|
|
||||||
{
|
|
||||||
list.Add(100);
|
|
||||||
await writer.YieldAsync(10);
|
|
||||||
|
|
||||||
list.Add(200);
|
|
||||||
await writer.YieldAsync(20);
|
|
||||||
|
|
||||||
list.Add(300);
|
|
||||||
await writer.YieldAsync(30);
|
|
||||||
|
|
||||||
list.Add(400);
|
|
||||||
});
|
|
||||||
|
|
||||||
list.Should().BeEmpty();
|
|
||||||
var e = xs.GetAsyncEnumerator();
|
|
||||||
|
|
||||||
list.Should().BeEmpty();
|
|
||||||
|
|
||||||
await e.MoveNextAsync();
|
|
||||||
list.Should().Equal(100);
|
|
||||||
e.Current.Should().Be(10);
|
|
||||||
|
|
||||||
await e.MoveNextAsync();
|
|
||||||
list.Should().Equal(100, 200);
|
|
||||||
e.Current.Should().Be(20);
|
|
||||||
|
|
||||||
await e.MoveNextAsync();
|
|
||||||
list.Should().Equal(100, 200, 300);
|
|
||||||
e.Current.Should().Be(30);
|
|
||||||
|
|
||||||
(await e.MoveNextAsync()).Should().BeFalse();
|
|
||||||
list.Should().Equal(100, 200, 300, 400);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SyncExceptionFirst()
|
|
||||||
{
|
|
||||||
var from = 10;
|
|
||||||
var count = 100;
|
|
||||||
|
|
||||||
var xs = UniTaskAsyncEnumerable.Create<int>(async (writer, token) =>
|
|
||||||
{
|
|
||||||
for (int i = 0; i < count; i++)
|
|
||||||
{
|
|
||||||
throw new UniTaskTestException();
|
|
||||||
await writer.YieldAsync(from + i);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<UniTaskTestException>(async () => await xs.ToArrayAsync());
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SyncException()
|
|
||||||
{
|
|
||||||
var from = 10;
|
|
||||||
var count = 100;
|
|
||||||
|
|
||||||
var xs = UniTaskAsyncEnumerable.Create<int>(async (writer, token) =>
|
|
||||||
{
|
|
||||||
for (int i = 0; i < count; i++)
|
|
||||||
{
|
|
||||||
await writer.YieldAsync(from + i);
|
|
||||||
|
|
||||||
if (i == 15)
|
|
||||||
{
|
|
||||||
throw new UniTaskTestException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<UniTaskTestException>(async () => await xs.ToArrayAsync());
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task ASyncManually()
|
|
||||||
{
|
|
||||||
var list = new List<int>();
|
|
||||||
var xs = UniTaskAsyncEnumerable.Create<int>(async (writer, token) =>
|
|
||||||
{
|
|
||||||
await UniTask.Yield();
|
|
||||||
|
|
||||||
list.Add(100);
|
|
||||||
await writer.YieldAsync(10);
|
|
||||||
|
|
||||||
await UniTask.Yield();
|
|
||||||
|
|
||||||
list.Add(200);
|
|
||||||
await writer.YieldAsync(20);
|
|
||||||
|
|
||||||
await UniTask.Yield();
|
|
||||||
list.Add(300);
|
|
||||||
await UniTask.Yield();
|
|
||||||
await writer.YieldAsync(30);
|
|
||||||
|
|
||||||
await UniTask.Yield();
|
|
||||||
|
|
||||||
list.Add(400);
|
|
||||||
});
|
|
||||||
|
|
||||||
list.Should().BeEmpty();
|
|
||||||
var e = xs.GetAsyncEnumerator();
|
|
||||||
|
|
||||||
list.Should().BeEmpty();
|
|
||||||
|
|
||||||
await e.MoveNextAsync();
|
|
||||||
list.Should().Equal(100);
|
|
||||||
e.Current.Should().Be(10);
|
|
||||||
|
|
||||||
await e.MoveNextAsync();
|
|
||||||
list.Should().Equal(100, 200);
|
|
||||||
e.Current.Should().Be(20);
|
|
||||||
|
|
||||||
await e.MoveNextAsync();
|
|
||||||
list.Should().Equal(100, 200, 300);
|
|
||||||
e.Current.Should().Be(30);
|
|
||||||
|
|
||||||
(await e.MoveNextAsync()).Should().BeFalse();
|
|
||||||
list.Should().Equal(100, 200, 300, 400);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AwaitForeachBreak()
|
|
||||||
{
|
|
||||||
var finallyCalled = false;
|
|
||||||
var enumerable = UniTaskAsyncEnumerable.Create<int>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
await writer.YieldAsync(1);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
finallyCalled = true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
await foreach (var x in enumerable)
|
|
||||||
{
|
|
||||||
x.Should().Be(1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
finallyCalled.Should().BeTrue();
|
|
||||||
}
|
|
||||||
|
|
||||||
async IAsyncEnumerable<int> Range(int from, int count)
|
|
||||||
{
|
|
||||||
for (int i = 0; i < count; i++)
|
|
||||||
{
|
|
||||||
yield return from + i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -22,7 +22,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(start, count).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(start, count).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(start, count).ToArray();
|
var ys = Enumerable.Range(start, count).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory]
|
[Theory]
|
||||||
|
@ -36,7 +36,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Repeat(value, count).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Repeat(value, count).ToArrayAsync();
|
||||||
var ys = Enumerable.Repeat(value, count).ToArray();
|
var ys = Enumerable.Repeat(value, count).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -45,7 +45,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Empty<int>().ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Empty<int>().ToArrayAsync();
|
||||||
var ys = Enumerable.Empty<int>().ToArray();
|
var ys = Enumerable.Empty<int>().ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory]
|
[Theory]
|
||||||
|
|
|
@ -23,26 +23,26 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var a = await src.Where(x => x % 2 == 0).ToArrayAsync();
|
var a = await src.Where(x => x % 2 == 0).ToArrayAsync();
|
||||||
var expected = range.Where(x => x % 2 == 0).ToArray();
|
var expected = range.Where(x => x % 2 == 0).ToArray();
|
||||||
a.Should().Equal(expected);
|
a.Should().BeEquivalentTo(expected);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var a = await src.Where((x, i) => (x + i) % 2 == 0).ToArrayAsync();
|
var a = await src.Where((x, i) => (x + i) % 2 == 0).ToArrayAsync();
|
||||||
var expected = range.Where((x, i) => (x + i) % 2 == 0).ToArray();
|
var expected = range.Where((x, i) => (x + i) % 2 == 0).ToArray();
|
||||||
a.Should().Equal(expected);
|
a.Should().BeEquivalentTo(expected);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var a = await src.WhereAwait(x => UniTask.Run(() => x % 2 == 0)).ToArrayAsync();
|
var a = await src.WhereAwait(x => UniTask.Run(() => x % 2 == 0)).ToArrayAsync();
|
||||||
var b = await src.WhereAwait(x => UniTask.FromResult(x % 2 == 0)).ToArrayAsync();
|
var b = await src.WhereAwait(x => UniTask.FromResult(x % 2 == 0)).ToArrayAsync();
|
||||||
var expected = range.Where(x => x % 2 == 0).ToArray();
|
var expected = range.Where(x => x % 2 == 0).ToArray();
|
||||||
a.Should().Equal(expected);
|
a.Should().BeEquivalentTo(expected);
|
||||||
b.Should().Equal(expected);
|
b.Should().BeEquivalentTo(expected);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var a = await src.WhereAwait((x, i) => UniTask.Run(() => (x + i) % 2 == 0)).ToArrayAsync();
|
var a = await src.WhereAwait((x, i) => UniTask.Run(() => (x + i) % 2 == 0)).ToArrayAsync();
|
||||||
var b = await src.WhereAwait((x, i) => UniTask.FromResult((x + i) % 2 == 0)).ToArrayAsync();
|
var b = await src.WhereAwait((x, i) => UniTask.FromResult((x + i) % 2 == 0)).ToArrayAsync();
|
||||||
var expected = range.Where((x, i) => (x + i) % 2 == 0).ToArray();
|
var expected = range.Where((x, i) => (x + i) % 2 == 0).ToArray();
|
||||||
a.Should().Equal(expected);
|
a.Should().BeEquivalentTo(expected);
|
||||||
b.Should().Equal(expected);
|
b.Should().BeEquivalentTo(expected);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -79,7 +79,7 @@ namespace NetCoreTests.Linq
|
||||||
var a = await data.ToUniTaskAsyncEnumerable().OfType<int>().ToArrayAsync();
|
var a = await data.ToUniTaskAsyncEnumerable().OfType<int>().ToArrayAsync();
|
||||||
var b = data.OfType<int>().ToArray();
|
var b = data.OfType<int>().ToArray();
|
||||||
|
|
||||||
a.Should().Equal(b);
|
a.Should().BeEquivalentTo(b);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -101,7 +101,7 @@ namespace NetCoreTests.Linq
|
||||||
var a = await data.ToUniTaskAsyncEnumerable().Cast<int>().ToArrayAsync();
|
var a = await data.ToUniTaskAsyncEnumerable().Cast<int>().ToArrayAsync();
|
||||||
var b = data.Cast<int>().ToArray();
|
var b = data.Cast<int>().ToArray();
|
||||||
|
|
||||||
a.Should().Equal(b);
|
a.Should().BeEquivalentTo(b);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -38,7 +38,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await outer.ToUniTaskAsyncEnumerable().Join(inner.ToUniTaskAsyncEnumerable(), x => x, x => x, (x, y) => (x, y)).ToArrayAsync();
|
var xs = await outer.ToUniTaskAsyncEnumerable().Join(inner.ToUniTaskAsyncEnumerable(), x => x, x => x, (x, y) => (x, y)).ToArrayAsync();
|
||||||
var ys = outer.Join(inner, x => x, x => x, (x, y) => (x, y)).ToArray();
|
var ys = outer.Join(inner, x => x, x => x, (x, y) => (x, y)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -66,7 +66,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await outer.ToUniTaskAsyncEnumerable().JoinAwait(inner.ToUniTaskAsyncEnumerable(), x => RandomRun(x), x => RandomRun(x), (x, y) => RandomRun((x, y))).ToArrayAsync();
|
var xs = await outer.ToUniTaskAsyncEnumerable().JoinAwait(inner.ToUniTaskAsyncEnumerable(), x => RandomRun(x), x => RandomRun(x), (x, y) => RandomRun((x, y))).ToArrayAsync();
|
||||||
var ys = outer.Join(inner, x => x, x => x, (x, y) => (x, y)).ToArray();
|
var ys = outer.Join(inner, x => x, x => x, (x, y) => (x, y)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -94,7 +94,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await outer.ToUniTaskAsyncEnumerable().JoinAwaitWithCancellation(inner.ToUniTaskAsyncEnumerable(), (x, _) => RandomRun(x), (x, _) => RandomRun(x), (x, y, _) => RandomRun((x, y))).ToArrayAsync();
|
var xs = await outer.ToUniTaskAsyncEnumerable().JoinAwaitWithCancellation(inner.ToUniTaskAsyncEnumerable(), (x, _) => RandomRun(x), (x, _) => RandomRun(x), (x, y, _) => RandomRun((x, y))).ToArrayAsync();
|
||||||
var ys = outer.Join(inner, x => x, x => x, (x, y) => (x, y)).ToArray();
|
var ys = outer.Join(inner, x => x, x => x, (x, y) => (x, y)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -123,7 +123,7 @@ namespace NetCoreTests.Linq
|
||||||
var ys = arr.GroupBy(x => x).ToArray();
|
var ys = arr.GroupBy(x => x).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.Should().BeEquivalentTo(ys);
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -131,7 +131,7 @@ namespace NetCoreTests.Linq
|
||||||
var ys = arr.GroupBy(x => x, (key, xs) => (key, xs.ToArray())).ToArray();
|
var ys = arr.GroupBy(x => x, (key, xs) => (key, xs.ToArray())).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.OrderBy(x => x.key).SelectMany(x => x.Item2).Should().Equal(ys.OrderBy(x => x.key).SelectMany(x => x.Item2));
|
xs.OrderBy(x => x.key).SelectMany(x => x.Item2).Should().BeEquivalentTo(ys.OrderBy(x => x.key).SelectMany(x => x.Item2));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -139,7 +139,7 @@ namespace NetCoreTests.Linq
|
||||||
var ys = arr.GroupBy(x => x).ToArray();
|
var ys = arr.GroupBy(x => x).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.Should().BeEquivalentTo(ys);
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -147,7 +147,7 @@ namespace NetCoreTests.Linq
|
||||||
var ys = arr.GroupBy(x => x, (key, xs) => (key, xs.ToArray())).ToArray();
|
var ys = arr.GroupBy(x => x, (key, xs) => (key, xs.ToArray())).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.OrderBy(x => x.key).SelectMany(x => x.Item2).Should().Equal(ys.OrderBy(x => x.key).SelectMany(x => x.Item2));
|
xs.OrderBy(x => x.key).SelectMany(x => x.Item2).Should().BeEquivalentTo(ys.OrderBy(x => x.key).SelectMany(x => x.Item2));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -155,7 +155,7 @@ namespace NetCoreTests.Linq
|
||||||
var ys = arr.GroupBy(x => x).ToArray();
|
var ys = arr.GroupBy(x => x).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.Should().BeEquivalentTo(ys);
|
xs.OrderBy(x => x.Key).Should().BeEquivalentTo(ys.OrderBy(x => x.Key));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -163,7 +163,7 @@ namespace NetCoreTests.Linq
|
||||||
var ys = arr.GroupBy(x => x, (key, xs) => (key, xs.ToArray())).ToArray();
|
var ys = arr.GroupBy(x => x, (key, xs) => (key, xs.ToArray())).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.OrderBy(x => x.key).SelectMany(x => x.Item2).Should().Equal(ys.OrderBy(x => x.key).SelectMany(x => x.Item2));
|
xs.OrderBy(x => x.key).SelectMany(x => x.Item2).Should().BeEquivalentTo(ys.OrderBy(x => x.key).SelectMany(x => x.Item2));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -199,21 +199,21 @@ namespace NetCoreTests.Linq
|
||||||
var ys = outer.GroupJoin(inner, x => x, x => x, (x, y) => (x, string.Join(", ", y))).ToArray();
|
var ys = outer.GroupJoin(inner, x => x, x => x, (x, y) => (x, string.Join(", ", y))).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await outer.ToUniTaskAsyncEnumerable().GroupJoinAwait(inner.ToUniTaskAsyncEnumerable(), x => RandomRun(x), x => RandomRun(x), (x, y) => RandomRun((x, string.Join(", ", y)))).ToArrayAsync();
|
var xs = await outer.ToUniTaskAsyncEnumerable().GroupJoinAwait(inner.ToUniTaskAsyncEnumerable(), x => RandomRun(x), x => RandomRun(x), (x, y) => RandomRun((x, string.Join(", ", y)))).ToArrayAsync();
|
||||||
var ys = outer.GroupJoin(inner, x => x, x => x, (x, y) => (x, string.Join(", ", y))).ToArray();
|
var ys = outer.GroupJoin(inner, x => x, x => x, (x, y) => (x, string.Join(", ", y))).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await outer.ToUniTaskAsyncEnumerable().GroupJoinAwaitWithCancellation(inner.ToUniTaskAsyncEnumerable(), (x, _) => RandomRun(x), (x, _) => RandomRun(x), (x, y, _) => RandomRun((x, string.Join(", ", y)))).ToArrayAsync();
|
var xs = await outer.ToUniTaskAsyncEnumerable().GroupJoinAwaitWithCancellation(inner.ToUniTaskAsyncEnumerable(), (x, _) => RandomRun(x), (x, _) => RandomRun(x), (x, y, _) => RandomRun((x, string.Join(", ", y)))).ToArrayAsync();
|
||||||
var ys = outer.GroupJoin(inner, x => x, x => x, (x, y) => (x, string.Join(", ", y))).ToArray();
|
var ys = outer.GroupJoin(inner, x => x, x => x, (x, y) => (x, string.Join(", ", y))).ToArray();
|
||||||
|
|
||||||
xs.Length.Should().Be(ys.Length);
|
xs.Length.Should().Be(ys.Length);
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,139 +0,0 @@
|
||||||
#pragma warning disable CS1998
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using FluentAssertions;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests.Linq
|
|
||||||
{
|
|
||||||
public class MergeTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task TwoSource()
|
|
||||||
{
|
|
||||||
var semaphore = new SemaphoreSlim(1, 1);
|
|
||||||
|
|
||||||
var a = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await UniTask.SwitchToThreadPool();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("A1");
|
|
||||||
semaphore.Release();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("A2");
|
|
||||||
semaphore.Release();
|
|
||||||
});
|
|
||||||
|
|
||||||
var b = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await UniTask.SwitchToThreadPool();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("B1");
|
|
||||||
await writer.YieldAsync("B2");
|
|
||||||
semaphore.Release();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("B3");
|
|
||||||
semaphore.Release();
|
|
||||||
});
|
|
||||||
|
|
||||||
var result = await a.Merge(b).ToArrayAsync();
|
|
||||||
result.Should().Equal("A1", "B1", "B2", "A2", "B3");
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task ThreeSource()
|
|
||||||
{
|
|
||||||
var semaphore = new SemaphoreSlim(0, 1);
|
|
||||||
|
|
||||||
var a = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await UniTask.SwitchToThreadPool();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("A1");
|
|
||||||
semaphore.Release();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("A2");
|
|
||||||
semaphore.Release();
|
|
||||||
});
|
|
||||||
|
|
||||||
var b = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await UniTask.SwitchToThreadPool();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("B1");
|
|
||||||
await writer.YieldAsync("B2");
|
|
||||||
semaphore.Release();
|
|
||||||
|
|
||||||
await semaphore.WaitAsync();
|
|
||||||
await writer.YieldAsync("B3");
|
|
||||||
semaphore.Release();
|
|
||||||
});
|
|
||||||
|
|
||||||
var c = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await UniTask.SwitchToThreadPool();
|
|
||||||
|
|
||||||
await writer.YieldAsync("C1");
|
|
||||||
semaphore.Release();
|
|
||||||
});
|
|
||||||
|
|
||||||
var result = await a.Merge(b, c).ToArrayAsync();
|
|
||||||
result.Should().Equal("C1", "A1", "B1", "B2", "A2", "B3");
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Throw()
|
|
||||||
{
|
|
||||||
var a = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await writer.YieldAsync("A1");
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
var b = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
throw new UniTaskTestException();
|
|
||||||
});
|
|
||||||
|
|
||||||
var enumerator = a.Merge(b).GetAsyncEnumerator();
|
|
||||||
(await enumerator.MoveNextAsync()).Should().Be(true);
|
|
||||||
enumerator.Current.Should().Be("A1");
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<UniTaskTestException>(async () => await enumerator.MoveNextAsync());
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Cancel()
|
|
||||||
{
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var a = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await writer.YieldAsync("A1");
|
|
||||||
});
|
|
||||||
|
|
||||||
var b = UniTaskAsyncEnumerable.Create<string>(async (writer, _) =>
|
|
||||||
{
|
|
||||||
await writer.YieldAsync("B1");
|
|
||||||
});
|
|
||||||
|
|
||||||
var enumerator = a.Merge(b).GetAsyncEnumerator(cts.Token);
|
|
||||||
(await enumerator.MoveNextAsync()).Should().Be(true);
|
|
||||||
enumerator.Current.Should().Be("A1");
|
|
||||||
|
|
||||||
cts.Cancel();
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await enumerator.MoveNextAsync());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -27,7 +27,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).Skip(skipCount).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).Skip(skipCount).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).Skip(skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).Skip(skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -52,7 +52,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipLast(skipCount).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipLast(skipCount).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).SkipLast(skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).SkipLast(skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -77,7 +77,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeLast(takeCount).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeLast(takeCount).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).TakeLast(takeCount).ToArray();
|
var ys = Enumerable.Range(1, collection).TakeLast(takeCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -103,7 +103,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).Take(takeCount).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).Take(takeCount).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).Take(takeCount).ToArray();
|
var ys = Enumerable.Range(1, collection).Take(takeCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -130,37 +130,37 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArray();
|
var ys = Enumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwait(x => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwait(x => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwait((x, i) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwait((x, i) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArray();
|
var ys = Enumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwaitWithCancellation((x, _) => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwaitWithCancellation((x, _) => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).SkipWhile(x => x < skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwaitWithCancellation((x, i, _) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).SkipWhileAwaitWithCancellation((x, i, _) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArray();
|
var ys = Enumerable.Range(1, collection).SkipWhile((x, i) => x < (skipCount - i)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -213,37 +213,37 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArray();
|
var ys = Enumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwait(x => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwait(x => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwait((x, i) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwait((x, i) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArray();
|
var ys = Enumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwaitWithCancellation((x, _) => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwaitWithCancellation((x, _) => UniTask.Run(() => x < skipCount)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArray();
|
var ys = Enumerable.Range(1, collection).TakeWhile(x => x < skipCount).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwaitWithCancellation((x, i, _) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, collection).TakeWhileAwaitWithCancellation((x, i, _) => UniTask.Run(() => x < (skipCount - i))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArray();
|
var ys = Enumerable.Range(1, collection).TakeWhile((x, i) => x < (skipCount - i)).ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await Enumerable.Range(start, count).ToUniTaskAsyncEnumerable().Reverse().ToArrayAsync();
|
var xs = await Enumerable.Range(start, count).ToUniTaskAsyncEnumerable().Reverse().ToArrayAsync();
|
||||||
var ys = Enumerable.Range(start, count).Reverse().ToArray();
|
var ys = Enumerable.Range(start, count).Reverse().ToArray();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -46,18 +46,18 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, count).Select(x => x * x).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, count).Select(x => x * x).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, count).Select(x => x * x).ToArray();
|
var ys = Enumerable.Range(1, count).Select(x => x * x).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
|
|
||||||
var zs = await UniTaskAsyncEnumerable.Range(1, count).SelectAwait((x) => UniTask.Run(() => x * x)).ToArrayAsync();
|
var zs = await UniTaskAsyncEnumerable.Range(1, count).SelectAwait((x) => UniTask.Run(() => x * x)).ToArrayAsync();
|
||||||
zs.Should().Equal(ys);
|
zs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, count).Select((x, i) => x * x * i).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, count).Select((x, i) => x * x * i).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, count).Select((x, i) => x * x * i).ToArray();
|
var ys = Enumerable.Range(1, count).Select((x, i) => x * x * i).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
|
|
||||||
var zs = await UniTaskAsyncEnumerable.Range(1, count).SelectAwait((x, i) => UniTask.Run(() => x * x * i)).ToArrayAsync();
|
var zs = await UniTaskAsyncEnumerable.Range(1, count).SelectAwait((x, i) => UniTask.Run(() => x * x * i)).ToArrayAsync();
|
||||||
zs.Should().Equal(ys);
|
zs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -98,22 +98,22 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany(x => UniTaskAsyncEnumerable.Range(99, rightCount * x)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany(x => UniTaskAsyncEnumerable.Range(99, rightCount * x)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany((i, x) => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany((i, x) => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany(x => UniTaskAsyncEnumerable.Range(99, rightCount * x), (x, y) => x * y).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany(x => UniTaskAsyncEnumerable.Range(99, rightCount * x), (x, y) => x * y).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x), (x, y) => x * y).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x), (x, y) => x * y).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany((i, x) => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectMany((i, x) => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
// await
|
// await
|
||||||
|
@ -121,22 +121,22 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait(x => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait(x => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait((i, x) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait((i, x) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait(x => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x)), (x, y) => UniTask.Run(() => x * y)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait(x => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x)), (x, y) => UniTask.Run(() => x * y)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x), (x, y) => x * y).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x), (x, y) => x * y).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait((i, x) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x)), (x, y) => UniTask.Run(() => x * y)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwait((i, x) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x)), (x, y) => UniTask.Run(() => x * y)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
// with cancel
|
// with cancel
|
||||||
|
@ -144,22 +144,22 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((i, x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((i, x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x)), (x, y, _) => UniTask.Run(() => x * y)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99, rightCount * x)), (x, y, _) => UniTask.Run(() => x * y)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x), (x, y) => x * y).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany(x => Enumerable.Range(99, rightCount * x), (x, y) => x * y).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((i, x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x)), (x, y, _) => UniTask.Run(() => x * y)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).SelectManyAwaitWithCancellation((i, x, _) => UniTask.Run(() => UniTaskAsyncEnumerable.Range(99 * i, rightCount * x)), (x, y, _) => UniTask.Run(() => x * y)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArray();
|
var ys = Enumerable.Range(1, leftCount).SelectMany((i, x) => Enumerable.Range(99 * i, rightCount * x), (x, y) => x * y).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -219,17 +219,17 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).Zip(UniTaskAsyncEnumerable.Range(99, rightCount)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).Zip(UniTaskAsyncEnumerable.Range(99, rightCount)).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).Zip(Enumerable.Range(99, rightCount)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).Zip(Enumerable.Range(99, rightCount)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).ZipAwait(UniTaskAsyncEnumerable.Range(99, rightCount), (x, y) => UniTask.Run(() => (x, y))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).ZipAwait(UniTaskAsyncEnumerable.Range(99, rightCount), (x, y) => UniTask.Run(() => (x, y))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).Zip(Enumerable.Range(99, rightCount)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).Zip(Enumerable.Range(99, rightCount)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).ZipAwaitWithCancellation(UniTaskAsyncEnumerable.Range(99, rightCount), (x, y, _) => UniTask.Run(() => (x, y))).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(1, leftCount).ZipAwaitWithCancellation(UniTaskAsyncEnumerable.Range(99, rightCount), (x, y, _) => UniTask.Run(() => (x, y))).ToArrayAsync();
|
||||||
var ys = Enumerable.Range(1, leftCount).Zip(Enumerable.Range(99, rightCount)).ToArray();
|
var ys = Enumerable.Range(1, leftCount).Zip(Enumerable.Range(99, rightCount)).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -288,7 +288,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
||||||
var ys = await AsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
var ys = await AsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory]
|
[Theory]
|
||||||
|
@ -305,7 +305,7 @@ namespace NetCoreTests.Linq
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount, skipCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
var xs = await UniTaskAsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount, skipCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
||||||
var ys = await AsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount, skipCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
var ys = await AsyncEnumerable.Range(0, rangeCount).Buffer(bufferCount, skipCount).Select(x => string.Join(",", x)).ToArrayAsync();
|
||||||
|
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@ -319,119 +319,5 @@ namespace NetCoreTests.Linq
|
||||||
await Assert.ThrowsAsync<UniTaskTestException>(async () => await ys);
|
await Assert.ThrowsAsync<UniTaskTestException>(async () => await ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task CombineLatestOK()
|
|
||||||
{
|
|
||||||
var a = new AsyncReactiveProperty<int>(0);
|
|
||||||
var b = new AsyncReactiveProperty<int>(0);
|
|
||||||
|
|
||||||
var list = new List<(int, int)>();
|
|
||||||
var complete = a.WithoutCurrent().CombineLatest(b.WithoutCurrent(), (x, y) => (x, y)).ForEachAsync(x => list.Add(x));
|
|
||||||
|
|
||||||
list.Count.Should().Be(0);
|
|
||||||
|
|
||||||
a.Value = 10;
|
|
||||||
list.Count.Should().Be(0);
|
|
||||||
|
|
||||||
a.Value = 20;
|
|
||||||
list.Count.Should().Be(0);
|
|
||||||
|
|
||||||
b.Value = 1;
|
|
||||||
list.Count.Should().Be(1);
|
|
||||||
|
|
||||||
list[0].Should().Be((20, 1));
|
|
||||||
|
|
||||||
a.Value = 30;
|
|
||||||
list.Last().Should().Be((30, 1));
|
|
||||||
|
|
||||||
b.Value = 2;
|
|
||||||
list.Last().Should().Be((30, 2));
|
|
||||||
|
|
||||||
a.Dispose();
|
|
||||||
b.Value = 3;
|
|
||||||
list.Last().Should().Be((30, 3));
|
|
||||||
|
|
||||||
b.Dispose();
|
|
||||||
|
|
||||||
await complete;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task CombineLatestLong()
|
|
||||||
{
|
|
||||||
var a = UniTaskAsyncEnumerable.Range(1, 100000);
|
|
||||||
var b = new AsyncReactiveProperty<int>(0);
|
|
||||||
|
|
||||||
var list = new List<(int, int)>();
|
|
||||||
var complete = a.CombineLatest(b.WithoutCurrent(), (x, y) => (x, y)).ForEachAsync(x => list.Add(x));
|
|
||||||
|
|
||||||
b.Value = 1;
|
|
||||||
|
|
||||||
list[0].Should().Be((100000, 1));
|
|
||||||
|
|
||||||
b.Dispose();
|
|
||||||
|
|
||||||
await complete;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task CombineLatestError()
|
|
||||||
{
|
|
||||||
var a = new AsyncReactiveProperty<int>(0);
|
|
||||||
var b = new AsyncReactiveProperty<int>(0);
|
|
||||||
|
|
||||||
var list = new List<(int, int)>();
|
|
||||||
var complete = a.WithoutCurrent()
|
|
||||||
.Select(x => { if (x == 0) { throw new MyException(); } return x; })
|
|
||||||
.CombineLatest(b.WithoutCurrent(), (x, y) => (x, y)).ForEachAsync(x => list.Add(x));
|
|
||||||
|
|
||||||
|
|
||||||
a.Value = 10;
|
|
||||||
b.Value = 1;
|
|
||||||
list.Last().Should().Be((10, 1));
|
|
||||||
|
|
||||||
a.Value = 0;
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<MyException>(async () => await complete);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task PariwiseImmediate()
|
|
||||||
{
|
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(1, 5).Pairwise().ToArrayAsync();
|
|
||||||
xs.Should().Equal((1, 2), (2, 3), (3, 4), (4, 5));
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Pariwise()
|
|
||||||
{
|
|
||||||
var a = new AsyncReactiveProperty<int>(0);
|
|
||||||
|
|
||||||
var list = new List<(int, int)>();
|
|
||||||
var complete = a.WithoutCurrent().Pairwise().ForEachAsync(x => list.Add(x));
|
|
||||||
|
|
||||||
list.Count.Should().Be(0);
|
|
||||||
a.Value = 10;
|
|
||||||
list.Count.Should().Be(0);
|
|
||||||
a.Value = 20;
|
|
||||||
list.Count.Should().Be(1);
|
|
||||||
a.Value = 30;
|
|
||||||
a.Value = 40;
|
|
||||||
a.Value = 50;
|
|
||||||
|
|
||||||
a.Dispose();
|
|
||||||
|
|
||||||
await complete;
|
|
||||||
|
|
||||||
list.Should().Equal((10, 20), (20, 30), (30, 40), (40, 50));
|
|
||||||
}
|
|
||||||
|
|
||||||
class MyException : Exception
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,78 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests.Linq
|
|
||||||
{
|
|
||||||
public class PublishTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Normal()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var multicast = rp.Publish();
|
|
||||||
|
|
||||||
var a = multicast.ToArrayAsync();
|
|
||||||
var b = multicast.Take(2).ToArrayAsync();
|
|
||||||
|
|
||||||
var disp = multicast.Connect();
|
|
||||||
|
|
||||||
rp.Value = 2;
|
|
||||||
|
|
||||||
(await b).Should().Equal(1, 2);
|
|
||||||
|
|
||||||
var c = multicast.ToArrayAsync();
|
|
||||||
|
|
||||||
rp.Value = 3;
|
|
||||||
rp.Value = 4;
|
|
||||||
rp.Value = 5;
|
|
||||||
|
|
||||||
rp.Dispose();
|
|
||||||
|
|
||||||
(await a).Should().Equal(1, 2, 3, 4, 5);
|
|
||||||
(await c).Should().Equal(3, 4, 5);
|
|
||||||
|
|
||||||
disp.Dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Cancel()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var multicast = rp.Publish();
|
|
||||||
|
|
||||||
var a = multicast.ToArrayAsync();
|
|
||||||
var b = multicast.Take(2).ToArrayAsync();
|
|
||||||
|
|
||||||
var disp = multicast.Connect();
|
|
||||||
|
|
||||||
rp.Value = 2;
|
|
||||||
|
|
||||||
(await b).Should().Equal(1, 2);
|
|
||||||
|
|
||||||
var c = multicast.ToArrayAsync();
|
|
||||||
|
|
||||||
rp.Value = 3;
|
|
||||||
|
|
||||||
disp.Dispose();
|
|
||||||
|
|
||||||
rp.Value = 4;
|
|
||||||
rp.Value = 5;
|
|
||||||
|
|
||||||
rp.Dispose();
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await a);
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await c);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,29 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests.Linq
|
|
||||||
{
|
|
||||||
public class QueueTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Q()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(100);
|
|
||||||
|
|
||||||
var l = new List<int>();
|
|
||||||
await rp.Take(10).Queue().ForEachAsync(x =>
|
|
||||||
{
|
|
||||||
rp.Value += 10;
|
|
||||||
l.Add(x);
|
|
||||||
});
|
|
||||||
|
|
||||||
l.Should().Equal(100, 110, 120, 130, 140, 150, 160, 170, 180, 190);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -34,10 +34,10 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var ys = array.Distinct().ToArray();
|
var ys = array.Distinct().ToArray();
|
||||||
{
|
{
|
||||||
(await array.ToUniTaskAsyncEnumerable().Distinct().ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().Distinct().ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
(await array.ToUniTaskAsyncEnumerable().Distinct(x => x).ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().Distinct(x => x).ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
(await array.ToUniTaskAsyncEnumerable().DistinctAwait(x => UniTask.Run(() => x)).ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().DistinctAwait(x => UniTask.Run(() => x)).ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
(await array.ToUniTaskAsyncEnumerable().DistinctAwaitWithCancellation((x, _) => UniTask.Run(() => x)).ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().DistinctAwaitWithCancellation((x, _) => UniTask.Run(() => x)).ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,10 +71,10 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var ys = await array.ToAsyncEnumerable().DistinctUntilChanged().ToArrayAsync();
|
var ys = await array.ToAsyncEnumerable().DistinctUntilChanged().ToArrayAsync();
|
||||||
{
|
{
|
||||||
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChanged().ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChanged().ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChanged(x => x).ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChanged(x => x).ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChangedAwait(x => UniTask.Run(() => x)).ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChangedAwait(x => UniTask.Run(() => x)).ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChangedAwaitWithCancellation((x, _) => UniTask.Run(() => x)).ToArrayAsync()).Should().Equal(ys);
|
(await array.ToUniTaskAsyncEnumerable().DistinctUntilChangedAwaitWithCancellation((x, _) => UniTask.Run(() => x)).ToArrayAsync()).Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -112,7 +112,7 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await a1.ToUniTaskAsyncEnumerable().Except(a2.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
var xs = await a1.ToUniTaskAsyncEnumerable().Except(a2.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
||||||
var ys = a1.Except(a2).ToArray();
|
var ys = a1.Except(a2).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -141,7 +141,7 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await a1.ToUniTaskAsyncEnumerable().Intersect(a2.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
var xs = await a1.ToUniTaskAsyncEnumerable().Intersect(a2.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
||||||
var ys = a1.Intersect(a2).ToArray();
|
var ys = a1.Intersect(a2).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -170,7 +170,7 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await a1.ToUniTaskAsyncEnumerable().Union(a2.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
var xs = await a1.ToUniTaskAsyncEnumerable().Union(a2.ToUniTaskAsyncEnumerable()).ToArrayAsync();
|
||||||
var ys = a1.Union(a2).ToArray();
|
var ys = a1.Union(a2).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,32 +59,32 @@ namespace NetCoreTests.Linq
|
||||||
{
|
{
|
||||||
var xs = await array.ToUniTaskAsyncEnumerable().OrderBy(x => x).ToArrayAsync();
|
var xs = await array.ToUniTaskAsyncEnumerable().OrderBy(x => x).ToArrayAsync();
|
||||||
var ys = array.OrderBy(x => x).ToArray();
|
var ys = array.OrderBy(x => x).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await array.ToUniTaskAsyncEnumerable().OrderByDescending(x => x).ToArrayAsync();
|
var xs = await array.ToUniTaskAsyncEnumerable().OrderByDescending(x => x).ToArrayAsync();
|
||||||
var ys = array.OrderByDescending(x => x).ToArray();
|
var ys = array.OrderByDescending(x => x).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await array.ToUniTaskAsyncEnumerable().OrderByAwait(RandomRun).ToArrayAsync();
|
var xs = await array.ToUniTaskAsyncEnumerable().OrderByAwait(RandomRun).ToArrayAsync();
|
||||||
var ys = array.OrderBy(x => x).ToArray();
|
var ys = array.OrderBy(x => x).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwait(RandomRun).ToArrayAsync();
|
var xs = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwait(RandomRun).ToArrayAsync();
|
||||||
var ys = array.OrderByDescending(x => x).ToArray();
|
var ys = array.OrderByDescending(x => x).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await array.ToUniTaskAsyncEnumerable().OrderByAwaitWithCancellation(RandomRun).ToArrayAsync();
|
var xs = await array.ToUniTaskAsyncEnumerable().OrderByAwaitWithCancellation(RandomRun).ToArrayAsync();
|
||||||
var ys = array.OrderBy(x => x).ToArray();
|
var ys = array.OrderBy(x => x).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwaitWithCancellation(RandomRun).ToArrayAsync();
|
var xs = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwaitWithCancellation(RandomRun).ToArrayAsync();
|
||||||
var ys = array.OrderByDescending(x => x).ToArray();
|
var ys = array.OrderByDescending(x => x).ToArray();
|
||||||
xs.Should().Equal(ys);
|
xs.Should().BeEquivalentTo(ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -125,14 +125,14 @@ namespace NetCoreTests.Linq
|
||||||
var g2 = await array.ToUniTaskAsyncEnumerable().OrderByDescending(x => x.Age).ThenByDescending(x => x.FirstName).ThenBy(x => x.LastName).ToArrayAsync();
|
var g2 = await array.ToUniTaskAsyncEnumerable().OrderByDescending(x => x.Age).ThenByDescending(x => x.FirstName).ThenBy(x => x.LastName).ToArrayAsync();
|
||||||
var h2 = await array.ToUniTaskAsyncEnumerable().OrderByDescending(x => x.Age).ThenByDescending(x => x.FirstName).ThenByDescending(x => x.LastName).ToArrayAsync();
|
var h2 = await array.ToUniTaskAsyncEnumerable().OrderByDescending(x => x.Age).ThenByDescending(x => x.FirstName).ThenByDescending(x => x.LastName).ToArrayAsync();
|
||||||
|
|
||||||
a.Should().Equal(a2);
|
a.Should().BeEquivalentTo(a2);
|
||||||
b.Should().Equal(b2);
|
b.Should().BeEquivalentTo(b2);
|
||||||
c.Should().Equal(c2);
|
c.Should().BeEquivalentTo(c2);
|
||||||
d.Should().Equal(d2);
|
d.Should().BeEquivalentTo(d2);
|
||||||
e.Should().Equal(e2);
|
e.Should().BeEquivalentTo(e2);
|
||||||
f.Should().Equal(f2);
|
f.Should().BeEquivalentTo(f2);
|
||||||
g.Should().Equal(g2);
|
g.Should().BeEquivalentTo(g2);
|
||||||
h.Should().Equal(h2);
|
h.Should().BeEquivalentTo(h2);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var a2 = await array.ToUniTaskAsyncEnumerable().OrderByAwait(x => RandomRun(x.Age)).ThenByAwait(x => RandomRun(x.FirstName)).ThenByAwait(x => RandomRun(x.LastName)).ToArrayAsync();
|
var a2 = await array.ToUniTaskAsyncEnumerable().OrderByAwait(x => RandomRun(x.Age)).ThenByAwait(x => RandomRun(x.FirstName)).ThenByAwait(x => RandomRun(x.LastName)).ToArrayAsync();
|
||||||
|
@ -144,14 +144,14 @@ namespace NetCoreTests.Linq
|
||||||
var g2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwait(x => RandomRun(x.Age)).ThenByDescendingAwait(x => RandomRun(x.FirstName)).ThenByAwait(x => RandomRun(x.LastName)).ToArrayAsync();
|
var g2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwait(x => RandomRun(x.Age)).ThenByDescendingAwait(x => RandomRun(x.FirstName)).ThenByAwait(x => RandomRun(x.LastName)).ToArrayAsync();
|
||||||
var h2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwait(x => RandomRun(x.Age)).ThenByDescendingAwait(x => RandomRun(x.FirstName)).ThenByDescendingAwait(x => RandomRun(x.LastName)).ToArrayAsync();
|
var h2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwait(x => RandomRun(x.Age)).ThenByDescendingAwait(x => RandomRun(x.FirstName)).ThenByDescendingAwait(x => RandomRun(x.LastName)).ToArrayAsync();
|
||||||
|
|
||||||
a.Should().Equal(a2);
|
a.Should().BeEquivalentTo(a2);
|
||||||
b.Should().Equal(b2);
|
b.Should().BeEquivalentTo(b2);
|
||||||
c.Should().Equal(c2);
|
c.Should().BeEquivalentTo(c2);
|
||||||
d.Should().Equal(d2);
|
d.Should().BeEquivalentTo(d2);
|
||||||
e.Should().Equal(e2);
|
e.Should().BeEquivalentTo(e2);
|
||||||
f.Should().Equal(f2);
|
f.Should().BeEquivalentTo(f2);
|
||||||
g.Should().Equal(g2);
|
g.Should().BeEquivalentTo(g2);
|
||||||
h.Should().Equal(h2);
|
h.Should().BeEquivalentTo(h2);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var a2 = await array.ToUniTaskAsyncEnumerable().OrderByAwaitWithCancellation((x, ct) => RandomRun(x.Age)).ThenByAwaitWithCancellation((x, ct) => RandomRun(x.FirstName)).ThenByAwaitWithCancellation((x, ct) => RandomRun(x.LastName)).ToArrayAsync();
|
var a2 = await array.ToUniTaskAsyncEnumerable().OrderByAwaitWithCancellation((x, ct) => RandomRun(x.Age)).ThenByAwaitWithCancellation((x, ct) => RandomRun(x.FirstName)).ThenByAwaitWithCancellation((x, ct) => RandomRun(x.LastName)).ToArrayAsync();
|
||||||
|
@ -163,14 +163,14 @@ namespace NetCoreTests.Linq
|
||||||
var g2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.Age)).ThenByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.FirstName)).ThenByAwaitWithCancellation((x, ct) => RandomRun(x.LastName)).ToArrayAsync();
|
var g2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.Age)).ThenByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.FirstName)).ThenByAwaitWithCancellation((x, ct) => RandomRun(x.LastName)).ToArrayAsync();
|
||||||
var h2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.Age)).ThenByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.FirstName)).ThenByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.LastName)).ToArrayAsync();
|
var h2 = await array.ToUniTaskAsyncEnumerable().OrderByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.Age)).ThenByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.FirstName)).ThenByDescendingAwaitWithCancellation((x, ct) => RandomRun(x.LastName)).ToArrayAsync();
|
||||||
|
|
||||||
a.Should().Equal(a2);
|
a.Should().BeEquivalentTo(a2);
|
||||||
b.Should().Equal(b2);
|
b.Should().BeEquivalentTo(b2);
|
||||||
c.Should().Equal(c2);
|
c.Should().BeEquivalentTo(c2);
|
||||||
d.Should().Equal(d2);
|
d.Should().BeEquivalentTo(d2);
|
||||||
e.Should().Equal(e2);
|
e.Should().BeEquivalentTo(e2);
|
||||||
f.Should().Equal(f2);
|
f.Should().BeEquivalentTo(f2);
|
||||||
g.Should().Equal(g2);
|
g.Should().BeEquivalentTo(g2);
|
||||||
h.Should().Equal(h2);
|
h.Should().BeEquivalentTo(h2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,165 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests.Linq
|
|
||||||
{
|
|
||||||
public class TakeInfinityTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Take()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var xs = rp.Take(5).ToArrayAsync();
|
|
||||||
|
|
||||||
rp.Value = 2;
|
|
||||||
rp.Value = 3;
|
|
||||||
rp.Value = 4;
|
|
||||||
rp.Value = 5;
|
|
||||||
|
|
||||||
(await xs).Should().Equal(1, 2, 3, 4, 5);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task TakeWhile()
|
|
||||||
{
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var xs = rp.TakeWhile(x => x != 5).ToArrayAsync();
|
|
||||||
|
|
||||||
rp.Value = 2;
|
|
||||||
rp.Value = 3;
|
|
||||||
rp.Value = 4;
|
|
||||||
rp.Value = 5;
|
|
||||||
|
|
||||||
(await xs).Should().Equal(1, 2, 3, 4);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task TakeUntilCanceled()
|
|
||||||
{
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var xs = rp.TakeUntilCanceled(cts.Token).ToArrayAsync();
|
|
||||||
|
|
||||||
var c = CancelAsync();
|
|
||||||
|
|
||||||
await c;
|
|
||||||
var foo = await xs;
|
|
||||||
|
|
||||||
foo.Should().Equal(new[] { 1, 10, 20 });
|
|
||||||
|
|
||||||
async Task CancelAsync()
|
|
||||||
{
|
|
||||||
rp.Value = 10;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 20;
|
|
||||||
await Task.Yield();
|
|
||||||
cts.Cancel();
|
|
||||||
rp.Value = 30;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 40;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SkipUntilCanceled()
|
|
||||||
{
|
|
||||||
var cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var xs = rp.SkipUntilCanceled(cts.Token).ToArrayAsync();
|
|
||||||
|
|
||||||
var c = CancelAsync();
|
|
||||||
|
|
||||||
await c;
|
|
||||||
var foo = await xs;
|
|
||||||
|
|
||||||
foo.Should().Equal(new[] { 20, 30, 40 });
|
|
||||||
|
|
||||||
async Task CancelAsync()
|
|
||||||
{
|
|
||||||
rp.Value = 10;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 20;
|
|
||||||
await Task.Yield();
|
|
||||||
cts.Cancel();
|
|
||||||
rp.Value = 30;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 40;
|
|
||||||
|
|
||||||
rp.Dispose(); // complete.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task TakeUntil()
|
|
||||||
{
|
|
||||||
var cts = new AsyncReactiveProperty<int>(0);
|
|
||||||
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var xs = rp.TakeUntil(cts.WaitAsync()).ToArrayAsync();
|
|
||||||
|
|
||||||
var c = CancelAsync();
|
|
||||||
|
|
||||||
await c;
|
|
||||||
var foo = await xs;
|
|
||||||
|
|
||||||
foo.Should().Equal(new[] { 1, 10, 20 });
|
|
||||||
|
|
||||||
async Task CancelAsync()
|
|
||||||
{
|
|
||||||
rp.Value = 10;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 20;
|
|
||||||
await Task.Yield();
|
|
||||||
cts.Value = 9999;
|
|
||||||
rp.Value = 30;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 40;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SkipUntil()
|
|
||||||
{
|
|
||||||
var cts = new AsyncReactiveProperty<int>(0);
|
|
||||||
|
|
||||||
var rp = new AsyncReactiveProperty<int>(1);
|
|
||||||
|
|
||||||
var xs = rp.SkipUntil(cts.WaitAsync()).ToArrayAsync();
|
|
||||||
|
|
||||||
var c = CancelAsync();
|
|
||||||
|
|
||||||
await c;
|
|
||||||
var foo = await xs;
|
|
||||||
|
|
||||||
foo.Should().Equal(new[] { 20, 30, 40 });
|
|
||||||
|
|
||||||
async Task CancelAsync()
|
|
||||||
{
|
|
||||||
rp.Value = 10;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 20;
|
|
||||||
await Task.Yield();
|
|
||||||
cts.Value = 9999;
|
|
||||||
rp.Value = 30;
|
|
||||||
await Task.Yield();
|
|
||||||
rp.Value = 40;
|
|
||||||
|
|
||||||
rp.Dispose(); // complete.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,301 +0,0 @@
|
||||||
#pragma warning disable CS1998
|
|
||||||
|
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class UniTaskBuilderTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Empty()
|
|
||||||
{
|
|
||||||
await Core();
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task EmptyThrow()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
throw new TaskTestException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Task_Done()
|
|
||||||
{
|
|
||||||
await Core();
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter(true, UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Task_Fail()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter(true, UniTaskStatus.Faulted);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Task_Cancel()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter(true, UniTaskStatus.Canceled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AwaitUnsafeOnCompletedCall_Task_SetResult()
|
|
||||||
{
|
|
||||||
await Core();
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
|
||||||
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
|
||||||
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AwaitUnsafeOnCompletedCall_Task_SetException()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
|
||||||
await new TestAwaiter(false, UniTaskStatus.Faulted);
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AwaitUnsafeOnCompletedCall_Task_SetCancelException()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
|
||||||
await new TestAwaiter(false, UniTaskStatus.Canceled);
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class UniTask_T_BuilderTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Empty()
|
|
||||||
{
|
|
||||||
(await Core()).Should().Be(10);
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
return 10;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task EmptyThrow()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
throw new TaskTestException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Task_Done()
|
|
||||||
{
|
|
||||||
(await Core()).Should().Be(10);
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
return await new TestAwaiter<int>(true, UniTaskStatus.Succeeded, 10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Task_Fail()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
return await new TestAwaiter<int>(true, UniTaskStatus.Faulted, 10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Task_Cancel()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
return await new TestAwaiter<int>(true, UniTaskStatus.Canceled, 10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AwaitUnsafeOnCompletedCall_Task_SetResult()
|
|
||||||
{
|
|
||||||
(await Core()).Should().Be(6);
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
var sum = 0;
|
|
||||||
sum += await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 1);
|
|
||||||
sum += await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 2);
|
|
||||||
sum += await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 3);
|
|
||||||
return sum;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AwaitUnsafeOnCompletedCall_Task_SetException()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 10);
|
|
||||||
await new TestAwaiter<int>(false, UniTaskStatus.Faulted, 10);
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task AwaitUnsafeOnCompletedCall_Task_SetCancelException()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
|
||||||
|
|
||||||
static async UniTask<int> Core()
|
|
||||||
{
|
|
||||||
await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 10);
|
|
||||||
await new TestAwaiter<int>(false, UniTaskStatus.Canceled, 10);
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class TaskTestException : Exception
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct TestAwaiter : ICriticalNotifyCompletion
|
|
||||||
{
|
|
||||||
readonly UniTaskStatus status;
|
|
||||||
readonly bool isCompleted;
|
|
||||||
|
|
||||||
public TestAwaiter(bool isCompleted, UniTaskStatus status)
|
|
||||||
{
|
|
||||||
this.isCompleted = isCompleted;
|
|
||||||
this.status = status;
|
|
||||||
}
|
|
||||||
|
|
||||||
public TestAwaiter GetAwaiter() => this;
|
|
||||||
|
|
||||||
public bool IsCompleted => isCompleted;
|
|
||||||
|
|
||||||
public void GetResult()
|
|
||||||
{
|
|
||||||
switch (status)
|
|
||||||
{
|
|
||||||
case UniTaskStatus.Faulted:
|
|
||||||
throw new TaskTestException();
|
|
||||||
case UniTaskStatus.Canceled:
|
|
||||||
throw new OperationCanceledException();
|
|
||||||
case UniTaskStatus.Pending:
|
|
||||||
case UniTaskStatus.Succeeded:
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
ThreadPool.QueueUserWorkItem(_ => continuation(), null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UnsafeOnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
ThreadPool.UnsafeQueueUserWorkItem(_ => continuation(), null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct TestAwaiter<T> : ICriticalNotifyCompletion
|
|
||||||
{
|
|
||||||
readonly UniTaskStatus status;
|
|
||||||
readonly bool isCompleted;
|
|
||||||
readonly T value;
|
|
||||||
|
|
||||||
public TestAwaiter(bool isCompleted, UniTaskStatus status, T value)
|
|
||||||
{
|
|
||||||
this.isCompleted = isCompleted;
|
|
||||||
this.status = status;
|
|
||||||
this.value = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public TestAwaiter<T> GetAwaiter() => this;
|
|
||||||
|
|
||||||
public bool IsCompleted => isCompleted;
|
|
||||||
|
|
||||||
public T GetResult()
|
|
||||||
{
|
|
||||||
switch (status)
|
|
||||||
{
|
|
||||||
case UniTaskStatus.Faulted:
|
|
||||||
throw new TaskTestException();
|
|
||||||
case UniTaskStatus.Canceled:
|
|
||||||
throw new OperationCanceledException();
|
|
||||||
case UniTaskStatus.Pending:
|
|
||||||
case UniTaskStatus.Succeeded:
|
|
||||||
default:
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
ThreadPool.QueueUserWorkItem(_ => continuation(), null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UnsafeOnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
ThreadPool.UnsafeQueueUserWorkItem(_ => continuation(), null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,45 +0,0 @@
|
||||||
#pragma warning disable CS1998
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class TaskExtensionsTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task PropagateException()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<InvalidOperationException>(async () =>
|
|
||||||
{
|
|
||||||
await ThrowAsync().AsUniTask();
|
|
||||||
});
|
|
||||||
|
|
||||||
await Assert.ThrowsAsync<InvalidOperationException>(async () =>
|
|
||||||
{
|
|
||||||
await ThrowOrValueAsync().AsUniTask();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task PropagateExceptionWhenAll()
|
|
||||||
{
|
|
||||||
await Assert.ThrowsAsync<InvalidOperationException>(async () =>
|
|
||||||
{
|
|
||||||
await Task.WhenAll(ThrowAsync(), ThrowAsync()).AsUniTask();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task ThrowAsync()
|
|
||||||
{
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task<int> ThrowOrValueAsync()
|
|
||||||
{
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,637 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Channels;
|
|
||||||
using Cysharp.Threading.Tasks.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class TriggerEventTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public void SimpleAdd()
|
|
||||||
{
|
|
||||||
var ev = new TriggerEvent<int>();
|
|
||||||
|
|
||||||
// do nothing
|
|
||||||
ev.SetResult(0);
|
|
||||||
ev.SetError(null);
|
|
||||||
ev.SetCompleted();
|
|
||||||
ev.SetCanceled(default);
|
|
||||||
|
|
||||||
{
|
|
||||||
var one = new TestEvent(1);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
|
|
||||||
ev.SetCompleted();
|
|
||||||
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
|
|
||||||
// do nothing
|
|
||||||
ev.SetResult(0);
|
|
||||||
ev.SetError(null);
|
|
||||||
ev.SetCompleted();
|
|
||||||
ev.SetCanceled(default);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
}
|
|
||||||
// after removed, onemore
|
|
||||||
{
|
|
||||||
var one = new TestEvent(1);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
|
|
||||||
ev.SetCompleted();
|
|
||||||
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
|
|
||||||
// do nothing
|
|
||||||
ev.SetResult(0);
|
|
||||||
ev.SetError(null);
|
|
||||||
ev.SetCompleted();
|
|
||||||
ev.SetCanceled(default);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void AddFour()
|
|
||||||
{
|
|
||||||
var ev = new TriggerEvent<int>();
|
|
||||||
|
|
||||||
// do nothing
|
|
||||||
ev.SetResult(0);
|
|
||||||
ev.SetError(null);
|
|
||||||
ev.SetCompleted();
|
|
||||||
ev.SetCanceled(default);
|
|
||||||
|
|
||||||
{
|
|
||||||
var one = new TestEvent(1);
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
var four = new TestEvent(4);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
ev.Add(four);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
four.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
|
|
||||||
ev.SetCompleted();
|
|
||||||
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
two.CompletedCalled.Count.Should().Be(1);
|
|
||||||
three.CompletedCalled.Count.Should().Be(1);
|
|
||||||
|
|
||||||
|
|
||||||
// do nothing
|
|
||||||
ev.SetResult(0);
|
|
||||||
ev.SetError(null);
|
|
||||||
ev.SetCompleted();
|
|
||||||
ev.SetCanceled(default);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.CompletedCalled.Count.Should().Be(1);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.CompletedCalled.Count.Should().Be(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// after removed, onemore.
|
|
||||||
{
|
|
||||||
var one = new TestEvent(1);
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
var four = new TestEvent(4);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
ev.Add(four);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
ev.Add(four);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
four.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
|
|
||||||
ev.SetCompleted();
|
|
||||||
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
two.CompletedCalled.Count.Should().Be(1);
|
|
||||||
three.CompletedCalled.Count.Should().Be(1);
|
|
||||||
|
|
||||||
|
|
||||||
// do nothing
|
|
||||||
ev.SetResult(0);
|
|
||||||
ev.SetError(null);
|
|
||||||
ev.SetCompleted();
|
|
||||||
ev.SetCanceled(default);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
one.CompletedCalled.Count.Should().Be(1);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.CompletedCalled.Count.Should().Be(1);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.CompletedCalled.Count.Should().Be(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void OneRemove()
|
|
||||||
{
|
|
||||||
var ev = new TriggerEvent<int>();
|
|
||||||
{
|
|
||||||
var one = new TestEvent(1);
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
|
|
||||||
ev.Remove(one);
|
|
||||||
|
|
||||||
ev.SetResult(40);
|
|
||||||
ev.SetResult(50);
|
|
||||||
ev.SetResult(60);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30, 40, 50, 60);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30, 40, 50, 60);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
[Fact]
|
|
||||||
public void TwoRemove()
|
|
||||||
{
|
|
||||||
var ev = new TriggerEvent<int>();
|
|
||||||
{
|
|
||||||
var one = new TestEvent(1);
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
|
|
||||||
ev.Remove(two);
|
|
||||||
|
|
||||||
ev.SetResult(40);
|
|
||||||
ev.SetResult(50);
|
|
||||||
ev.SetResult(60);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30, 40, 50, 60);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30, 40, 50, 60);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
[Fact]
|
|
||||||
public void ThreeRemove()
|
|
||||||
{
|
|
||||||
var ev = new TriggerEvent<int>();
|
|
||||||
{
|
|
||||||
var one = new TestEvent(1);
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
|
|
||||||
ev.Remove(three);
|
|
||||||
|
|
||||||
ev.SetResult(40);
|
|
||||||
ev.SetResult(50);
|
|
||||||
ev.SetResult(60);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30, 40, 50, 60);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30, 40, 50, 60);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void RemoveSelf()
|
|
||||||
{
|
|
||||||
new RemoveMe().Run1();
|
|
||||||
new RemoveMe().Run2();
|
|
||||||
new RemoveMe().Run3();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void RemoveNextInIterating()
|
|
||||||
{
|
|
||||||
new RemoveNext().Run1();
|
|
||||||
new RemoveNext().Run2();
|
|
||||||
new RemoveNext().Run3();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void RemoveNextNextTest()
|
|
||||||
{
|
|
||||||
new RemoveNextNext().Run1();
|
|
||||||
new RemoveNextNext().Run2();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void AddTest()
|
|
||||||
{
|
|
||||||
new AddMe().Run1();
|
|
||||||
new AddMe().Run2();
|
|
||||||
}
|
|
||||||
|
|
||||||
public class RemoveMe
|
|
||||||
{
|
|
||||||
TriggerEvent<int> ev;
|
|
||||||
|
|
||||||
public void Run1()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
one = new TestEvent(1, () => ev.Remove(one));
|
|
||||||
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Run2()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
one = new TestEvent(1, () => ev.Remove(one));
|
|
||||||
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(one); // add second.
|
|
||||||
ev.Add(three);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Run3()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
one = new TestEvent(1, () => ev.Remove(one));
|
|
||||||
|
|
||||||
var two = new TestEvent(2);
|
|
||||||
var three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
ev.Add(one); // add thired.
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10);
|
|
||||||
two.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class RemoveNext
|
|
||||||
{
|
|
||||||
TriggerEvent<int> ev;
|
|
||||||
|
|
||||||
public void Run1()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
TestEvent two = default;
|
|
||||||
TestEvent three = default;
|
|
||||||
one = new TestEvent(1, () => ev.Remove(two));
|
|
||||||
two = new TestEvent(2);
|
|
||||||
three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Count.Should().Be(0);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Run2()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
TestEvent two = default;
|
|
||||||
TestEvent three = default;
|
|
||||||
one = new TestEvent(1, () => ev.Remove(two));
|
|
||||||
two = new TestEvent(2);
|
|
||||||
three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(one); // add second
|
|
||||||
ev.Add(three);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Run3()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
TestEvent two = default;
|
|
||||||
TestEvent three = default;
|
|
||||||
one = new TestEvent(1, () => ev.Remove(two));
|
|
||||||
two = new TestEvent(2);
|
|
||||||
three = new TestEvent(3);
|
|
||||||
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
ev.Add(one); // add thired.
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Should().Equal(10);
|
|
||||||
three.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class RemoveNextNext
|
|
||||||
{
|
|
||||||
TriggerEvent<int> ev;
|
|
||||||
|
|
||||||
public void Run1()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
TestEvent two = default;
|
|
||||||
TestEvent three = default;
|
|
||||||
TestEvent four = default;
|
|
||||||
one = new TestEvent(1, () => { ev.Remove(two); ev.Remove(three); });
|
|
||||||
two = new TestEvent(2);
|
|
||||||
three = new TestEvent(3);
|
|
||||||
four = new TestEvent(4);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
ev.Add(four);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
two.NextCalled.Count.Should().Be(0);
|
|
||||||
three.NextCalled.Count.Should().Be(0);
|
|
||||||
four.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Run2()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
TestEvent two = default;
|
|
||||||
TestEvent three = default;
|
|
||||||
TestEvent four = default;
|
|
||||||
one = new TestEvent(1, () => { ev.Remove(one); ev.Remove(two); ev.Remove(three); });
|
|
||||||
two = new TestEvent(2);
|
|
||||||
three = new TestEvent(3);
|
|
||||||
four = new TestEvent(4);
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
ev.Add(two);
|
|
||||||
ev.Add(three);
|
|
||||||
ev.Add(four);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10);
|
|
||||||
two.NextCalled.Count.Should().Be(0);
|
|
||||||
three.NextCalled.Count.Should().Be(0);
|
|
||||||
four.NextCalled.Should().Equal(10, 20, 30);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public class AddMe
|
|
||||||
{
|
|
||||||
TriggerEvent<int> ev;
|
|
||||||
|
|
||||||
public void Run1()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
TestEvent two = default;
|
|
||||||
TestEvent three = default;
|
|
||||||
TestEvent four = default;
|
|
||||||
|
|
||||||
one = new TestEvent(1, () =>
|
|
||||||
{
|
|
||||||
if (two == null)
|
|
||||||
{
|
|
||||||
ev.Add(two = new TestEvent(2));
|
|
||||||
}
|
|
||||||
else if (three == null)
|
|
||||||
{
|
|
||||||
ev.Add(three = new TestEvent(3));
|
|
||||||
}
|
|
||||||
else if (four == null)
|
|
||||||
{
|
|
||||||
ev.Add(four = new TestEvent(4));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
ev.SetResult(40);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30, 40);
|
|
||||||
two.NextCalled.Should().Equal(20, 30, 40);
|
|
||||||
three.NextCalled.Should().Equal(30, 40);
|
|
||||||
four.NextCalled.Should().Equal(40);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Run2()
|
|
||||||
{
|
|
||||||
TestEvent one = default;
|
|
||||||
TestEvent two = default;
|
|
||||||
TestEvent three = default;
|
|
||||||
TestEvent four = default;
|
|
||||||
|
|
||||||
one = new TestEvent(1, () =>
|
|
||||||
{
|
|
||||||
if (two == null)
|
|
||||||
{
|
|
||||||
ev.Add(two = new TestEvent(2, () =>
|
|
||||||
{
|
|
||||||
if (three == null)
|
|
||||||
{
|
|
||||||
ev.Add(three = new TestEvent(3, () =>
|
|
||||||
{
|
|
||||||
if (four == null)
|
|
||||||
{
|
|
||||||
ev.Add(four = new TestEvent(4));
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
ev.Add(one);
|
|
||||||
|
|
||||||
ev.SetResult(10);
|
|
||||||
ev.SetResult(20);
|
|
||||||
ev.SetResult(30);
|
|
||||||
ev.SetResult(40);
|
|
||||||
|
|
||||||
one.NextCalled.Should().Equal(10, 20, 30, 40);
|
|
||||||
two.NextCalled.Should().Equal(20, 30, 40);
|
|
||||||
three.NextCalled.Should().Equal(30, 40);
|
|
||||||
four.NextCalled.Should().Equal(40);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class TestEvent : ITriggerHandler<int>
|
|
||||||
{
|
|
||||||
public readonly int Id;
|
|
||||||
readonly Action iteratingEvent;
|
|
||||||
|
|
||||||
public TestEvent(int id)
|
|
||||||
{
|
|
||||||
this.Id = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
public TestEvent(int id, Action iteratingEvent)
|
|
||||||
{
|
|
||||||
this.Id = id;
|
|
||||||
this.iteratingEvent = iteratingEvent;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<int> NextCalled = new List<int>();
|
|
||||||
public List<Exception> ErrorCalled = new List<Exception>();
|
|
||||||
public List<object> CompletedCalled = new List<object>();
|
|
||||||
public List<CancellationToken> CancelCalled = new List<CancellationToken>();
|
|
||||||
|
|
||||||
public ITriggerHandler<int> Prev { get; set; }
|
|
||||||
public ITriggerHandler<int> Next { get; set; }
|
|
||||||
|
|
||||||
public void OnCanceled(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
CancelCalled.Add(cancellationToken);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted()
|
|
||||||
{
|
|
||||||
CompletedCalled.Add(new object());
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnError(Exception ex)
|
|
||||||
{
|
|
||||||
ErrorCalled.Add(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnNext(int value)
|
|
||||||
{
|
|
||||||
NextCalled.Add(value);
|
|
||||||
iteratingEvent?.Invoke();
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
return Id.ToString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,7 +1,7 @@
|
||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>net7.0</TargetFramework>
|
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||||
|
|
||||||
<IsPackable>false</IsPackable>
|
<IsPackable>false</IsPackable>
|
||||||
|
|
||||||
|
|
|
@ -1,80 +0,0 @@
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using NetCoreTests.Linq;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class AutoResetUniTaskCompletionSourceTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task SetResultAfterReturn()
|
|
||||||
{
|
|
||||||
var source1 = AutoResetUniTaskCompletionSource.Create();
|
|
||||||
source1.TrySetResult();
|
|
||||||
await source1.Task;
|
|
||||||
|
|
||||||
source1.TrySetResult().Should().BeFalse();
|
|
||||||
|
|
||||||
var source2 = AutoResetUniTaskCompletionSource.Create();
|
|
||||||
source2.TrySetResult();
|
|
||||||
await source2.Task;
|
|
||||||
|
|
||||||
source2.TrySetResult().Should().BeFalse();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SetCancelAfterReturn()
|
|
||||||
{
|
|
||||||
var source = AutoResetUniTaskCompletionSource.Create();
|
|
||||||
source.TrySetResult();
|
|
||||||
await source.Task;
|
|
||||||
|
|
||||||
source.TrySetCanceled().Should().BeFalse();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SetExceptionAfterReturn()
|
|
||||||
{
|
|
||||||
var source = AutoResetUniTaskCompletionSource.Create();
|
|
||||||
source.TrySetResult();
|
|
||||||
await source.Task;
|
|
||||||
|
|
||||||
source.TrySetException(new UniTaskTestException()).Should().BeFalse();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SetResultWithValueAfterReturn()
|
|
||||||
{
|
|
||||||
var source1 = AutoResetUniTaskCompletionSource<int>.Create();
|
|
||||||
source1.TrySetResult(100);
|
|
||||||
(await source1.Task).Should().Be(100);
|
|
||||||
|
|
||||||
source1.TrySetResult(100).Should().BeFalse();
|
|
||||||
|
|
||||||
var source2 = AutoResetUniTaskCompletionSource.Create();
|
|
||||||
source2.TrySetResult();
|
|
||||||
await source2.Task;
|
|
||||||
source2.TrySetResult().Should().BeFalse();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SetCancelWithValueAfterReturn()
|
|
||||||
{
|
|
||||||
var source = AutoResetUniTaskCompletionSource<int>.Create();
|
|
||||||
source.TrySetResult(100);
|
|
||||||
(await source.Task).Should().Be(100);
|
|
||||||
source.TrySetCanceled().Should().BeFalse();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task SetExceptionWithValueAfterReturn()
|
|
||||||
{
|
|
||||||
var source = AutoResetUniTaskCompletionSource<int>.Create();
|
|
||||||
source.TrySetResult(100);
|
|
||||||
(await source.Task).Should().Be(100);
|
|
||||||
source.TrySetException(new UniTaskTestException()).Should().BeFalse();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,69 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class WhenEachTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Each()
|
|
||||||
{
|
|
||||||
var a = Delay(1, 3000);
|
|
||||||
var b = Delay(2, 1000);
|
|
||||||
var c = Delay(3, 2000);
|
|
||||||
|
|
||||||
var l = new List<int>();
|
|
||||||
await foreach (var item in UniTask.WhenEach(a, b, c))
|
|
||||||
{
|
|
||||||
l.Add(item.Result);
|
|
||||||
}
|
|
||||||
|
|
||||||
l.Should().Equal(2, 3, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Error()
|
|
||||||
{
|
|
||||||
var a = Delay2(1, 3000);
|
|
||||||
var b = Delay2(2, 1000);
|
|
||||||
var c = Delay2(3, 2000);
|
|
||||||
|
|
||||||
var l = new List<WhenEachResult<int>>();
|
|
||||||
await foreach (var item in UniTask.WhenEach(a, b, c))
|
|
||||||
{
|
|
||||||
l.Add(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
l[0].IsCompletedSuccessfully.Should().BeTrue();
|
|
||||||
l[0].IsFaulted.Should().BeFalse();
|
|
||||||
l[0].Result.Should().Be(2);
|
|
||||||
|
|
||||||
l[1].IsCompletedSuccessfully.Should().BeFalse();
|
|
||||||
l[1].IsFaulted.Should().BeTrue();
|
|
||||||
l[1].Exception.Message.Should().Be("ERROR");
|
|
||||||
|
|
||||||
l[2].IsCompletedSuccessfully.Should().BeTrue();
|
|
||||||
l[2].IsFaulted.Should().BeFalse();
|
|
||||||
l[2].Result.Should().Be(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
async UniTask<int> Delay(int id, int sleep)
|
|
||||||
{
|
|
||||||
await Task.Delay(sleep);
|
|
||||||
return id;
|
|
||||||
}
|
|
||||||
|
|
||||||
async UniTask<int> Delay2(int id, int sleep)
|
|
||||||
{
|
|
||||||
await Task.Delay(sleep);
|
|
||||||
if (id == 3) throw new Exception("ERROR");
|
|
||||||
return id;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,40 +0,0 @@
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using FluentAssertions;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace NetCoreTests
|
|
||||||
{
|
|
||||||
public class WithCancellationTest
|
|
||||||
{
|
|
||||||
[Fact]
|
|
||||||
public async Task Standard()
|
|
||||||
{
|
|
||||||
CancellationTokenSource cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var v = await UniTask.Run(() => 10).AttachExternalCancellation(cts.Token);
|
|
||||||
|
|
||||||
v.Should().Be(10);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task Cancel()
|
|
||||||
{
|
|
||||||
CancellationTokenSource cts = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var t = UniTask.Create(async () =>
|
|
||||||
{
|
|
||||||
await Task.Delay(TimeSpan.FromSeconds(1));
|
|
||||||
return 10;
|
|
||||||
}).AttachExternalCancellation(cts.Token);
|
|
||||||
|
|
||||||
cts.Cancel();
|
|
||||||
|
|
||||||
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await t)).CancellationToken.Should().Be(cts.Token);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,33 +0,0 @@
|
||||||
#if UNITY_EDITOR
|
|
||||||
|
|
||||||
using Cysharp.Threading.Tasks;
|
|
||||||
using System;
|
|
||||||
using System.IO;
|
|
||||||
using System.Linq;
|
|
||||||
using UnityEditor;
|
|
||||||
using UnityEngine;
|
|
||||||
using UnityEngine.Networking;
|
|
||||||
|
|
||||||
public static class EditorRunnerChecker
|
|
||||||
{
|
|
||||||
[MenuItem("Tools/UniTaskEditorRunnerChecker")]
|
|
||||||
public static void RunUniTaskAsync()
|
|
||||||
{
|
|
||||||
RunCore().Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
static async UniTaskVoid RunCore()
|
|
||||||
{
|
|
||||||
Debug.Log("Start");
|
|
||||||
|
|
||||||
//var r = await UnityWebRequest.Get("https://bing.com/").SendWebRequest().ToUniTask();
|
|
||||||
//Debug.Log(r.downloadHandler.text.Substring(0, 100));
|
|
||||||
//await UniTask.Yield();
|
|
||||||
|
|
||||||
await UniTask.DelayFrame(30);
|
|
||||||
|
|
||||||
Debug.Log("End");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -10,7 +10,7 @@
|
||||||
"allowUnsafeCode": false,
|
"allowUnsafeCode": false,
|
||||||
"overrideReferences": false,
|
"overrideReferences": false,
|
||||||
"precompiledReferences": [],
|
"precompiledReferences": [],
|
||||||
"autoReferenced": false,
|
"autoReferenced": true,
|
||||||
"defineConstraints": [],
|
"defineConstraints": [],
|
||||||
"versionDefines": [],
|
"versionDefines": [],
|
||||||
"noEngineReferences": false
|
"noEngineReferences": false
|
||||||
|
|
|
@ -7,239 +7,113 @@ namespace Cysharp.Threading.Tasks
|
||||||
{
|
{
|
||||||
public class AsyncLazy
|
public class AsyncLazy
|
||||||
{
|
{
|
||||||
static Action<object> continuation = SetCompletionSource;
|
Func<UniTask> valueFactory;
|
||||||
|
UniTask target;
|
||||||
Func<UniTask> taskFactory;
|
|
||||||
UniTaskCompletionSource completionSource;
|
|
||||||
UniTask.Awaiter awaiter;
|
|
||||||
|
|
||||||
object syncLock;
|
object syncLock;
|
||||||
bool initialized;
|
bool initialized;
|
||||||
|
|
||||||
public AsyncLazy(Func<UniTask> taskFactory)
|
public AsyncLazy(Func<UniTask> valueFactory)
|
||||||
{
|
{
|
||||||
this.taskFactory = taskFactory;
|
this.valueFactory = valueFactory;
|
||||||
this.completionSource = new UniTaskCompletionSource();
|
this.target = default;
|
||||||
this.syncLock = new object();
|
this.syncLock = new object();
|
||||||
this.initialized = false;
|
this.initialized = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal AsyncLazy(UniTask task)
|
internal AsyncLazy(UniTask value)
|
||||||
{
|
{
|
||||||
this.taskFactory = null;
|
this.valueFactory = null;
|
||||||
this.completionSource = new UniTaskCompletionSource();
|
this.target = value;
|
||||||
this.syncLock = null;
|
this.syncLock = null;
|
||||||
this.initialized = true;
|
this.initialized = true;
|
||||||
|
|
||||||
var awaiter = task.GetAwaiter();
|
|
||||||
if (awaiter.IsCompleted)
|
|
||||||
{
|
|
||||||
SetCompletionSource(awaiter);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
this.awaiter = awaiter;
|
|
||||||
awaiter.SourceOnCompleted(continuation, this);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public UniTask Task
|
public UniTask Task => EnsureInitialized();
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
EnsureInitialized();
|
|
||||||
return completionSource.Task;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
public UniTask.Awaiter GetAwaiter() => EnsureInitialized().GetAwaiter();
|
||||||
|
|
||||||
public UniTask.Awaiter GetAwaiter() => Task.GetAwaiter();
|
UniTask EnsureInitialized()
|
||||||
|
|
||||||
void EnsureInitialized()
|
|
||||||
{
|
{
|
||||||
if (Volatile.Read(ref initialized))
|
if (Volatile.Read(ref initialized))
|
||||||
{
|
{
|
||||||
return;
|
return target;
|
||||||
}
|
}
|
||||||
|
|
||||||
EnsureInitializedCore();
|
return EnsureInitializedCore();
|
||||||
}
|
}
|
||||||
|
|
||||||
void EnsureInitializedCore()
|
UniTask EnsureInitializedCore()
|
||||||
{
|
{
|
||||||
lock (syncLock)
|
lock (syncLock)
|
||||||
{
|
{
|
||||||
if (!Volatile.Read(ref initialized))
|
if (!Volatile.Read(ref initialized))
|
||||||
{
|
{
|
||||||
var f = Interlocked.Exchange(ref taskFactory, null);
|
var f = Interlocked.Exchange(ref valueFactory, null);
|
||||||
if (f != null)
|
if (f != null)
|
||||||
{
|
{
|
||||||
var task = f();
|
target = f().Preserve(); // with preserve(allow multiple await).
|
||||||
var awaiter = task.GetAwaiter();
|
|
||||||
if (awaiter.IsCompleted)
|
|
||||||
{
|
|
||||||
SetCompletionSource(awaiter);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
this.awaiter = awaiter;
|
|
||||||
awaiter.SourceOnCompleted(continuation, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
Volatile.Write(ref initialized, true);
|
Volatile.Write(ref initialized, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void SetCompletionSource(in UniTask.Awaiter awaiter)
|
return target;
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
awaiter.GetResult();
|
|
||||||
completionSource.TrySetResult();
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
completionSource.TrySetException(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void SetCompletionSource(object state)
|
|
||||||
{
|
|
||||||
var self = (AsyncLazy)state;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
self.awaiter.GetResult();
|
|
||||||
self.completionSource.TrySetResult();
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
self.completionSource.TrySetException(ex);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
self.awaiter = default;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class AsyncLazy<T>
|
public class AsyncLazy<T>
|
||||||
{
|
{
|
||||||
static Action<object> continuation = SetCompletionSource;
|
Func<UniTask<T>> valueFactory;
|
||||||
|
UniTask<T> target;
|
||||||
Func<UniTask<T>> taskFactory;
|
|
||||||
UniTaskCompletionSource<T> completionSource;
|
|
||||||
UniTask<T>.Awaiter awaiter;
|
|
||||||
|
|
||||||
object syncLock;
|
object syncLock;
|
||||||
bool initialized;
|
bool initialized;
|
||||||
|
|
||||||
public AsyncLazy(Func<UniTask<T>> taskFactory)
|
public AsyncLazy(Func<UniTask<T>> valueFactory)
|
||||||
{
|
{
|
||||||
this.taskFactory = taskFactory;
|
this.valueFactory = valueFactory;
|
||||||
this.completionSource = new UniTaskCompletionSource<T>();
|
this.target = default;
|
||||||
this.syncLock = new object();
|
this.syncLock = new object();
|
||||||
this.initialized = false;
|
this.initialized = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal AsyncLazy(UniTask<T> task)
|
internal AsyncLazy(UniTask<T> value)
|
||||||
{
|
{
|
||||||
this.taskFactory = null;
|
this.valueFactory = null;
|
||||||
this.completionSource = new UniTaskCompletionSource<T>();
|
this.target = value;
|
||||||
this.syncLock = null;
|
this.syncLock = null;
|
||||||
this.initialized = true;
|
this.initialized = true;
|
||||||
|
|
||||||
var awaiter = task.GetAwaiter();
|
|
||||||
if (awaiter.IsCompleted)
|
|
||||||
{
|
|
||||||
SetCompletionSource(awaiter);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
this.awaiter = awaiter;
|
|
||||||
awaiter.SourceOnCompleted(continuation, this);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public UniTask<T> Task
|
public UniTask<T> Task => EnsureInitialized();
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
EnsureInitialized();
|
|
||||||
return completionSource.Task;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
public UniTask<T>.Awaiter GetAwaiter() => EnsureInitialized().GetAwaiter();
|
||||||
|
|
||||||
public UniTask<T>.Awaiter GetAwaiter() => Task.GetAwaiter();
|
UniTask<T> EnsureInitialized()
|
||||||
|
|
||||||
void EnsureInitialized()
|
|
||||||
{
|
{
|
||||||
if (Volatile.Read(ref initialized))
|
if (Volatile.Read(ref initialized))
|
||||||
{
|
{
|
||||||
return;
|
return target;
|
||||||
}
|
}
|
||||||
|
|
||||||
EnsureInitializedCore();
|
return EnsureInitializedCore();
|
||||||
}
|
}
|
||||||
|
|
||||||
void EnsureInitializedCore()
|
UniTask<T> EnsureInitializedCore()
|
||||||
{
|
{
|
||||||
lock (syncLock)
|
lock (syncLock)
|
||||||
{
|
{
|
||||||
if (!Volatile.Read(ref initialized))
|
if (!Volatile.Read(ref initialized))
|
||||||
{
|
{
|
||||||
var f = Interlocked.Exchange(ref taskFactory, null);
|
var f = Interlocked.Exchange(ref valueFactory, null);
|
||||||
if (f != null)
|
if (f != null)
|
||||||
{
|
{
|
||||||
var task = f();
|
target = f().Preserve(); // with preserve(allow multiple await).
|
||||||
var awaiter = task.GetAwaiter();
|
|
||||||
if (awaiter.IsCompleted)
|
|
||||||
{
|
|
||||||
SetCompletionSource(awaiter);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
this.awaiter = awaiter;
|
|
||||||
awaiter.SourceOnCompleted(continuation, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
Volatile.Write(ref initialized, true);
|
Volatile.Write(ref initialized, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void SetCompletionSource(in UniTask<T>.Awaiter awaiter)
|
return target;
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var result = awaiter.GetResult();
|
|
||||||
completionSource.TrySetResult(result);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
completionSource.TrySetException(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void SetCompletionSource(object state)
|
|
||||||
{
|
|
||||||
var self = (AsyncLazy<T>)state;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var result = self.awaiter.GetResult();
|
|
||||||
self.completionSource.TrySetResult(result);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
self.completionSource.TrySetException(ex);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
self.awaiter = default;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,644 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public interface IReadOnlyAsyncReactiveProperty<T> : IUniTaskAsyncEnumerable<T>
|
|
||||||
{
|
|
||||||
T Value { get; }
|
|
||||||
IUniTaskAsyncEnumerable<T> WithoutCurrent();
|
|
||||||
UniTask<T> WaitAsync(CancellationToken cancellationToken = default);
|
|
||||||
}
|
|
||||||
|
|
||||||
public interface IAsyncReactiveProperty<T> : IReadOnlyAsyncReactiveProperty<T>
|
|
||||||
{
|
|
||||||
new T Value { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
[Serializable]
|
|
||||||
public class AsyncReactiveProperty<T> : IAsyncReactiveProperty<T>, IDisposable
|
|
||||||
{
|
|
||||||
TriggerEvent<T> triggerEvent;
|
|
||||||
|
|
||||||
#if UNITY_2018_3_OR_NEWER
|
|
||||||
[UnityEngine.SerializeField]
|
|
||||||
#endif
|
|
||||||
T latestValue;
|
|
||||||
|
|
||||||
public T Value
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return latestValue;
|
|
||||||
}
|
|
||||||
set
|
|
||||||
{
|
|
||||||
this.latestValue = value;
|
|
||||||
triggerEvent.SetResult(value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public AsyncReactiveProperty(T value)
|
|
||||||
{
|
|
||||||
this.latestValue = value;
|
|
||||||
this.triggerEvent = default;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerable<T> WithoutCurrent()
|
|
||||||
{
|
|
||||||
return new WithoutCurrentEnumerable(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new Enumerator(this, cancellationToken, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
triggerEvent.SetCompleted();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static implicit operator T(AsyncReactiveProperty<T> value)
|
|
||||||
{
|
|
||||||
return value.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
if (isValueType) return latestValue.ToString();
|
|
||||||
return latestValue?.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask<T> WaitAsync(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
return new UniTask<T>(WaitAsyncSource.Create(this, cancellationToken, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool isValueType;
|
|
||||||
|
|
||||||
static AsyncReactiveProperty()
|
|
||||||
{
|
|
||||||
isValueType = typeof(T).IsValueType;
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class WaitAsyncSource : IUniTaskSource<T>, ITriggerHandler<T>, ITaskPoolNode<WaitAsyncSource>
|
|
||||||
{
|
|
||||||
static Action<object> cancellationCallback = CancellationCallback;
|
|
||||||
|
|
||||||
static TaskPool<WaitAsyncSource> pool;
|
|
||||||
WaitAsyncSource nextNode;
|
|
||||||
ref WaitAsyncSource ITaskPoolNode<WaitAsyncSource>.NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static WaitAsyncSource()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(WaitAsyncSource), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
AsyncReactiveProperty<T> parent;
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
CancellationTokenRegistration cancellationTokenRegistration;
|
|
||||||
UniTaskCompletionSourceCore<T> core;
|
|
||||||
|
|
||||||
WaitAsyncSource()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskSource<T> Create(AsyncReactiveProperty<T> parent, CancellationToken cancellationToken, out short token)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
return AutoResetUniTaskCompletionSource<T>.CreateFromCanceled(cancellationToken, out token);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new WaitAsyncSource();
|
|
||||||
}
|
|
||||||
|
|
||||||
result.parent = parent;
|
|
||||||
result.cancellationToken = cancellationToken;
|
|
||||||
|
|
||||||
if (cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationCallback, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
result.parent.triggerEvent.Add(result);
|
|
||||||
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
token = result.core.Version;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryReturn()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
cancellationTokenRegistration.Dispose();
|
|
||||||
cancellationTokenRegistration = default;
|
|
||||||
parent.triggerEvent.Remove(this);
|
|
||||||
parent = null;
|
|
||||||
cancellationToken = default;
|
|
||||||
return pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CancellationCallback(object state)
|
|
||||||
{
|
|
||||||
var self = (WaitAsyncSource)state;
|
|
||||||
self.OnCanceled(self.cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
// IUniTaskSource
|
|
||||||
|
|
||||||
public T GetResult(short token)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return core.GetResult(token);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
TryReturn();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
// ITriggerHandler
|
|
||||||
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Prev { get; set; }
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Next { get; set; }
|
|
||||||
|
|
||||||
public void OnCanceled(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted()
|
|
||||||
{
|
|
||||||
// Complete as Cancel.
|
|
||||||
core.TrySetCanceled(CancellationToken.None);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnError(Exception ex)
|
|
||||||
{
|
|
||||||
core.TrySetException(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnNext(T value)
|
|
||||||
{
|
|
||||||
core.TrySetResult(value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class WithoutCurrentEnumerable : IUniTaskAsyncEnumerable<T>
|
|
||||||
{
|
|
||||||
readonly AsyncReactiveProperty<T> parent;
|
|
||||||
|
|
||||||
public WithoutCurrentEnumerable(AsyncReactiveProperty<T> parent)
|
|
||||||
{
|
|
||||||
this.parent = parent;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
return new Enumerator(parent, cancellationToken, false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<T>, ITriggerHandler<T>
|
|
||||||
{
|
|
||||||
static Action<object> cancellationCallback = CancellationCallback;
|
|
||||||
|
|
||||||
readonly AsyncReactiveProperty<T> parent;
|
|
||||||
readonly CancellationToken cancellationToken;
|
|
||||||
readonly CancellationTokenRegistration cancellationTokenRegistration;
|
|
||||||
T value;
|
|
||||||
bool isDisposed;
|
|
||||||
bool firstCall;
|
|
||||||
|
|
||||||
public Enumerator(AsyncReactiveProperty<T> parent, CancellationToken cancellationToken, bool publishCurrentValue)
|
|
||||||
{
|
|
||||||
this.parent = parent;
|
|
||||||
this.cancellationToken = cancellationToken;
|
|
||||||
this.firstCall = publishCurrentValue;
|
|
||||||
|
|
||||||
parent.triggerEvent.Add(this);
|
|
||||||
TaskTracker.TrackActiveTask(this, 3);
|
|
||||||
|
|
||||||
if (cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationCallback, this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public T Current => value;
|
|
||||||
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Prev { get; set; }
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Next { get; set; }
|
|
||||||
|
|
||||||
public UniTask<bool> MoveNextAsync()
|
|
||||||
{
|
|
||||||
// raise latest value on first call.
|
|
||||||
if (firstCall)
|
|
||||||
{
|
|
||||||
firstCall = false;
|
|
||||||
value = parent.Value;
|
|
||||||
return CompletedTasks.True;
|
|
||||||
}
|
|
||||||
|
|
||||||
completionSource.Reset();
|
|
||||||
return new UniTask<bool>(this, completionSource.Version);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask DisposeAsync()
|
|
||||||
{
|
|
||||||
if (!isDisposed)
|
|
||||||
{
|
|
||||||
isDisposed = true;
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
completionSource.TrySetCanceled(cancellationToken);
|
|
||||||
parent.triggerEvent.Remove(this);
|
|
||||||
}
|
|
||||||
return default;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnNext(T value)
|
|
||||||
{
|
|
||||||
this.value = value;
|
|
||||||
completionSource.TrySetResult(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCanceled(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
DisposeAsync().Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted()
|
|
||||||
{
|
|
||||||
completionSource.TrySetResult(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnError(Exception ex)
|
|
||||||
{
|
|
||||||
completionSource.TrySetException(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CancellationCallback(object state)
|
|
||||||
{
|
|
||||||
var self = (Enumerator)state;
|
|
||||||
self.DisposeAsync().Forget();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class ReadOnlyAsyncReactiveProperty<T> : IReadOnlyAsyncReactiveProperty<T>, IDisposable
|
|
||||||
{
|
|
||||||
TriggerEvent<T> triggerEvent;
|
|
||||||
|
|
||||||
T latestValue;
|
|
||||||
IUniTaskAsyncEnumerator<T> enumerator;
|
|
||||||
|
|
||||||
public T Value
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return latestValue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public ReadOnlyAsyncReactiveProperty(T initialValue, IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
latestValue = initialValue;
|
|
||||||
ConsumeEnumerator(source, cancellationToken).Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
public ReadOnlyAsyncReactiveProperty(IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
ConsumeEnumerator(source, cancellationToken).Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
async UniTaskVoid ConsumeEnumerator(IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
enumerator = source.GetAsyncEnumerator(cancellationToken);
|
|
||||||
try
|
|
||||||
{
|
|
||||||
while (await enumerator.MoveNextAsync())
|
|
||||||
{
|
|
||||||
var value = enumerator.Current;
|
|
||||||
this.latestValue = value;
|
|
||||||
triggerEvent.SetResult(value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
await enumerator.DisposeAsync();
|
|
||||||
enumerator = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerable<T> WithoutCurrent()
|
|
||||||
{
|
|
||||||
return new WithoutCurrentEnumerable(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new Enumerator(this, cancellationToken, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
if (enumerator != null)
|
|
||||||
{
|
|
||||||
enumerator.DisposeAsync().Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
triggerEvent.SetCompleted();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static implicit operator T(ReadOnlyAsyncReactiveProperty<T> value)
|
|
||||||
{
|
|
||||||
return value.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
if (isValueType) return latestValue.ToString();
|
|
||||||
return latestValue?.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask<T> WaitAsync(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
return new UniTask<T>(WaitAsyncSource.Create(this, cancellationToken, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool isValueType;
|
|
||||||
|
|
||||||
static ReadOnlyAsyncReactiveProperty()
|
|
||||||
{
|
|
||||||
isValueType = typeof(T).IsValueType;
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class WaitAsyncSource : IUniTaskSource<T>, ITriggerHandler<T>, ITaskPoolNode<WaitAsyncSource>
|
|
||||||
{
|
|
||||||
static Action<object> cancellationCallback = CancellationCallback;
|
|
||||||
|
|
||||||
static TaskPool<WaitAsyncSource> pool;
|
|
||||||
WaitAsyncSource nextNode;
|
|
||||||
ref WaitAsyncSource ITaskPoolNode<WaitAsyncSource>.NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static WaitAsyncSource()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(WaitAsyncSource), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
ReadOnlyAsyncReactiveProperty<T> parent;
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
CancellationTokenRegistration cancellationTokenRegistration;
|
|
||||||
UniTaskCompletionSourceCore<T> core;
|
|
||||||
|
|
||||||
WaitAsyncSource()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskSource<T> Create(ReadOnlyAsyncReactiveProperty<T> parent, CancellationToken cancellationToken, out short token)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
return AutoResetUniTaskCompletionSource<T>.CreateFromCanceled(cancellationToken, out token);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new WaitAsyncSource();
|
|
||||||
}
|
|
||||||
|
|
||||||
result.parent = parent;
|
|
||||||
result.cancellationToken = cancellationToken;
|
|
||||||
|
|
||||||
if (cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationCallback, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
result.parent.triggerEvent.Add(result);
|
|
||||||
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
token = result.core.Version;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryReturn()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
cancellationTokenRegistration.Dispose();
|
|
||||||
cancellationTokenRegistration = default;
|
|
||||||
parent.triggerEvent.Remove(this);
|
|
||||||
parent = null;
|
|
||||||
cancellationToken = default;
|
|
||||||
return pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CancellationCallback(object state)
|
|
||||||
{
|
|
||||||
var self = (WaitAsyncSource)state;
|
|
||||||
self.OnCanceled(self.cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
// IUniTaskSource
|
|
||||||
|
|
||||||
public T GetResult(short token)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return core.GetResult(token);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
TryReturn();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
// ITriggerHandler
|
|
||||||
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Prev { get; set; }
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Next { get; set; }
|
|
||||||
|
|
||||||
public void OnCanceled(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted()
|
|
||||||
{
|
|
||||||
// Complete as Cancel.
|
|
||||||
core.TrySetCanceled(CancellationToken.None);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnError(Exception ex)
|
|
||||||
{
|
|
||||||
core.TrySetException(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnNext(T value)
|
|
||||||
{
|
|
||||||
core.TrySetResult(value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class WithoutCurrentEnumerable : IUniTaskAsyncEnumerable<T>
|
|
||||||
{
|
|
||||||
readonly ReadOnlyAsyncReactiveProperty<T> parent;
|
|
||||||
|
|
||||||
public WithoutCurrentEnumerable(ReadOnlyAsyncReactiveProperty<T> parent)
|
|
||||||
{
|
|
||||||
this.parent = parent;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
return new Enumerator(parent, cancellationToken, false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<T>, ITriggerHandler<T>
|
|
||||||
{
|
|
||||||
static Action<object> cancellationCallback = CancellationCallback;
|
|
||||||
|
|
||||||
readonly ReadOnlyAsyncReactiveProperty<T> parent;
|
|
||||||
readonly CancellationToken cancellationToken;
|
|
||||||
readonly CancellationTokenRegistration cancellationTokenRegistration;
|
|
||||||
T value;
|
|
||||||
bool isDisposed;
|
|
||||||
bool firstCall;
|
|
||||||
|
|
||||||
public Enumerator(ReadOnlyAsyncReactiveProperty<T> parent, CancellationToken cancellationToken, bool publishCurrentValue)
|
|
||||||
{
|
|
||||||
this.parent = parent;
|
|
||||||
this.cancellationToken = cancellationToken;
|
|
||||||
this.firstCall = publishCurrentValue;
|
|
||||||
|
|
||||||
parent.triggerEvent.Add(this);
|
|
||||||
TaskTracker.TrackActiveTask(this, 3);
|
|
||||||
|
|
||||||
if (cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationCallback, this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public T Current => value;
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Prev { get; set; }
|
|
||||||
ITriggerHandler<T> ITriggerHandler<T>.Next { get; set; }
|
|
||||||
|
|
||||||
public UniTask<bool> MoveNextAsync()
|
|
||||||
{
|
|
||||||
// raise latest value on first call.
|
|
||||||
if (firstCall)
|
|
||||||
{
|
|
||||||
firstCall = false;
|
|
||||||
value = parent.Value;
|
|
||||||
return CompletedTasks.True;
|
|
||||||
}
|
|
||||||
|
|
||||||
completionSource.Reset();
|
|
||||||
return new UniTask<bool>(this, completionSource.Version);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask DisposeAsync()
|
|
||||||
{
|
|
||||||
if (!isDisposed)
|
|
||||||
{
|
|
||||||
isDisposed = true;
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
completionSource.TrySetCanceled(cancellationToken);
|
|
||||||
parent.triggerEvent.Remove(this);
|
|
||||||
}
|
|
||||||
return default;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnNext(T value)
|
|
||||||
{
|
|
||||||
this.value = value;
|
|
||||||
completionSource.TrySetResult(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCanceled(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
DisposeAsync().Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted()
|
|
||||||
{
|
|
||||||
completionSource.TrySetResult(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnError(Exception ex)
|
|
||||||
{
|
|
||||||
completionSource.TrySetException(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CancellationCallback(object state)
|
|
||||||
{
|
|
||||||
var self = (Enumerator)state;
|
|
||||||
self.DisposeAsync().Forget();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class StateExtensions
|
|
||||||
{
|
|
||||||
public static ReadOnlyAsyncReactiveProperty<T> ToReadOnlyAsyncReactiveProperty<T>(this IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new ReadOnlyAsyncReactiveProperty<T>(source, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static ReadOnlyAsyncReactiveProperty<T> ToReadOnlyAsyncReactiveProperty<T>(this IUniTaskAsyncEnumerable<T> source, T initialValue, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new ReadOnlyAsyncReactiveProperty<T>(initialValue, source, cancellationToken);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -4,7 +4,7 @@ using System;
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
namespace Cysharp.Threading.Tasks
|
||||||
{
|
{
|
||||||
public readonly struct AsyncUnit : IEquatable<AsyncUnit>
|
public struct AsyncUnit : IEquatable<AsyncUnit>
|
||||||
{
|
{
|
||||||
public static readonly AsyncUnit Default = new AsyncUnit();
|
public static readonly AsyncUnit Default = new AsyncUnit();
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
namespace Cysharp.Threading.Tasks
|
||||||
|
@ -9,66 +8,16 @@ namespace Cysharp.Threading.Tasks
|
||||||
public static class CancellationTokenExtensions
|
public static class CancellationTokenExtensions
|
||||||
{
|
{
|
||||||
static readonly Action<object> cancellationTokenCallback = Callback;
|
static readonly Action<object> cancellationTokenCallback = Callback;
|
||||||
static readonly Action<object> disposeCallback = DisposeCallback;
|
|
||||||
|
|
||||||
public static CancellationToken ToCancellationToken(this UniTask task)
|
public static (UniTask, CancellationTokenRegistration) ToUniTask(this CancellationToken cts)
|
||||||
{
|
{
|
||||||
var cts = new CancellationTokenSource();
|
if (cts.IsCancellationRequested)
|
||||||
ToCancellationTokenCore(task, cts).Forget();
|
|
||||||
return cts.Token;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static CancellationToken ToCancellationToken(this UniTask task, CancellationToken linkToken)
|
|
||||||
{
|
|
||||||
if (linkToken.IsCancellationRequested)
|
|
||||||
{
|
{
|
||||||
return linkToken;
|
return (UniTask.FromCanceled(cts), default(CancellationTokenRegistration));
|
||||||
}
|
|
||||||
|
|
||||||
if (!linkToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
return ToCancellationToken(task);
|
|
||||||
}
|
|
||||||
|
|
||||||
var cts = CancellationTokenSource.CreateLinkedTokenSource(linkToken);
|
|
||||||
ToCancellationTokenCore(task, cts).Forget();
|
|
||||||
|
|
||||||
return cts.Token;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static CancellationToken ToCancellationToken<T>(this UniTask<T> task)
|
|
||||||
{
|
|
||||||
return ToCancellationToken(task.AsUniTask());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static CancellationToken ToCancellationToken<T>(this UniTask<T> task, CancellationToken linkToken)
|
|
||||||
{
|
|
||||||
return ToCancellationToken(task.AsUniTask(), linkToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
static async UniTaskVoid ToCancellationTokenCore(UniTask task, CancellationTokenSource cts)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
await task;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
UniTaskScheduler.PublishUnobservedTaskException(ex);
|
|
||||||
}
|
|
||||||
cts.Cancel();
|
|
||||||
cts.Dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static (UniTask, CancellationTokenRegistration) ToUniTask(this CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
return (UniTask.FromCanceled(cancellationToken), default(CancellationTokenRegistration));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var promise = new UniTaskCompletionSource();
|
var promise = new UniTaskCompletionSource();
|
||||||
return (promise.Task, cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationTokenCallback, promise));
|
return (promise.Task, cts.RegisterWithoutCaptureExecutionContext(cancellationTokenCallback, promise));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void Callback(object state)
|
static void Callback(object state)
|
||||||
|
@ -77,11 +26,6 @@ namespace Cysharp.Threading.Tasks
|
||||||
promise.TrySetResult();
|
promise.TrySetResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static CancellationTokenAwaitable WaitUntilCanceled(this CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new CancellationTokenAwaitable(cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static CancellationTokenRegistration RegisterWithoutCaptureExecutionContext(this CancellationToken cancellationToken, Action callback)
|
public static CancellationTokenRegistration RegisterWithoutCaptureExecutionContext(this CancellationToken cancellationToken, Action callback)
|
||||||
{
|
{
|
||||||
var restoreFlow = false;
|
var restoreFlow = false;
|
||||||
|
@ -125,58 +69,6 @@ namespace Cysharp.Threading.Tasks
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static CancellationTokenRegistration AddTo(this IDisposable disposable, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return cancellationToken.RegisterWithoutCaptureExecutionContext(disposeCallback, disposable);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void DisposeCallback(object state)
|
|
||||||
{
|
|
||||||
var d = (IDisposable)state;
|
|
||||||
d.Dispose();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct CancellationTokenAwaitable
|
|
||||||
{
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
|
|
||||||
public CancellationTokenAwaitable(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
this.cancellationToken = cancellationToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Awaiter GetAwaiter()
|
|
||||||
{
|
|
||||||
return new Awaiter(cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct Awaiter : ICriticalNotifyCompletion
|
|
||||||
{
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
|
|
||||||
public Awaiter(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
this.cancellationToken = cancellationToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsCompleted => !cancellationToken.CanBeCanceled || cancellationToken.IsCancellationRequested;
|
|
||||||
|
|
||||||
public void GetResult()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
UnsafeOnCompleted(continuation);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UnsafeOnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
cancellationToken.RegisterWithoutCaptureExecutionContext(continuation);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4,29 +4,31 @@ using System.Threading;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
using Cysharp.Threading.Tasks.Triggers;
|
using Cysharp.Threading.Tasks.Triggers;
|
||||||
using System;
|
using System;
|
||||||
using Cysharp.Threading.Tasks.Internal;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
namespace Cysharp.Threading.Tasks
|
||||||
{
|
{
|
||||||
|
public static class CancellationTokenSourceExtensions
|
||||||
public static partial class CancellationTokenSourceExtensions
|
|
||||||
{
|
{
|
||||||
readonly static Action<object> CancelCancellationTokenSourceStateDelegate = new Action<object>(CancelCancellationTokenSourceState);
|
public static void CancelAfterSlim(this CancellationTokenSource cts, int millisecondsDelay, bool ignoreTimeScale = false, PlayerLoopTiming delayTiming = PlayerLoopTiming.Update)
|
||||||
|
|
||||||
static void CancelCancellationTokenSourceState(object state)
|
|
||||||
{
|
{
|
||||||
var cts = (CancellationTokenSource)state;
|
var delay = UniTask.Delay(millisecondsDelay, ignoreTimeScale, delayTiming, cts.Token);
|
||||||
cts.Cancel();
|
CancelAfterCore(cts, delay).Forget();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static IDisposable CancelAfterSlim(this CancellationTokenSource cts, int millisecondsDelay, DelayType delayType = DelayType.DeltaTime, PlayerLoopTiming delayTiming = PlayerLoopTiming.Update)
|
public static void CancelAfterSlim(this CancellationTokenSource cts, TimeSpan delayTimeSpan, bool ignoreTimeScale = false, PlayerLoopTiming delayTiming = PlayerLoopTiming.Update)
|
||||||
{
|
{
|
||||||
return CancelAfterSlim(cts, TimeSpan.FromMilliseconds(millisecondsDelay), delayType, delayTiming);
|
var delay = UniTask.Delay(delayTimeSpan, ignoreTimeScale, delayTiming, cts.Token);
|
||||||
|
CancelAfterCore(cts, delay).Forget();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static IDisposable CancelAfterSlim(this CancellationTokenSource cts, TimeSpan delayTimeSpan, DelayType delayType = DelayType.DeltaTime, PlayerLoopTiming delayTiming = PlayerLoopTiming.Update)
|
static async UniTaskVoid CancelAfterCore(CancellationTokenSource cts, UniTask delayTask)
|
||||||
{
|
{
|
||||||
return PlayerLoopTimer.StartNew(delayTimeSpan, false, delayType, delayTiming, cts.Token, CancelCancellationTokenSourceStateDelegate, cts);
|
var alreadyCanceled = await delayTask.SuppressCancellationThrow();
|
||||||
|
if (!alreadyCanceled)
|
||||||
|
{
|
||||||
|
cts.Cancel();
|
||||||
|
cts.Dispose();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void RegisterRaiseCancelOnDestroy(this CancellationTokenSource cts, Component component)
|
public static void RegisterRaiseCancelOnDestroy(this CancellationTokenSource cts, Component component)
|
||||||
|
@ -37,7 +39,11 @@ namespace Cysharp.Threading.Tasks
|
||||||
public static void RegisterRaiseCancelOnDestroy(this CancellationTokenSource cts, GameObject gameObject)
|
public static void RegisterRaiseCancelOnDestroy(this CancellationTokenSource cts, GameObject gameObject)
|
||||||
{
|
{
|
||||||
var trigger = gameObject.GetAsyncDestroyTrigger();
|
var trigger = gameObject.GetAsyncDestroyTrigger();
|
||||||
trigger.CancellationToken.RegisterWithoutCaptureExecutionContext(CancelCancellationTokenSourceStateDelegate, cts);
|
trigger.CancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
||||||
|
{
|
||||||
|
var cts2 = (CancellationTokenSource)state;
|
||||||
|
cts2.Cancel();
|
||||||
|
}, cts);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,450 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public static class Channel
|
|
||||||
{
|
|
||||||
public static Channel<T> CreateSingleConsumerUnbounded<T>()
|
|
||||||
{
|
|
||||||
return new SingleConsumerUnboundedChannel<T>();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract class Channel<TWrite, TRead>
|
|
||||||
{
|
|
||||||
public ChannelReader<TRead> Reader { get; protected set; }
|
|
||||||
public ChannelWriter<TWrite> Writer { get; protected set; }
|
|
||||||
|
|
||||||
public static implicit operator ChannelReader<TRead>(Channel<TWrite, TRead> channel) => channel.Reader;
|
|
||||||
public static implicit operator ChannelWriter<TWrite>(Channel<TWrite, TRead> channel) => channel.Writer;
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract class Channel<T> : Channel<T, T>
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract class ChannelReader<T>
|
|
||||||
{
|
|
||||||
public abstract bool TryRead(out T item);
|
|
||||||
public abstract UniTask<bool> WaitToReadAsync(CancellationToken cancellationToken = default(CancellationToken));
|
|
||||||
|
|
||||||
public abstract UniTask Completion { get; }
|
|
||||||
|
|
||||||
public virtual UniTask<T> ReadAsync(CancellationToken cancellationToken = default(CancellationToken))
|
|
||||||
{
|
|
||||||
if (this.TryRead(out var item))
|
|
||||||
{
|
|
||||||
return UniTask.FromResult(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ReadAsyncCore(cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
async UniTask<T> ReadAsyncCore(CancellationToken cancellationToken = default(CancellationToken))
|
|
||||||
{
|
|
||||||
if (await WaitToReadAsync(cancellationToken))
|
|
||||||
{
|
|
||||||
if (TryRead(out var item))
|
|
||||||
{
|
|
||||||
return item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new ChannelClosedException();
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract IUniTaskAsyncEnumerable<T> ReadAllAsync(CancellationToken cancellationToken = default(CancellationToken));
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract class ChannelWriter<T>
|
|
||||||
{
|
|
||||||
public abstract bool TryWrite(T item);
|
|
||||||
public abstract bool TryComplete(Exception error = null);
|
|
||||||
|
|
||||||
public void Complete(Exception error = null)
|
|
||||||
{
|
|
||||||
if (!TryComplete(error))
|
|
||||||
{
|
|
||||||
throw new ChannelClosedException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public partial class ChannelClosedException : InvalidOperationException
|
|
||||||
{
|
|
||||||
public ChannelClosedException() :
|
|
||||||
base("Channel is already closed.")
|
|
||||||
{ }
|
|
||||||
|
|
||||||
public ChannelClosedException(string message) : base(message) { }
|
|
||||||
|
|
||||||
public ChannelClosedException(Exception innerException) :
|
|
||||||
base("Channel is already closed", innerException)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
public ChannelClosedException(string message, Exception innerException) : base(message, innerException) { }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal class SingleConsumerUnboundedChannel<T> : Channel<T>
|
|
||||||
{
|
|
||||||
readonly Queue<T> items;
|
|
||||||
readonly SingleConsumerUnboundedChannelReader readerSource;
|
|
||||||
UniTaskCompletionSource completedTaskSource;
|
|
||||||
UniTask completedTask;
|
|
||||||
|
|
||||||
Exception completionError;
|
|
||||||
bool closed;
|
|
||||||
|
|
||||||
public SingleConsumerUnboundedChannel()
|
|
||||||
{
|
|
||||||
items = new Queue<T>();
|
|
||||||
Writer = new SingleConsumerUnboundedChannelWriter(this);
|
|
||||||
readerSource = new SingleConsumerUnboundedChannelReader(this);
|
|
||||||
Reader = readerSource;
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class SingleConsumerUnboundedChannelWriter : ChannelWriter<T>
|
|
||||||
{
|
|
||||||
readonly SingleConsumerUnboundedChannel<T> parent;
|
|
||||||
|
|
||||||
public SingleConsumerUnboundedChannelWriter(SingleConsumerUnboundedChannel<T> parent)
|
|
||||||
{
|
|
||||||
this.parent = parent;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool TryWrite(T item)
|
|
||||||
{
|
|
||||||
bool waiting;
|
|
||||||
lock (parent.items)
|
|
||||||
{
|
|
||||||
if (parent.closed) return false;
|
|
||||||
|
|
||||||
parent.items.Enqueue(item);
|
|
||||||
waiting = parent.readerSource.isWaiting;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (waiting)
|
|
||||||
{
|
|
||||||
parent.readerSource.SingalContinuation();
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool TryComplete(Exception error = null)
|
|
||||||
{
|
|
||||||
bool waiting;
|
|
||||||
lock (parent.items)
|
|
||||||
{
|
|
||||||
if (parent.closed) return false;
|
|
||||||
parent.closed = true;
|
|
||||||
waiting = parent.readerSource.isWaiting;
|
|
||||||
|
|
||||||
if (parent.items.Count == 0)
|
|
||||||
{
|
|
||||||
if (error == null)
|
|
||||||
{
|
|
||||||
if (parent.completedTaskSource != null)
|
|
||||||
{
|
|
||||||
parent.completedTaskSource.TrySetResult();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
parent.completedTask = UniTask.CompletedTask;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (parent.completedTaskSource != null)
|
|
||||||
{
|
|
||||||
parent.completedTaskSource.TrySetException(error);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
parent.completedTask = UniTask.FromException(error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (waiting)
|
|
||||||
{
|
|
||||||
parent.readerSource.SingalCompleted(error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
parent.completionError = error;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class SingleConsumerUnboundedChannelReader : ChannelReader<T>, IUniTaskSource<bool>
|
|
||||||
{
|
|
||||||
readonly Action<object> CancellationCallbackDelegate = CancellationCallback;
|
|
||||||
readonly SingleConsumerUnboundedChannel<T> parent;
|
|
||||||
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
CancellationTokenRegistration cancellationTokenRegistration;
|
|
||||||
UniTaskCompletionSourceCore<bool> core;
|
|
||||||
internal bool isWaiting;
|
|
||||||
|
|
||||||
public SingleConsumerUnboundedChannelReader(SingleConsumerUnboundedChannel<T> parent)
|
|
||||||
{
|
|
||||||
this.parent = parent;
|
|
||||||
|
|
||||||
TaskTracker.TrackActiveTask(this, 4);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override UniTask Completion
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
if (parent.completedTaskSource != null) return parent.completedTaskSource.Task;
|
|
||||||
|
|
||||||
if (parent.closed)
|
|
||||||
{
|
|
||||||
return parent.completedTask;
|
|
||||||
}
|
|
||||||
|
|
||||||
parent.completedTaskSource = new UniTaskCompletionSource();
|
|
||||||
return parent.completedTaskSource.Task;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool TryRead(out T item)
|
|
||||||
{
|
|
||||||
lock (parent.items)
|
|
||||||
{
|
|
||||||
if (parent.items.Count != 0)
|
|
||||||
{
|
|
||||||
item = parent.items.Dequeue();
|
|
||||||
|
|
||||||
// complete when all value was consumed.
|
|
||||||
if (parent.closed && parent.items.Count == 0)
|
|
||||||
{
|
|
||||||
if (parent.completionError != null)
|
|
||||||
{
|
|
||||||
if (parent.completedTaskSource != null)
|
|
||||||
{
|
|
||||||
parent.completedTaskSource.TrySetException(parent.completionError);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
parent.completedTask = UniTask.FromException(parent.completionError);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (parent.completedTaskSource != null)
|
|
||||||
{
|
|
||||||
parent.completedTaskSource.TrySetResult();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
parent.completedTask = UniTask.CompletedTask;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
item = default;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override UniTask<bool> WaitToReadAsync(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
return UniTask.FromCanceled<bool>(cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
lock (parent.items)
|
|
||||||
{
|
|
||||||
if (parent.items.Count != 0)
|
|
||||||
{
|
|
||||||
return CompletedTasks.True;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (parent.closed)
|
|
||||||
{
|
|
||||||
if (parent.completionError == null)
|
|
||||||
{
|
|
||||||
return CompletedTasks.False;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return UniTask.FromException<bool>(parent.completionError);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
cancellationTokenRegistration.Dispose();
|
|
||||||
|
|
||||||
core.Reset();
|
|
||||||
isWaiting = true;
|
|
||||||
|
|
||||||
this.cancellationToken = cancellationToken;
|
|
||||||
if (this.cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
cancellationTokenRegistration = this.cancellationToken.RegisterWithoutCaptureExecutionContext(CancellationCallbackDelegate, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new UniTask<bool>(this, core.Version);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SingalContinuation()
|
|
||||||
{
|
|
||||||
core.TrySetResult(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SingalCancellation(CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SingalCompleted(Exception error)
|
|
||||||
{
|
|
||||||
if (error != null)
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.TrySetException(error);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.TrySetResult(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public override IUniTaskAsyncEnumerable<T> ReadAllAsync(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
return new ReadAllAsyncEnumerable(this, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IUniTaskSource<bool>.GetResult(short token)
|
|
||||||
{
|
|
||||||
return core.GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
core.GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
UniTaskStatus IUniTaskSource.GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
UniTaskStatus IUniTaskSource.UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CancellationCallback(object state)
|
|
||||||
{
|
|
||||||
var self = (SingleConsumerUnboundedChannelReader)state;
|
|
||||||
self.SingalCancellation(self.cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class ReadAllAsyncEnumerable : IUniTaskAsyncEnumerable<T>, IUniTaskAsyncEnumerator<T>
|
|
||||||
{
|
|
||||||
readonly Action<object> CancellationCallback1Delegate = CancellationCallback1;
|
|
||||||
readonly Action<object> CancellationCallback2Delegate = CancellationCallback2;
|
|
||||||
|
|
||||||
readonly SingleConsumerUnboundedChannelReader parent;
|
|
||||||
CancellationToken cancellationToken1;
|
|
||||||
CancellationToken cancellationToken2;
|
|
||||||
CancellationTokenRegistration cancellationTokenRegistration1;
|
|
||||||
CancellationTokenRegistration cancellationTokenRegistration2;
|
|
||||||
|
|
||||||
T current;
|
|
||||||
bool cacheValue;
|
|
||||||
bool running;
|
|
||||||
|
|
||||||
public ReadAllAsyncEnumerable(SingleConsumerUnboundedChannelReader parent, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
this.parent = parent;
|
|
||||||
this.cancellationToken1 = cancellationToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
if (running)
|
|
||||||
{
|
|
||||||
throw new InvalidOperationException("Enumerator is already running, does not allow call GetAsyncEnumerator twice.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.cancellationToken1 != cancellationToken)
|
|
||||||
{
|
|
||||||
this.cancellationToken2 = cancellationToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.cancellationToken1.CanBeCanceled)
|
|
||||||
{
|
|
||||||
this.cancellationTokenRegistration1 = this.cancellationToken1.RegisterWithoutCaptureExecutionContext(CancellationCallback1Delegate, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.cancellationToken2.CanBeCanceled)
|
|
||||||
{
|
|
||||||
this.cancellationTokenRegistration2 = this.cancellationToken2.RegisterWithoutCaptureExecutionContext(CancellationCallback2Delegate, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
running = true;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public T Current
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
if (cacheValue)
|
|
||||||
{
|
|
||||||
return current;
|
|
||||||
}
|
|
||||||
parent.TryRead(out current);
|
|
||||||
return current;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask<bool> MoveNextAsync()
|
|
||||||
{
|
|
||||||
cacheValue = false;
|
|
||||||
return parent.WaitToReadAsync(CancellationToken.None); // ok to use None, registered in ctor.
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask DisposeAsync()
|
|
||||||
{
|
|
||||||
cancellationTokenRegistration1.Dispose();
|
|
||||||
cancellationTokenRegistration2.Dispose();
|
|
||||||
return default;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CancellationCallback1(object state)
|
|
||||||
{
|
|
||||||
var self = (ReadAllAsyncEnumerable)state;
|
|
||||||
self.parent.SingalCancellation(self.cancellationToken1);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CancellationCallback2(object state)
|
|
||||||
{
|
|
||||||
var self = (ReadAllAsyncEnumerable)state;
|
|
||||||
self.parent.SingalCancellation(self.cancellationToken2);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -12,8 +12,9 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
[StructLayout(LayoutKind.Auto)]
|
[StructLayout(LayoutKind.Auto)]
|
||||||
public struct AsyncUniTaskMethodBuilder
|
public struct AsyncUniTaskMethodBuilder
|
||||||
{
|
{
|
||||||
IStateMachineRunnerPromise runnerPromise;
|
// cache items.
|
||||||
Exception ex;
|
AutoResetUniTaskCompletionSource promise;
|
||||||
|
IMoveNextRunner runner;
|
||||||
|
|
||||||
// 1. Static Create method.
|
// 1. Static Create method.
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
@ -30,81 +31,98 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
get
|
get
|
||||||
{
|
{
|
||||||
if (runnerPromise != null)
|
if (promise != null)
|
||||||
{
|
{
|
||||||
return runnerPromise.Task;
|
return promise.Task;
|
||||||
}
|
}
|
||||||
else if (ex != null)
|
|
||||||
{
|
if (runner == null)
|
||||||
return UniTask.FromException(ex);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
return UniTask.CompletedTask;
|
return UniTask.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
promise = AutoResetUniTaskCompletionSource.Create();
|
||||||
|
return promise.Task;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. SetException
|
// 3. SetException
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void SetException(Exception exception)
|
public void SetException(Exception exception)
|
||||||
{
|
{
|
||||||
if (runnerPromise == null)
|
// runner is finished, return first.
|
||||||
|
if (runner != null)
|
||||||
{
|
{
|
||||||
ex = exception;
|
runner.Return();
|
||||||
|
runner = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (promise != null)
|
||||||
|
{
|
||||||
|
promise.TrySetException(exception);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
runnerPromise.SetException(exception);
|
promise = AutoResetUniTaskCompletionSource.CreateFromException(exception, out _);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. SetResult
|
// 4. SetResult
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void SetResult()
|
public void SetResult()
|
||||||
{
|
{
|
||||||
if (runnerPromise != null)
|
// runner is finished, return first.
|
||||||
|
if (runner != null)
|
||||||
{
|
{
|
||||||
runnerPromise.SetResult();
|
runner.Return();
|
||||||
|
runner = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (promise != null)
|
||||||
|
{
|
||||||
|
promise.TrySetResult();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. AwaitOnCompleted
|
// 5. AwaitOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : INotifyCompletion
|
where TAwaiter : INotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runnerPromise == null)
|
if (promise == null)
|
||||||
{
|
{
|
||||||
AsyncUniTask<TStateMachine>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
promise = AutoResetUniTaskCompletionSource.Create();
|
||||||
|
}
|
||||||
|
if (runner == null)
|
||||||
|
{
|
||||||
|
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runnerPromise.MoveNext);
|
awaiter.OnCompleted(runner.CallMoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. AwaitUnsafeOnCompleted
|
// 6. AwaitUnsafeOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[SecuritySafeCritical]
|
[SecuritySafeCritical]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : ICriticalNotifyCompletion
|
where TAwaiter : ICriticalNotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runnerPromise == null)
|
if (promise == null)
|
||||||
{
|
{
|
||||||
AsyncUniTask<TStateMachine>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
promise = AutoResetUniTaskCompletionSource.Create();
|
||||||
|
}
|
||||||
|
if (runner == null)
|
||||||
|
{
|
||||||
|
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.UnsafeOnCompleted(runnerPromise.MoveNext);
|
awaiter.OnCompleted(runner.CallMoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7. Start
|
// 7. Start
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
|
@ -138,8 +156,9 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
[StructLayout(LayoutKind.Auto)]
|
[StructLayout(LayoutKind.Auto)]
|
||||||
public struct AsyncUniTaskMethodBuilder<T>
|
public struct AsyncUniTaskMethodBuilder<T>
|
||||||
{
|
{
|
||||||
IStateMachineRunnerPromise<T> runnerPromise;
|
// cache items.
|
||||||
Exception ex;
|
AutoResetUniTaskCompletionSource<T> promise;
|
||||||
|
IMoveNextRunner runner;
|
||||||
T result;
|
T result;
|
||||||
|
|
||||||
// 1. Static Create method.
|
// 1. Static Create method.
|
||||||
|
@ -151,91 +170,106 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. TaskLike Task property.
|
// 2. TaskLike Task property.
|
||||||
|
[DebuggerHidden]
|
||||||
public UniTask<T> Task
|
public UniTask<T> Task
|
||||||
{
|
{
|
||||||
[DebuggerHidden]
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
get
|
get
|
||||||
{
|
{
|
||||||
if (runnerPromise != null)
|
if (promise != null)
|
||||||
{
|
{
|
||||||
return runnerPromise.Task;
|
return promise.Task;
|
||||||
}
|
}
|
||||||
else if (ex != null)
|
|
||||||
{
|
if (runner == null)
|
||||||
return UniTask.FromException<T>(ex);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
return UniTask.FromResult(result);
|
return UniTask.FromResult(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
promise = AutoResetUniTaskCompletionSource<T>.Create();
|
||||||
|
return promise.Task;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. SetException
|
// 3. SetException
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void SetException(Exception exception)
|
public void SetException(Exception exception)
|
||||||
{
|
{
|
||||||
if (runnerPromise == null)
|
// runner is finished, return first.
|
||||||
|
if (runner != null)
|
||||||
{
|
{
|
||||||
ex = exception;
|
runner.Return();
|
||||||
|
runner = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (promise == null)
|
||||||
|
{
|
||||||
|
promise = AutoResetUniTaskCompletionSource<T>.CreateFromException(exception, out _);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
runnerPromise.SetException(exception);
|
promise.TrySetException(exception);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. SetResult
|
// 4. SetResult
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void SetResult(T result)
|
public void SetResult(T result)
|
||||||
{
|
{
|
||||||
if (runnerPromise == null)
|
// runner is finished, return first.
|
||||||
|
if (runner != null)
|
||||||
|
{
|
||||||
|
runner.Return();
|
||||||
|
runner = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (promise == null)
|
||||||
{
|
{
|
||||||
this.result = result;
|
this.result = result;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
promise.TrySetResult(result);
|
||||||
runnerPromise.SetResult(result);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. AwaitOnCompleted
|
// 5. AwaitOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : INotifyCompletion
|
where TAwaiter : INotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runnerPromise == null)
|
if (promise == null)
|
||||||
{
|
{
|
||||||
AsyncUniTask<TStateMachine, T>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
promise = AutoResetUniTaskCompletionSource<T>.Create();
|
||||||
|
}
|
||||||
|
if (runner == null)
|
||||||
|
{
|
||||||
|
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runnerPromise.MoveNext);
|
awaiter.OnCompleted(runner.CallMoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. AwaitUnsafeOnCompleted
|
// 6. AwaitUnsafeOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[SecuritySafeCritical]
|
[SecuritySafeCritical]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : ICriticalNotifyCompletion
|
where TAwaiter : ICriticalNotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runnerPromise == null)
|
if (promise == null)
|
||||||
{
|
{
|
||||||
AsyncUniTask<TStateMachine, T>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
promise = AutoResetUniTaskCompletionSource<T>.Create();
|
||||||
|
}
|
||||||
|
if (runner == null)
|
||||||
|
{
|
||||||
|
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.UnsafeOnCompleted(runnerPromise.MoveNext);
|
awaiter.OnCompleted(runner.CallMoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7. Start
|
// 7. Start
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
|
|
|
@ -4,15 +4,13 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
using System.Security;
|
using System.Security;
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.CompilerServices
|
namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
{
|
{
|
||||||
[StructLayout(LayoutKind.Auto)]
|
|
||||||
public struct AsyncUniTaskVoidMethodBuilder
|
public struct AsyncUniTaskVoidMethodBuilder
|
||||||
{
|
{
|
||||||
IStateMachineRunner runner;
|
IMoveNextRunner runner;
|
||||||
|
|
||||||
// 1. Static Create method.
|
// 1. Static Create method.
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
@ -35,18 +33,12 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
|
|
||||||
// 3. SetException
|
// 3. SetException
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void SetException(Exception exception)
|
public void SetException(Exception exception)
|
||||||
{
|
{
|
||||||
// runner is finished, return first.
|
// runner is finished, return first.
|
||||||
if (runner != null)
|
if (runner != null)
|
||||||
{
|
{
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
// workaround for IL2CPP bug.
|
|
||||||
PlayerLoopHelper.AddContinuation(PlayerLoopTiming.LastPostLateUpdate, runner.ReturnAction);
|
|
||||||
#else
|
|
||||||
runner.Return();
|
runner.Return();
|
||||||
#endif
|
|
||||||
runner = null;
|
runner = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -55,51 +47,43 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
|
|
||||||
// 4. SetResult
|
// 4. SetResult
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void SetResult()
|
public void SetResult()
|
||||||
{
|
{
|
||||||
// runner is finished, return.
|
// runner is finished, return.
|
||||||
if (runner != null)
|
if (runner != null)
|
||||||
{
|
{
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
// workaround for IL2CPP bug.
|
|
||||||
PlayerLoopHelper.AddContinuation(PlayerLoopTiming.LastPostLateUpdate, runner.ReturnAction);
|
|
||||||
#else
|
|
||||||
runner.Return();
|
runner.Return();
|
||||||
#endif
|
|
||||||
runner = null;
|
runner = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. AwaitOnCompleted
|
// 5. AwaitOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : INotifyCompletion
|
where TAwaiter : INotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runner == null)
|
if (runner == null)
|
||||||
{
|
{
|
||||||
AsyncUniTaskVoid<TStateMachine>.SetStateMachine(ref stateMachine, ref runner);
|
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runner.MoveNext);
|
awaiter.OnCompleted(runner.CallMoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. AwaitUnsafeOnCompleted
|
// 6. AwaitUnsafeOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[SecuritySafeCritical]
|
[SecuritySafeCritical]
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : ICriticalNotifyCompletion
|
where TAwaiter : ICriticalNotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runner == null)
|
if (runner == null)
|
||||||
{
|
{
|
||||||
AsyncUniTaskVoid<TStateMachine>.SetStateMachine(ref stateMachine, ref runner);
|
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.UnsafeOnCompleted(runner.MoveNext);
|
awaiter.OnCompleted(runner.CallMoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7. Start
|
// 7. Start
|
||||||
|
|
|
@ -0,0 +1,57 @@
|
||||||
|
|
||||||
|
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
|
{
|
||||||
|
internal interface IMoveNextRunner
|
||||||
|
{
|
||||||
|
Action CallMoveNext { get; }
|
||||||
|
void Return();
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed class MoveNextRunner<TStateMachine> : IMoveNextRunner, IPromisePoolItem
|
||||||
|
where TStateMachine : IAsyncStateMachine
|
||||||
|
{
|
||||||
|
static PromisePool<MoveNextRunner<TStateMachine>> pool = new PromisePool<MoveNextRunner<TStateMachine>>();
|
||||||
|
|
||||||
|
TStateMachine stateMachine;
|
||||||
|
internal readonly Action callMoveNext;
|
||||||
|
|
||||||
|
public Action CallMoveNext => callMoveNext;
|
||||||
|
|
||||||
|
MoveNextRunner()
|
||||||
|
{
|
||||||
|
callMoveNext = MoveNext;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static MoveNextRunner<TStateMachine> Create(ref TStateMachine stateMachine)
|
||||||
|
{
|
||||||
|
var result = pool.TryRent() ?? new MoveNextRunner<TStateMachine>();
|
||||||
|
result.stateMachine = stateMachine;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
void MoveNext()
|
||||||
|
{
|
||||||
|
stateMachine.MoveNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Return()
|
||||||
|
{
|
||||||
|
pool.TryReturn(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void IPromisePoolItem.Reset()
|
||||||
|
{
|
||||||
|
stateMachine = default;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -1,380 +0,0 @@
|
||||||
#pragma warning disable CS1591
|
|
||||||
|
|
||||||
using Cysharp.Threading.Tasks.Internal;
|
|
||||||
using System;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.CompilerServices
|
|
||||||
{
|
|
||||||
// #ENABLE_IL2CPP in this file is to avoid bug of IL2CPP VM.
|
|
||||||
// Issue is tracked on https://issuetracker.unity3d.com/issues/il2cpp-incorrect-results-when-calling-a-method-from-outside-class-in-a-struct
|
|
||||||
// but currently it is labeled `Won't Fix`.
|
|
||||||
|
|
||||||
internal interface IStateMachineRunner
|
|
||||||
{
|
|
||||||
Action MoveNext { get; }
|
|
||||||
void Return();
|
|
||||||
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
Action ReturnAction { get; }
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
internal interface IStateMachineRunnerPromise : IUniTaskSource
|
|
||||||
{
|
|
||||||
Action MoveNext { get; }
|
|
||||||
UniTask Task { get; }
|
|
||||||
void SetResult();
|
|
||||||
void SetException(Exception exception);
|
|
||||||
}
|
|
||||||
|
|
||||||
internal interface IStateMachineRunnerPromise<T> : IUniTaskSource<T>
|
|
||||||
{
|
|
||||||
Action MoveNext { get; }
|
|
||||||
UniTask<T> Task { get; }
|
|
||||||
void SetResult(T result);
|
|
||||||
void SetException(Exception exception);
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static class StateMachineUtility
|
|
||||||
{
|
|
||||||
// Get AsyncStateMachine internal state to check IL2CPP bug
|
|
||||||
public static int GetState(IAsyncStateMachine stateMachine)
|
|
||||||
{
|
|
||||||
var info = stateMachine.GetType().GetFields(System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance)
|
|
||||||
.First(x => x.Name.EndsWith("__state"));
|
|
||||||
return (int)info.GetValue(stateMachine);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal sealed class AsyncUniTaskVoid<TStateMachine> : IStateMachineRunner, ITaskPoolNode<AsyncUniTaskVoid<TStateMachine>>, IUniTaskSource
|
|
||||||
where TStateMachine : IAsyncStateMachine
|
|
||||||
{
|
|
||||||
static TaskPool<AsyncUniTaskVoid<TStateMachine>> pool;
|
|
||||||
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
public Action ReturnAction { get; }
|
|
||||||
#endif
|
|
||||||
|
|
||||||
TStateMachine stateMachine;
|
|
||||||
|
|
||||||
public Action MoveNext { get; }
|
|
||||||
|
|
||||||
public AsyncUniTaskVoid()
|
|
||||||
{
|
|
||||||
MoveNext = Run;
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
ReturnAction = Return;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void SetStateMachine(ref TStateMachine stateMachine, ref IStateMachineRunner runnerFieldRef)
|
|
||||||
{
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new AsyncUniTaskVoid<TStateMachine>();
|
|
||||||
}
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
runnerFieldRef = result; // set runner before copied.
|
|
||||||
result.stateMachine = stateMachine; // copy struct StateMachine(in release build).
|
|
||||||
}
|
|
||||||
|
|
||||||
static AsyncUniTaskVoid()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(AsyncUniTaskVoid<TStateMachine>), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
AsyncUniTaskVoid<TStateMachine> nextNode;
|
|
||||||
public ref AsyncUniTaskVoid<TStateMachine> NextNode => ref nextNode;
|
|
||||||
|
|
||||||
public void Return()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
stateMachine = default;
|
|
||||||
pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
void Run()
|
|
||||||
{
|
|
||||||
stateMachine.MoveNext();
|
|
||||||
}
|
|
||||||
|
|
||||||
// dummy interface implementation for TaskTracker.
|
|
||||||
|
|
||||||
UniTaskStatus IUniTaskSource.GetStatus(short token)
|
|
||||||
{
|
|
||||||
return UniTaskStatus.Pending;
|
|
||||||
}
|
|
||||||
|
|
||||||
UniTaskStatus IUniTaskSource.UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return UniTaskStatus.Pending;
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal sealed class AsyncUniTask<TStateMachine> : IStateMachineRunnerPromise, IUniTaskSource, ITaskPoolNode<AsyncUniTask<TStateMachine>>
|
|
||||||
where TStateMachine : IAsyncStateMachine
|
|
||||||
{
|
|
||||||
static TaskPool<AsyncUniTask<TStateMachine>> pool;
|
|
||||||
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
readonly Action returnDelegate;
|
|
||||||
#endif
|
|
||||||
public Action MoveNext { get; }
|
|
||||||
|
|
||||||
TStateMachine stateMachine;
|
|
||||||
UniTaskCompletionSourceCore<AsyncUnit> core;
|
|
||||||
|
|
||||||
AsyncUniTask()
|
|
||||||
{
|
|
||||||
MoveNext = Run;
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
returnDelegate = Return;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void SetStateMachine(ref TStateMachine stateMachine, ref IStateMachineRunnerPromise runnerPromiseFieldRef)
|
|
||||||
{
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new AsyncUniTask<TStateMachine>();
|
|
||||||
}
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
runnerPromiseFieldRef = result; // set runner before copied.
|
|
||||||
result.stateMachine = stateMachine; // copy struct StateMachine(in release build).
|
|
||||||
}
|
|
||||||
|
|
||||||
AsyncUniTask<TStateMachine> nextNode;
|
|
||||||
public ref AsyncUniTask<TStateMachine> NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static AsyncUniTask()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(AsyncUniTask<TStateMachine>), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Return()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
stateMachine = default;
|
|
||||||
pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryReturn()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
stateMachine = default;
|
|
||||||
return pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
void Run()
|
|
||||||
{
|
|
||||||
stateMachine.MoveNext();
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask Task
|
|
||||||
{
|
|
||||||
[DebuggerHidden]
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return new UniTask(this, core.Version);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public void SetResult()
|
|
||||||
{
|
|
||||||
core.TrySetResult(AsyncUnit.Default);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public void SetException(Exception exception)
|
|
||||||
{
|
|
||||||
core.TrySetException(exception);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public void GetResult(short token)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
core.GetResult(token);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
// workaround for IL2CPP bug.
|
|
||||||
PlayerLoopHelper.AddContinuation(PlayerLoopTiming.LastPostLateUpdate, returnDelegate);
|
|
||||||
#else
|
|
||||||
TryReturn();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal sealed class AsyncUniTask<TStateMachine, T> : IStateMachineRunnerPromise<T>, IUniTaskSource<T>, ITaskPoolNode<AsyncUniTask<TStateMachine, T>>
|
|
||||||
where TStateMachine : IAsyncStateMachine
|
|
||||||
{
|
|
||||||
static TaskPool<AsyncUniTask<TStateMachine, T>> pool;
|
|
||||||
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
readonly Action returnDelegate;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
public Action MoveNext { get; }
|
|
||||||
|
|
||||||
TStateMachine stateMachine;
|
|
||||||
UniTaskCompletionSourceCore<T> core;
|
|
||||||
|
|
||||||
AsyncUniTask()
|
|
||||||
{
|
|
||||||
MoveNext = Run;
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
returnDelegate = Return;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void SetStateMachine(ref TStateMachine stateMachine, ref IStateMachineRunnerPromise<T> runnerPromiseFieldRef)
|
|
||||||
{
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new AsyncUniTask<TStateMachine, T>();
|
|
||||||
}
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
runnerPromiseFieldRef = result; // set runner before copied.
|
|
||||||
result.stateMachine = stateMachine; // copy struct StateMachine(in release build).
|
|
||||||
}
|
|
||||||
|
|
||||||
AsyncUniTask<TStateMachine, T> nextNode;
|
|
||||||
public ref AsyncUniTask<TStateMachine, T> NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static AsyncUniTask()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(AsyncUniTask<TStateMachine, T>), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Return()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
stateMachine = default;
|
|
||||||
pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryReturn()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
stateMachine = default;
|
|
||||||
return pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
void Run()
|
|
||||||
{
|
|
||||||
// UnityEngine.Debug.Log($"MoveNext State:" + StateMachineUtility.GetState(stateMachine));
|
|
||||||
stateMachine.MoveNext();
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTask<T> Task
|
|
||||||
{
|
|
||||||
[DebuggerHidden]
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return new UniTask<T>(this, core.Version);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public void SetResult(T result)
|
|
||||||
{
|
|
||||||
core.TrySetResult(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public void SetException(Exception exception)
|
|
||||||
{
|
|
||||||
core.TrySetException(exception);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public T GetResult(short token)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return core.GetResult(token);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
#if ENABLE_IL2CPP
|
|
||||||
// workaround for IL2CPP bug.
|
|
||||||
PlayerLoopHelper.AddContinuation(PlayerLoopTiming.LastPostLateUpdate, returnDelegate);
|
|
||||||
#else
|
|
||||||
TryReturn();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
void IUniTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -12,55 +12,27 @@ namespace Cysharp.Threading.Tasks
|
||||||
{
|
{
|
||||||
public static class EnumeratorAsyncExtensions
|
public static class EnumeratorAsyncExtensions
|
||||||
{
|
{
|
||||||
public static UniTask.Awaiter GetAwaiter<T>(this T enumerator)
|
public static UniTask.Awaiter GetAwaiter(this IEnumerator enumerator)
|
||||||
where T : IEnumerator
|
|
||||||
{
|
{
|
||||||
var e = (IEnumerator)enumerator;
|
return new UniTask(EnumeratorPromise.Create(enumerator, PlayerLoopTiming.Update, CancellationToken.None, out var token), token).GetAwaiter();
|
||||||
Error.ThrowArgumentNullException(e, nameof(enumerator));
|
|
||||||
return new UniTask(EnumeratorPromise.Create(e, PlayerLoopTiming.Update, CancellationToken.None, out var token), token).GetAwaiter();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask WithCancellation(this IEnumerator enumerator, CancellationToken cancellationToken)
|
public static UniTask ToUniTask(this IEnumerator enumerator)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(enumerator, nameof(enumerator));
|
return new UniTask(EnumeratorPromise.Create(enumerator, PlayerLoopTiming.Update, CancellationToken.None, out var token), token);
|
||||||
return new UniTask(EnumeratorPromise.Create(enumerator, PlayerLoopTiming.Update, cancellationToken, out var token), token);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ToUniTask(this IEnumerator enumerator, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken))
|
public static UniTask ConfigureAwait(this IEnumerator enumerator, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken))
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(enumerator, nameof(enumerator));
|
|
||||||
return new UniTask(EnumeratorPromise.Create(enumerator, timing, cancellationToken, out var token), token);
|
return new UniTask(EnumeratorPromise.Create(enumerator, timing, cancellationToken, out var token), token);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ToUniTask(this IEnumerator enumerator, MonoBehaviour coroutineRunner)
|
class EnumeratorPromise : IUniTaskSource, IPlayerLoopItem, IPromisePoolItem
|
||||||
{
|
{
|
||||||
var source = AutoResetUniTaskCompletionSource.Create();
|
static readonly PromisePool<EnumeratorPromise> pool = new PromisePool<EnumeratorPromise>();
|
||||||
coroutineRunner.StartCoroutine(Core(enumerator, coroutineRunner, source));
|
|
||||||
return source.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
static IEnumerator Core(IEnumerator inner, MonoBehaviour coroutineRunner, AutoResetUniTaskCompletionSource source)
|
|
||||||
{
|
|
||||||
yield return coroutineRunner.StartCoroutine(inner);
|
|
||||||
source.TrySetResult();
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class EnumeratorPromise : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<EnumeratorPromise>
|
|
||||||
{
|
|
||||||
static TaskPool<EnumeratorPromise> pool;
|
|
||||||
EnumeratorPromise nextNode;
|
|
||||||
public ref EnumeratorPromise NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static EnumeratorPromise()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(EnumeratorPromise), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
IEnumerator innerEnumerator;
|
IEnumerator innerEnumerator;
|
||||||
CancellationToken cancellationToken;
|
CancellationToken cancellationToken;
|
||||||
int initialFrame;
|
|
||||||
bool loopRunning;
|
|
||||||
bool calledGetResult;
|
|
||||||
|
|
||||||
UniTaskCompletionSourceCore<object> core;
|
UniTaskCompletionSourceCore<object> core;
|
||||||
|
|
||||||
|
@ -75,26 +47,16 @@ namespace Cysharp.Threading.Tasks
|
||||||
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!pool.TryPop(out var result))
|
var result = pool.TryRent() ?? new EnumeratorPromise();
|
||||||
{
|
|
||||||
result = new EnumeratorPromise();
|
|
||||||
}
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
result.innerEnumerator = ConsumeEnumerator(innerEnumerator);
|
result.innerEnumerator = ConsumeEnumerator(innerEnumerator);
|
||||||
result.cancellationToken = cancellationToken;
|
result.cancellationToken = cancellationToken;
|
||||||
result.loopRunning = true;
|
|
||||||
result.calledGetResult = false;
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
result.initialFrame = -1;
|
|
||||||
|
PlayerLoopHelper.AddAction(timing, result);
|
||||||
|
|
||||||
token = result.core.Version;
|
token = result.core.Version;
|
||||||
|
|
||||||
// run immediately.
|
|
||||||
if (result.MoveNext())
|
|
||||||
{
|
|
||||||
PlayerLoopHelper.AddAction(timing, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,15 +64,12 @@ namespace Cysharp.Threading.Tasks
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
calledGetResult = true;
|
TaskTracker.RemoveTracking(this);
|
||||||
core.GetResult(token);
|
core.GetResult(token);
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
if (!loopRunning)
|
pool.TryReturn(this);
|
||||||
{
|
|
||||||
TryReturn();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -131,38 +90,12 @@ namespace Cysharp.Threading.Tasks
|
||||||
|
|
||||||
public bool MoveNext()
|
public bool MoveNext()
|
||||||
{
|
{
|
||||||
if (calledGetResult)
|
|
||||||
{
|
|
||||||
loopRunning = false;
|
|
||||||
TryReturn();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (innerEnumerator == null) // invalid status, returned but loop running?
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
if (cancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
loopRunning = false;
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
core.TrySetCanceled(cancellationToken);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (initialFrame == -1)
|
|
||||||
{
|
|
||||||
// Time can not touch in threadpool.
|
|
||||||
if (PlayerLoopHelper.IsMainThread)
|
|
||||||
{
|
|
||||||
initialFrame = Time.frameCount;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (initialFrame == Time.frameCount)
|
|
||||||
{
|
|
||||||
return true; // already executed in first frame, skip.
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (innerEnumerator.MoveNext())
|
if (innerEnumerator.MoveNext())
|
||||||
|
@ -172,24 +105,27 @@ namespace Cysharp.Threading.Tasks
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
loopRunning = false;
|
|
||||||
core.TrySetException(ex);
|
core.TrySetException(ex);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
loopRunning = false;
|
|
||||||
core.TrySetResult(null);
|
core.TrySetResult(null);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TryReturn()
|
public void Reset()
|
||||||
{
|
{
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
core.Reset();
|
||||||
innerEnumerator = default;
|
innerEnumerator = default;
|
||||||
cancellationToken = default;
|
cancellationToken = default;
|
||||||
|
}
|
||||||
|
|
||||||
return pool.TryPush(this);
|
~EnumeratorPromise()
|
||||||
|
{
|
||||||
|
if (pool.TryReturn(this))
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Unwrap YieldInstructions
|
// Unwrap YieldInstructions
|
||||||
|
@ -203,10 +139,11 @@ namespace Cysharp.Threading.Tasks
|
||||||
{
|
{
|
||||||
yield return null;
|
yield return null;
|
||||||
}
|
}
|
||||||
else if (current is CustomYieldInstruction cyi)
|
else if (current is CustomYieldInstruction)
|
||||||
{
|
{
|
||||||
// WWW, WaitForSecondsRealtime
|
// WWW, WaitForSecondsRealtime
|
||||||
while (cyi.keepWaiting)
|
var e2 = UnwrapWaitCustomYieldInstruction((CustomYieldInstruction)current);
|
||||||
|
while (e2.MoveNext())
|
||||||
{
|
{
|
||||||
yield return null;
|
yield return null;
|
||||||
}
|
}
|
||||||
|
@ -232,7 +169,7 @@ namespace Cysharp.Threading.Tasks
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
goto WARN;
|
yield return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (current is IEnumerator e3)
|
else if (current is IEnumerator e3)
|
||||||
|
@ -245,14 +182,17 @@ namespace Cysharp.Threading.Tasks
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
goto WARN;
|
// WaitForEndOfFrame, WaitForFixedUpdate, others.
|
||||||
|
yield return null;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
continue;
|
// WWW and others as CustomYieldInstruction.
|
||||||
|
static IEnumerator UnwrapWaitCustomYieldInstruction(CustomYieldInstruction yieldInstruction)
|
||||||
WARN:
|
{
|
||||||
// WaitForEndOfFrame, WaitForFixedUpdate, others.
|
while (yieldInstruction.keepWaiting)
|
||||||
UnityEngine.Debug.LogWarning($"yield {current.GetType().Name} is not supported on await IEnumerator or IEnumerator.ToUniTask(), please use ToUniTask(MonoBehaviour coroutineRunner) instead.");
|
{
|
||||||
yield return null;
|
yield return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -262,12 +202,12 @@ namespace Cysharp.Threading.Tasks
|
||||||
static IEnumerator UnwrapWaitForSeconds(WaitForSeconds waitForSeconds)
|
static IEnumerator UnwrapWaitForSeconds(WaitForSeconds waitForSeconds)
|
||||||
{
|
{
|
||||||
var second = (float)waitForSeconds_Seconds.GetValue(waitForSeconds);
|
var second = (float)waitForSeconds_Seconds.GetValue(waitForSeconds);
|
||||||
var elapsed = 0.0f;
|
var startTime = DateTimeOffset.UtcNow;
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
yield return null;
|
yield return null;
|
||||||
|
|
||||||
elapsed += Time.deltaTime;
|
var elapsed = (DateTimeOffset.UtcNow - startTime).TotalSeconds;
|
||||||
if (elapsed >= second)
|
if (elapsed >= second)
|
||||||
{
|
{
|
||||||
break;
|
break;
|
||||||
|
@ -284,4 +224,5 @@ namespace Cysharp.Threading.Tasks
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
373
src/UniTask/Assets/Plugins/UniTask/Runtime/External/AddressableAsyncExtensions.cs
vendored
Normal file
373
src/UniTask/Assets/Plugins/UniTask/Runtime/External/AddressableAsyncExtensions.cs
vendored
Normal file
|
@ -0,0 +1,373 @@
|
||||||
|
// asmdef Version Defines, enabled when com.unity.addressables is imported.
|
||||||
|
|
||||||
|
#if UNITASK_ADDRESSABLE_SUPPORT
|
||||||
|
|
||||||
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
using System;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Runtime.ExceptionServices;
|
||||||
|
using System.Threading;
|
||||||
|
using UnityEngine.ResourceManagement.AsyncOperations;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks
|
||||||
|
{
|
||||||
|
public static class AddressableAsyncExtensions
|
||||||
|
{
|
||||||
|
#region AsyncOperationHandle
|
||||||
|
|
||||||
|
public static AsyncOperationHandleAwaiter GetAwaiter(this AsyncOperationHandle handle)
|
||||||
|
{
|
||||||
|
return new AsyncOperationHandleAwaiter(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static UniTask ToUniTask(this AsyncOperationHandle handle)
|
||||||
|
{
|
||||||
|
return new UniTask(AsyncOperationHandleConfiguredSource.Create(handle, PlayerLoopTiming.Update, null, CancellationToken.None, out var token), token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static UniTask ConfigureAwait(this AsyncOperationHandle handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellation = default(CancellationToken))
|
||||||
|
{
|
||||||
|
return new UniTask(AsyncOperationHandleConfiguredSource.Create(handle, timing, progress, cancellation, out var token), token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct AsyncOperationHandleAwaiter : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
AsyncOperationHandle handle;
|
||||||
|
Action<AsyncOperationHandle> continuationAction;
|
||||||
|
|
||||||
|
public AsyncOperationHandleAwaiter(AsyncOperationHandle handle)
|
||||||
|
{
|
||||||
|
this.handle = handle;
|
||||||
|
this.continuationAction = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsCompleted => handle.IsDone;
|
||||||
|
|
||||||
|
public void GetResult()
|
||||||
|
{
|
||||||
|
if (continuationAction != null)
|
||||||
|
{
|
||||||
|
handle.Completed -= continuationAction;
|
||||||
|
continuationAction = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (handle.Status == AsyncOperationStatus.Failed)
|
||||||
|
{
|
||||||
|
var e = handle.OperationException;
|
||||||
|
handle = default;
|
||||||
|
ExceptionDispatchInfo.Capture(e).Throw();
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = handle.Result;
|
||||||
|
handle = default;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
UnsafeOnCompleted(continuation);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
Error.ThrowWhenContinuationIsAlreadyRegistered(continuationAction);
|
||||||
|
continuationAction = continuation.AsFuncOfT<AsyncOperationHandle>(); // allocate delegate.
|
||||||
|
handle.Completed += continuationAction;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class AsyncOperationHandleConfiguredSource : IUniTaskSource, IPlayerLoopItem, IPromisePoolItem
|
||||||
|
{
|
||||||
|
static readonly PromisePool<AsyncOperationHandleConfiguredSource> pool = new PromisePool<AsyncOperationHandleConfiguredSource>();
|
||||||
|
|
||||||
|
AsyncOperationHandle handle;
|
||||||
|
IProgress<float> progress;
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
UniTaskCompletionSourceCore<AsyncUnit> core;
|
||||||
|
|
||||||
|
AsyncOperationHandleConfiguredSource()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IUniTaskSource Create(AsyncOperationHandle handle, PlayerLoopTiming timing, IProgress<float> progress, CancellationToken cancellationToken, out short token)
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = pool.TryRent() ?? new AsyncOperationHandleConfiguredSource();
|
||||||
|
|
||||||
|
result.handle = handle;
|
||||||
|
result.progress = progress;
|
||||||
|
result.cancellationToken = cancellationToken;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
PlayerLoopHelper.AddAction(timing, result);
|
||||||
|
|
||||||
|
token = result.core.Version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void GetResult(short token)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.GetResult(token);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
pool.TryReturn(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool MoveNext()
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
core.TrySetCanceled(cancellationToken);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (progress != null)
|
||||||
|
{
|
||||||
|
progress.Report(handle.PercentComplete);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (handle.IsDone)
|
||||||
|
{
|
||||||
|
if (handle.Status == AsyncOperationStatus.Failed)
|
||||||
|
{
|
||||||
|
core.TrySetException(handle.OperationException);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
core.TrySetResult(AsyncUnit.Default);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Reset()
|
||||||
|
{
|
||||||
|
core.Reset();
|
||||||
|
handle = default;
|
||||||
|
progress = default;
|
||||||
|
cancellationToken = default;
|
||||||
|
}
|
||||||
|
|
||||||
|
~AsyncOperationHandleConfiguredSource()
|
||||||
|
{
|
||||||
|
if (pool.TryReturn(this))
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region AsyncOperationHandle_T
|
||||||
|
|
||||||
|
public static AsyncOperationHandleAwaiter<T> GetAwaiter<T>(this AsyncOperationHandle<T> handle)
|
||||||
|
{
|
||||||
|
return new AsyncOperationHandleAwaiter<T>(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static UniTask<T> ToUniTask<T>(this AsyncOperationHandle<T> handle)
|
||||||
|
{
|
||||||
|
return new UniTask<T>(AsyncOperationHandleConfiguredSource<T>.Create(handle, PlayerLoopTiming.Update, null, CancellationToken.None, out var token), token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static UniTask<T> ConfigureAwait<T>(this AsyncOperationHandle<T> handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellation = default(CancellationToken))
|
||||||
|
{
|
||||||
|
return new UniTask<T>(AsyncOperationHandleConfiguredSource<T>.Create(handle, timing, progress, cancellation, out var token), token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct AsyncOperationHandleAwaiter<T> : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
AsyncOperationHandle<T> handle;
|
||||||
|
Action<AsyncOperationHandle> continuationAction;
|
||||||
|
|
||||||
|
public AsyncOperationHandleAwaiter(AsyncOperationHandle<T> handle)
|
||||||
|
{
|
||||||
|
this.handle = handle;
|
||||||
|
this.continuationAction = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsCompleted => handle.IsDone;
|
||||||
|
|
||||||
|
public T GetResult()
|
||||||
|
{
|
||||||
|
if (continuationAction != null)
|
||||||
|
{
|
||||||
|
handle.CompletedTypeless -= continuationAction;
|
||||||
|
continuationAction = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (handle.Status == AsyncOperationStatus.Failed)
|
||||||
|
{
|
||||||
|
var e = handle.OperationException;
|
||||||
|
handle = default;
|
||||||
|
ExceptionDispatchInfo.Capture(e).Throw();
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = handle.Result;
|
||||||
|
handle = default;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
UnsafeOnCompleted(continuation);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
Error.ThrowWhenContinuationIsAlreadyRegistered(continuationAction);
|
||||||
|
continuationAction = continuation.AsFuncOfT<AsyncOperationHandle>(); // allocate delegate.
|
||||||
|
handle.CompletedTypeless += continuationAction;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class AsyncOperationHandleConfiguredSource<T> : IUniTaskSource<T>, IPlayerLoopItem, IPromisePoolItem
|
||||||
|
{
|
||||||
|
static readonly PromisePool<AsyncOperationHandleConfiguredSource<T>> pool = new PromisePool<AsyncOperationHandleConfiguredSource<T>>();
|
||||||
|
|
||||||
|
AsyncOperationHandle<T> handle;
|
||||||
|
IProgress<float> progress;
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
UniTaskCompletionSourceCore<T> core;
|
||||||
|
|
||||||
|
AsyncOperationHandleConfiguredSource()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IUniTaskSource<T> Create(AsyncOperationHandle<T> handle, PlayerLoopTiming timing, IProgress<float> progress, CancellationToken cancellationToken, out short token)
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
return AutoResetUniTaskCompletionSource<T>.CreateFromCanceled(cancellationToken, out token);
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = pool.TryRent() ?? new AsyncOperationHandleConfiguredSource<T>();
|
||||||
|
|
||||||
|
result.handle = handle;
|
||||||
|
result.progress = progress;
|
||||||
|
result.cancellationToken = cancellationToken;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
PlayerLoopHelper.AddAction(timing, result);
|
||||||
|
|
||||||
|
token = result.core.Version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public T GetResult(short token)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
|
||||||
|
return core.GetResult(token);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
pool.TryReturn(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void IUniTaskSource.GetResult(short token)
|
||||||
|
{
|
||||||
|
GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool MoveNext()
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
core.TrySetCanceled(cancellationToken);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (progress != null)
|
||||||
|
{
|
||||||
|
progress.Report(handle.PercentComplete);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (handle.IsDone)
|
||||||
|
{
|
||||||
|
if (handle.Status == AsyncOperationStatus.Failed)
|
||||||
|
{
|
||||||
|
core.TrySetException(handle.OperationException);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
core.TrySetResult(handle.Result);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Reset()
|
||||||
|
{
|
||||||
|
core.Reset();
|
||||||
|
handle = default;
|
||||||
|
progress = default;
|
||||||
|
cancellationToken = default;
|
||||||
|
}
|
||||||
|
|
||||||
|
~AsyncOperationHandleConfiguredSource()
|
||||||
|
{
|
||||||
|
if (pool.TryReturn(this))
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -1,8 +0,0 @@
|
||||||
fileFormatVersion: 2
|
|
||||||
guid: a5b9231662e24c942b544bd85d4b39cb
|
|
||||||
folderAsset: yes
|
|
||||||
DefaultImporter:
|
|
||||||
externalObjects: {}
|
|
||||||
userData:
|
|
||||||
assetBundleName:
|
|
||||||
assetBundleVariant:
|
|
|
@ -1,483 +0,0 @@
|
||||||
// asmdef Version Defines, enabled when com.unity.addressables is imported.
|
|
||||||
|
|
||||||
#if UNITASK_ADDRESSABLE_SUPPORT
|
|
||||||
|
|
||||||
using Cysharp.Threading.Tasks.Internal;
|
|
||||||
using System;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Runtime.ExceptionServices;
|
|
||||||
using System.Threading;
|
|
||||||
using UnityEngine.AddressableAssets;
|
|
||||||
using UnityEngine.ResourceManagement.AsyncOperations;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public static class AddressablesAsyncExtensions
|
|
||||||
{
|
|
||||||
#region AsyncOperationHandle
|
|
||||||
|
|
||||||
public static UniTask.Awaiter GetAwaiter(this AsyncOperationHandle handle)
|
|
||||||
{
|
|
||||||
return ToUniTask(handle).GetAwaiter();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask WithCancellation(this AsyncOperationHandle handle, CancellationToken cancellationToken, bool cancelImmediately = false, bool autoReleaseWhenCanceled = false)
|
|
||||||
{
|
|
||||||
return ToUniTask(handle, cancellationToken: cancellationToken, cancelImmediately: cancelImmediately, autoReleaseWhenCanceled: autoReleaseWhenCanceled);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask ToUniTask(this AsyncOperationHandle handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken), bool cancelImmediately = false, bool autoReleaseWhenCanceled = false)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested) return UniTask.FromCanceled(cancellationToken);
|
|
||||||
|
|
||||||
if (!handle.IsValid())
|
|
||||||
{
|
|
||||||
// autoReleaseHandle:true handle is invalid(immediately internal handle == null) so return completed.
|
|
||||||
return UniTask.CompletedTask;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (handle.IsDone)
|
|
||||||
{
|
|
||||||
if (handle.Status == AsyncOperationStatus.Failed)
|
|
||||||
{
|
|
||||||
return UniTask.FromException(handle.OperationException);
|
|
||||||
}
|
|
||||||
return UniTask.CompletedTask;
|
|
||||||
}
|
|
||||||
|
|
||||||
return new UniTask(AsyncOperationHandleConfiguredSource.Create(handle, timing, progress, cancellationToken, cancelImmediately, autoReleaseWhenCanceled, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct AsyncOperationHandleAwaiter : ICriticalNotifyCompletion
|
|
||||||
{
|
|
||||||
AsyncOperationHandle handle;
|
|
||||||
Action<AsyncOperationHandle> continuationAction;
|
|
||||||
|
|
||||||
public AsyncOperationHandleAwaiter(AsyncOperationHandle handle)
|
|
||||||
{
|
|
||||||
this.handle = handle;
|
|
||||||
this.continuationAction = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsCompleted => handle.IsDone;
|
|
||||||
|
|
||||||
public void GetResult()
|
|
||||||
{
|
|
||||||
if (continuationAction != null)
|
|
||||||
{
|
|
||||||
handle.Completed -= continuationAction;
|
|
||||||
continuationAction = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (handle.Status == AsyncOperationStatus.Failed)
|
|
||||||
{
|
|
||||||
var e = handle.OperationException;
|
|
||||||
handle = default;
|
|
||||||
ExceptionDispatchInfo.Capture(e).Throw();
|
|
||||||
}
|
|
||||||
|
|
||||||
var result = handle.Result;
|
|
||||||
handle = default;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
UnsafeOnCompleted(continuation);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UnsafeOnCompleted(Action continuation)
|
|
||||||
{
|
|
||||||
Error.ThrowWhenContinuationIsAlreadyRegistered(continuationAction);
|
|
||||||
continuationAction = PooledDelegate<AsyncOperationHandle>.Create(continuation);
|
|
||||||
handle.Completed += continuationAction;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class AsyncOperationHandleConfiguredSource : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<AsyncOperationHandleConfiguredSource>
|
|
||||||
{
|
|
||||||
static TaskPool<AsyncOperationHandleConfiguredSource> pool;
|
|
||||||
AsyncOperationHandleConfiguredSource nextNode;
|
|
||||||
public ref AsyncOperationHandleConfiguredSource NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static AsyncOperationHandleConfiguredSource()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(AsyncOperationHandleConfiguredSource), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly Action<AsyncOperationHandle> completedCallback;
|
|
||||||
AsyncOperationHandle handle;
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
CancellationTokenRegistration cancellationTokenRegistration;
|
|
||||||
IProgress<float> progress;
|
|
||||||
bool autoReleaseWhenCanceled;
|
|
||||||
bool cancelImmediately;
|
|
||||||
bool completed;
|
|
||||||
|
|
||||||
UniTaskCompletionSourceCore<AsyncUnit> core;
|
|
||||||
|
|
||||||
AsyncOperationHandleConfiguredSource()
|
|
||||||
{
|
|
||||||
completedCallback = HandleCompleted;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskSource Create(AsyncOperationHandle handle, PlayerLoopTiming timing, IProgress<float> progress, CancellationToken cancellationToken, bool cancelImmediately, bool autoReleaseWhenCanceled, out short token)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new AsyncOperationHandleConfiguredSource();
|
|
||||||
}
|
|
||||||
|
|
||||||
result.handle = handle;
|
|
||||||
result.progress = progress;
|
|
||||||
result.cancellationToken = cancellationToken;
|
|
||||||
result.cancelImmediately = cancelImmediately;
|
|
||||||
result.autoReleaseWhenCanceled = autoReleaseWhenCanceled;
|
|
||||||
result.completed = false;
|
|
||||||
|
|
||||||
if (cancelImmediately && cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
|
||||||
{
|
|
||||||
var promise = (AsyncOperationHandleConfiguredSource)state;
|
|
||||||
if (promise.autoReleaseWhenCanceled && promise.handle.IsValid())
|
|
||||||
{
|
|
||||||
Addressables.Release(promise.handle);
|
|
||||||
}
|
|
||||||
promise.core.TrySetCanceled(promise.cancellationToken);
|
|
||||||
}, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
PlayerLoopHelper.AddAction(timing, result);
|
|
||||||
|
|
||||||
handle.Completed += result.completedCallback;
|
|
||||||
|
|
||||||
token = result.core.Version;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HandleCompleted(AsyncOperationHandle _)
|
|
||||||
{
|
|
||||||
if (handle.IsValid())
|
|
||||||
{
|
|
||||||
handle.Completed -= completedCallback;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (completed)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
completed = true;
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
if (autoReleaseWhenCanceled && handle.IsValid())
|
|
||||||
{
|
|
||||||
Addressables.Release(handle);
|
|
||||||
}
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
}
|
|
||||||
else if (handle.Status == AsyncOperationStatus.Failed)
|
|
||||||
{
|
|
||||||
core.TrySetException(handle.OperationException);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
core.TrySetResult(AsyncUnit.Default);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void GetResult(short token)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
core.GetResult(token);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
if (!(cancelImmediately && cancellationToken.IsCancellationRequested))
|
|
||||||
{
|
|
||||||
TryReturn();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool MoveNext()
|
|
||||||
{
|
|
||||||
if (completed)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
completed = true;
|
|
||||||
if (autoReleaseWhenCanceled && handle.IsValid())
|
|
||||||
{
|
|
||||||
Addressables.Release(handle);
|
|
||||||
}
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (progress != null && handle.IsValid())
|
|
||||||
{
|
|
||||||
progress.Report(handle.GetDownloadStatus().Percent);
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryReturn()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
handle = default;
|
|
||||||
progress = default;
|
|
||||||
cancellationToken = default;
|
|
||||||
cancellationTokenRegistration.Dispose();
|
|
||||||
return pool.TryPush(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region AsyncOperationHandle_T
|
|
||||||
|
|
||||||
public static UniTask<T>.Awaiter GetAwaiter<T>(this AsyncOperationHandle<T> handle)
|
|
||||||
{
|
|
||||||
return ToUniTask(handle).GetAwaiter();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<T> WithCancellation<T>(this AsyncOperationHandle<T> handle, CancellationToken cancellationToken, bool cancelImmediately = false, bool autoReleaseWhenCanceled = false)
|
|
||||||
{
|
|
||||||
return ToUniTask(handle, cancellationToken: cancellationToken, cancelImmediately: cancelImmediately, autoReleaseWhenCanceled: autoReleaseWhenCanceled);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<T> ToUniTask<T>(this AsyncOperationHandle<T> handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken), bool cancelImmediately = false, bool autoReleaseWhenCanceled = false)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested) return UniTask.FromCanceled<T>(cancellationToken);
|
|
||||||
|
|
||||||
if (!handle.IsValid())
|
|
||||||
{
|
|
||||||
throw new Exception("Attempting to use an invalid operation handle");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (handle.IsDone)
|
|
||||||
{
|
|
||||||
if (handle.Status == AsyncOperationStatus.Failed)
|
|
||||||
{
|
|
||||||
return UniTask.FromException<T>(handle.OperationException);
|
|
||||||
}
|
|
||||||
return UniTask.FromResult(handle.Result);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new UniTask<T>(AsyncOperationHandleConfiguredSource<T>.Create(handle, timing, progress, cancellationToken, cancelImmediately, autoReleaseWhenCanceled, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class AsyncOperationHandleConfiguredSource<T> : IUniTaskSource<T>, IPlayerLoopItem, ITaskPoolNode<AsyncOperationHandleConfiguredSource<T>>
|
|
||||||
{
|
|
||||||
static TaskPool<AsyncOperationHandleConfiguredSource<T>> pool;
|
|
||||||
AsyncOperationHandleConfiguredSource<T> nextNode;
|
|
||||||
public ref AsyncOperationHandleConfiguredSource<T> NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static AsyncOperationHandleConfiguredSource()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(AsyncOperationHandleConfiguredSource<T>), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly Action<AsyncOperationHandle<T>> completedCallback;
|
|
||||||
AsyncOperationHandle<T> handle;
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
CancellationTokenRegistration cancellationTokenRegistration;
|
|
||||||
IProgress<float> progress;
|
|
||||||
bool autoReleaseWhenCanceled;
|
|
||||||
bool cancelImmediately;
|
|
||||||
bool completed;
|
|
||||||
|
|
||||||
UniTaskCompletionSourceCore<T> core;
|
|
||||||
|
|
||||||
AsyncOperationHandleConfiguredSource()
|
|
||||||
{
|
|
||||||
completedCallback = HandleCompleted;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskSource<T> Create(AsyncOperationHandle<T> handle, PlayerLoopTiming timing, IProgress<float> progress, CancellationToken cancellationToken, bool cancelImmediately, bool autoReleaseWhenCanceled, out short token)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
return AutoResetUniTaskCompletionSource<T>.CreateFromCanceled(cancellationToken, out token);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new AsyncOperationHandleConfiguredSource<T>();
|
|
||||||
}
|
|
||||||
|
|
||||||
result.handle = handle;
|
|
||||||
result.cancellationToken = cancellationToken;
|
|
||||||
result.completed = false;
|
|
||||||
result.progress = progress;
|
|
||||||
result.autoReleaseWhenCanceled = autoReleaseWhenCanceled;
|
|
||||||
result.cancelImmediately = cancelImmediately;
|
|
||||||
|
|
||||||
if (cancelImmediately && cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
|
||||||
{
|
|
||||||
var promise = (AsyncOperationHandleConfiguredSource<T>)state;
|
|
||||||
if (promise.autoReleaseWhenCanceled && promise.handle.IsValid())
|
|
||||||
{
|
|
||||||
Addressables.Release(promise.handle);
|
|
||||||
}
|
|
||||||
promise.core.TrySetCanceled(promise.cancellationToken);
|
|
||||||
}, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
PlayerLoopHelper.AddAction(timing, result);
|
|
||||||
|
|
||||||
handle.Completed += result.completedCallback;
|
|
||||||
|
|
||||||
token = result.core.Version;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HandleCompleted(AsyncOperationHandle<T> argHandle)
|
|
||||||
{
|
|
||||||
if (handle.IsValid())
|
|
||||||
{
|
|
||||||
handle.Completed -= completedCallback;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (completed)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
completed = true;
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
if (autoReleaseWhenCanceled && handle.IsValid())
|
|
||||||
{
|
|
||||||
Addressables.Release(handle);
|
|
||||||
}
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
}
|
|
||||||
else if (argHandle.Status == AsyncOperationStatus.Failed)
|
|
||||||
{
|
|
||||||
core.TrySetException(argHandle.OperationException);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
core.TrySetResult(argHandle.Result);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public T GetResult(short token)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return core.GetResult(token);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
if (!(cancelImmediately && cancellationToken.IsCancellationRequested))
|
|
||||||
{
|
|
||||||
TryReturn();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool MoveNext()
|
|
||||||
{
|
|
||||||
if (completed)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
completed = true;
|
|
||||||
if (autoReleaseWhenCanceled && handle.IsValid())
|
|
||||||
{
|
|
||||||
Addressables.Release(handle);
|
|
||||||
}
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (progress != null && handle.IsValid())
|
|
||||||
{
|
|
||||||
progress.Report(handle.GetDownloadStatus().Percent);
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryReturn()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
handle = default;
|
|
||||||
progress = default;
|
|
||||||
cancellationToken = default;
|
|
||||||
cancellationTokenRegistration.Dispose();
|
|
||||||
return pool.TryPush(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,28 +0,0 @@
|
||||||
{
|
|
||||||
"name": "UniTask.Addressables",
|
|
||||||
"references": [
|
|
||||||
"UniTask",
|
|
||||||
"Unity.ResourceManager",
|
|
||||||
"Unity.Addressables"
|
|
||||||
],
|
|
||||||
"includePlatforms": [],
|
|
||||||
"excludePlatforms": [],
|
|
||||||
"allowUnsafeCode": false,
|
|
||||||
"overrideReferences": false,
|
|
||||||
"precompiledReferences": [],
|
|
||||||
"autoReferenced": true,
|
|
||||||
"defineConstraints": [],
|
|
||||||
"versionDefines": [
|
|
||||||
{
|
|
||||||
"name": "com.unity.addressables",
|
|
||||||
"expression": "",
|
|
||||||
"define": "UNITASK_ADDRESSABLE_SUPPORT"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "com.unity.addressables.cn",
|
|
||||||
"expression": "",
|
|
||||||
"define": "UNITASK_ADDRESSABLE_SUPPORT"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"noEngineReferences": false
|
|
||||||
}
|
|
|
@ -1,436 +0,0 @@
|
||||||
// asmdef Version Defines, enabled when com.demigiant.dotween is imported.
|
|
||||||
|
|
||||||
#if UNITASK_DOTWEEN_SUPPORT
|
|
||||||
|
|
||||||
using Cysharp.Threading.Tasks.Internal;
|
|
||||||
using DG.Tweening;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Concurrent;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public enum TweenCancelBehaviour
|
|
||||||
{
|
|
||||||
Kill,
|
|
||||||
KillWithCompleteCallback,
|
|
||||||
Complete,
|
|
||||||
CompleteWithSequenceCallback,
|
|
||||||
CancelAwait,
|
|
||||||
|
|
||||||
// AndCancelAwait
|
|
||||||
KillAndCancelAwait,
|
|
||||||
KillWithCompleteCallbackAndCancelAwait,
|
|
||||||
CompleteAndCancelAwait,
|
|
||||||
CompleteWithSequenceCallbackAndCancelAwait
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class DOTweenAsyncExtensions
|
|
||||||
{
|
|
||||||
enum CallbackType
|
|
||||||
{
|
|
||||||
Kill,
|
|
||||||
Complete,
|
|
||||||
Pause,
|
|
||||||
Play,
|
|
||||||
Rewind,
|
|
||||||
StepComplete
|
|
||||||
}
|
|
||||||
|
|
||||||
public static TweenAwaiter GetAwaiter(this Tween tween)
|
|
||||||
{
|
|
||||||
return new TweenAwaiter(tween);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask WithCancellation(this Tween tween, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
Error.ThrowArgumentNullException(tween, nameof(tween));
|
|
||||||
|
|
||||||
if (!tween.IsActive()) return UniTask.CompletedTask;
|
|
||||||
return new UniTask(TweenConfiguredSource.Create(tween, TweenCancelBehaviour.Kill, cancellationToken, CallbackType.Kill, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask ToUniTask(this Tween tween, TweenCancelBehaviour tweenCancelBehaviour = TweenCancelBehaviour.Kill, CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
Error.ThrowArgumentNullException(tween, nameof(tween));
|
|
||||||
|
|
||||||
if (!tween.IsActive()) return UniTask.CompletedTask;
|
|
||||||
return new UniTask(TweenConfiguredSource.Create(tween, tweenCancelBehaviour, cancellationToken, CallbackType.Kill, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask AwaitForComplete(this Tween tween, TweenCancelBehaviour tweenCancelBehaviour = TweenCancelBehaviour.Kill, CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
Error.ThrowArgumentNullException(tween, nameof(tween));
|
|
||||||
|
|
||||||
if (!tween.IsActive()) return UniTask.CompletedTask;
|
|
||||||
return new UniTask(TweenConfiguredSource.Create(tween, tweenCancelBehaviour, cancellationToken, CallbackType.Complete, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask AwaitForPause(this Tween tween, TweenCancelBehaviour tweenCancelBehaviour = TweenCancelBehaviour.Kill, CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
Error.ThrowArgumentNullException(tween, nameof(tween));
|
|
||||||
|
|
||||||
if (!tween.IsActive()) return UniTask.CompletedTask;
|
|
||||||
return new UniTask(TweenConfiguredSource.Create(tween, tweenCancelBehaviour, cancellationToken, CallbackType.Pause, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask AwaitForPlay(this Tween tween, TweenCancelBehaviour tweenCancelBehaviour = TweenCancelBehaviour.Kill, CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
Error.ThrowArgumentNullException(tween, nameof(tween));
|
|
||||||
|
|
||||||
if (!tween.IsActive()) return UniTask.CompletedTask;
|
|
||||||
return new UniTask(TweenConfiguredSource.Create(tween, tweenCancelBehaviour, cancellationToken, CallbackType.Play, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask AwaitForRewind(this Tween tween, TweenCancelBehaviour tweenCancelBehaviour = TweenCancelBehaviour.Kill, CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
Error.ThrowArgumentNullException(tween, nameof(tween));
|
|
||||||
|
|
||||||
if (!tween.IsActive()) return UniTask.CompletedTask;
|
|
||||||
return new UniTask(TweenConfiguredSource.Create(tween, tweenCancelBehaviour, cancellationToken, CallbackType.Rewind, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask AwaitForStepComplete(this Tween tween, TweenCancelBehaviour tweenCancelBehaviour = TweenCancelBehaviour.Kill, CancellationToken cancellationToken = default)
|
|
||||||
{
|
|
||||||
Error.ThrowArgumentNullException(tween, nameof(tween));
|
|
||||||
|
|
||||||
if (!tween.IsActive()) return UniTask.CompletedTask;
|
|
||||||
return new UniTask(TweenConfiguredSource.Create(tween, tweenCancelBehaviour, cancellationToken, CallbackType.StepComplete, out var token), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct TweenAwaiter : ICriticalNotifyCompletion
|
|
||||||
{
|
|
||||||
readonly Tween tween;
|
|
||||||
|
|
||||||
// killed(non active) as completed.
|
|
||||||
public bool IsCompleted => !tween.IsActive();
|
|
||||||
|
|
||||||
public TweenAwaiter(Tween tween)
|
|
||||||
{
|
|
||||||
this.tween = tween;
|
|
||||||
}
|
|
||||||
|
|
||||||
public TweenAwaiter GetAwaiter()
|
|
||||||
{
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void GetResult()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(System.Action continuation)
|
|
||||||
{
|
|
||||||
UnsafeOnCompleted(continuation);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UnsafeOnCompleted(System.Action continuation)
|
|
||||||
{
|
|
||||||
// onKill is called after OnCompleted, both Complete(false/true) and Kill(false/true).
|
|
||||||
tween.onKill = PooledTweenCallback.Create(continuation);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class TweenConfiguredSource : IUniTaskSource, ITaskPoolNode<TweenConfiguredSource>
|
|
||||||
{
|
|
||||||
static TaskPool<TweenConfiguredSource> pool;
|
|
||||||
TweenConfiguredSource nextNode;
|
|
||||||
public ref TweenConfiguredSource NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static TweenConfiguredSource()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(TweenConfiguredSource), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly TweenCallback onCompleteCallbackDelegate;
|
|
||||||
|
|
||||||
Tween tween;
|
|
||||||
TweenCancelBehaviour cancelBehaviour;
|
|
||||||
CancellationToken cancellationToken;
|
|
||||||
CancellationTokenRegistration cancellationRegistration;
|
|
||||||
CallbackType callbackType;
|
|
||||||
bool canceled;
|
|
||||||
|
|
||||||
TweenCallback originalCompleteAction;
|
|
||||||
UniTaskCompletionSourceCore<AsyncUnit> core;
|
|
||||||
|
|
||||||
TweenConfiguredSource()
|
|
||||||
{
|
|
||||||
onCompleteCallbackDelegate = OnCompleteCallbackDelegate;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskSource Create(Tween tween, TweenCancelBehaviour cancelBehaviour, CancellationToken cancellationToken, CallbackType callbackType, out short token)
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
DoCancelBeforeCreate(tween, cancelBehaviour);
|
|
||||||
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!pool.TryPop(out var result))
|
|
||||||
{
|
|
||||||
result = new TweenConfiguredSource();
|
|
||||||
}
|
|
||||||
|
|
||||||
result.tween = tween;
|
|
||||||
result.cancelBehaviour = cancelBehaviour;
|
|
||||||
result.cancellationToken = cancellationToken;
|
|
||||||
result.callbackType = callbackType;
|
|
||||||
result.canceled = false;
|
|
||||||
|
|
||||||
switch (callbackType)
|
|
||||||
{
|
|
||||||
case CallbackType.Kill:
|
|
||||||
result.originalCompleteAction = tween.onKill;
|
|
||||||
tween.onKill = result.onCompleteCallbackDelegate;
|
|
||||||
break;
|
|
||||||
case CallbackType.Complete:
|
|
||||||
result.originalCompleteAction = tween.onComplete;
|
|
||||||
tween.onComplete = result.onCompleteCallbackDelegate;
|
|
||||||
break;
|
|
||||||
case CallbackType.Pause:
|
|
||||||
result.originalCompleteAction = tween.onPause;
|
|
||||||
tween.onPause = result.onCompleteCallbackDelegate;
|
|
||||||
break;
|
|
||||||
case CallbackType.Play:
|
|
||||||
result.originalCompleteAction = tween.onPlay;
|
|
||||||
tween.onPlay = result.onCompleteCallbackDelegate;
|
|
||||||
break;
|
|
||||||
case CallbackType.Rewind:
|
|
||||||
result.originalCompleteAction = tween.onRewind;
|
|
||||||
tween.onRewind = result.onCompleteCallbackDelegate;
|
|
||||||
break;
|
|
||||||
case CallbackType.StepComplete:
|
|
||||||
result.originalCompleteAction = tween.onStepComplete;
|
|
||||||
tween.onStepComplete = result.onCompleteCallbackDelegate;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result.originalCompleteAction == result.onCompleteCallbackDelegate)
|
|
||||||
{
|
|
||||||
result.originalCompleteAction = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cancellationToken.CanBeCanceled)
|
|
||||||
{
|
|
||||||
result.cancellationRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(x =>
|
|
||||||
{
|
|
||||||
var source = (TweenConfiguredSource)x;
|
|
||||||
switch (source.cancelBehaviour)
|
|
||||||
{
|
|
||||||
case TweenCancelBehaviour.Kill:
|
|
||||||
default:
|
|
||||||
source.tween.Kill(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.KillAndCancelAwait:
|
|
||||||
source.canceled = true;
|
|
||||||
source.tween.Kill(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.KillWithCompleteCallback:
|
|
||||||
source.tween.Kill(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.KillWithCompleteCallbackAndCancelAwait:
|
|
||||||
source.canceled = true;
|
|
||||||
source.tween.Kill(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.Complete:
|
|
||||||
source.tween.Complete(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CompleteAndCancelAwait:
|
|
||||||
source.canceled = true;
|
|
||||||
source.tween.Complete(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CompleteWithSequenceCallback:
|
|
||||||
source.tween.Complete(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CompleteWithSequenceCallbackAndCancelAwait:
|
|
||||||
source.canceled = true;
|
|
||||||
source.tween.Complete(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CancelAwait:
|
|
||||||
source.RestoreOriginalCallback();
|
|
||||||
source.core.TrySetCanceled(source.cancellationToken);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
token = result.core.Version;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
void OnCompleteCallbackDelegate()
|
|
||||||
{
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
if (this.cancelBehaviour == TweenCancelBehaviour.KillAndCancelAwait
|
|
||||||
|| this.cancelBehaviour == TweenCancelBehaviour.KillWithCompleteCallbackAndCancelAwait
|
|
||||||
|| this.cancelBehaviour == TweenCancelBehaviour.CompleteAndCancelAwait
|
|
||||||
|| this.cancelBehaviour == TweenCancelBehaviour.CompleteWithSequenceCallbackAndCancelAwait
|
|
||||||
|| this.cancelBehaviour == TweenCancelBehaviour.CancelAwait)
|
|
||||||
{
|
|
||||||
canceled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (canceled)
|
|
||||||
{
|
|
||||||
core.TrySetCanceled(cancellationToken);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
originalCompleteAction?.Invoke();
|
|
||||||
core.TrySetResult(AsyncUnit.Default);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void DoCancelBeforeCreate(Tween tween, TweenCancelBehaviour tweenCancelBehaviour)
|
|
||||||
{
|
|
||||||
|
|
||||||
switch (tweenCancelBehaviour)
|
|
||||||
{
|
|
||||||
case TweenCancelBehaviour.Kill:
|
|
||||||
default:
|
|
||||||
tween.Kill(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.KillAndCancelAwait:
|
|
||||||
tween.Kill(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.KillWithCompleteCallback:
|
|
||||||
tween.Kill(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.KillWithCompleteCallbackAndCancelAwait:
|
|
||||||
tween.Kill(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.Complete:
|
|
||||||
tween.Complete(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CompleteAndCancelAwait:
|
|
||||||
tween.Complete(false);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CompleteWithSequenceCallback:
|
|
||||||
tween.Complete(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CompleteWithSequenceCallbackAndCancelAwait:
|
|
||||||
tween.Complete(true);
|
|
||||||
break;
|
|
||||||
case TweenCancelBehaviour.CancelAwait:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void GetResult(short token)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
core.GetResult(token);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
TryReturn();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return core.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return core.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
core.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryReturn()
|
|
||||||
{
|
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.Reset();
|
|
||||||
cancellationRegistration.Dispose();
|
|
||||||
|
|
||||||
RestoreOriginalCallback();
|
|
||||||
|
|
||||||
tween = default;
|
|
||||||
cancellationToken = default;
|
|
||||||
originalCompleteAction = default;
|
|
||||||
return pool.TryPush(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
void RestoreOriginalCallback()
|
|
||||||
{
|
|
||||||
switch (callbackType)
|
|
||||||
{
|
|
||||||
case CallbackType.Kill:
|
|
||||||
tween.onKill = originalCompleteAction;
|
|
||||||
break;
|
|
||||||
case CallbackType.Complete:
|
|
||||||
tween.onComplete = originalCompleteAction;
|
|
||||||
break;
|
|
||||||
case CallbackType.Pause:
|
|
||||||
tween.onPause = originalCompleteAction;
|
|
||||||
break;
|
|
||||||
case CallbackType.Play:
|
|
||||||
tween.onPlay = originalCompleteAction;
|
|
||||||
break;
|
|
||||||
case CallbackType.Rewind:
|
|
||||||
tween.onRewind = originalCompleteAction;
|
|
||||||
break;
|
|
||||||
case CallbackType.StepComplete:
|
|
||||||
tween.onStepComplete = originalCompleteAction;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sealed class PooledTweenCallback
|
|
||||||
{
|
|
||||||
static readonly ConcurrentQueue<PooledTweenCallback> pool = new ConcurrentQueue<PooledTweenCallback>();
|
|
||||||
|
|
||||||
readonly TweenCallback runDelegate;
|
|
||||||
|
|
||||||
Action continuation;
|
|
||||||
|
|
||||||
|
|
||||||
PooledTweenCallback()
|
|
||||||
{
|
|
||||||
runDelegate = Run;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public static TweenCallback Create(Action continuation)
|
|
||||||
{
|
|
||||||
if (!pool.TryDequeue(out var item))
|
|
||||||
{
|
|
||||||
item = new PooledTweenCallback();
|
|
||||||
}
|
|
||||||
|
|
||||||
item.continuation = continuation;
|
|
||||||
return item.runDelegate;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
void Run()
|
|
||||||
{
|
|
||||||
var call = continuation;
|
|
||||||
continuation = null;
|
|
||||||
if (call != null)
|
|
||||||
{
|
|
||||||
pool.Enqueue(this);
|
|
||||||
call.Invoke();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,22 +0,0 @@
|
||||||
{
|
|
||||||
"name": "UniTask.DOTween",
|
|
||||||
"references": [
|
|
||||||
"UniTask",
|
|
||||||
"DOTween.Modules"
|
|
||||||
],
|
|
||||||
"includePlatforms": [],
|
|
||||||
"excludePlatforms": [],
|
|
||||||
"allowUnsafeCode": false,
|
|
||||||
"overrideReferences": false,
|
|
||||||
"precompiledReferences": [],
|
|
||||||
"autoReferenced": true,
|
|
||||||
"defineConstraints": [],
|
|
||||||
"versionDefines": [
|
|
||||||
{
|
|
||||||
"name": "com.demigiant.dotween",
|
|
||||||
"expression": "",
|
|
||||||
"define": "UNITASK_DOTWEEN_SUPPORT"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"noEngineReferences": false
|
|
||||||
}
|
|
|
@ -1,7 +0,0 @@
|
||||||
fileFormatVersion: 2
|
|
||||||
guid: 029c1c1b674aaae47a6841a0b89ad80e
|
|
||||||
AssemblyDefinitionImporter:
|
|
||||||
externalObjects: {}
|
|
||||||
userData:
|
|
||||||
assetBundleName:
|
|
||||||
assetBundleVariant:
|
|
|
@ -1,224 +0,0 @@
|
||||||
#if UNITASK_TEXTMESHPRO_SUPPORT
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Threading;
|
|
||||||
using TMPro;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public static partial class TextMeshProAsyncExtensions
|
|
||||||
{
|
|
||||||
public static IAsyncValueChangedEventHandler<string> GetAsyncValueChangedEventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onValueChanged, inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncValueChangedEventHandler<string> GetAsyncValueChangedEventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onValueChanged, cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnValueChangedAsync(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onValueChanged, inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnValueChangedAsync(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onValueChanged, cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnValueChangedAsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onValueChanged, inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnValueChangedAsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onValueChanged, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncEndEditEventHandler<string> GetAsyncEndEditEventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onEndEdit, inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncEndEditEventHandler<string> GetAsyncEndEditEventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onEndEdit, cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnEndEditAsync(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onEndEdit, inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnEndEditAsync(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onEndEdit, cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnEndEditAsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onEndEdit, inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnEndEditAsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onEndEdit, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncEndTextSelectionEventHandler<(string, int, int)> GetAsyncEndTextSelectionEventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onEndTextSelection), inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncEndTextSelectionEventHandler<(string, int, int)> GetAsyncEndTextSelectionEventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onEndTextSelection), cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<(string, int, int)> OnEndTextSelectionAsync(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onEndTextSelection), inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<(string, int, int)> OnEndTextSelectionAsync(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onEndTextSelection), cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<(string, int, int)> OnEndTextSelectionAsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<(string, int, int)>(new TextSelectionEventConverter(inputField.onEndTextSelection), inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<(string, int, int)> OnEndTextSelectionAsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<(string, int, int)>(new TextSelectionEventConverter(inputField.onEndTextSelection), cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncTextSelectionEventHandler<(string, int, int)> GetAsyncTextSelectionEventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onTextSelection), inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncTextSelectionEventHandler<(string, int, int)> GetAsyncTextSelectionEventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onTextSelection), cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<(string, int, int)> OnTextSelectionAsync(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onTextSelection), inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<(string, int, int)> OnTextSelectionAsync(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<(string, int, int)>(new TextSelectionEventConverter(inputField.onTextSelection), cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<(string, int, int)> OnTextSelectionAsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<(string, int, int)>(new TextSelectionEventConverter(inputField.onTextSelection), inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<(string, int, int)> OnTextSelectionAsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<(string, int, int)>(new TextSelectionEventConverter(inputField.onTextSelection), cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncDeselectEventHandler<string> GetAsyncDeselectEventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onDeselect, inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncDeselectEventHandler<string> GetAsyncDeselectEventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onDeselect, cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnDeselectAsync(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onDeselect, inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnDeselectAsync(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onDeselect, cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnDeselectAsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onDeselect, inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnDeselectAsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onDeselect, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncSelectEventHandler<string> GetAsyncSelectEventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSelect, inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncSelectEventHandler<string> GetAsyncSelectEventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSelect, cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnSelectAsync(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSelect, inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnSelectAsync(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSelect, cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnSelectAsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onSelect, inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnSelectAsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onSelect, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncSubmitEventHandler<string> GetAsyncSubmitEventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSubmit, inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsyncSubmitEventHandler<string> GetAsyncSubmitEventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSubmit, cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnSubmitAsync(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSubmit, inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<string> OnSubmitAsync(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<string>(inputField.onSubmit, cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnSubmitAsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onSubmit, inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<string> OnSubmitAsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<string>(inputField.onSubmit, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,11 +0,0 @@
|
||||||
fileFormatVersion: 2
|
|
||||||
guid: 79f4f2475e0b2c44e97ed1dee760627b
|
|
||||||
MonoImporter:
|
|
||||||
externalObjects: {}
|
|
||||||
serializedVersion: 2
|
|
||||||
defaultReferences: []
|
|
||||||
executionOrder: 0
|
|
||||||
icon: {instanceID: 0}
|
|
||||||
userData:
|
|
||||||
assetBundleName:
|
|
||||||
assetBundleVariant:
|
|
|
@ -1,66 +0,0 @@
|
||||||
<#@ template debug="false" hostspecific="false" language="C#" #>
|
|
||||||
<#@ assembly name="System.Core" #>
|
|
||||||
<#@ import namespace="System.Linq" #>
|
|
||||||
<#@ import namespace="System.Text" #>
|
|
||||||
<#@ import namespace="System.Collections.Generic" #>
|
|
||||||
<#@ output extension=".cs" #>
|
|
||||||
<#
|
|
||||||
var handlers = new (string name, string type)[] {
|
|
||||||
("ValueChanged", "string"),
|
|
||||||
("EndEdit", "string"),
|
|
||||||
("EndTextSelection", "(string, int, int)"),
|
|
||||||
("TextSelection", "(string, int, int)"),
|
|
||||||
("Deselect", "string"),
|
|
||||||
("Select", "string"),
|
|
||||||
("Submit", "string"),
|
|
||||||
};
|
|
||||||
|
|
||||||
Func<string, bool> shouldConvert = x => x.EndsWith("TextSelection");
|
|
||||||
Func<string, string> eventName = x => shouldConvert(x) ? $"new TextSelectionEventConverter(inputField.on{x})" : $"inputField.on{x}";
|
|
||||||
#>
|
|
||||||
#if UNITASK_TEXTMESHPRO_SUPPORT
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Threading;
|
|
||||||
using TMPro;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public static partial class TextMeshProAsyncExtensions
|
|
||||||
{
|
|
||||||
<# foreach(var (name, type) in handlers) { #>
|
|
||||||
public static IAsync<#= (name) #>EventHandler<<#= type #>> GetAsync<#= (name) #>EventHandler(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<<#= type #>>(<#= eventName(name) #>, inputField.GetCancellationTokenOnDestroy(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IAsync<#= (name) #>EventHandler<<#= type #>> GetAsync<#= (name) #>EventHandler(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<<#= type #>>(<#= eventName(name) #>, cancellationToken, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<<#= type #>> On<#= (name) #>Async(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<<#= type #>>(<#= eventName(name) #>, inputField.GetCancellationTokenOnDestroy(), true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static UniTask<<#= type #>> On<#= (name) #>Async(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new AsyncUnityEventHandler<<#= type #>>(<#= eventName(name) #>, cancellationToken, true).OnInvokeAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<<#= type #>> On<#= (name) #>AsAsyncEnumerable(this TMP_InputField inputField)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<<#= type #>>(<#= eventName(name) #>, inputField.GetCancellationTokenOnDestroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<<#= type #>> On<#= (name) #>AsAsyncEnumerable(this TMP_InputField inputField, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
return new UnityEventHandlerAsyncEnumerable<<#= type #>>(<#= eventName(name) #>, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
<# } #>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,7 +0,0 @@
|
||||||
fileFormatVersion: 2
|
|
||||||
guid: e9bb9fc551a975d44a7180e022a2debe
|
|
||||||
DefaultImporter:
|
|
||||||
externalObjects: {}
|
|
||||||
userData:
|
|
||||||
assetBundleName:
|
|
||||||
assetBundleVariant:
|
|
|
@ -1,130 +0,0 @@
|
||||||
#if UNITASK_TEXTMESHPRO_SUPPORT
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Threading;
|
|
||||||
using TMPro;
|
|
||||||
using UnityEngine.Events;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
|
||||||
{
|
|
||||||
public static partial class TextMeshProAsyncExtensions
|
|
||||||
{
|
|
||||||
// <string> -> Text
|
|
||||||
public static void BindTo(this IUniTaskAsyncEnumerable<string> source, TMP_Text text, bool rebindOnError = true)
|
|
||||||
{
|
|
||||||
BindToCore(source, text, text.GetCancellationTokenOnDestroy(), rebindOnError).Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void BindTo(this IUniTaskAsyncEnumerable<string> source, TMP_Text text, CancellationToken cancellationToken, bool rebindOnError = true)
|
|
||||||
{
|
|
||||||
BindToCore(source, text, cancellationToken, rebindOnError).Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
static async UniTaskVoid BindToCore(IUniTaskAsyncEnumerable<string> source, TMP_Text text, CancellationToken cancellationToken, bool rebindOnError)
|
|
||||||
{
|
|
||||||
var repeat = false;
|
|
||||||
BIND_AGAIN:
|
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
|
||||||
try
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
bool moveNext;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
moveNext = await e.MoveNextAsync();
|
|
||||||
repeat = false;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
if (ex is OperationCanceledException) return;
|
|
||||||
|
|
||||||
if (rebindOnError && !repeat)
|
|
||||||
{
|
|
||||||
repeat = true;
|
|
||||||
goto BIND_AGAIN;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!moveNext) return;
|
|
||||||
|
|
||||||
text.text = e.Current;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
if (e != null)
|
|
||||||
{
|
|
||||||
await e.DisposeAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// <T> -> Text
|
|
||||||
|
|
||||||
public static void BindTo<T>(this IUniTaskAsyncEnumerable<T> source, TMP_Text text, bool rebindOnError = true)
|
|
||||||
{
|
|
||||||
BindToCore(source, text, text.GetCancellationTokenOnDestroy(), rebindOnError).Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void BindTo<T>(this IUniTaskAsyncEnumerable<T> source, TMP_Text text, CancellationToken cancellationToken, bool rebindOnError = true)
|
|
||||||
{
|
|
||||||
BindToCore(source, text, cancellationToken, rebindOnError).Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void BindTo<T>(this AsyncReactiveProperty<T> source, TMP_Text text, bool rebindOnError = true)
|
|
||||||
{
|
|
||||||
BindToCore(source, text, text.GetCancellationTokenOnDestroy(), rebindOnError).Forget();
|
|
||||||
}
|
|
||||||
|
|
||||||
static async UniTaskVoid BindToCore<T>(IUniTaskAsyncEnumerable<T> source, TMP_Text text, CancellationToken cancellationToken, bool rebindOnError)
|
|
||||||
{
|
|
||||||
var repeat = false;
|
|
||||||
BIND_AGAIN:
|
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
|
||||||
try
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
bool moveNext;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
moveNext = await e.MoveNextAsync();
|
|
||||||
repeat = false;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
if (ex is OperationCanceledException) return;
|
|
||||||
|
|
||||||
if (rebindOnError && !repeat)
|
|
||||||
{
|
|
||||||
repeat = true;
|
|
||||||
goto BIND_AGAIN;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!moveNext) return;
|
|
||||||
|
|
||||||
text.text = e.Current.ToString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
if (e != null)
|
|
||||||
{
|
|
||||||
await e.DisposeAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,11 +0,0 @@
|
||||||
fileFormatVersion: 2
|
|
||||||
guid: b6ba480edafb67d4e91bb10feb64fae5
|
|
||||||
MonoImporter:
|
|
||||||
externalObjects: {}
|
|
||||||
serializedVersion: 2
|
|
||||||
defaultReferences: []
|
|
||||||
executionOrder: 0
|
|
||||||
icon: {instanceID: 0}
|
|
||||||
userData:
|
|
||||||
assetBundleName:
|
|
||||||
assetBundleVariant:
|
|
|
@ -1,27 +0,0 @@
|
||||||
{
|
|
||||||
"name": "UniTask.TextMeshPro",
|
|
||||||
"references": [
|
|
||||||
"UniTask",
|
|
||||||
"Unity.TextMeshPro"
|
|
||||||
],
|
|
||||||
"includePlatforms": [],
|
|
||||||
"excludePlatforms": [],
|
|
||||||
"allowUnsafeCode": false,
|
|
||||||
"overrideReferences": false,
|
|
||||||
"precompiledReferences": [],
|
|
||||||
"autoReferenced": true,
|
|
||||||
"defineConstraints": [],
|
|
||||||
"versionDefines": [
|
|
||||||
{
|
|
||||||
"name": "com.unity.textmeshpro",
|
|
||||||
"expression": "",
|
|
||||||
"define": "UNITASK_TEXTMESHPRO_SUPPORT"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "com.unity.ugui",
|
|
||||||
"expression": "2.0.0",
|
|
||||||
"define": "UNITASK_TEXTMESHPRO_SUPPORT"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"noEngineReferences": false
|
|
||||||
}
|
|
|
@ -1,7 +0,0 @@
|
||||||
fileFormatVersion: 2
|
|
||||||
guid: dc47925d1a5fa2946bdd37746b2b5d48
|
|
||||||
AssemblyDefinitionImporter:
|
|
||||||
externalObjects: {}
|
|
||||||
userData:
|
|
||||||
assetBundleName:
|
|
||||||
assetBundleVariant:
|
|
|
@ -28,12 +28,6 @@ namespace Cysharp.Threading.Tasks
|
||||||
IUniTaskOrderedAsyncEnumerable<TElement> CreateOrderedEnumerable<TKey>(Func<TElement, CancellationToken, UniTask<TKey>> keySelector, IComparer<TKey> comparer, bool descending);
|
IUniTaskOrderedAsyncEnumerable<TElement> CreateOrderedEnumerable<TKey>(Func<TElement, CancellationToken, UniTask<TKey>> keySelector, IComparer<TKey> comparer, bool descending);
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface IConnectableUniTaskAsyncEnumerable<out T> : IUniTaskAsyncEnumerable<T>
|
|
||||||
{
|
|
||||||
IDisposable Connect();
|
|
||||||
}
|
|
||||||
|
|
||||||
// don't use AsyncGrouping.
|
|
||||||
//public interface IUniTaskAsyncGrouping<out TKey, out TElement> : IUniTaskAsyncEnumerable<TElement>
|
//public interface IUniTaskAsyncGrouping<out TKey, out TElement> : IUniTaskAsyncEnumerable<TElement>
|
||||||
//{
|
//{
|
||||||
// TKey Key { get; }
|
// TKey Key { get; }
|
||||||
|
|
|
@ -1,9 +1,4 @@
|
||||||
#pragma warning disable CS1591
|
#pragma warning disable CS1591
|
||||||
#pragma warning disable CS0108
|
|
||||||
|
|
||||||
#if (UNITASK_NETCORE && !NETSTANDARD2_0) || UNITY_2022_3_OR_NEWER
|
|
||||||
#define SUPPORT_VALUETASK
|
|
||||||
#endif
|
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
|
@ -24,73 +19,17 @@ namespace Cysharp.Threading.Tasks
|
||||||
|
|
||||||
// similar as IValueTaskSource
|
// similar as IValueTaskSource
|
||||||
public interface IUniTaskSource
|
public interface IUniTaskSource
|
||||||
#if SUPPORT_VALUETASK
|
|
||||||
: System.Threading.Tasks.Sources.IValueTaskSource
|
|
||||||
#endif
|
|
||||||
{
|
{
|
||||||
UniTaskStatus GetStatus(short token);
|
UniTaskStatus GetStatus(short token);
|
||||||
void OnCompleted(Action<object> continuation, object state, short token);
|
void OnCompleted(Action<object> continuation, object state, short token);
|
||||||
void GetResult(short token);
|
void GetResult(short token);
|
||||||
|
|
||||||
UniTaskStatus UnsafeGetStatus(); // only for debug use.
|
UniTaskStatus UnsafeGetStatus(); // only for debug use.
|
||||||
|
|
||||||
#if SUPPORT_VALUETASK
|
|
||||||
|
|
||||||
System.Threading.Tasks.Sources.ValueTaskSourceStatus System.Threading.Tasks.Sources.IValueTaskSource.GetStatus(short token)
|
|
||||||
{
|
|
||||||
return (System.Threading.Tasks.Sources.ValueTaskSourceStatus)(int)((IUniTaskSource)this).GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
void System.Threading.Tasks.Sources.IValueTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
((IUniTaskSource)this).GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
void System.Threading.Tasks.Sources.IValueTaskSource.OnCompleted(Action<object> continuation, object state, short token, System.Threading.Tasks.Sources.ValueTaskSourceOnCompletedFlags flags)
|
|
||||||
{
|
|
||||||
// ignore flags, always none.
|
|
||||||
((IUniTaskSource)this).OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface IUniTaskSource<out T> : IUniTaskSource
|
public interface IUniTaskSource<out T> : IUniTaskSource
|
||||||
#if SUPPORT_VALUETASK
|
|
||||||
, System.Threading.Tasks.Sources.IValueTaskSource<T>
|
|
||||||
#endif
|
|
||||||
{
|
{
|
||||||
new T GetResult(short token);
|
new T GetResult(short token);
|
||||||
|
|
||||||
#if SUPPORT_VALUETASK
|
|
||||||
|
|
||||||
new public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return ((IUniTaskSource)this).GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
new public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
((IUniTaskSource)this).OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
System.Threading.Tasks.Sources.ValueTaskSourceStatus System.Threading.Tasks.Sources.IValueTaskSource<T>.GetStatus(short token)
|
|
||||||
{
|
|
||||||
return (System.Threading.Tasks.Sources.ValueTaskSourceStatus)(int)((IUniTaskSource)this).GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
T System.Threading.Tasks.Sources.IValueTaskSource<T>.GetResult(short token)
|
|
||||||
{
|
|
||||||
return ((IUniTaskSource<T>)this).GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
void System.Threading.Tasks.Sources.IValueTaskSource<T>.OnCompleted(Action<object> continuation, object state, short token, System.Threading.Tasks.Sources.ValueTaskSourceOnCompletedFlags flags)
|
|
||||||
{
|
|
||||||
// ignore flags, always none.
|
|
||||||
((IUniTaskSource)this).OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class UniTaskStatusExtensions
|
public static class UniTaskStatusExtensions
|
||||||
|
|
|
@ -10,9 +10,7 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
const int MaxArrayLength = 0X7FEFFFFF;
|
const int MaxArrayLength = 0X7FEFFFFF;
|
||||||
const int InitialSize = 16;
|
const int InitialSize = 16;
|
||||||
|
|
||||||
readonly PlayerLoopTiming timing;
|
SpinLock gate = new SpinLock();
|
||||||
|
|
||||||
SpinLock gate = new SpinLock(false);
|
|
||||||
bool dequing = false;
|
bool dequing = false;
|
||||||
|
|
||||||
int actionListCount = 0;
|
int actionListCount = 0;
|
||||||
|
@ -21,11 +19,6 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
int waitingListCount = 0;
|
int waitingListCount = 0;
|
||||||
Action[] waitingList = new Action[InitialSize];
|
Action[] waitingList = new Action[InitialSize];
|
||||||
|
|
||||||
public ContinuationQueue(PlayerLoopTiming timing)
|
|
||||||
{
|
|
||||||
this.timing = timing;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Enqueue(Action continuation)
|
public void Enqueue(Action continuation)
|
||||||
{
|
{
|
||||||
bool lockTaken = false;
|
bool lockTaken = false;
|
||||||
|
@ -70,105 +63,16 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Clear()
|
public void Clear()
|
||||||
{
|
{
|
||||||
var rest = actionListCount + waitingListCount;
|
|
||||||
|
|
||||||
actionListCount = 0;
|
actionListCount = 0;
|
||||||
actionList = new Action[InitialSize];
|
actionList = new Action[InitialSize];
|
||||||
|
|
||||||
waitingListCount = 0;
|
waitingListCount = 0;
|
||||||
waitingList = new Action[InitialSize];
|
waitingList = new Action[InitialSize];
|
||||||
|
|
||||||
return rest;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// delegate entrypoint.
|
|
||||||
public void Run()
|
public void Run()
|
||||||
{
|
|
||||||
// for debugging, create named stacktrace.
|
|
||||||
#if DEBUG
|
|
||||||
switch (timing)
|
|
||||||
{
|
|
||||||
case PlayerLoopTiming.Initialization:
|
|
||||||
Initialization();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastInitialization:
|
|
||||||
LastInitialization();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.EarlyUpdate:
|
|
||||||
EarlyUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastEarlyUpdate:
|
|
||||||
LastEarlyUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.FixedUpdate:
|
|
||||||
FixedUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastFixedUpdate:
|
|
||||||
LastFixedUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.PreUpdate:
|
|
||||||
PreUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastPreUpdate:
|
|
||||||
LastPreUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.Update:
|
|
||||||
Update();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastUpdate:
|
|
||||||
LastUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.PreLateUpdate:
|
|
||||||
PreLateUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastPreLateUpdate:
|
|
||||||
LastPreLateUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.PostLateUpdate:
|
|
||||||
PostLateUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastPostLateUpdate:
|
|
||||||
LastPostLateUpdate();
|
|
||||||
break;
|
|
||||||
#if UNITY_2020_2_OR_NEWER
|
|
||||||
case PlayerLoopTiming.TimeUpdate:
|
|
||||||
TimeUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastTimeUpdate:
|
|
||||||
LastTimeUpdate();
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
RunCore();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void Initialization() => RunCore();
|
|
||||||
void LastInitialization() => RunCore();
|
|
||||||
void EarlyUpdate() => RunCore();
|
|
||||||
void LastEarlyUpdate() => RunCore();
|
|
||||||
void FixedUpdate() => RunCore();
|
|
||||||
void LastFixedUpdate() => RunCore();
|
|
||||||
void PreUpdate() => RunCore();
|
|
||||||
void LastPreUpdate() => RunCore();
|
|
||||||
void Update() => RunCore();
|
|
||||||
void LastUpdate() => RunCore();
|
|
||||||
void PreLateUpdate() => RunCore();
|
|
||||||
void LastPreLateUpdate() => RunCore();
|
|
||||||
void PostLateUpdate() => RunCore();
|
|
||||||
void LastPostLateUpdate() => RunCore();
|
|
||||||
#if UNITY_2020_2_OR_NEWER
|
|
||||||
void TimeUpdate() => RunCore();
|
|
||||||
void LastTimeUpdate() => RunCore();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
[System.Diagnostics.DebuggerHidden]
|
|
||||||
void RunCore()
|
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
bool lockTaken = false;
|
bool lockTaken = false;
|
||||||
|
@ -186,17 +90,10 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
|
|
||||||
for (int i = 0; i < actionListCount; i++)
|
for (int i = 0; i < actionListCount; i++)
|
||||||
{
|
{
|
||||||
|
|
||||||
var action = actionList[i];
|
var action = actionList[i];
|
||||||
actionList[i] = null;
|
actionList[i] = null;
|
||||||
try
|
|
||||||
{
|
action();
|
||||||
action();
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
UnityEngine.Debug.LogException(ex);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
|
|
@ -148,7 +148,7 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
|
|
||||||
foreach (var candidateMethod in methods)
|
foreach (var candidateMethod in methods)
|
||||||
{
|
{
|
||||||
var attributes = candidateMethod.GetCustomAttributes<StateMachineAttribute>(false);
|
var attributes = candidateMethod.GetCustomAttributes<StateMachineAttribute>();
|
||||||
if (attributes == null)
|
if (attributes == null)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
|
@ -239,7 +239,7 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
var fname = fi.FullName.Replace(Path.DirectorySeparatorChar, '/').Replace(PlayerLoopHelper.ApplicationDataPath, "");
|
var fname = fi.FullName.Replace(Path.DirectorySeparatorChar, '/').Replace(Application.dataPath, "");
|
||||||
var withAssetsPath = "Assets/" + fname;
|
var withAssetsPath = "Assets/" + fname;
|
||||||
return "<a href=\"" + withAssetsPath + "\" line=\"" + line + "\">" + withAssetsPath + ":" + line + "</a>";
|
return "<a href=\"" + withAssetsPath + "\" line=\"" + line + "\">" + withAssetsPath + ":" + line + "</a>";
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,7 +39,7 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
}
|
}
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.NoInlining)]
|
[MethodImpl(MethodImplOptions.NoInlining)]
|
||||||
public static void ThrowArgumentException(string message)
|
public static void ThrowArgumentException<T>(string message)
|
||||||
{
|
{
|
||||||
throw new ArgumentException(message);
|
throw new ArgumentException(message);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
#if NET_4_6 || NET_STANDARD_2_0 || CSHARP_7_OR_LATER
|
||||||
|
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.Internal
|
||||||
|
{
|
||||||
|
internal static class FuncExtensions
|
||||||
|
{
|
||||||
|
// avoid lambda capture
|
||||||
|
|
||||||
|
internal static Action<T> AsFuncOfT<T>(this Action action)
|
||||||
|
{
|
||||||
|
return new Action<T>(action.Invoke);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void Invoke<T>(this Action action, T unused)
|
||||||
|
{
|
||||||
|
action();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -1,5 +1,5 @@
|
||||||
fileFormatVersion: 2
|
fileFormatVersion: 2
|
||||||
guid: e51b78c06cb410f42b36e0af9de3b065
|
guid: 4d5a9a3e1f0f069478969f752fde29a9
|
||||||
MonoImporter:
|
MonoImporter:
|
||||||
externalObjects: {}
|
externalObjects: {}
|
||||||
serializedVersion: 2
|
serializedVersion: 2
|
|
@ -8,7 +8,6 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
{
|
{
|
||||||
const int InitialSize = 16;
|
const int InitialSize = 16;
|
||||||
|
|
||||||
readonly PlayerLoopTiming timing;
|
|
||||||
readonly object runningAndQueueLock = new object();
|
readonly object runningAndQueueLock = new object();
|
||||||
readonly object arrayLock = new object();
|
readonly object arrayLock = new object();
|
||||||
readonly Action<Exception> unhandledExceptionCallback;
|
readonly Action<Exception> unhandledExceptionCallback;
|
||||||
|
@ -18,12 +17,9 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
IPlayerLoopItem[] loopItems = new IPlayerLoopItem[InitialSize];
|
IPlayerLoopItem[] loopItems = new IPlayerLoopItem[InitialSize];
|
||||||
MinimumQueue<IPlayerLoopItem> waitQueue = new MinimumQueue<IPlayerLoopItem>(InitialSize);
|
MinimumQueue<IPlayerLoopItem> waitQueue = new MinimumQueue<IPlayerLoopItem>(InitialSize);
|
||||||
|
|
||||||
|
public PlayerLoopRunner()
|
||||||
|
|
||||||
public PlayerLoopRunner(PlayerLoopTiming timing)
|
|
||||||
{
|
{
|
||||||
this.unhandledExceptionCallback = ex => Debug.LogException(ex);
|
this.unhandledExceptionCallback = ex => Debug.LogException(ex);
|
||||||
this.timing = timing;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void AddAction(IPlayerLoopItem item)
|
public void AddAction(IPlayerLoopItem item)
|
||||||
|
@ -48,113 +44,18 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Clear()
|
public void Clear()
|
||||||
{
|
{
|
||||||
lock (arrayLock)
|
lock (arrayLock)
|
||||||
{
|
{
|
||||||
var rest = 0;
|
|
||||||
|
|
||||||
for (var index = 0; index < loopItems.Length; index++)
|
for (var index = 0; index < loopItems.Length; index++)
|
||||||
{
|
{
|
||||||
if (loopItems[index] != null)
|
|
||||||
{
|
|
||||||
rest++;
|
|
||||||
}
|
|
||||||
|
|
||||||
loopItems[index] = null;
|
loopItems[index] = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
tail = 0;
|
|
||||||
return rest;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// delegate entrypoint.
|
|
||||||
public void Run()
|
public void Run()
|
||||||
{
|
|
||||||
// for debugging, create named stacktrace.
|
|
||||||
#if DEBUG
|
|
||||||
switch (timing)
|
|
||||||
{
|
|
||||||
case PlayerLoopTiming.Initialization:
|
|
||||||
Initialization();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastInitialization:
|
|
||||||
LastInitialization();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.EarlyUpdate:
|
|
||||||
EarlyUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastEarlyUpdate:
|
|
||||||
LastEarlyUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.FixedUpdate:
|
|
||||||
FixedUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastFixedUpdate:
|
|
||||||
LastFixedUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.PreUpdate:
|
|
||||||
PreUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastPreUpdate:
|
|
||||||
LastPreUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.Update:
|
|
||||||
Update();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastUpdate:
|
|
||||||
LastUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.PreLateUpdate:
|
|
||||||
PreLateUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastPreLateUpdate:
|
|
||||||
LastPreLateUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.PostLateUpdate:
|
|
||||||
PostLateUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastPostLateUpdate:
|
|
||||||
LastPostLateUpdate();
|
|
||||||
break;
|
|
||||||
#if UNITY_2020_2_OR_NEWER
|
|
||||||
case PlayerLoopTiming.TimeUpdate:
|
|
||||||
TimeUpdate();
|
|
||||||
break;
|
|
||||||
case PlayerLoopTiming.LastTimeUpdate:
|
|
||||||
LastTimeUpdate();
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
RunCore();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void Initialization() => RunCore();
|
|
||||||
void LastInitialization() => RunCore();
|
|
||||||
void EarlyUpdate() => RunCore();
|
|
||||||
void LastEarlyUpdate() => RunCore();
|
|
||||||
void FixedUpdate() => RunCore();
|
|
||||||
void LastFixedUpdate() => RunCore();
|
|
||||||
void PreUpdate() => RunCore();
|
|
||||||
void LastPreUpdate() => RunCore();
|
|
||||||
void Update() => RunCore();
|
|
||||||
void LastUpdate() => RunCore();
|
|
||||||
void PreLateUpdate() => RunCore();
|
|
||||||
void LastPreLateUpdate() => RunCore();
|
|
||||||
void PostLateUpdate() => RunCore();
|
|
||||||
void LastPostLateUpdate() => RunCore();
|
|
||||||
#if UNITY_2020_2_OR_NEWER
|
|
||||||
void TimeUpdate() => RunCore();
|
|
||||||
void LastTimeUpdate() => RunCore();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
[System.Diagnostics.DebuggerHidden]
|
|
||||||
void RunCore()
|
|
||||||
{
|
{
|
||||||
lock (runningAndQueueLock)
|
lock (runningAndQueueLock)
|
||||||
{
|
{
|
||||||
|
@ -165,6 +66,7 @@ namespace Cysharp.Threading.Tasks.Internal
|
||||||
{
|
{
|
||||||
var j = tail - 1;
|
var j = tail - 1;
|
||||||
|
|
||||||
|
// eliminate array-bound check for i
|
||||||
for (int i = 0; i < loopItems.Length; i++)
|
for (int i = 0; i < loopItems.Length; i++)
|
||||||
{
|
{
|
||||||
var action = loopItems[i];
|
var action = loopItems[i];
|
||||||
|
|
|
@ -1,50 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.Internal
|
|
||||||
{
|
|
||||||
internal sealed class PooledDelegate<T> : ITaskPoolNode<PooledDelegate<T>>
|
|
||||||
{
|
|
||||||
static TaskPool<PooledDelegate<T>> pool;
|
|
||||||
|
|
||||||
PooledDelegate<T> nextNode;
|
|
||||||
public ref PooledDelegate<T> NextNode => ref nextNode;
|
|
||||||
|
|
||||||
static PooledDelegate()
|
|
||||||
{
|
|
||||||
TaskPool.RegisterSizeGetter(typeof(PooledDelegate<T>), () => pool.Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly Action<T> runDelegate;
|
|
||||||
Action continuation;
|
|
||||||
|
|
||||||
PooledDelegate()
|
|
||||||
{
|
|
||||||
runDelegate = Run;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public static Action<T> Create(Action continuation)
|
|
||||||
{
|
|
||||||
if (!pool.TryPop(out var item))
|
|
||||||
{
|
|
||||||
item = new PooledDelegate<T>();
|
|
||||||
}
|
|
||||||
|
|
||||||
item.continuation = continuation;
|
|
||||||
return item.runDelegate;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
void Run(T _)
|
|
||||||
{
|
|
||||||
var call = continuation;
|
|
||||||
continuation = null;
|
|
||||||
if (call != null)
|
|
||||||
{
|
|
||||||
pool.TryPush(this);
|
|
||||||
call.Invoke();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,11 +0,0 @@
|
||||||
fileFormatVersion: 2
|
|
||||||
guid: 8932579438742fa40b010edd412dbfba
|
|
||||||
MonoImporter:
|
|
||||||
externalObjects: {}
|
|
||||||
serializedVersion: 2
|
|
||||||
defaultReferences: []
|
|
||||||
executionOrder: 0
|
|
||||||
icon: {instanceID: 0}
|
|
||||||
userData:
|
|
||||||
assetBundleName:
|
|
||||||
assetBundleVariant:
|
|
|
@ -0,0 +1,55 @@
|
||||||
|
using System.Collections.Concurrent;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.Internal
|
||||||
|
{
|
||||||
|
// public, allow to user create custom operator with pool.
|
||||||
|
|
||||||
|
public interface IPromisePoolItem
|
||||||
|
{
|
||||||
|
void Reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
public class PromisePool<T>
|
||||||
|
where T : class, IPromisePoolItem
|
||||||
|
{
|
||||||
|
int count = 0;
|
||||||
|
readonly ConcurrentQueue<T> queue = new ConcurrentQueue<T>();
|
||||||
|
readonly int maxSize;
|
||||||
|
|
||||||
|
public PromisePool(int maxSize = 256)
|
||||||
|
{
|
||||||
|
this.maxSize = maxSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public T TryRent()
|
||||||
|
{
|
||||||
|
if (queue.TryDequeue(out var value))
|
||||||
|
{
|
||||||
|
Interlocked.Decrement(ref count);
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryReturn(T value)
|
||||||
|
{
|
||||||
|
value.Reset(); // reset when return.
|
||||||
|
|
||||||
|
if (count < maxSize)
|
||||||
|
{
|
||||||
|
queue.Enqueue(value);
|
||||||
|
Interlocked.Increment(ref count);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue