123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947 |
- // Licensed to the .NET Foundation under one or more agreements.
- // The .NET Foundation licenses this file to you under the MIT License.
- // See the LICENSE file in the project root for more information.
- using System;
- using System.Collections.Generic;
- using System.Diagnostics;
- using System.Linq;
- using System.Reactive;
- using System.Reactive.Concurrency;
- using System.Reactive.Linq;
- using System.Threading;
- using System.Threading.Tasks;
- using Microsoft.Reactive.Testing;
- using ReactiveTests.Dummies;
- using Microsoft.VisualStudio.TestTools.UnitTesting;
- using Assert = Xunit.Assert;
- namespace ReactiveTests.Tests
- {
- [TestClass]
- public class ConcatTest : ReactiveTest
- {
- [TestMethod]
- public void Concat_ArgumentChecking()
- {
- var xs = DummyObservable<int>.Instance;
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat(xs, null));
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat(null, xs));
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat((IObservable<int>[])null));
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat((IEnumerable<IObservable<int>>)null));
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat((IObservable<int>[])null));
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat((IEnumerable<IObservable<int>>)null));
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat(xs, null));
- ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Concat(null, xs));
- }
- [TestMethod]
- public void Concat_DefaultScheduler()
- {
- var evt = new ManualResetEvent(false);
- var sum = 0;
- Observable.Concat(Observable.Return(1), Observable.Return(2), Observable.Return(3)).Subscribe(n =>
- {
- sum += n;
- }, () => evt.Set());
- evt.WaitOne();
- Assert.Equal(6, sum);
- }
- [TestMethod]
- public void Concat_IEofIO_DefaultScheduler()
- {
- var evt = new ManualResetEvent(false);
- IEnumerable<IObservable<int>> sources = [Observable.Return(1), Observable.Return(2), Observable.Return(3)];
- var sum = 0;
- Observable.Concat(sources).Subscribe(n =>
- {
- sum += n;
- }, () => evt.Set());
- evt.WaitOne();
- Assert.Equal(6, sum);
- }
- [TestMethod]
- public void Concat_IEofIO_GetEnumeratorThrows()
- {
- var ex = new Exception();
- var scheduler = new TestScheduler();
- var xss = new RogueEnumerable<IObservable<int>>(ex);
- var res = scheduler.Start(() =>
- Observable.Concat(xss)
- );
- res.Messages.AssertEqual(
- OnError<int>(200, ex)
- );
- }
- [TestMethod]
- public void Concat_IEofIO()
- {
- var scheduler = new TestScheduler();
- var xs1 = scheduler.CreateColdObservable(
- OnNext(10, 1),
- OnNext(20, 2),
- OnNext(30, 3),
- OnCompleted<int>(40)
- );
- var xs2 = scheduler.CreateColdObservable(
- OnNext(10, 4),
- OnNext(20, 5),
- OnCompleted<int>(30)
- );
- var xs3 = scheduler.CreateColdObservable(
- OnNext(10, 6),
- OnNext(20, 7),
- OnNext(30, 8),
- OnNext(40, 9),
- OnCompleted<int>(50)
- );
- var res = scheduler.Start(() =>
- Observable.Concat([xs1, xs2, xs3])
- );
- res.Messages.AssertEqual(
- OnNext(210, 1),
- OnNext(220, 2),
- OnNext(230, 3),
- OnNext(250, 4),
- OnNext(260, 5),
- OnNext(280, 6),
- OnNext(290, 7),
- OnNext(300, 8),
- OnNext(310, 9),
- OnCompleted<int>(320)
- );
- xs1.Subscriptions.AssertEqual(
- Subscribe(200, 240)
- );
- xs2.Subscriptions.AssertEqual(
- Subscribe(240, 270)
- );
- xs3.Subscriptions.AssertEqual(
- Subscribe(270, 320)
- );
- }
- [TestMethod]
- public void Concat_EmptyEmpty()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(250)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnCompleted<int>(250)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 250)
- );
- }
- [TestMethod]
- public void Concat_EmptyNever()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 1000)
- );
- }
- [TestMethod]
- public void Concat_NeverEmpty()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(230)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 1000)
- );
- o2.Subscriptions.AssertEqual(
- );
- }
- [TestMethod]
- public void Concat_NeverNever()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 1000)
- );
- o2.Subscriptions.AssertEqual(
- );
- }
- [TestMethod]
- public void Concat_EmptyThrow()
- {
- var scheduler = new TestScheduler();
- var ex = new Exception();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnError<int>(250, ex)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnError<int>(250, ex)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 250)
- );
- }
- [TestMethod]
- public void Concat_ThrowEmpty()
- {
- var scheduler = new TestScheduler();
- var ex = new Exception();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnError<int>(230, ex)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(250)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnError<int>(230, ex)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- );
- }
- [TestMethod]
- public void Concat_ThrowThrow()
- {
- var scheduler = new TestScheduler();
- var ex = new Exception();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnError<int>(230, ex)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnError<int>(250, new Exception())
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnError<int>(230, ex)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- );
- }
- [TestMethod]
- public void Concat_ReturnEmpty()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(250)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnNext(210, 2),
- OnCompleted<int>(250)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 250)
- );
- }
- [TestMethod]
- public void Concat_EmptyReturn()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(240, 2),
- OnCompleted<int>(250)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnNext(240, 2),
- OnCompleted<int>(250)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 250)
- );
- }
- [TestMethod]
- public void Concat_ReturnNever()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnNext(210, 2)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 1000)
- );
- }
- [TestMethod]
- public void Concat_NeverReturn()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnCompleted<int>(230)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 1000)
- );
- o2.Subscriptions.AssertEqual(
- );
- }
- [TestMethod]
- public void Concat_ReturnReturn()
- {
- var scheduler = new TestScheduler();
- var ex = new Exception();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(220, 2),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(240, 3),
- OnCompleted<int>(250)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnNext(220, 2),
- OnNext(240, 3),
- OnCompleted<int>(250)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 250)
- );
- }
- [TestMethod]
- public void Concat_ThrowReturn()
- {
- var scheduler = new TestScheduler();
- var ex = new Exception();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnError<int>(230, ex)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(240, 2),
- OnCompleted<int>(250)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnError<int>(230, ex)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- );
- }
- [TestMethod]
- public void Concat_ReturnThrow()
- {
- var scheduler = new TestScheduler();
- var ex = new Exception();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(220, 2),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnError<int>(250, ex)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnNext(220, 2),
- OnError<int>(250, ex)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 250)
- );
- }
- [TestMethod]
- public void Concat_SomeDataSomeData()
- {
- var scheduler = new TestScheduler();
- var ex = new Exception();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnNext(220, 3),
- OnCompleted<int>(225)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(230, 4),
- OnNext(240, 5),
- OnCompleted<int>(250)
- );
- var res = scheduler.Start(() =>
- o1.Concat(o2)
- );
- res.Messages.AssertEqual(
- OnNext(210, 2),
- OnNext(220, 3),
- OnNext(230, 4),
- OnNext(240, 5),
- OnCompleted<int>(250)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 225)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(225, 250)
- );
- }
- [TestMethod]
- public void Concat_EnumerableThrows()
- {
- var scheduler = new TestScheduler();
- var o = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnNext(220, 3),
- OnCompleted<int>(225)
- );
- var ex = new Exception();
- var xss = new MockEnumerable<IObservable<int>>(scheduler, GetObservablesForConcatThrow(o, ex));
- var res = scheduler.Start(() =>
- xss.Concat()
- );
- res.Messages.AssertEqual(
- OnNext(210, 2),
- OnNext(220, 3),
- OnError<int>(225, ex)
- );
- o.Subscriptions.AssertEqual(
- Subscribe(200, 225)
- );
- xss.Subscriptions.AssertEqual(
- Subscribe(200, 225)
- );
- }
- private IEnumerable<IObservable<int>> GetObservablesForConcatThrow(IObservable<int> first, Exception ex)
- {
- yield return first;
- throw ex;
- }
- [TestMethod]
- public void Concat_EnumerableTiming()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2), // !
- OnNext(220, 3), // !
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateColdObservable(
- OnNext(50, 4), // !
- OnNext(60, 5), // !
- OnNext(70, 6), // !
- OnCompleted<int>(80)
- );
- var o3 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(200, 2),
- OnNext(210, 3),
- OnNext(220, 4),
- OnNext(230, 5),
- OnNext(270, 6),
- OnNext(320, 7), // !
- OnNext(330, 8), // !
- OnCompleted<int>(340)
- );
- var xss = new MockEnumerable<ITestableObservable<int>>(scheduler, [o1, o2, o3, o2]);
- var res = scheduler.Start(() =>
- xss.Select(xs => (IObservable<int>)xs).Concat()
- );
- res.Messages.AssertEqual(
- OnNext(210, 2),
- OnNext(220, 3),
- OnNext(280, 4),
- OnNext(290, 5),
- OnNext(300, 6),
- OnNext(320, 7),
- OnNext(330, 8),
- OnNext(390, 4),
- OnNext(400, 5),
- OnNext(410, 6),
- OnCompleted<int>(420)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 310),
- Subscribe(340, 420)
- );
- o3.Subscriptions.AssertEqual(
- Subscribe(310, 340)
- );
- xss.Subscriptions.AssertEqual(
- Subscribe(200, 420)
- );
- }
- [TestMethod]
- public void Concat_Enumerable_Dispose()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnNext(220, 3),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(200, 2),
- OnNext(210, 3),
- OnNext(240, 4),
- OnNext(270, 5),
- OnNext(320, 6),
- OnNext(330, 7),
- OnCompleted<int>(340)
- );
- var xss = new MockEnumerable<ITestableObservable<int>>(scheduler, [o1, o2]);
- var res = scheduler.Start(() =>
- xss.Select(xs => (IObservable<int>)xs).Concat(),
- 300
- );
- res.Messages.AssertEqual(
- OnNext(210, 2),
- OnNext(220, 3),
- OnNext(240, 4),
- OnNext(270, 5)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 300)
- );
- xss.Subscriptions.AssertEqual(
- Subscribe(200, 300)
- );
- }
- [TestMethod]
- public void Concat_Optimization_DeferEvalTiming()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnNext(220, 3),
- OnCompleted<int>(230)
- );
- var o2 = scheduler.CreateColdObservable(
- OnNext(10, 4),
- OnNext(20, 5),
- OnNext(30, 6),
- OnCompleted<int>(40)
- );
- var invoked = default(long);
- var xs = o1;
- var ys = Observable.Defer(() => { invoked = scheduler.Clock; return o2; });
- var res = scheduler.Start(() =>
- xs.Concat(ys)
- );
- res.Messages.AssertEqual(
- OnNext(210, 2),
- OnNext(220, 3),
- OnNext(240, 4),
- OnNext(250, 5),
- OnNext(260, 6),
- OnCompleted<int>(270)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 230)
- );
- o2.Subscriptions.AssertEqual(
- Subscribe(230, 270)
- );
- Assert.Equal(230, invoked);
- }
- [TestMethod]
- public void Concat_Optimization_DeferExceptionPropagation()
- {
- var scheduler = new TestScheduler();
- var o1 = scheduler.CreateHotObservable(
- OnNext(150, 1),
- OnNext(210, 2),
- OnCompleted<int>(220)
- );
- var ex = new Exception();
- var invoked = default(long);
- var xs = o1;
- var ys = Observable.Defer(new Func<IObservable<int>>(() => { invoked = scheduler.Clock; throw ex; }));
- var res = scheduler.Start(() =>
- xs.Concat(ys)
- );
- res.Messages.AssertEqual(
- OnNext(210, 2),
- OnError<int>(220, ex)
- );
- o1.Subscriptions.AssertEqual(
- Subscribe(200, 220)
- );
- Assert.Equal(220, invoked);
- }
- #if !NO_PERF
- [TestMethod]
- public void Concat_TailRecursive1()
- {
- var create = 0L;
- var start = 200L;
- var end = 1000L;
- var scheduler = new TestScheduler();
- var o = scheduler.CreateColdObservable(
- OnNext(10, 1),
- OnNext(20, 2),
- OnNext(30, 3),
- OnCompleted<int>(40)
- );
- IObservable<int> f() => Observable.Defer(() => o.Concat(f()));
- var res = scheduler.Start(() => f(), create, start, end);
- var expected = new List<Recorded<Notification<int>>>();
- var t = start;
- while (t <= end)
- {
- var n = (t - start) / 10;
- if (n % 4 != 0)
- {
- expected.Add(OnNext(t, (int)(n % 4)));
- }
- t += 10;
- }
- res.Messages.AssertEqual(expected);
- }
- [TestMethod]
- public void Concat_TailRecursive2()
- {
- IObservable<int> f(int x) => Observable.Defer(() => Observable.Return(x, ThreadPoolScheduler.Instance).Concat(f(x + 1)));
- var lst = new List<int>();
- f(0).Select(x => new StackTrace().FrameCount).Take(10).ForEach(lst.Add);
- Assert.True(lst.Last() - lst.First() < 10);
- }
- #endif
- [TestMethod]
- public void Concat_Task()
- {
- var tss = Observable.Concat(new[] { Task.Factory.StartNew(() => 1), Task.Factory.StartNew(() => 2), Task.Factory.StartNew(() => 3) }.ToObservable());
- var res = tss.ToArray().Single();
- Assert.True(res.SequenceEqual([1, 2, 3]));
- }
- }
- }
|