1 | // License: GPL. For details, see LICENSE file.
|
---|
2 | package org.openstreetmap.josm.data.validation.tests;
|
---|
3 |
|
---|
4 | import static org.junit.Assert.assertEquals;
|
---|
5 | import static org.junit.Assert.assertFalse;
|
---|
6 | import static org.junit.Assert.assertNotNull;
|
---|
7 | import static org.junit.Assert.assertTrue;
|
---|
8 |
|
---|
9 | import java.io.InputStream;
|
---|
10 | import java.io.StringReader;
|
---|
11 | import java.util.Collection;
|
---|
12 | import java.util.HashSet;
|
---|
13 | import java.util.Iterator;
|
---|
14 | import java.util.LinkedHashSet;
|
---|
15 | import java.util.List;
|
---|
16 | import java.util.Set;
|
---|
17 |
|
---|
18 | import org.junit.Ignore;
|
---|
19 | import org.junit.Rule;
|
---|
20 | import org.junit.Test;
|
---|
21 | import org.openstreetmap.josm.TestUtils;
|
---|
22 | import org.openstreetmap.josm.command.ChangePropertyCommand;
|
---|
23 | import org.openstreetmap.josm.command.ChangePropertyKeyCommand;
|
---|
24 | import org.openstreetmap.josm.command.Command;
|
---|
25 | import org.openstreetmap.josm.command.PseudoCommand;
|
---|
26 | import org.openstreetmap.josm.command.SequenceCommand;
|
---|
27 | import org.openstreetmap.josm.data.osm.DataSet;
|
---|
28 | import org.openstreetmap.josm.data.osm.OsmPrimitive;
|
---|
29 | import org.openstreetmap.josm.data.osm.OsmUtils;
|
---|
30 | import org.openstreetmap.josm.data.validation.Severity;
|
---|
31 | import org.openstreetmap.josm.data.validation.TestError;
|
---|
32 | import org.openstreetmap.josm.data.validation.tests.MapCSSTagChecker.ParseResult;
|
---|
33 | import org.openstreetmap.josm.data.validation.tests.MapCSSTagChecker.TagCheck;
|
---|
34 | import org.openstreetmap.josm.gui.mappaint.MapPaintStyles;
|
---|
35 | import org.openstreetmap.josm.gui.mappaint.mapcss.MapCSSStyleSource;
|
---|
36 | import org.openstreetmap.josm.gui.mappaint.mapcss.parsergen.ParseException;
|
---|
37 | import org.openstreetmap.josm.io.OsmReader;
|
---|
38 | import org.openstreetmap.josm.testutils.JOSMTestRules;
|
---|
39 | import org.openstreetmap.josm.tools.Logging;
|
---|
40 |
|
---|
41 | import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
|
---|
42 |
|
---|
43 | /**
|
---|
44 | * JUnit Test of {@link MapCSSTagChecker}.
|
---|
45 | */
|
---|
46 | public class MapCSSTagCheckerTest {
|
---|
47 |
|
---|
48 | /**
|
---|
49 | * Setup test.
|
---|
50 | */
|
---|
51 | @Rule
|
---|
52 | @SuppressFBWarnings(value = "URF_UNREAD_PUBLIC_OR_PROTECTED_FIELD")
|
---|
53 | public JOSMTestRules test = new JOSMTestRules().projection().platform();
|
---|
54 |
|
---|
55 | static MapCSSTagChecker buildTagChecker(String css) throws ParseException {
|
---|
56 | final MapCSSTagChecker test = new MapCSSTagChecker();
|
---|
57 | test.checks.putAll("test", TagCheck.readMapCSS(new StringReader(css)).parseChecks);
|
---|
58 | return test;
|
---|
59 | }
|
---|
60 |
|
---|
61 | /**
|
---|
62 | * Test {@code natural=marsh}.
|
---|
63 | * @throws ParseException if a parsing error occurs
|
---|
64 | */
|
---|
65 | @Test
|
---|
66 | public void testNaturalMarsh() throws ParseException {
|
---|
67 | ParseResult result = TagCheck.readMapCSS(new StringReader(
|
---|
68 | "*[natural=marsh] {\n" +
|
---|
69 | " group: tr(\"deprecated\");\n" +
|
---|
70 | " throwWarning: tr(\"{0}={1} is deprecated\", \"{0.key}\", tag(\"natural\"));\n" +
|
---|
71 | " fixRemove: \"{0.key}\";\n" +
|
---|
72 | " fixAdd: \"natural=wetland\";\n" +
|
---|
73 | " fixAdd: \"wetland=marsh\";\n" +
|
---|
74 | "}"));
|
---|
75 | final List<TagCheck> checks = result.parseChecks;
|
---|
76 | assertEquals(1, checks.size());
|
---|
77 | assertTrue(result.parseErrors.isEmpty());
|
---|
78 | final TagCheck check = checks.get(0);
|
---|
79 | assertNotNull(check);
|
---|
80 | assertEquals("{0.key}=null is deprecated", check.getDescription(null));
|
---|
81 | assertEquals("fixRemove: {0.key}", check.fixCommands.get(0).toString());
|
---|
82 | assertEquals("fixAdd: natural=wetland", check.fixCommands.get(1).toString());
|
---|
83 | assertEquals("fixAdd: wetland=marsh", check.fixCommands.get(2).toString());
|
---|
84 | final OsmPrimitive n1 = OsmUtils.createPrimitive("node natural=marsh");
|
---|
85 | final OsmPrimitive n2 = OsmUtils.createPrimitive("node natural=wood");
|
---|
86 | new DataSet(n1, n2);
|
---|
87 | assertTrue(check.test(n1));
|
---|
88 | assertEquals("deprecated", check.getErrorForPrimitive(n1).getMessage());
|
---|
89 | assertEquals("natural=marsh is deprecated", check.getErrorForPrimitive(n1).getDescription());
|
---|
90 | assertEquals(Severity.WARNING, check.getErrorForPrimitive(n1).getSeverity());
|
---|
91 | assertEquals("Sequence: Fix of natural=marsh is deprecated", check.fixPrimitive(n1).getDescriptionText());
|
---|
92 | assertEquals("{natural=}", ((ChangePropertyCommand) check.fixPrimitive(n1).getChildren().iterator().next()).getTags().toString());
|
---|
93 | assertFalse(check.test(n2));
|
---|
94 | assertEquals("The key is natural and the value is marsh",
|
---|
95 | TagCheck.insertArguments(check.rule.selectors.get(0), "The key is {0.key} and the value is {0.value}", null));
|
---|
96 | }
|
---|
97 |
|
---|
98 | /**
|
---|
99 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/10913">Bug #10913</a>.
|
---|
100 | * @throws ParseException if a parsing error occurs
|
---|
101 | */
|
---|
102 | @Test
|
---|
103 | public void testTicket10913() throws ParseException {
|
---|
104 | final OsmPrimitive p = OsmUtils.createPrimitive("way highway=tertiary construction=yes");
|
---|
105 | final TagCheck check = TagCheck.readMapCSS(new StringReader("way {" +
|
---|
106 | "throwError: \"error\";" +
|
---|
107 | "fixChangeKey: \"highway => construction\";\n" +
|
---|
108 | "fixAdd: \"highway=construction\";\n" +
|
---|
109 | "}")).parseChecks.get(0);
|
---|
110 | new DataSet(p);
|
---|
111 | final Command command = check.fixPrimitive(p);
|
---|
112 | assertTrue(command instanceof SequenceCommand);
|
---|
113 | final Iterator<PseudoCommand> it = command.getChildren().iterator();
|
---|
114 | assertTrue(it.next() instanceof ChangePropertyKeyCommand);
|
---|
115 | assertTrue(it.next() instanceof ChangePropertyCommand);
|
---|
116 | }
|
---|
117 |
|
---|
118 | /**
|
---|
119 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/9782">Bug #9782</a>.
|
---|
120 | * @throws ParseException if a parsing error occurs
|
---|
121 | */
|
---|
122 | @Test
|
---|
123 | public void testTicket9782() throws ParseException {
|
---|
124 | final MapCSSTagChecker test = buildTagChecker("*[/.+_name/][!name] {" +
|
---|
125 | "throwWarning: tr(\"has {0} but not {1}\", \"{0.key}\", \"{1.key}\");}");
|
---|
126 | final OsmPrimitive p = OsmUtils.createPrimitive("way alt_name=Foo");
|
---|
127 | final Collection<TestError> errors = test.getErrorsForPrimitive(p, false);
|
---|
128 | assertEquals(1, errors.size());
|
---|
129 | assertEquals("has alt_name but not name", errors.iterator().next().getMessage());
|
---|
130 | assertEquals("3000_*[.+_name][!name]", errors.iterator().next().getIgnoreSubGroup());
|
---|
131 | }
|
---|
132 |
|
---|
133 | /**
|
---|
134 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/10859">Bug #10859</a>.
|
---|
135 | * @throws ParseException if a parsing error occurs
|
---|
136 | */
|
---|
137 | @Test
|
---|
138 | public void testTicket10859() throws ParseException {
|
---|
139 | final MapCSSTagChecker test = buildTagChecker("way[highway=footway][foot?!] {\n" +
|
---|
140 | " throwWarning: tr(\"{0} used with {1}\", \"{0.value}\", \"{1.tag}\");}");
|
---|
141 | final OsmPrimitive p = OsmUtils.createPrimitive("way highway=footway foot=no");
|
---|
142 | final Collection<TestError> errors = test.getErrorsForPrimitive(p, false);
|
---|
143 | assertEquals(1, errors.size());
|
---|
144 | assertEquals("footway used with foot=no", errors.iterator().next().getMessage());
|
---|
145 | assertEquals("3000_way[highway=footway][foot]", errors.iterator().next().getIgnoreSubGroup());
|
---|
146 | }
|
---|
147 |
|
---|
148 | /**
|
---|
149 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/13630">Bug #13630</a>.
|
---|
150 | * @throws ParseException if a parsing error occurs
|
---|
151 | */
|
---|
152 | @Test
|
---|
153 | public void testTicket13630() throws ParseException {
|
---|
154 | ParseResult result = TagCheck.readMapCSS(new StringReader(
|
---|
155 | "node[crossing=zebra] {fixRemove: \"crossing=zebra\";}"));
|
---|
156 | assertTrue(result.parseChecks.isEmpty());
|
---|
157 | assertEquals(1, result.parseErrors.size());
|
---|
158 | }
|
---|
159 |
|
---|
160 | /**
|
---|
161 | * Unit test of {@code min-josm-version} processing.
|
---|
162 | * @throws ParseException if a parsing error occurs
|
---|
163 | */
|
---|
164 | @Test
|
---|
165 | public void testPreprocessing() throws ParseException {
|
---|
166 | final MapCSSTagChecker test = buildTagChecker("" +
|
---|
167 | "@supports (min-josm-version: 1) { *[foo] { throwWarning: \"!\"; } }\n" +
|
---|
168 | "@supports (min-josm-version: 2147483647) { *[bar] { throwWarning: \"!\"; } }\n");
|
---|
169 | assertEquals(1, test.getErrorsForPrimitive(OsmUtils.createPrimitive("way foo=1"), false).size());
|
---|
170 | assertEquals(0, test.getErrorsForPrimitive(OsmUtils.createPrimitive("way bar=1"), false).size());
|
---|
171 | }
|
---|
172 |
|
---|
173 | /**
|
---|
174 | * Unit test of {@link MapCSSTagChecker#initialize}.
|
---|
175 | * @throws Exception if an error occurs
|
---|
176 | */
|
---|
177 | @Test
|
---|
178 | public void testInit() throws Exception {
|
---|
179 | MapCSSTagChecker c = new MapCSSTagChecker();
|
---|
180 | c.initialize();
|
---|
181 |
|
---|
182 | Set<String> assertionErrors = new LinkedHashSet<>();
|
---|
183 | for (Set<TagCheck> schecks : c.checks.values()) {
|
---|
184 | assertionErrors.addAll(c.checkAsserts(schecks));
|
---|
185 | }
|
---|
186 | for (String msg : assertionErrors) {
|
---|
187 | Logging.error(msg);
|
---|
188 | }
|
---|
189 | assertTrue("not all assertions included in the tests are met", assertionErrors.isEmpty());
|
---|
190 | }
|
---|
191 |
|
---|
192 | /**
|
---|
193 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/13762">Bug #13762</a>.
|
---|
194 | * @throws ParseException if a parsing error occurs
|
---|
195 | */
|
---|
196 | @Test
|
---|
197 | public void testTicket13762() throws ParseException {
|
---|
198 | final ParseResult parseResult = TagCheck.readMapCSS(new StringReader("" +
|
---|
199 | "meta[lang=de] {\n" +
|
---|
200 | " title: \"Deutschlandspezifische Regeln\";" +
|
---|
201 | "}"));
|
---|
202 | assertTrue(parseResult.parseErrors.isEmpty());
|
---|
203 | }
|
---|
204 |
|
---|
205 | /**
|
---|
206 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/14287">Bug #14287</a>.
|
---|
207 | * @throws Exception if an error occurs
|
---|
208 | */
|
---|
209 | @Test
|
---|
210 | @Ignore("not fixed yet")
|
---|
211 | public void testTicket14287() throws Exception {
|
---|
212 | final MapCSSTagChecker test = buildTagChecker(
|
---|
213 | "node[amenity=parking] ∈ *[amenity=parking] {" +
|
---|
214 | " throwWarning: tr(\"{0} inside {1}\", \"amenity=parking\", \"amenity=parking\");" +
|
---|
215 | "}");
|
---|
216 | try (InputStream is = TestUtils.getRegressionDataStream(14287, "example.osm")) {
|
---|
217 | test.visit(OsmReader.parseDataSet(is, null).allPrimitives());
|
---|
218 | assertEquals(6, test.getErrors().size());
|
---|
219 | }
|
---|
220 | }
|
---|
221 |
|
---|
222 | private void doTestNaturalWood(int ticket, String filename, int errorsCount, int setsCount) throws Exception {
|
---|
223 | final MapCSSTagChecker test = buildTagChecker(
|
---|
224 | "area:closed:areaStyle[tag(\"natural\") = parent_tag(\"natural\")] ⧉ area:closed:areaStyle[natural] {" +
|
---|
225 | " throwWarning: tr(\"Overlapping Identical Natural Areas\");" +
|
---|
226 | "}");
|
---|
227 | final MapCSSStyleSource style = new MapCSSStyleSource(
|
---|
228 | "area[natural=wood] {" +
|
---|
229 | " fill-color: woodarea#008000;" +
|
---|
230 | "}");
|
---|
231 | MapPaintStyles.addStyle(style);
|
---|
232 | try (InputStream is = TestUtils.getRegressionDataStream(ticket, filename)) {
|
---|
233 | test.visit(OsmReader.parseDataSet(is, null).allPrimitives());
|
---|
234 | List<TestError> errors = test.getErrors();
|
---|
235 | assertEquals(errorsCount, errors.size());
|
---|
236 | Set<Set<OsmPrimitive>> primitives = new HashSet<>();
|
---|
237 | for (TestError e : errors) {
|
---|
238 | primitives.add(new HashSet<>(e.getPrimitives()));
|
---|
239 | }
|
---|
240 | assertEquals(setsCount, primitives.size());
|
---|
241 | } finally {
|
---|
242 | MapPaintStyles.removeStyle(style);
|
---|
243 | }
|
---|
244 | }
|
---|
245 |
|
---|
246 | /**
|
---|
247 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/12627">Bug #12627</a>.
|
---|
248 | * @throws Exception if an error occurs
|
---|
249 | */
|
---|
250 | @Test
|
---|
251 | @Ignore("not fixed yet")
|
---|
252 | public void testTicket12627() throws Exception {
|
---|
253 | doTestNaturalWood(12627, "overlapping.osm", 1, 1);
|
---|
254 | }
|
---|
255 |
|
---|
256 | /**
|
---|
257 | * Non-regression test for <a href="https://josm.openstreetmap.de/ticket/14289">Bug #14289</a>.
|
---|
258 | * @throws Exception if an error occurs
|
---|
259 | */
|
---|
260 | @Test
|
---|
261 | @Ignore("not fixed yet")
|
---|
262 | public void testTicket14289() throws Exception {
|
---|
263 | doTestNaturalWood(14289, "example2.osm", 3, 3);
|
---|
264 | }
|
---|
265 | }
|
---|