Subversion Repositories mkgmap

Rev

Rev 1548 | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
 * Copyright (c) 2009.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 3 or
 * version 2 as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 */


package uk.me.parabola.mkgmap.osmstyle;

import java.io.Reader;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.List;

import uk.me.parabola.mkgmap.general.LevelInfo;
import uk.me.parabola.mkgmap.reader.osm.Element;
import uk.me.parabola.mkgmap.reader.osm.GType;
import uk.me.parabola.mkgmap.reader.osm.Rule;
import uk.me.parabola.mkgmap.reader.osm.TypeResult;
import uk.me.parabola.mkgmap.reader.osm.Way;

import org.junit.Test;

import static org.junit.Assert.*;

/**
 * More tests for rule sets. Mostly concentrating on ordering issues and
 * not on the resulting type.
 *
 * @see RuleFileReaderTest
 */

public class RuleSetTest {
        private final String MAXSPEED_EXAMPLE = "highway=* & maxspeed=40mph {set mcssl=40}" +
                        "highway=primary & mcssl=40 [0x01]" +
                        "highway=* & mcssl=40 [0x02]" +
                        "highway=primary [0x3]";

        /**
         * A test for matching in the correct order with a simple set
         * of tags.  See also the next test.
         */

        @Test
        public void testFirstMatch1() {
                RuleSet rs = makeRuleSet("c=d & a=b [0x1]" +
                                "a=b & c=d [0x2]" +
                                "a=b [0x3]");

                Way el = new Way(1);
                el.addTag("a", "b");
                el.addTag("c", "d");
                GType type = getFirstType(rs, el);
                assertNotNull("should be found", type);
                assertEquals("first matching rule wins", 1, type.getType());
        }
        /**
         * As previous test but with order reversed.  Depending on the order
         * that the tags iterate from the way, you might get different results.
         */

        @Test
        public void testFirstMatch2() {
                RuleSet rs = makeRuleSet("a=b & c=d [0x1]" +
                                "c=d & a=b [0x2]" +
                                "a=b [0x3]");

                Way el = new Way(1);
                el.addTag("a", "b");
                el.addTag("c", "d");
                GType type = getFirstType(rs, el);
                assertNotNull("should be found", type);
                assertEquals("first matching rule wins", 1, type.getType());
        }

        /**
         * An action variable is set on a rule that starts with an exists clause.
         * We then attempt to match on value that it is
         * @throws Exception
         */

        @Test
        public void testActionVarSetOnExistsRule1() throws Exception {
                RuleSet rs = makeRuleSet(MAXSPEED_EXAMPLE);

                Way el = new Way(1);
                el.addTag("highway", "primary");
                el.addTag("maxspeed", "40mph");
                el.addTag("ref", "A123");
                el.addTag("name", "Long Lane");

                GType type = getFirstType(rs, el);
                assertEquals("should match first", 1, type.getType());
        }

        @Test
        public void testActionVarSetOnExistsRule2() throws Exception {
                RuleSet rs = makeRuleSet(MAXSPEED_EXAMPLE);

                Way el = new Way(1);
                el.addTag("highway", "unclassified");
                el.addTag("maxspeed", "40mph");
                el.addTag("ref", "A123");
                el.addTag("name", "Long Lane");

                GType type = getFirstType(rs, el);
                assertEquals("should match first", 2, type.getType());
        }

        /**
         * Check that actions are run in the order given.  Use the add command
         * to set a variable.  The first add that is run will be the value of
         * the variable.
         */

        @Test
        public void testActionOrder() {
                RuleSet rs = makeRuleSet("b=c {add fred=1}" +
                                "a=b {add fred=2}" +
                                "c=d {add fred=3}" +
                                "a=b [0x1]");

                // All of the conditions are set.
                Way el = new Way(1);
                el.addTag("a", "b");
                el.addTag("b", "c");
                el.addTag("c", "d");

                getFirstType(rs, el);
                assertEquals("b=c was first action", "1", el.getTag("fred"));
        }

        /**
         * Match on a tag that was set in a previous action rule and was not
         * on the original element.
         */

