forked from MonoGame/MonoGame
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathContextScopeFactoryTests.cs
134 lines (119 loc) · 4.84 KB
/
ContextScopeFactoryTests.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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
// MonoGame - Copyright (C) MonoGame Foundation, Inc
// This file is subject to the terms and conditions defined in
// file 'LICENSE.txt', which is part of this source code package.
using System;
using System.Collections.Generic;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.Xna.Framework.Content.Pipeline;
using MonoGame.Framework.Content;
using NUnit.Framework;
namespace MonoGame.Tools.Tests;
public class ContextScopeFactoryTests
{
/// <summary>
/// This test is attempting to show that multiple tasks can inherit the active context
/// from a top-level task.
/// </summary>
[Test]
public async Task ContextScopeFactory_HierarchicalAccess()
{
var tasks = new List<Task>();
long errorCount = 0;
var ctx = new NoOpContext(0);
ContextScopeFactory.BeginContext(ctx);
for (var i = 0; i < 100; i++)
{
var task = Task.Run(async () =>
{
try
{
await Task.Delay(10);
// get the context and make sure it is the
// the scope from the original task
var retrievedCtx = ContextScopeFactory.ActiveContext;
Assert.That(ctx, Is.EqualTo(retrievedCtx));
}
catch (Exception ex)
{
Interlocked.Increment(ref errorCount);
Console.WriteLine($"exception {ex.GetType().Name} - {ex.Message} \n {ex.StackTrace}");
}
});
tasks.Add(task);
}
await Task.WhenAll(tasks);
ctx.Dispose();
Assert.That(errorCount, Is.EqualTo(0), "there were exceptions thrown in the task looping.");
}
/// <summary>
/// This test is trying to show that
/// - there can be an active context _per_ Task, and that if
/// code fetches the active context, it'll find the one on
/// the current task. This is useful if/when the Content Builder
/// is running multiple import jobs at once
/// - there can still be nested contexts. If you start an import-job,
/// and then have a sub-task to import a sub-asset, the ActiveContext
/// will use the sub-task's context and then switch back to the top
/// level one
/// </summary>
[Test]
public async Task ContextScopeFactory_AsyncAccessNestedAccess()
{
var tasks = new List<Task>();
long errorCount = 0;
for (var i = 0; i < 100; i++)
{
var capturedIndex = i;
var task = Task.Run(async () =>
{
try
{
// spawn a context to be used
using var ctx = new NoOpContext(capturedIndex);
ContextScopeFactory.BeginContext(ctx);
// wait some "arbitrary" amount of time
await Task.Delay(10);
using (var nestedCtx = new NoOpContext(capturedIndex * 100))
{
ContextScopeFactory.BeginContext(nestedCtx);
// wait some more "arbitrary" amount of time
await Task.Delay(10);
// get the context and make sure it is the same as the one we expect
var retrievedNestedCtx = ContextScopeFactory.ActiveContext;
Assert.That(nestedCtx, Is.EqualTo(retrievedNestedCtx));
}
// get the context and make sure it is the same as the one we expect
var retrievedCtx = ContextScopeFactory.ActiveContext;
Assert.That(ctx, Is.EqualTo(retrievedCtx));
}
catch (Exception ex)
{
Interlocked.Increment(ref errorCount);
Console.WriteLine($"exception {ex.GetType().Name} - {ex.Message} \n {ex.StackTrace}");
}
});
tasks.Add(task);
}
await Task.WhenAll(tasks);
Assert.That(errorCount, Is.EqualTo(0), "there were exceptions thrown in the task looping.");
}
/// <summary>
/// This is a dummy class to act as a stand-in for tests that
/// just need to check the _identity_ of the context being used
/// </summary>
class NoOpContext : ContextScopeFactory.ContextScope
{
public readonly int Id;
public override string IntermediateDirectory { get; }
public override ContentBuildLogger Logger { get; }
public override ContentIdentity SourceIdentity { get; }
public NoOpContext(int id)
{
Id = id;
}
// helpful during debugging or broken tests to be able
// to see that one ToString()'d instance is different than another
public override string ToString() => "test-ctx-" + Id;
}
}