// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information. #if !NO_PERF using System; namespace System.Reactive.Linq.Observαble { class SingleAsync : Producer { private readonly IObservable _source; private readonly Func _predicate; private readonly bool _throwOnEmpty; public SingleAsync(IObservable source, Func predicate, bool throwOnEmpty) { _source = source; _predicate = predicate; _throwOnEmpty = throwOnEmpty; } protected override IDisposable Run(IObserver observer, IDisposable cancel, Action setSink) { if (_predicate != null) { var sink = new π(this, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } else { var sink = new _(this, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } } class _ : Sink, IObserver { private readonly SingleAsync _parent; private TSource _value; private bool _seenValue; public _(SingleAsync parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { _parent = parent; _value = default(TSource); _seenValue = false; } public void OnNext(TSource value) { if (_seenValue) { base._observer.OnError(new InvalidOperationException(Strings_Linq.MORE_THAN_ONE_ELEMENT)); base.Dispose(); return; } _value = value; _seenValue = true; } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { if (!_seenValue && _parent._throwOnEmpty) { base._observer.OnError(new InvalidOperationException(Strings_Linq.NO_ELEMENTS)); } else { base._observer.OnNext(_value); base._observer.OnCompleted(); } base.Dispose(); } } class π : Sink, IObserver { private readonly SingleAsync _parent; private TSource _value; private bool _seenValue; public π(SingleAsync parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { _parent = parent; _value = default(TSource); _seenValue = false; } public void OnNext(TSource value) { var b = false; try { b = _parent._predicate(value); } catch (Exception ex) { base._observer.OnError(ex); base.Dispose(); return; } if (b) { if (_seenValue) { base._observer.OnError(new InvalidOperationException(Strings_Linq.MORE_THAN_ONE_MATCHING_ELEMENT)); base.Dispose(); return; } _value = value; _seenValue = true; } } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { if (!_seenValue && _parent._throwOnEmpty) { base._observer.OnError(new InvalidOperationException(Strings_Linq.NO_MATCHING_ELEMENTS)); } else { base._observer.OnNext(_value); base._observer.OnCompleted(); } base.Dispose(); } } } } #endif