        @Test
        public void testMatchOnSetTag() {
                RuleSet rs = makeRuleSet("highway=yes {set abcxyz = 1}" +
                                "abcxyz=1 [0x1]");

                Way el = new Way(1);
                el.addTag("highway", "yes");

                GType type = getFirstType(rs, el);
                assertNotNull("type matched on previously set tag", type);
        }

        /**
         * A chain of rules, some of which contain tags from the element and
         * some that contain only tags that are set in previous rules.
         */

        @Test
        public void testOrderChain() {
                RuleSet rs = makeRuleSet("z=1 {add fred=1;}" +
                                "fred=1 {add abba=1}" +
                                "z=1 & abba=1 {add destiny=1}" +
                                "destiny=1 [0x1]" +
                                "z=1 [0x2]");

                Way el = new Way(1);
                el.addTag("z", "1");

                GType type = getFirstType(rs, el);
                assertNotNull("chain of commands", type);
                assertEquals("'destiny' should be selected", 1, type.getType());
        }

        /**
         * A chain of rules, some of which contain tags from the element and
         * some that contain only tags that are set in previous rules.
         */

        @Test
        public void testOrderChain2() {
                RuleSet rs = makeRuleSet("z=1 {add fred=1;}" +
                                "fred=1 {add abba=1}" +
                                "abba=1 {add destiny=1}" +
                                "destiny=1 [0x1]");

                Way el = new Way(1);
                el.addTag("z", "1");

                GType type = getFirstType(rs, el);
                assertNotNull("chain of commands", type);
        }

        /**
         * Append to a variable in the correct order as in the rule set.
         */

        @Test
        public void testAppendInOrder() {
                RuleSet rs = makeRuleSet("highway=primary {set R='${R} a'}" +
                                "ref=A1 {set R='${R} b'}" +
                                "z=1 {set R='${R} c'}" +
                                "a=1 {set R='${R} d'}");

                Way el = new Way(1);
                el.addTag("R", "init");
                el.addTag("highway", "primary");
                el.addTag("ref", "A1");
                el.addTag("z", "1");
                el.addTag("a", "1");

                getFirstType(rs, el);
                String s = el.getTag("R");
                assertEquals("appended value", "init a b c d", s);
        }

        /**
         * Rules should only be evaluated once for an element.  Because of the
         * way that we handle rules that may get run after tags are set in actions
         * it is possible that a rule would get run twice if not careful.
         *
         * It is not that easy to trigger, as this is the second attempt at
         * showing it is possible...
         */

        @Test
        public void testRuleEvaluatedOnce() {
                RuleSet rs = makeRuleSet("highway=primary " +
                                "  {set highway=primary; set result='${result} 1';}" +
                                "highway='primary' {set result='${result} 2'");
                Way el = new Way(1);
                el.addTag("highway", "primary");
                el.addTag("result", "0");

                getFirstType(rs, el);
                assertEquals("rules run once", "0 1 2", el.getTag("result"));
        }

        /**
         * The example that was in the checkin comment, make sure it actually
         * does work ;)
         */

        @Test
        public void testCheckinExample() {
                RuleSet rs = makeRuleSet("highway=motorway  {set blue=true;}\n" +
                                "blue=true  [0x1 ]\n" +
                                "highway=motorway [0x2]");

                Way el = new Way(1);
                el.addTag("highway", "motorway");

                GType type = getFirstType(rs, el);
                assertEquals("first match is on blue", 1, type.getType());
        }

        @Test
        public void testActionsMixedWithTypes() {
                RuleSet rs = makeRuleSet("highway=primary {set marker=1}" +
                                "marker=2 [0x1]" +
                                "highway=primary {set marker=2}" +
                                "marker=2 [0x2]");

                Way el = new Way(1);
                el.addTag("highway", "primary");

                GType type = getFirstType(rs, el);
                assertEquals("second marker rule", 2, type.getType());
        }

