FuncChainTaskBase.cs
58 lines
| 1.8 KiB
| text/x-csharp
|
CSharpLexer
/ Implab / FuncChainTaskBase.cs
|
|
r145 | using System; | ||
| using System.Threading; | ||||
| namespace Implab { | ||||
| public class FuncChainTaskBase<TResult> : AbstractPromise<TResult> { | ||||
| readonly Func<Exception, IPromise<TResult>> m_error; | ||||
| readonly Func<Exception, IPromise<TResult>> m_cancel; | ||||
| int m_cancelationLock; | ||||
|
|
r149 | protected FuncChainTaskBase( Func<Exception, IPromise<TResult>> error, Func<Exception, IPromise<TResult>> cancel, bool autoCancellable) { | ||
|
|
r145 | m_error = error; | ||
| m_cancel = cancel; | ||||
|
|
r149 | if (autoCancellable) | ||
| CancellationRequested(CancelOperation); | ||||
|
|
r145 | } | ||
| public void Reject(Exception error) { | ||||
| if (LockCancelation()) | ||||
| HandleErrorInternal(error); | ||||
| } | ||||
| public override void CancelOperation(Exception reason) { | ||||
|
|
r149 | if (LockCancelation()) { | ||
| if (m_cancel != null) { | ||||
| try { | ||||
| m_cancel(reason).On(SetResult, HandleErrorInternal, SetCancelled); | ||||
| } catch (Exception err) { | ||||
| HandleErrorInternal(err); | ||||
| } | ||||
| } else { | ||||
| SetCancelled(reason); | ||||
|
|
r145 | } | ||
| } | ||||
| } | ||||
| protected void HandleErrorInternal(Exception error) { | ||||
| if (m_error != null) { | ||||
| try { | ||||
|
|
r149 | var operation = m_error(error); | ||
| operation.On(SetResult, SetError, SetCancelled); | ||||
| CancellationRequested(operation.Cancel); | ||||
|
|
r145 | } catch(Exception err) { | ||
| SetError(err); | ||||
| } | ||||
| } else { | ||||
| SetError(error); | ||||
| } | ||||
| } | ||||
| protected bool LockCancelation() { | ||||
| return 0 == Interlocked.CompareExchange(ref m_cancelationLock, 1, 0); | ||||
| } | ||||
| } | ||||
| } | ||||
