forked from Lilliklave/Hundregistret
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHR2_2_DogNameComparatorTest.java
254 lines (212 loc) · 9.8 KB
/
HR2_2_DogNameComparatorTest.java
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
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
import static org.junit.jupiter.api.Assertions.*;
import java.util.Arrays;
import org.junit.jupiter.api.*;
import org.junit.jupiter.api.MethodOrderer.OrderAnnotation;
/**
* Testfall för jämförelsefunktionen för namn i uppgift HR2.2.
* <p>
* Beskrivningen av testfallens uppgift, styrka och svagheter från
* <code>{@link HR1_1_OwnerTest}</code> gäller (naturligvis) också för
* testfallen i denna klass. Var speciellt uppmärksam på att testfallen kan
* uppdateras när som helst, inklusive <em>efter</em> deadline.
*
* @author Henrik Bergström
* @version 2023-12-11 14:56
* @see HR1_1_OwnerTest
*/
@TestMethodOrder(OrderAnnotation.class)
@DisplayName("HR2.2: Testfall för jämförelsefunktionen för namn")
public class HR2_2_DogNameComparatorTest {
public static final String VERSION = "2023-12-11 14:56";
private static final String DEFAULT_BREED = "Breed";
private static final int DEFAULT_AGE = 3;
private static final int DEFAULT_WEIGHT = 7;
private static final Dog FIRST_DOG_IN_ALPHABETIC_ORDER = new Dog("Fido", DEFAULT_BREED, DEFAULT_AGE,
DEFAULT_WEIGHT);
private static final Dog SECOND_DOG_IN_ALPHABETIC_ORDER = new Dog("Karo", DEFAULT_BREED, DEFAULT_AGE,
DEFAULT_WEIGHT);
private static final Dog ANY_DOG = FIRST_DOG_IN_ALPHABETIC_ORDER;
@Test
@Order(10)
@DisplayName("Implementerad enligt instruktionerna")
public void validateImplementation() {
new DogNameComparatorImplementationValidator().execute();
}
@Test
@Order(20)
@DisplayName("Jämförelse med samma hund ger resultatet 0")
public void aDogIsEqualToItSelf() {
DogNameComparator sut = new DogNameComparator();
assertEquals(0, sut.compare(ANY_DOG, ANY_DOG));
}
@Test
@Order(30)
@DisplayName("Jämförelse med annan hund med samma namn ger resultatet 0")
public void aDogIsEqualToAnotherDogWithTheSameName() {
DogNameComparator sut = new DogNameComparator();
assertEquals(0, sut.compare(ANY_DOG,
new Dog(new String(ANY_DOG.getName()), DEFAULT_BREED, DEFAULT_AGE, DEFAULT_WEIGHT)));
}
@Test
@Order(40)
@DisplayName("Jämförelse med ett namn tidigare i bokstavsordning och ett efter ger ett resultat under 0")
public void theFirstDogComesBeforeTheSecondInAlphabeticOrder() {
DogNameComparator sut = new DogNameComparator();
assertTrue(sut.compare(FIRST_DOG_IN_ALPHABETIC_ORDER, SECOND_DOG_IN_ALPHABETIC_ORDER) < 0);
}
@Test
@Order(50)
@DisplayName("Jämförelse med ett namn efter i bokstavsordning och ett tidigare ger ett resultat över 0")
public void theFirstDogComesAfterTheSecondInAlphabeticOrder() {
DogNameComparator sut = new DogNameComparator();
assertTrue(sut.compare(SECOND_DOG_IN_ALPHABETIC_ORDER, FIRST_DOG_IN_ALPHABETIC_ORDER) > 0);
}
@Test
@Order(60)
@DisplayName("Jämförelse med en kortare och en längre version av samma namn ger ett resultat under 0")
public void theFirstDogHasShorterVersionOfTheNameOfTheSecond() {
DogNameComparator sut = new DogNameComparator();
Dog fido = new Dog("Fido", DEFAULT_BREED, DEFAULT_AGE, DEFAULT_WEIGHT);
Dog fidolina = new Dog("Fidolina", DEFAULT_BREED, DEFAULT_AGE, DEFAULT_WEIGHT);
assertTrue(sut.compare(fido, fidolina) < 0);
}
@Test
@Order(70)
@DisplayName("Jämförelse med en längre och en kortare version av samma namn ger ett resultat över 0")
public void theFirstDogHasLongerVersionOfTheNameOfTheSecond() {
DogNameComparator sut = new DogNameComparator();
Dog fido = new Dog("Fido", DEFAULT_BREED, DEFAULT_AGE, DEFAULT_WEIGHT);
Dog fidolina = new Dog("Fidolina", DEFAULT_BREED, DEFAULT_AGE, DEFAULT_WEIGHT);
assertTrue(sut.compare(fidolina, fido) > 0);
}
@Test
@Order(80)
@DisplayName("Normaliserade namn används vid jämförelsen")
public void normalizedNamesUsed() {
DogNameComparator sut = new DogNameComparator();
assertEquals(0, sut.compare(ANY_DOG,
new Dog(ANY_DOG.getName().toLowerCase(), DEFAULT_BREED, DEFAULT_AGE, DEFAULT_WEIGHT)));
}
@Test
@Order(90)
@DisplayName("Sortera hundar med hjälp av comparatorn")
public void sortDogsUsingComparator() {
var bamse = new Dog("Bamse", "Dachshund", 2, 4); // svans=3,7
var doris = new Dog("Doris", "Pudel", 20, 11); // svans=22,0
var fido = new Dog("Fido", "Tax", 2, 15); // svans=3,7
var karo = new Dog("Karo", "Dachshund", 17, 20); // svans=3,7
var lassie = new Dog("Lassie", "Shih tzu", 20, 8); // svans=16,0
var milou = new Dog("Milou", "Grand danois", 12, 8); // svans=9,6
var ratata = new Dog("Ratata", "Boxer", 5, 15); // svans=7,5
var rex = new Dog("Rex", "Boxer", 10, 7); // svans=7,0
var snobben = new Dog("Snobben", "Puli", 15, 9); // svans=13,5
var wilma = new Dog("Wilma", "Golden retriever", 16, 15); // svans=24,0
Dog[] expected = { bamse, doris, fido, karo, lassie, milou, ratata, rex, snobben, wilma };
Dog[] actual = { doris, snobben, wilma, lassie, milou, bamse, ratata, fido, karo, rex };
var sut = new DogNameComparator();
Arrays.sort(actual, sut);
assertArrayEquals(expected, actual);
}
/**
* Denna record (klass) används för att kontrollera att bara de förväntade
* metoderna finns i det publika gränssnittet i de klasser du implementerat.
* Kopior av MethodHeader finns i de flesta testfalls-filerna. Duplicerad kod är
* normalt en dålig idé eftersom det leder till svårigheter med underhåll när
* man måste hitta alla ställen koden finns på och applicera ändringar överallt.
* <p>
* Här är detta acceptabelt eftersom det gör att varje testklass innehåller allt
* som behövs för att kunna köra den, och för att alla ändringar automatiskt
* läggs in i alla klasser.
*/
public record MethodHeader(boolean isStatic, String returnType, java.util.regex.Pattern namePattern,
String[] parameters) {
public MethodHeader(boolean isStatic, String returnType, String namePattern, String... parameters) {
this(isStatic, returnType, java.util.regex.Pattern.compile(namePattern), parameters);
}
public boolean matches(java.lang.reflect.Method m) {
if (isStatic != java.lang.reflect.Modifier.isStatic(m.getModifiers()))
return false;
if (!returnType.equals(m.getReturnType().getSimpleName()))
return false;
if (!namePattern.matcher(m.getName()).matches())
return false;
var actualParams = java.util.stream.Stream.of(m.getParameterTypes()).map(p -> p.getSimpleName()).toList()
.toArray(new String[] {});
if (!java.util.Arrays.equals(parameters, actualParams))
return false;
return true;
}
@Override
public String toString() {
return "%s%s %s%s".formatted(isStatic ? "static " : "", returnType, namePattern,
java.util.Arrays.toString(parameters));
}
}
// @formatter:off
/**
* Denna klass är automatiskt genererad. Ändringar i den kommer att skrivas
* över vid nästa uppdatering.
*/
public class DogNameComparatorImplementationValidator {
private final Class<?> cut = DogNameComparator.class;
private static final java.util.Collection<MethodHeader> EXPECTED_PUBLIC_METHODS = new java.util.ArrayList<>();
/**
* Denna lista är automatiskt genererad från en referens-implementation.
* Ordningen på metoderna är fast, men har inget med den förväntade ordningen
* i klassen att göra.
*/
static {
EXPECTED_PUBLIC_METHODS.add(new MethodHeader(false, "int", "compare", "Dog", "Dog"));
EXPECTED_PUBLIC_METHODS.add(new MethodHeader(false, "int", "compare", "Object", "Object"));
}
public void execute() {
org.junit.jupiter.api.Assertions.assertAll(this::onlyPrivateFields, this::allConstructorsPublicOrPrivate,
this::expectedNumberOfPublicConstructors, this::allMethodsPublicOrPrivate, this::onlyExpectedPublicMethods);
}
private void onlyPrivateFields() {
for (var f : cut.getDeclaredFields()) {
org.junit.jupiter.api.Assertions.assertTrue(isPrivate(f),
"Fältet %s i %s har inte en korrekt skyddsnivå. Endast privat är tillåten.".formatted(f.getName(),
cut.getSimpleName()));
}
}
private void expectedNumberOfPublicConstructors() {
org.junit.jupiter.api.Assertions.assertEquals(1, cut.getDeclaredConstructors().length,
"Fel antal publika konstruktorer i klassen %s".formatted(cut.getSimpleName()));
}
private void allConstructorsPublicOrPrivate() {
for (var c : cut.getDeclaredConstructors()) {
org.junit.jupiter.api.Assertions.assertTrue(isPublic(c) || isPrivate(c),
"Konstruktorn %s i %s har inte en korrekt skyddsnivå. Endast publik eller privat är tillåtna."
.formatted(c.getName(), cut.getSimpleName()));
}
}
private void allMethodsPublicOrPrivate() {
for (var m : cut.getDeclaredMethods()) {
org.junit.jupiter.api.Assertions.assertTrue(isPublic(m) || isPrivate(m),
"Metoden %s i %s har inte en korrekt skyddsnivå. Endast publik eller privat är tillåtna."
.formatted(m.getName(), cut.getSimpleName()));
}
}
public void onlyExpectedPublicMethods() {
var matches = unexpectedPublicMethods();
org.junit.jupiter.api.Assertions.assertTrue(matches.isEmpty(),
"Det finns publika metoder i %s som testen inte förväntar sig:\n %s\n".formatted(cut.getSimpleName(),
String.join("\n ", matches.stream().map(m -> m.toString()).toList())));
}
private java.util.List<java.lang.reflect.Method> unexpectedPublicMethods() {
return java.util.stream.Stream.of(cut.getDeclaredMethods())
.filter(m -> isPublic(m) && !isExpectedPublicMethod(m)).toList();
}
private boolean isExpectedPublicMethod(java.lang.reflect.Method m) {
return EXPECTED_PUBLIC_METHODS.stream().anyMatch(mh -> mh.matches(m));
}
private static boolean isPublic(java.lang.reflect.Member m) {
return java.lang.reflect.Modifier.isPublic(m.getModifiers());
}
private static boolean isPrivate(java.lang.reflect.Member m) {
return java.lang.reflect.Modifier.isPrivate(m.getModifiers());
}
}
// @formatter:on
}