I am using Tasks to run long running server calls in my ViewModel and the results are marshalled back on Dispatcher
using TaskScheduler.FromSyncronizationContext()
. For example:
var context = TaskScheduler.FromCurrentSynchronizationContext(); this.Message = "Loading..."; Task task = Task.Factory.StartNew(() => { ... }) .ContinueWith(x => this.Message = "Completed" , context);
This works fine when I execute the application. But when I run my NUnit
tests on Resharper
I get the error message on the call to FromCurrentSynchronizationContext
as:
The current SynchronizationContext may not be used as a TaskScheduler.
I guess this is because the tests are run on worker threads. How can I ensure the tests are run on main thread ? Any other suggestions are welcome.
You need to provide a SynchronizationContext. This is how I handle it:
[SetUp] public void TestSetUp() { SynchronizationContext.SetSynchronizationContext(new SynchronizationContext()); }
Ritch Melton's solution did not work for me. This is because my TestInitialize
function is async, as are my tests, so with every await
the current SynchronizationContext
is lost. This is because as MSDN points out, the SynchronizationContext
class is "dumb" and just queues all work to the thread pool.
What worked for me is actually just skipping over the FromCurrentSynchronizationContext
call when there isn't a SynchronizationContext
(that is, if the current context is null). If there's no UI thread, I don't need to synchronize with it in the first place.
TaskScheduler syncContextScheduler; if (SynchronizationContext.Current != null) { syncContextScheduler = TaskScheduler.FromCurrentSynchronizationContext(); } else { // If there is no SyncContext for this thread (e.g. we are in a unit test // or console scenario instead of running in an app), then just use the // default scheduler because there is no UI thread to sync with. syncContextScheduler = TaskScheduler.Current; }
I found this solution more straightforward than the alternatives, which where:
TaskScheduler
to the ViewModel (via dependency injection)SynchronizationContext
and a "fake" UI thread for the tests to run on - way more trouble for me that it's worthI lose some of the threading nuance, but I am not explicitly testing that my OnPropertyChanged callbacks trigger on a specific thread so I am okay with that. The other answers using new SynchronizationContext()
don't really do any better for that goal anyway.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With