Subversion Repositories mkgmap

Rev

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

/*
 * Copyright (C) 2007 Steve Ratcliffe
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License 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.
 *
 *
 * Author: Steve Ratcliffe
 * Create date: Feb 17, 2008
 */

package uk.me.parabola.mkgmap.osmstyle;

import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.Reader;
import java.io.Writer;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Properties;
import java.util.Set;
import java.util.regex.Pattern;

import uk.me.parabola.imgfmt.Utils;
import uk.me.parabola.log.Logger;
import uk.me.parabola.mkgmap.Option;
import uk.me.parabola.mkgmap.OptionProcessor;
import uk.me.parabola.mkgmap.Options;
import uk.me.parabola.mkgmap.general.LevelInfo;
import uk.me.parabola.mkgmap.general.LineAdder;
import uk.me.parabola.mkgmap.general.MapLine;
import uk.me.parabola.mkgmap.osmstyle.actions.Action;
import uk.me.parabola.mkgmap.osmstyle.actions.NameAction;
import uk.me.parabola.mkgmap.osmstyle.eval.EqualsOp;
import uk.me.parabola.mkgmap.osmstyle.eval.ExistsOp;
import uk.me.parabola.mkgmap.osmstyle.eval.Op;
import uk.me.parabola.mkgmap.osmstyle.eval.ValueOp;
import uk.me.parabola.mkgmap.reader.osm.GType;
import uk.me.parabola.mkgmap.reader.osm.Rule;
import uk.me.parabola.mkgmap.reader.osm.Style;
import uk.me.parabola.mkgmap.reader.osm.StyleInfo;
import uk.me.parabola.mkgmap.scan.SyntaxException;
import uk.me.parabola.mkgmap.scan.TokenScanner;

/**
 * A style is a collection of files that describe the mapping between the OSM
 * features and the garmin features.  This file reads in those files and
 * provides methods for using the information.
 *
 * The files are either contained in a directory, in a package or in a zip'ed
 * file.
 *
 * @author Steve Ratcliffe
 */

public class StyleImpl implements Style {
        private static final Logger log = Logger.getLogger(StyleImpl.class);

        // This is max the version that we understand
        private static final int VERSION = 1;

        // General options just have a value and don't need any special processing.
        private static final Collection<String> OPTION_LIST = new ArrayList<String>(
                        Arrays.asList("levels", "extra-used-tags"));

        // Options that should not be overridden from the command line if the
        // value is empty.
        private static final Collection<String> DONT_OVERRIDE = new ArrayList<String>(
                        Arrays.asList("levels"));

        // File names
        private static final String FILE_VERSION = "version";
        private static final String FILE_INFO = "info";
        private static final String FILE_FEATURES = "map-features.csv";
        private static final String FILE_OPTIONS = "options";
        private static final String FILE_OVERLAYS = "overlays";

        // Patterns
        private static final Pattern COMMA_OR_SPACE_PATTERN = Pattern.compile("[,\\s]+");
        private static final Pattern EQUAL_PATTERN = Pattern.compile("=");

        // A handle on the style directory or file.
        private final StyleFileLoader fileLoader;
        private final String location;

        // The general information in the 'info' file.
        private StyleInfo info = new StyleInfo();

        // Set if this style is based on another one.
        private final List<StyleImpl> baseStyles = new ArrayList<StyleImpl>();

        // A list of tag names to be used as the element name
        private String[] nameTagList;

        // Options from the option file that are used outside this file.
        private final Map<String, String> generalOptions = new HashMap<String, String>();

        private final RuleSet lines = new RuleSet();
        private final RuleSet polygons = new RuleSet();
        private final RuleSet nodes = new RuleSet();
        private final RuleSet relations = new RuleSet();

        private OverlayReader overlays;

        /**
         * Create a style from the given location and name.
         * @param loc The location of the style. Can be null to mean just check
         * the classpath.
         * @param name The name.  Can be null if the location isn't.  If it is
         * null then we just check for the first version file that can be found.
         * @throws FileNotFoundException If the file doesn't exist.  This can
         * include the version file being missing.
         */

