Subversion Repositories mkgmap

Rev

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

/*
 * Copyright (C) 2006 - 2012.
 *
 * 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.reader.osm.xml;

import uk.me.parabola.imgfmt.FormatException;
import uk.me.parabola.imgfmt.app.Coord;
import uk.me.parabola.log.Logger;
import uk.me.parabola.mkgmap.reader.osm.Element;
import uk.me.parabola.mkgmap.reader.osm.FakeIdGenerator;
import uk.me.parabola.mkgmap.reader.osm.GeneralRelation;
import uk.me.parabola.mkgmap.reader.osm.Node;
import uk.me.parabola.mkgmap.reader.osm.OsmHandler;
import uk.me.parabola.mkgmap.reader.osm.Relation;
import uk.me.parabola.mkgmap.reader.osm.Way;

import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;

import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;

import org.xml.sax.Attributes;
import org.xml.sax.ContentHandler;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;
import org.xml.sax.helpers.DefaultHandler;

/**
 * Reads and parses the OSM XML format.
 *
 * Creates the nodes/ways and relations that are read from the file and passes
 * them to the OsmCollector.
 *
 * It should not examine tags, or do anything else.
 *
 * @author Steve Ratcliffe
 */

public class OsmXmlHandler extends OsmHandler {
        private static final Logger log = Logger.getLogger(OsmXmlHandler.class);

        // Set to the currently processing element.
        private int mode;

        // Values for mode above.
        private static final int MODE_NODE = 1;
        private static final int MODE_WAY = 2;
        private static final int MODE_BOUND = 3;
        private static final int MODE_RELATION = 4;
        private static final int MODE_BOUNDS = 5;

        // Current state.
        private Node currentNode;
        private Way currentWay;
        private Relation currentRelation;
        private long currentElementId;
        private final Map<String, Long> fakeIdMap = new HashMap<String, Long>();

        public OsmXmlHandler() {
        }

        @Override
        public boolean isFileSupported(String name) {
                // This is the default format so say supported if we get this far,
                // this one must always be last for this reason.
                return true;
        }

        @Override
        public void parse(InputStream is) throws FormatException {
                try {
                        SAXParserFactory parserFactory = SAXParserFactory.newInstance();
                        parserFactory.setXIncludeAware(true);
                        parserFactory.setNamespaceAware(true);
                        SAXParser parser = parserFactory.newSAXParser();

                        try {
                                SaxHandler saxHandler = new SaxHandler();

                                // parse the xml file
                                parser.parse(is, saxHandler);

                        } catch (IOException e) {
                                throw new FormatException("Error reading file", e);
                        }
                } catch (SAXException e) {
                        throw new FormatException("Error parsing file", e);
                } catch (ParserConfigurationException e) {
                        throw new FormatException("Internal error configuring xml parser", e);
                }
        }
        /**
         * Convert an id as a string to a number. If the id is not a number, then create
         * a unique number instead.
         * @param id The id as a string. Does not have to be a numeric quantity.
         * @return A long id, either parsed from the input, or a unique id generated internally.
         */

        private long idVal(String id) {
                try {
                        // attempt to parse id as a number
                        return Long.parseLong(id);
                } catch (NumberFormatException e) {
                        // if that fails, fake a (hopefully) unique value
                        Long fakeIdVal = fakeIdMap.get(id);
                        if(fakeIdVal == null) {
                                fakeIdVal = FakeIdGenerator.makeFakeId();
                                fakeIdMap.put(id, fakeIdVal);
                        }
                        //System.out.printf("%s = 0x%016x\n", id, fakeIdVal);
                        return fakeIdVal;
                }
        }

        /**
         * The XML handler callbacks.
         *
         * Need an inner class here so that the top class can inherit from OsmHandler.
         */

        public class SaxHandler extends DefaultHandler {

                /**
                 * Receive notification of the start of an element.
                 *
                 * @param uri           The Namespace URI, or the empty string if the
                 *                   element has no Namespace URI or if Namespace
                 *                   processing is not being performed.
                 * @param localName  The local name (without prefix), or the
                 *                   empty string if Namespace processing is not being
                 *                   performed.
                 * @param qName   The qualified name (with prefix), or the
                 *                   empty string if qualified names are not available.
                 * @param attributes The attributes attached to the element.  If
                 *                   there are no attributes, it shall be an empty
                 *                   Attributes object.
                 * @throws SAXException Any SAX exception, possibly
                 *                                  wrapping another exception.
                 * @see ContentHandler#startElement
                 */

                public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
                        if (mode == 0) {
                                if (qName.equals("node")) {
                                        mode = MODE_NODE;
                                        startNode(attributes.getValue("id"),
                                                        attributes.getValue("lat"),
                                                        attributes.getValue("lon"));

                                } else if (qName.equals("way")) {
                                        mode = MODE_WAY;
                                        startWay(attributes.getValue("id"));

                                } else if (qName.equals("relation")) {
                                        mode = MODE_RELATION;
                                        currentRelation = new GeneralRelation(idVal(attributes.getValue("id")));

                                } else if (qName.equals("bound")) {
                                        mode = MODE_BOUND;
                                        if(!isIgnoreBounds()) {
                                                String box = attributes.getValue("box");
                                                setupBBoxFromBound(box);
                                        }

                                } else if (qName.equals("bounds")) {
                                        mode = MODE_BOUNDS;
                                        if(!isIgnoreBounds())
                                                setupBBoxFromBounds(attributes);
                                }

                        } else if (mode == MODE_NODE) {
                                startInNode(qName, attributes);
                        } else if (mode == MODE_WAY) {
                                startInWay(qName, attributes);
                        } else if (mode == MODE_RELATION) {
                                startInRelation(qName, attributes);
                        }
                }

                /**
                 * Receive notification of the end of an element.
                 *
                 * @param uri      The Namespace URI, or the empty string if the
                 *                  element has no Namespace URI or if Namespace
                 *                  processing is not being performed.
                 * @param localName The local name (without prefix), or the
                 *                  empty string if Namespace processing is not being
                 *                  performed.
                 * @param qName  The qualified name (with prefix), or the
                 *                  empty string if qualified names are not available.
                 * @throws SAXException Any SAX exception, possibly
                 *                                  wrapping another exception.
                 * @see ContentHandler#endElement
                 */

                public void endElement(String uri, String localName, String qName) throws SAXException {
                        if (mode == MODE_NODE) {
                                if (qName.equals("node")) {
                                        mode = 0;
                                        if (currentNode != null) {
                                                saver.addNode(currentNode);
                                                hooks.onAddNode(currentNode);
                                        }
                                        currentElementId = 0;
                                        currentNode = null;
                                }

                        } else if (mode == MODE_WAY) {
                                if (qName.equals("way")) {
                                        mode = 0;

                                        endWay(currentWay);
                                        currentWay = null;
                                }

                        } else if (mode == MODE_BOUND) {
                                if (qName.equals("bound"))
                                        mode = 0;

                        } else if (mode == MODE_BOUNDS) {
                                if (qName.equals("bounds"))
                                        mode = 0;

                        } else if (mode == MODE_RELATION) {
                                if (qName.equals("relation")) {
                                        mode = 0;
                                        saver.addRelation(currentRelation);
                                }
                        }
                }

                /**
                 * Called on an XML error.  Attempt to print a line number to aid in
                 * working out the problem.
                 * @throws SAXException
                 */

                public void fatalError(SAXParseException e) throws SAXException {
                        System.err.println("Error at line " + e.getLineNumber() + ", col "
                                        + e.getColumnNumber());
                        super.fatalError(e);
                }
        }

        /**
         * A new tag has been started while we are inside a node element.
         * @param qName The new tag name.
         * @param attributes Its attributes.
         */

        private void startInNode(String qName, Attributes attributes) {
                if (qName.equals("tag")) {
                        String key = attributes.getValue("k");
                        String val = attributes.getValue("v");

                        // We only want to create a full node for nodes that are POI's
                        // and not just one point of a way.  Only create if it has tags that
                        // could be used in a POI.
                        key = keepTag(key, val);
                        if (key != null) {
                                if (currentNode == null) {
                                        Coord co = saver.getCoord(currentElementId);
                                        currentNode = new Node(currentElementId, co);
                                }

                                currentNode.addTagFromRawOSM(key, val);
                        }
                }
        }

        /**
         * A new tag has been started while we are inside a way element.
         * @param qName The new tag name.
         * @param attributes Its attributes.
         */

        private void startInWay(String qName, Attributes attributes) {
                if (qName.equals("nd")) {
                        long id = idVal(attributes.getValue("ref"));
                        addCoordToWay(currentWay, id);
                } else if (qName.equals("tag")) {
                        String key = attributes.getValue("k");
                        String val = attributes.getValue("v");
                        key = keepTag(key, val);
                        if (key != null)
                                currentWay.addTagFromRawOSM(key, val);
                }
        }

        /**
         * A new tag has been started while we are inside the relation tag.
         * @param qName The new tag name.
         * @param attributes Its attributes.
         */

        private void startInRelation(String qName, Attributes attributes) {
                if (qName.equals("member")) {
                        long id = idVal(attributes.getValue("ref"));
                        Element el;
                        String type = attributes.getValue("type");
                        if ("way".equals(type)){
                                el = saver.getWay(id);
                        } else if ("node".equals(type)) {
                                el = saver.getNode(id);
                                if(el == null) {
                                        // we didn't make a node for this point earlier,
                                        // do it now (if it exists)
                                        Coord co = saver.getCoord(id);
                                        if(co != null) {
                                                el = new Node(id, co);
                                                saver.addNode((Node)el);
                                        }
                                }
                        } else if ("relation".equals(type)) {
                                el = saver.getRelation(id);
                                if (el == null) {
                                        saver.deferRelation(id, currentRelation, attributes.getValue("role"));
                                }
                        } else
                                el = null;
                        if (el != null) // ignore non existing ways caused by splitting files
                                currentRelation.addElement(attributes.getValue("role"), el);
                } else if (qName.equals("tag")) {
                        String key = attributes.getValue("k");
                        String val = attributes.getValue("v");
                        // the type tag is required for relations - all other tags are filtered
                        if ("type".equals(key))
                                // intern the key
                                key = "type";
                        else
                                key = keepTag(key, val);
                        if (key == null) {
                                currentRelation.setTagsIncomplete(true);
                        } else {
                                currentRelation.addTagFromRawOSM(key, val);
                        }
                }
        }

        /**
         * Set a bounding box from the bounds element.
         * There are two ways of specifying a bounding box in the XML format, this
         * one uses attributes of the element to give the bounds.
         * @param xmlattr The bounds element attributes.
         */

        private void setupBBoxFromBounds(Attributes xmlattr) {
                try {
                        setBBox(Double.parseDouble(xmlattr.getValue("minlat")),
                                        Double.parseDouble(xmlattr.getValue("minlon")),
                                        Double.parseDouble(xmlattr.getValue("maxlat")),
                                        Double.parseDouble(xmlattr.getValue("maxlon")));
                } catch (NumberFormatException e) {
                        // just ignore it
                        log.warn("NumberformatException: Cannot read bbox");
                }
        }

        /**
         * Set a bounding box from the bound element.  There are two ways of
         * specifying a bounding box, this one has a single 'box' attribute that
         * is a comma separated list of the bounds values.
         * @param box The value of the box attribute.
         */

        private void setupBBoxFromBound(String box) {
                String[] f = box.split(",");
                try {
                        setBBox(Double.parseDouble(f[0]), Double.parseDouble(f[1]),
                                        Double.parseDouble(f[2]), Double.parseDouble(f[3]));
                } catch (NumberFormatException e) {
                        // just ignore it
                        log.warn("NumberformatException: Cannot read bbox");
                }
        }

        /**
         * Save node information.  Consists of a location specified by lat/long.
         *
         * @param sid The id as a string.
         * @param slat The lat as a string.
         * @param slon The longitude as a string.
         */

        private void startNode(String sid, String slat, String slon) {
                if (sid == null || slat == null || slon == null)
                        return;
               
                try {
                        long id = idVal(sid);

                        Coord co = new Coord(Double.parseDouble(slat), Double.parseDouble(slon));
                        saver.addPoint(id, co);
                        currentElementId = id;
                } catch (NumberFormatException e) {
                        // ignore bad numeric data. The coord will be discarded
                }
        }

        /**
         * A new way element has been seen.
         * @param sid The way id as a string.
         */

        private void startWay(String sid) {
                try {
                        long id = idVal(sid);
                        currentWay = startWay(id);
                } catch (NumberFormatException e) {
                        // ignore bad numeric data. The way will be discarded
                }
        }
}