-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathHierarchicalTopicMappingServiceTest.cs
204 lines (168 loc) · 10.9 KB
/
HierarchicalTopicMappingServiceTest.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
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
/*==============================================================================================================================
| Author Ignia, LLC
| Client Ignia, LLC
| Project Topics Library
\=============================================================================================================================*/
using OnTopic.Data.Caching;
using OnTopic.Mapping.Hierarchical;
using OnTopic.Repositories;
using OnTopic.TestDoubles;
using OnTopic.Tests.Fixtures;
using Xunit;
namespace OnTopic.Tests {
/*============================================================================================================================
| CLASS: HIERARCHICAL TOPIC MAPPING SERVICE TEST
\---------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Provides unit tests for the <see cref="HierarchicalTopicMappingService{T}"/>.
/// </summary>
[ExcludeFromCodeCoverage]
[Xunit.Collection("Shared Repository")]
public class HierarchicalTopicMappingServiceTest: IClassFixture<TopicInfrastructureFixture<StubTopicRepository>> {
/*==========================================================================================================================
| PRIVATE VARIABLES
\-------------------------------------------------------------------------------------------------------------------------*/
readonly ITopicRepository _topicRepository;
readonly Topic _topic;
/*==========================================================================================================================
| HIERARCHICAL TOPIC MAPPING SERVICE
\-------------------------------------------------------------------------------------------------------------------------*/
private readonly IHierarchicalTopicMappingService<NavigationTopicViewModel> _hierarchicalMappingService;
/*==========================================================================================================================
| CONSTRUCTOR
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Initializes a new instance of the <see cref="HierarchicalTopicMappingServiceTest"/> with shared resources.
/// </summary>
/// <remarks>
/// This uses the <see cref="StubTopicRepository"/> to provide data, and then <see cref="CachedTopicRepository"/> to
/// manage the in-memory representation of the data. While this introduces some overhead to the tests, the latter is a
/// relatively lightweight façade to any <see cref="ITopicRepository"/>, and prevents the need to duplicate logic for
/// crawling the object graph. In addition, it initializes a shared <see cref="Topic"/> reference to use for the various
/// tests.
/// </remarks>
public HierarchicalTopicMappingServiceTest(TopicInfrastructureFixture<StubTopicRepository> fixture) {
/*------------------------------------------------------------------------------------------------------------------------
| Validate parameters
\-----------------------------------------------------------------------------------------------------------------------*/
Contract.Requires(fixture, nameof(fixture));
/*------------------------------------------------------------------------------------------------------------------------
| Establish dependencies
\-----------------------------------------------------------------------------------------------------------------------*/
_topicRepository = fixture.CachedTopicRepository;
_topic = _topicRepository.Load("Root:Web:Web_3:Web_3_0")!;
/*------------------------------------------------------------------------------------------------------------------------
| Establish hierarchical topic mapping service
\-----------------------------------------------------------------------------------------------------------------------*/
_hierarchicalMappingService = new CachedHierarchicalTopicMappingService<NavigationTopicViewModel>(
new HierarchicalTopicMappingService<NavigationTopicViewModel>(
fixture.TopicRepository,
fixture.MappingService
)
);
}
/*==========================================================================================================================
| TEST: GET HIERARCHICAL ROOT: WITH NULL TOPIC: RETURNS DEFAULT ROOT
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Calls <see cref="HierarchicalTopicMappingService{T}.GetHierarchicalRoot(Topic?, Int32, String)"/> method with no
/// <c>currentTopic</c> and ensures it falls back to the <c>defaultRoot</c>.
/// </summary>
[Fact]
public void GetHierarchicalRoot_WithNullTopic_ReturnsDefaultRoot() {
var rootTopic = _hierarchicalMappingService.GetHierarchicalRoot(null, 2, "Configuration");
Assert.NotNull(rootTopic);
Assert.Equal("Configuration", rootTopic?.Key);
}
/*==========================================================================================================================
| TEST: GET HIERARCHICAL ROOT: WITH NULL TOPIC: THROWS EXCEPTION
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Calls <see cref="HierarchicalTopicMappingService{T}.GetHierarchicalRoot(Topic?, Int32, String)"/> method with no
/// <c>currentTopic</c> or <c>defaultRoot</c> and ensures it throws an <see cref="ArgumentNullException"/>.
/// </summary>
[Fact]
public void GetHierarchicalRoot_WithNullTopic_ThrowsException() =>
Assert.Throws<ArgumentNullException>(() =>
_hierarchicalMappingService.GetHierarchicalRoot(null, 2, "")
);
/*==========================================================================================================================
| TEST: GET HIERARCHICAL ROOT: WITH INVALID DEFAULT ROOT: TRHOWS EXCEPTION
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Calls <see cref="HierarchicalTopicMappingService{T}.GetHierarchicalRoot(Topic?, Int32, String)"/> method with no
/// <c>currentTopic</c> and throws an <see cref="ArgumentException"/> when it cannot identify the <c>defaultRoot</c>.
/// </summary>
[Fact]
public void GetHierarchicalRoot_WithInvalidDefaultRoot_ThrowsException() =>
Assert.Throws<ArgumentOutOfRangeException>(() =>
_hierarchicalMappingService.GetHierarchicalRoot(null, 2, "InvalidDefaultRoot")
);
/*==========================================================================================================================
| TEST: GET HIERARCHICAL ROOT: WITH DEEP TOPIC: RETURNS ROOT
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Calls <see cref="HierarchicalTopicMappingService{T}.GetHierarchicalRoot(Topic?, Int32, String)"/> method with a deeply
/// nested topic and ensures that it returns the expected root.
/// </summary>
[Fact]
public void GetHierarchicalRoot_WithDeepTopic_ReturnsRoot() {
var rootTopic = _hierarchicalMappingService.GetHierarchicalRoot(_topic, 2, "Configuration");
Assert.NotNull(rootTopic);
Assert.Equal("Web", rootTopic?.Key);
}
/*==========================================================================================================================
| TEST: GET VIEW MODEL: WITH TWO LEVELS: RETURNS GRAPH
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Calls <see cref="HierarchicalTopicMappingService{T}.GetViewModelAsync(Topic?, Int32, Func{Topic, Boolean}?)"/> method
/// and ensures that the expected data is returned.
/// </summary>
[Fact]
public async Task GetViewModel_WithTwoLevels_ReturnsGraph() {
var rootTopic = _topicRepository.Load("Root:Web");
var viewModel = await _hierarchicalMappingService.GetViewModelAsync(rootTopic, 1).ConfigureAwait(false);
Assert.NotNull(viewModel);
Assert.Equal(3, viewModel?.Children.Count);
Assert.Empty(viewModel?.Children[0].Children);
}
/*==========================================================================================================================
| TEST: GET VIEW MODEL: WITH VALIDATION DELEGATE: EXCLUDES TOPICS
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Calls <see cref="HierarchicalTopicMappingService{T}.GetViewModelAsync(Topic?, Int32, Func{Topic, Boolean}?)"/> method
/// with a <c>validationDelegate</c> and ensures that it correctly trims the topic graph.
/// </summary>
[Fact]
public async Task GetViewModel_WithValidationDelegate_ExcludesTopics() {
var rootTopic = _topicRepository.Load("Root:Web");
var viewModel = await _hierarchicalMappingService
.GetViewModelAsync(rootTopic, 2, (t) => t.Key.EndsWith("1", StringComparison.Ordinal))
.ConfigureAwait(false);
Assert.NotNull(viewModel);
Assert.Single(viewModel?.Children);
Assert.Single(viewModel?.Children[0].Children);
}
/*==========================================================================================================================
| TEST: GET VIEW MODEL: WITH DISABLED: EXCLUDES DISABLED
\-------------------------------------------------------------------------------------------------------------------------*/
/// <summary>
/// Calls <see cref="HierarchicalTopicMappingService{T}.GetViewModelAsync(Topic?, Int32, Func{Topic, Boolean}?)"/> method
/// with a <see cref="Topic.IsDisabled"/> topic in the graph, and ensures it is not returned.
/// </summary>
[Fact]
public async Task GetViewModel_WithDisabled_ExcludesDisabled() {
var rootTopic = _topicRepository.Load("Root:Web:Web_3")!;
var disabledTopic = _topicRepository.Load("Root:Web:Web_3:Web_3_0");
Contract.Assume(disabledTopic);
rootTopic.IsDisabled = true;
disabledTopic.IsDisabled = true;
var viewModel = await _hierarchicalMappingService.GetViewModelAsync(rootTopic, 1).ConfigureAwait(false);
Assert.NotNull(viewModel);
Assert.Single(viewModel?.Children);
//Revert state
rootTopic.IsDisabled = false;
disabledTopic.IsDisabled = false;
}
} //Class
} //Namespace