        public StyleImpl(String loc, String name) throws FileNotFoundException {
                location = loc;
                fileLoader = StyleFileLoader.createStyleLoader(loc, name);

                // There must be a version file, if not then we don't create the style.
                checkVersion();

                readInfo();

                for (String baseName : info.baseStyles())
                        readBaseStyle(baseName);

                for (StyleImpl baseStyle : baseStyles)
                        mergeOptions(baseStyle);

                readOptions();
                readRules();

                readOverlays();

                readMapFeatures();

                ListIterator<StyleImpl> listIterator = baseStyles.listIterator(baseStyles.size());
                while (listIterator.hasPrevious())
                        mergeRules(listIterator.previous());

                // OR: other way
                //for (StyleImpl s : baseStyles)
                //      mergeRules(s);
        }

        public String[] getNameTagList() {
                return nameTagList;
        }

        public String getOption(String name) {
                return generalOptions.get(name);
        }

        public StyleInfo getInfo() {
                return info;
        }

        /**
         * After the style is loaded we override any options that might
         * have been set in the style itself with the command line options.
         *
         * We may have to filter some options that we don't ever want to
         * set on the command line.
         *
         * @param config The command line options.
         */

        public void applyOptionOverride(Properties config) {
                Set<Entry<Object,Object>> entries = config.entrySet();
                for (Entry<Object,Object> ent : entries) {
                        String key = (String) ent.getKey();
                        String val = (String) ent.getValue();

                        if (!DONT_OVERRIDE.contains(key))
                                if (key.equals("name-tag-list")) {
                                        // The name-tag-list allows you to redefine what you want to use
                                        // as the name of a feature.  By default this is just 'name', but
                                        // you can supply a list of tags to use
                                        // instead eg. "name:en,int_name,name" or you could use some
                                        // completely different tag...
                                        nameTagList = COMMA_OR_SPACE_PATTERN.split(val);
                                } else if (OPTION_LIST.contains(key)) {
                                        // Simple options that have string value.  Perhaps we should allow
                                        // anything here?
                                        generalOptions.put(key, val);
                                }
                }
        }

        public Rule getNodeRules() {
                nodes.prepare();
                return nodes;
        }

        public Rule getWayRules() {
                RuleSet r = new RuleSet();
                r.addAll(lines);
                r.addAll(polygons);
                r.prepare();
                return r;
        }

        public Rule getLineRules() {
                lines.prepare();
                return lines;
        }

        public Rule getPolygonRules() {
                polygons.prepare();
                return polygons;
        }
       
        public Rule getRelationRules() {
                relations.prepare();
                return relations;
        }

        public LineAdder getOverlays(final LineAdder lineAdder) {
                LineAdder adder = null;

                if (overlays != null) {
                        adder = new LineAdder() {
                                public void add(MapLine element) {
                                        overlays.addLine(element, lineAdder);
                                }
                        };
                }
                return adder;
        }

        public Set<String> getUsedTags() {
                Set<String> set = new HashSet<String>();
                set.addAll(relations.getUsedTags());
                set.addAll(lines.getUsedTags());
                set.addAll(polygons.getUsedTags());
                set.addAll(nodes.getUsedTags());

                // this is to allow style authors to say that tags are really used even
                // if they are not found in the style file.  This is mostly to work
                // around situations that we haven't thought of - the style is expected
                // to get it right for itself.
                String s = getOption("extra-used-tags");
                if (s != null)
                        set.addAll(Arrays.asList(COMMA_OR_SPACE_PATTERN.split(s)));

                // These tags are passed on the command line and so must be added
                if (nameTagList != null)
                        set.addAll(Arrays.asList(nameTagList));

                // There are a lot of tags that are used within mkgmap that
                InputStream is = getClass().getResourceAsStream("/styles/builtin-tag-list");
                try {
                        if (is != null) {
                                BufferedReader br = new BufferedReader(new InputStreamReader(is));
                                //System.out.println("Got built in list");
                                String line;
                                while ((line = br.readLine()) != null) {
                                        line = line.trim();
                                        if (line.startsWith("#"))
                                                continue;
                                        //System.out.println("adding " + line);
                                        set.add(line);
                                }
                        }
                } catch (IOException e) {
                        // the file doesn't exist, this is ok but unlikely
                        System.err.println("warning: built in tag list not found");
                } finally {
                        Utils.closeFile(is);
                }
                return set;
        }

