Deferred`1.cs
53 lines
| 1.6 KiB
| text/x-csharp
|
CSharpLexer
/ Implab / Deferred`1.cs
cin
|
r246 | using System; | |
using System.Diagnostics; | |||
namespace Implab { | |||
public class Deferred<T> : IResolvable<T> { | |||
cin
|
r248 | readonly Promise<T> m_promise; | |
cin
|
r246 | readonly IDispatcher m_dispatcher; | |
cin
|
r248 | internal Deferred(IDispatcher dispatcher) : this(new Promise<T>(), dispatcher) { | |
} | |||
internal Deferred(Promise<T> promise, IDispatcher dispatcher) { | |||
cin
|
r246 | Debug.Assert(promise != null); | |
m_promise = promise; | |||
m_dispatcher = dispatcher; | |||
} | |||
public IPromise<T> Promise { | |||
get { return m_promise; } | |||
} | |||
public void Reject(Exception error) { | |||
cin
|
r248 | if (error is PromiseTransientException) | |
error = ((PromiseTransientException)error).InnerException; | |||
m_promise.RejectPromise(error); | |||
cin
|
r246 | } | |
public void Resolve(T value) { | |||
cin
|
r248 | m_promise.ResolvePromise(value); | |
cin
|
r246 | } | |
public void Resolve(IPromise<T> thenable) { | |||
if (thenable == null) | |||
Reject(new Exception("The promise or task are expected")); | |||
if (thenable == m_promise) | |||
Reject(new Exception("The promise cannot be resolved with oneself")); | |||
else if (m_dispatcher != null) | |||
// dispatch (see ecma-262/6.0: 25.4.1.3.2 Promise Resolve Functions) | |||
cin
|
r248 | m_dispatcher.Enqueue(Chain, thenable); | |
cin
|
r246 | else | |
Chain(thenable); | |||
} | |||
void Chain(IPromise<T> thenable) { | |||
try { | |||
thenable.Then(this); | |||
} catch (Exception err) { | |||
Reject(err); | |||
} | |||
} | |||
} | |||
} |