        @Test
        public void testContinueDefault() {
                RuleSet rs = makeRuleSet("highway=footway {set surface=good;} [0x1 continue]" +
                                "surface=good [0x20]" +
                                "surface=bad [0x30]");

                Way el = new Way(1);
                el.addTag("highway", "footway");
                el.addTag("surface", "bad");

                List<GType> list = resolveList(rs, el);
                assertEquals("number of lines returned", 2, list.size());
                assertEquals("surface setting not propagated", "bad", el.getTag("surface"));
                assertEquals("result type", 0x30, list.get(1).getType());
        }

        @Test
        public void testContinuePropagate() {
                RuleSet rs = makeRuleSet("highway=footway {set surface=good;} [0x1 continue propagate]" +
                                "surface=good [0x20]" +
                                "surface=bad [0x30]");

                Way el = new Way(1);
                el.addTag("highway", "footway");
                el.addTag("surface", "bad");

                List<GType> list = resolveList(rs, el);
                assertEquals("number of lines returned", 2, list.size());
                assertEquals("surface setting is propagated", "good", el.getTag("surface"));
                assertEquals("result type", 0x20, list.get(1).getType());
        }

        @Test
        public void testContinueNoPropagate() {
                RuleSet rs = makeRuleSet("highway=footway {set surface=good;} [0x1 continue no_propagate]" +
                                "surface=good [0x20]" +
                                "surface=bad [0x30]");

                Way el = new Way(1);
                el.addTag("highway", "footway");
                el.addTag("surface", "bad");

                List<GType> list = resolveList(rs, el);
                assertEquals("number of lines returned", 2, list.size());
                assertEquals("surface setting is not propagated", "bad", el.getTag("surface"));
                assertEquals("result type", 0x30, list.get(1).getType());
        }

        @Test
        public void testContinueWithActions() {
                RuleSet rs = makeRuleSet("highway=footway {set surface=good;} [0x1 continue with_actions]" +
                                "surface=good [0x20]" +
                                "surface=bad [0x30]");

                Way el = new Way(1);
                el.addTag("highway", "footway");
                el.addTag("surface", "bad");

                List<GType> list = resolveList(rs, el);
                assertEquals("number of lines returned", 2, list.size());
                assertEquals("surface setting is propagated", "good", el.getTag("surface"));
                assertEquals("result type", 0x20, list.get(1).getType());
        }

        @Test
        public void testContinueChangesTag() {
                RuleSet rs = makeRuleSet("highway=crossing & crossing=zebra_crossing" +
                                "    {set highway=deleted_crossing} [0x4004 resolution 24 continue propagate]" +
                                "highway=crossing [0x610f resolution 24 continue]" +
                                "highway=deleted_crossing [0x6 resolution 24 continue]"
                );

                Way el = new Way(1);
                el.addTag("highway", "crossing");
                el.addTag("crossing", "zebra_crossing");

                List<GType> list = resolveList(rs, el);
                assertEquals("first element", 0x4004, list.get(0).getType());
                assertEquals("second element", 0x6, list.get(1).getType());
        }

        private List<GType> resolveList(RuleSet rs, Way el) {
                final List<GType> list = new ArrayList<GType>();
                rs.resolveType(el, new TypeResult() {
                        public void add(Element el, GType type) {
                                list.add(type);
                        }
                });
                return list;
        }

        private GType getFirstType(Rule rs, Element el) {
                final List<GType> types = new ArrayList<GType>();
                rs.resolveType(el, new TypeResult() {
                        public void add(Element el, GType type) {
                                types.add(type);
                        }
                });
                if (types.isEmpty())
                        return null;
                else
                        return types.get(0);
        }

        /**
         * Create a rule set out of a string.  The string is processed
         * as if it were in a file and the levels spec had been set.
         */

        private RuleSet makeRuleSet(String in) {
                Reader r = new StringReader(in);

                RuleSet rs = new RuleSet();
                RuleFileReader rr = new RuleFileReader(GType.POLYLINE, LevelInfo.createFromString("0:24 1:20 2:18 3:16 4:14"), rs);
                rr.load(r, "string");
                return rs;
        }
}