        private void readRules() {
                String l = generalOptions.get("levels");
                if (l == null)
                        l = LevelInfo.DEFAULT_LEVELS;
                LevelInfo[] levels = LevelInfo.createFromString(l);

                try {
                        RuleFileReader reader = new RuleFileReader(0, levels, relations);
                        reader.load(fileLoader, "relations");
                } catch (FileNotFoundException e) {
                        // it is ok for this file to not exist.
                        log.debug("no relations file");
                }

                try {
                        RuleFileReader reader = new RuleFileReader(GType.POINT, levels, nodes);
                        reader.load(fileLoader, "points");
                } catch (FileNotFoundException e) {
                        // it is ok for this file to not exist.
                        log.debug("no points file");
                }

                try {
                        RuleFileReader reader = new RuleFileReader(GType.POLYLINE, levels, lines);
                        reader.load(fileLoader, "lines");
                } catch (FileNotFoundException e) {
                        log.debug("no lines file");
                }

                try {
                        RuleFileReader reader = new RuleFileReader(GType.POLYGON, levels, polygons);
                        reader.load(fileLoader, "polygons");
                } catch (FileNotFoundException e) {
                        log.debug("no polygons file");
                }
        }

        /**
         * Read the map-features file.  This is the old format of the mapping
         * between osm and garmin types.
         */

        private void readMapFeatures() {
                try {
                        Reader r = fileLoader.open(FILE_FEATURES);
                        MapFeatureReader mfr = new MapFeatureReader();
                        String l = generalOptions.get("levels");
                        if (l == null)
                                l = LevelInfo.DEFAULT_LEVELS;
                        mfr.setLevels(LevelInfo.createFromString(l));
                        mfr.readFeatures(new BufferedReader(r));
                        initFromMapFeatures(mfr);
                } catch (FileNotFoundException e) {
                        // optional file
                        log.debug("no map-features file");
                } catch (IOException e) {
                        log.error("could not read map features file");
                }
        }

        /**
         * Take the output of the map-features file and create rules for
         * each line and add to this style.  All rules in map-features are
         * unconditional, in other words the osm 'amenity=cinema' always
         * maps to the same garmin type.
         *
         * @param mfr The map feature file reader.
         */

        private void initFromMapFeatures(MapFeatureReader mfr) {
                addBackwardCompatibleRules();

                for (Entry<String, GType> me : mfr.getLineFeatures().entrySet())
                        lines.add(me.getKey(), createRule(me.getKey(), me.getValue()), Collections.<String>emptySet());

                for (Entry<String, GType> me : mfr.getShapeFeatures().entrySet())
                        polygons.add(me.getKey(), createRule(me.getKey(), me.getValue()), Collections.<String>emptySet());

                for (Entry<String, GType> me : mfr.getPointFeatures().entrySet())
                        nodes.add(me.getKey(), createRule(me.getKey(), me.getValue()), Collections.<String>emptySet());
        }

        /**
         * For backward compatibility, when we find a map-features file we add
         * rules for actions that were previously hard coded in the conversion.
         * These are added even if there was also a lines, points, etc file.
         */

