I have the following C# code.
var databaseRestore = new Microsoft.SqlServer.Management.Smo.Restore();
//databaseRestore.PercentComplete += CompletionStatusInPercent;
//databaseRestore.PercentCompleteNotification = 10;
//databaseRestore.Complete += Restore_Completed;
...
var complete = Observable
.FromEventPattern(databaseRestore, "Complete")
.Select(x=>x.EventArgs as ServerMessageEventArgs)
.Select(x=>x.Error.Message)
.Take(1)
.DumpLive("Complete");
var percentComplete = Observable
.FromEventPattern(databaseRestore, "PercentComplete")
.Select(x=>x.EventArgs as PercentCompleteEventArgs)
.Select(x=>x.Percent)
.TakeUntil(complete)
.DumpLive("PercentComplete");
...
databaseRestore.SqlRestore(server);
If I run it this, first comes always the output from handlers (if I uncomment them).
Then first, Linqpad shows the "Live Observables" result tab with
- "Complete" observable, already finished and with final result message.
- "PercentComplete", still waiting and with "-" (yet no entries?)
What I want is just to come away from events using reactive extenxions. First should come the "PercentComplete" observable updated with actual progress. Then "Complete" with final message.
The question: how do I set up the observables properly?
My guess is this is the fault of how DumpLive() uses a Dispatcher on the main thread.
Are you running the database restore on the main thread, or blocking the main thread waiting for a completion signal?
I experimented a bit and ran an Observable on a background thread with a DumpLive() - if the main thread was blocked (e.g. with a Wait or a Thread.Sleep) the UI did not update.
Even if it's on a different thread, it seems DumpLive() can only update the UI on the main thread. My guess it that it needs to render updates on a Dispatcher associated with the main thread.
When using DumpLive LINQPad won't terminate execution until the observables complete, so if you have any blocking code on main thread remove it and see if it works then.
Here's my experimental code. As soon as you add in either
DumpLive()
clause in the example below, you'll see the blocking behaviour: