主に、この質問のテスト駆動型非同期タスクのフォローアップとして、タスクを待機させない場合は機能するが、待機させると失敗するコードを考え出しました。
誰でも理由を説明できますか?
例外:
Stephen Cleary が彼のブログで書いたユーティリティ クラスのコンストラクターにコードがヒットすると、このエラーが発生します。
public ProgressReporter()
{
_scheduler = TaskScheduler.FromCurrentSynchronizationContext();
}
Test 'Smack.Core.Presentation.Tests.Threading.ProgressReporterTests.OnSuccessFullComplete_ExpectedResultIsReturned_JustWait' failed:
System.AggregateException : One or more errors occurred.
----> System.InvalidOperationException : The current SynchronizationContext may not be used as a TaskScheduler.
at System.Threading.Tasks.Task.ThrowIfExceptional(Boolean includeTaskCanceledExceptions)
at System.Threading.Tasks.Task.Wait(Int32 millisecondsTimeout, CancellationToken cancellationToken)
at System.Threading.Tasks.Task.Wait()
Threading\ProgressReporterTests.cs(142,0): at Smack.Core.Presentation.Tests.Threading.ProgressReporterTests.OnSuccessFullComplete_ExpectedResultIsReturned_JustWait()
--InvalidOperationException
at System.Threading.Tasks.SynchronizationContextTaskScheduler..ctor()
at System.Threading.Tasks.TaskScheduler.FromCurrentSynchronizationContext()
Threading\ProgressReporter.cs(24,0): at Smack.Core.Lib.Threading.ProgressReporter..ctor()
Threading\ProgressReporterTests.cs(52,0): at Smack.Core.Presentation.Tests.Threading.ProgressReporterTests._startBackgroundTask(Boolean causeError)
Threading\ProgressReporterTests.cs(141,0): at Smack.Core.Presentation.Tests.Threading.ProgressReporterTests.<OnSuccessFullComplete_ExpectedResultIsReturned_JustWait>b__a()
at System.Threading.Tasks.Task.InnerInvoke()
at System.Threading.Tasks.Task.Execute()
テスト (NUnit w/ TestDriven.Net ランナー):
private class MockSynchContext : SynchronizationContext{}
[Test]
public void OnSuccessFullComplete_ExpectedResultIsReturned_Wait()
{
var mc = new MockSynchContext();
SynchronizationContext.SetSynchronizationContext(mc);
Assert.That(SynchronizationContext.Current, Is.EqualTo(mc));
Assert.DoesNotThrow(() => TaskScheduler.FromCurrentSynchronizationContext());
var task = Task.Factory.StartNew(() => _startBackgroundTask(false));
task.Wait(2000);
_actualResult = 42;
}
SuT:
private void _startBackgroundTask(bool causeError)
{
_cancellationTokenSource = new CancellationTokenSource();
var cancellationToken = _cancellationTokenSource.Token;
_progressReporter = new ProgressReporter();
var task = Task.Factory.StartNew(() =>
{
for (var i = 0; i != 100; ++i) {
// Check for cancellation
cancellationToken.ThrowIfCancellationRequested();
Thread.Sleep(30); // Do some work.
// Report progress of the work.
_progressReporter.ReportProgress(
() =>
{
// Note: code passed to "ReportProgress" can access UI elements freely.
_currentProgress = i;
});
}
// After all that work, cause the error if requested.
if (causeError) {
throw new InvalidOperationException("Oops...");
}
// The answer, at last!
return 42;
},
cancellationToken);
// ProgressReporter can be used to report successful completion,
// cancelation, or failure to the UI thread.
_progressReporter.RegisterContinuation(task, () =>
{
// Update UI to reflect completion.
_currentProgress = 100;
// Display results.
if (task.Exception != null)
_actualErrorMessage = task.Exception.ToString();
else if (task.IsCanceled)
_wasCancelled = true;
else
_actualResult = task.Result;
// Reset UI.
_whenCompleted();
});
}
明確にするために: task.Wait をコメントアウトすると、そのテストは実際に成功します。何故ですか?
余分なポイント:
これは技術的には別の質問であることは知っていますが、これをすべて繰り返すのは残念です。
テストでは MockSynchContext が TaskScheduler.FromCurrentSynchronizationContext() で例外をスローしなかったのに、2 番目のタスクでは例外をスローしたのはなぜですか? さらに重要なのは、テストを適切に実行できるようにコンテキストを渡す方法はありますか?