        private void addBackwardCompatibleRules() {
                // Name rule for highways
                List<Action> l = new ArrayList<Action>();
                NameAction action = new NameAction();
                action.add("${name} (${ref})");
                action.add("${ref}");
                action.add("${name}");
                l.add(action);

                Op expr = new ExistsOp();
                expr.setFirst(new ValueOp("highway"));
                Rule rule = new ActionRule(expr, l);
                lines.add("highway=*", rule, Collections.<String>emptySet());

                // Name rule for contour lines
                l = new ArrayList<Action>();
                action = new NameAction();
                action.add("${ele|conv:m=>ft}");
                l.add(action);

                EqualsOp expr2 = new EqualsOp();
                expr2.setFirst(new ValueOp("contour"));
                expr2.setSecond(new ValueOp("elevation"));
                rule = new ActionRule(expr2, l);
                lines.add("contour=elevation", rule, Collections.<String>emptySet()); // "contour=elevation"

                expr2 = new EqualsOp();
                expr2.setFirst(new ValueOp("contour_ext"));
                expr2.setSecond(new ValueOp("elevation"));
                rule = new ActionRule(expr2, l);
                lines.add("contour_ext=elevation", rule, Collections.<String>emptySet()); // "contour_ext=elevation"
        }

        /**
         * Create a rule from a raw gtype. You get raw gtypes when you
         * have read the types from a map-features file.
         *
         * @return A rule that is conditional on the key string given.
         */

        private Rule createRule(String key, GType gt) {
                if (gt.getDefaultName() != null)
                        log.debug("set default name of", gt.getDefaultName(), "for", key);
                String[] tagval = EQUAL_PATTERN.split(key);
                EqualsOp op = new EqualsOp();
                op.setFirst(new ValueOp(tagval[0]));
                op.setSecond(new ValueOp(tagval[1]));
                return new ExpressionRule(op, gt);
        }

        /**
         * If there is an options file, then read it and keep options that
         * we are interested in.
         *
         * Only specific options can be set.
         */

        private void readOptions() {
                try {
                        Reader r = fileLoader.open(FILE_OPTIONS);
                        Options opts = new Options(new OptionProcessor() {
                                public void processOption(Option opt) {
                                        String key = opt.getOption();
                                        String val = opt.getValue();
                                        if (key.equals("name-tag-list")) {
                                                // The name-tag-list allows you to redefine what you want to use
                                                // as the name of a feature.  By default this is just 'name', but
                                                // you can supply a list of tags to use
                                                // instead eg. "name:en,int_name,name" or you could use some
                                                // completely different tag...
                                                nameTagList = COMMA_OR_SPACE_PATTERN.split(val);
                                        } else if (OPTION_LIST.contains(key)) {
                                                // Simple options that have string value.  Perhaps we should allow
                                                // anything here?
                                                generalOptions.put(key, val);
                                        }
                                }
                        });

                        opts.readOptionFile(r, FILE_OPTIONS);
                } catch (FileNotFoundException e) {
                        // the file is optional, so ignore if not present, or causes error
                        log.debug("no options file");
                }
        }

        /**
         * Read the info file.  This is just information about the style.
         */

        private void readInfo() {
                try {
                        Reader br = new BufferedReader(fileLoader.open(FILE_INFO));
                        info = new StyleInfo();

                        Options opts = new Options(new OptionProcessor() {
                                public void processOption(Option opt) {
                                        String word = opt.getOption();
                                        String value = opt.getValue();
                                        if (word.equals("summary"))
                                                info.setSummary(value);
                                        else if (word.equals("version")) {
                                                info.setVersion(value);
                                        } else if (word.equals("base-style")) {
                                                info.addBaseStyleName(value);
                                        } else if (word.equals("description")) {
                                                info.setLongDescription(value);
                                        }

                                }
                        });

                        opts.readOptionFile(br, FILE_INFO);

                } catch (FileNotFoundException e) {
                        // optional file..
                        log.debug("no info file");
                }
        }

        private void readOverlays() {
                try {
                        Reader r = fileLoader.open(FILE_OVERLAYS);
                        overlays = new OverlayReader(r, FILE_OVERLAYS);
                        overlays.readOverlays();
                } catch (FileNotFoundException e) {
                        // this is perfectly normal
                        log.debug("no overlay file");
                }
        }

