-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathMergeTests.cs
110 lines (85 loc) · 2.76 KB
/
MergeTests.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
namespace Open.ChannelExtensions.Tests;
public static class MergeTests
{
const int Total = 3000000;
const int Bounds = 100;
const int Count = 5;
private static Channel<int>[] GetChannels()
=> Enumerable.Range(0, Count).Select(_ => Channel.CreateBounded<int>(Bounds)).ToArray();
private static async Task BasicMergeTestCore(ChannelWriter<int>[] writers, ValueTask<List<int>> merging)
{
// Act
await Parallel.ForAsync(0, Total,
(i, token) => writers[i % Count].WriteAsync(i, token));
foreach (ChannelWriter<int> writer in writers)
writer.Complete();
List<int> merged = await merging;
merged.Sort();
// Assert
Assert.Equal(Total, merged.Count);
Assert.True(Enumerable.Range(0, Total).SequenceEqual(merged));
}
[Fact()]
public static async Task BasicMergeTest()
{
// 3 channels
Channel<int>[] c = GetChannels();
// 3 writers
ChannelWriter<int>[] writers = c.Select(e => e.Writer).ToArray();
// 3 readers
ValueTask<List<int>> merging = c.Select(e => e.Reader).Merge().ToListAsync(Total);
await BasicMergeTestCore(writers, merging);
}
[Fact()]
public static async Task MergeChainTest()
{
// 3 channels
Channel<int>[] c = GetChannels();
// 3 writers
ChannelWriter<int>[] writers = c.Select(e => e.Writer).ToArray();
ChannelReader<int> reader = c[0].Reader;
for (int i = 1; i < c.Length; i++)
reader = reader.Merge(c[i].Reader);
// 3 readers
ValueTask<List<int>> merging = reader.ToListAsync(Total);
await BasicMergeTestCore(writers, merging);
}
[Fact()]
public static async Task MergeChainTest2()
{
// 3 channels
Channel<int>[] c = GetChannels();
// 3 writers
ChannelWriter<int>[] writers = c.Select(e => e.Writer).ToArray();
MergingChannelReader<int> reader = c[0].Reader.Merge(c[1].Reader, c.Skip(2).Select(e => e.Reader).ToArray());
for (int i = 1; i < c.Length; i++)
reader = reader.Merge(c[i].Reader);
// 3 readers
ValueTask<List<int>> merging = reader.ToListAsync(Total);
await BasicMergeTestCore(writers, merging);
}
[Fact()]
public static async Task ExceptionPropagationTest()
{
// 3 channels
Channel<int>[] c = GetChannels();
// 3 writers
ChannelWriter<int>[] writers = c.Select(e => e.Writer).ToArray();
// 3 readers
MergingChannelReader<int> merging = c.Select(e => e.Reader).Merge();
ValueTask<List<int>> list = merging.ToListAsync(Total);
// Act
await Assert.ThrowsAsync<ChannelClosedException>(() => Parallel.ForAsync(0, Total,
async (i, token) =>
{
ChannelWriter<int> w = writers[i % 3];
if (i == Total / 2)
w.Complete(new Exception("Test"));
else
await w.WriteAsync(i, token).ConfigureAwait(false);
}));
// Assert
await Assert.ThrowsAsync<Exception>(list.AsTask);
await Assert.ThrowsAsync<Exception>(() => merging.Completion);
}
}