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
|
/*
* Copyright 2000-2016 Vaadin Ltd.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package com.vaadin.tests.tb3;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import org.junit.runners.Parameterized.Parameters;
import org.junit.runners.model.FrameworkMethod;
import org.junit.runners.model.InitializationError;
/**
* TestBench test runner which supports static @Parameters annotated methods
* providing parameters for the corresponding setter.
* <p>
* {@code @Parameters public static Collection<String> getThemes() } creates one
* permutation for each value returned by {@code getThemes()}. The value is
* automatically assigned to the test instance using {@code setTheme(String)}
* before invoking the test method
*
* @author Vaadin Ltd
*/
public class ParameterizedTB3Runner extends TB3Runner {
public ParameterizedTB3Runner(Class<?> klass) throws InitializationError {
super(klass);
}
@Override
protected List<FrameworkMethod> computeTestMethods() {
List<FrameworkMethod> methods = super.computeTestMethods();
Map<Method, Collection<String>> parameters = new LinkedHashMap<>();
// Find all @Parameters methods and invoke them to find out permutations
for (Method m : getTestClass().getJavaClass().getMethods()) {
Parameters p = m.getAnnotation(Parameters.class);
if (p == null) {
continue;
}
if (!m.getName().startsWith("get") || !m.getName().endsWith("s")) {
throw new IllegalStateException("Method " + m.getName()
+ " is annotated with @Parameter but is not named getSomeThings() as it should");
}
if (m.getParameterTypes().length != 0) {
throw new IllegalStateException("Method " + m.getName()
+ " annotated with @Parameter should not have any arguments");
}
if (!Modifier.isStatic(m.getModifiers())) {
throw new IllegalStateException("Method " + m.getName()
+ " annotated with @Parameter must be static");
}
// getThemes -> setTheme
String setter = "set" + m.getName().substring("get".length());
setter = setter.substring(0, setter.length() - 1);
// property = property.substring(0, 1).toLowerCase(Locale.ROOT)
// + property.substring(1);
Method setterMethod;
try {
setterMethod = getTestClass().getJavaClass().getMethod(setter,
String.class);
} catch (Exception e) {
throw new IllegalStateException(
"No setter " + setter + " found in "
+ getTestClass().getJavaClass().getName(),
e);
}
Collection<String> values;
try {
values = (Collection<String>) m.invoke(null);
if (!values.isEmpty()) {
// Ignore any empty collections to allow e.g. integration
// tests to use "/demo" path by default without adding that
// to the screenshot name
parameters.put(setterMethod, values);
}
} catch (Exception e) {
throw new IllegalStateException(
"The setter " + m.getName() + " could not be invoked",
e);
}
}
// Add method permutations for all @Parameters
for (Method setter : parameters.keySet()) {
List<FrameworkMethod> newMethods = new ArrayList<>();
for (FrameworkMethod m : methods) {
if (!(m instanceof TBMethod)) {
System.err.println(
"Unknown method type: " + m.getClass().getName());
newMethods.add(m);
continue;
}
// testFoo
// testBar
// ->
// testFoo[valo]
// testFoo[runo]
// testBar[valo]
// testBar[runo]
for (final String value : parameters.get(setter)) {
newMethods.add(new TBMethodWithBefore((TBMethod) m, setter,
value));
}
}
// Update methods so next parameters will use all expanded methods
methods = newMethods;
}
return methods;
}
public static class TBMethodWithBefore extends TBMethod {
private Method setter;
private String value;
private TBMethod parent;
public TBMethodWithBefore(TBMethod m, Method setter, String value) {
super(m.getMethod(), m.getCapabilities());
parent = m;
this.setter = setter;
this.value = value;
}
@Override
public Object invokeExplosively(Object target, Object... params)
throws Throwable {
setter.invoke(target, value);
return parent.invokeExplosively(target, params);
}
@Override
public String getName() {
return parent.getName() + "[" + value + "]";
};
@Override
public boolean equals(Object obj) {
if (!TBMethodWithBefore.class.isInstance(obj)) {
return false;
}
TBMethodWithBefore otherTbMethod = (TBMethodWithBefore) obj;
return super.equals(obj)
&& Objects.equals(otherTbMethod.parent, parent)
&& Objects.equals(otherTbMethod.setter, setter)
&& Objects.equals(otherTbMethod.value, value);
}
}
}
|