        /**
         * If this style is based upon another one then read it in now.  The rules
         * for merging styles are that it is as-if the style was read just after
         * the current styles 'info' section and any option or rule specified
         * in the current style will override any corresponding item in the
         * base style.
         * @param name The name of the base style
         */

        private void readBaseStyle(String name) {
                if (name == null)
                        return;

                try {
                        baseStyles.add(new StyleImpl(location, name));
                } catch (SyntaxException e) {
                        System.err.println("Error in style: " + e.getMessage());
                } catch (FileNotFoundException e) {
                        // not found, try on the classpath.  This is the common
                        // case where you have an external style, but want to
                        // base it on a built in one.
                        log.debug("could not open base style file", e);

                        try {
                                baseStyles.add(new StyleImpl(null, name));
                        } catch (SyntaxException se) {
                                System.err.println("Error in style: " + se.getMessage());
                        } catch (FileNotFoundException e1) {
                                log.error("Could not find base style", e);
                        }
                }
        }

        /**
         * Merge another style's options into this one.  The style will have a lower
         * priority, in other words any option set in 'other' and this style will
         * take the value given in this style.
         *
         * This is used to base styles on other ones, without having to repeat
         * everything.
         *
         * @see #mergeRules(StyleImpl)
         */

        private void mergeOptions(StyleImpl other) {
                this.nameTagList = other.nameTagList;
                for (Entry<String, String> ent : other.generalOptions.entrySet()) {
                        String opt = ent.getKey();
                        String val = ent.getValue();
                        if (opt.equals("name-tag-list")) {
                                // The name-tag-list allows you to redefine what you want to use
                                // as the name of a feature.  By default this is just 'name', but
                                // you can supply a list of tags to use
                                // instead eg. "name:en,int_name,name" or you could use some
                                // completely different tag...
                                nameTagList = COMMA_OR_SPACE_PATTERN.split(val);
                        } else if (OPTION_LIST.contains(opt)) {
                                // Simple options that have string value.  Perhaps we should allow
                                // anything here?
                                generalOptions.put(opt, val);
                        }
                }
        }

        /**
         * Merge rules from the base style.  This has to called after this
         * style's rules are read.
         *
         * The other rules have a lower priority than the rules in this file; it is as if they
         * were appended to the rule files of this style.
         *
         * @see #mergeOptions(StyleImpl)
         */

        private void mergeRules(StyleImpl other) {
                lines.merge(other.lines);
                polygons.merge(other.polygons);
                nodes.merge(other.nodes);
                relations.merge(other.relations);
        }

        private void checkVersion() throws FileNotFoundException {
                Reader r = fileLoader.open(FILE_VERSION);
                TokenScanner scan = new TokenScanner(FILE_VERSION, r);
                int version;
                try {
                        version = scan.nextInt();
                        log.debug("Got version", version);
                } catch (NumberFormatException e) {
                        // default to 0 if no version can be found.
                        version = 0;
                }

                if (version > VERSION) {
                        System.err.println("Warning: unrecognised style version " + version +
                        ", but only versions up to " + VERSION + " are understood");
                }
        }

        /**
         * Writes out this file to the given writer in the single file format.
         * This produces a valid style file, although it is mostly used
         * for testing.
         */

        void dumpToFile(Writer out) {
                StylePrinter stylePrinter = new StylePrinter(this);
                stylePrinter.setGeneralOptions(generalOptions);
                stylePrinter.setRelations(relations);
                stylePrinter.setLines(lines);
                stylePrinter.setNodes(nodes);
                stylePrinter.setPolygons(polygons);
                stylePrinter.dumpToFile(out);
        }

        public static void main(String[] args) throws FileNotFoundException {
                String file = args[0];
                String name = null;
                if (args.length > 1)
                        name = args[1];
                StyleImpl style = new StyleImpl(file, name);

                style.dumpToFile(new OutputStreamWriter(System.out));
        }
}