Subversion Repositories splitter

Rev

Rev 163 | 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 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.splitter;

import org.xmlpull.v1.XmlPullParserException;

/**
 * Parses an OSM file, calling the appropriate methods on a
 * {@code MapProcessor} as it progresses.
 */

class OSMParser extends AbstractXppParser implements MapReader {

        // How many elements to process before displaying a status update
        private static final int NODE_STATUS_UPDATE_THRESHOLD = 2500000;
        private static final int WAY_STATUS_UPDATE_THRESHOLD = 500000;
        private static final int RELATION_STATUS_UPDATE_THRESHOLD = 50000;

        private enum State {
                Node, Way, Relation, None
        }

        private Node currentNode = new Node(); 
        private Way currentWay = new Way();    
        private Relation currentRelation = new Relation();     

        private final MapProcessor processor;

        // There are mixed nodes and ways in the file
        private final boolean mixed;
        private final boolean startNodeOnly;

        private State state = State.None;
        private long nodeCount;
        private long wayCount;
        private long relationCount;
        private int minNodeId = Integer.MAX_VALUE;
        private int maxNodeId = Integer.MIN_VALUE;

        OSMParser(MapProcessor processor, boolean mixed) throws XmlPullParserException {
                this.processor = processor;
                this.startNodeOnly = processor.isStartNodeOnly();
                this.mixed = mixed;
        }
        /*
        @Override
        public long getNodeCount() {
                return nodeCount;
        }

        @Override
        public long getWayCount() {
                return wayCount;
        }

        @Override
        public long getRelationCount() {
                return relationCount;
        }

        @Override
        public int getMinNodeId() {
                return minNodeId;
        }

        @Override
        public int getMaxNodeId() {
                return maxNodeId;
        }
        */

        public void endMap() {
                processor.endMap();
        }
       
        /**
         * Receive notification of the start of an element.
         */

        @Override
        public boolean startElement(String name) {
                switch (state) {
                case None:
                        CharSequence action = getAttr("action");
                        if (action != null && action.equals("delete"))
                                return false;
                        if (name.equals("node")) {
                                startNode();
                        } else if (name.equals("way")) {
                                if (!startNodeOnly)
                                        startWay();
                                else if (!mixed)
                                        return true;
                        } else if (name.equals("relation")) {
                                if (!startNodeOnly)
                                        startRelation();
                        } else if (name.equals("bounds") || name.equals("bound")) {
                                processBounds();
                        }
                        break;
                case Node:
                        if (!startNodeOnly)
                                processNode(name);
                        break;
                case Way:
                        if (!startNodeOnly)
                                processWay(name);
                        break;
                case Relation:
                        if (!startNodeOnly)
                                processRelation(name);
                        break;
                }
                return false;
        }

        private void startNode() {
                String idStr = getAttr("id");
                String latStr = getAttr("lat");
                String lonStr = getAttr("lon");

                if (idStr == null || latStr == null || lonStr == null) {
                        // This should never happen - bad/corrupt .osm file?
                        System.err.println("Node encountered with missing data. Bad/corrupt osm file? id=" + idStr + ", lat=" + latStr + ", lon=" + lonStr + ". Ignoring this node");
                        return;
                }

                int id = Integer.parseInt(idStr);
                double lat = Convert.parseDouble(latStr);
                double lon = Convert.parseDouble(lonStr);

                if (id < minNodeId) {
                        minNodeId = id;
                }
                if (id > maxNodeId) {
                        maxNodeId = id;
                }

                currentNode = new Node();
                currentNode.set(id, lat, lon);
                state = State.Node;
        }

        private void startWay() {
                currentWay = new Way();
                currentWay.set(getIntAttr("id"));
                state = State.Way;
        }

        private void startRelation() {
                currentRelation = new Relation();
                currentRelation.set(getIntAttr("id"));
                state = State.Relation;
        }

        private void processNode(CharSequence name) {
                if (name.equals("tag")) {
                        currentNode.addTag(getAttr("k"), getAttr("v"));
                }
        }

        private void processWay(CharSequence name) {
                if (name.equals("nd")) {
                        currentWay.addRef(getIntAttr("ref"));
                } else if (name.equals("tag")) {
                        currentWay.addTag(getAttr("k"), getAttr("v"));
                }
        }

        private void processRelation(CharSequence name) {
                if (name.equals("tag")) {
                        currentRelation.addTag(getAttr("k"), getAttr("v"));
                } else if (name.equals("member")) {
                        String type = getAttr("type");
                        int id = getIntAttr("ref");
                        String role = getAttr("role");
                        if ("node".equals(type) || "way".equals(type)) {
                                currentRelation.addMember(type, id, role);
                        }
                }
        }

        private static final String[] BOUND_ATTRS = {"minlat", "minlon", "maxlat", "maxlon"};

        private void processBounds() {
                String[] split;
                String boxStr = getAttr("box");
                if (boxStr == null) {
                        split = new String[4];
                        for (int i = 0; i < BOUND_ATTRS.length; i++) {
                                split[i] = getAttr(BOUND_ATTRS[i]);
                                if (split[i] == null) {
                                        System.err.println("A <bounds/> tag was found but it has no 'box' attribute and no '" + BOUND_ATTRS[i] + "' attribute. Ignoring bounds");
                                        return;
                                }
                        }
                } else {
                        split = boxStr.split(",");
                        if (split.length != 4) {
                                System.err.println(
                                                                "A <bounds/> tag was found but its 'box' attribute contains an unexpected number of coordinates (expected 4, found " + split.length + "). Ignoring bounds");
                                return;
                        }
                }
                double[] coords = new double[4];
                int[] mapUnits = new int[4];
                for (int i = 0; i < 4; i++) {
                        try {
                                coords[i] = Double.parseDouble(split[i].trim());
                        } catch (NumberFormatException e) {
                                System.err.println("A <bounds/> tag was found but it contains unexpected data. Unable to parse '" + split[i] + "' as a double. Ignoring bounds");
                                return;
                        }
                        mapUnits[i] = Utils.toMapUnit(coords[i]);
                }
                Area bounds = new Area(mapUnits[0], mapUnits[1], mapUnits[2], mapUnits[3]);

                if (bounds.getMinLong() > bounds.getMaxLong()) {
                        System.out.println("A <bounds/> tag was found but it crosses +/-180 the latitude line (western edge=" +
                                                        Utils.toDegrees(bounds.getMinLong()) + ", eastern=" + Utils.toDegrees(bounds.getMaxLong()) +
                                                        "). The splitter isn't currently able to deal with this, so the bounds are being ignored");
                        return;
                }

                processor.boundTag(bounds);
                System.out.println("A <bounds/> tag was found. Area covered is " + bounds.toString());
        }

        /**
         * Receive notification of the end of an element.
         */

        @Override
        public void endElement(String name) {
                if (state == State.Node) {
                        if (name.equals("node")) {
                                processor.processNode(currentNode);
                                state = State.None;
                                nodeCount++;
                                if (nodeCount % NODE_STATUS_UPDATE_THRESHOLD == 0) {
                                        System.out.println(Utils.format(nodeCount) + " nodes processed...");
                                }
                        }
                } else if (state == State.Way) {
                        if (name.equals("way")) {
                                if (!startNodeOnly)
                                        processor.processWay(currentWay);
                                state = State.None;
                                wayCount++;
                                if (wayCount % WAY_STATUS_UPDATE_THRESHOLD == 0) {
                                        System.out.println(Utils.format(wayCount) + " ways processed...");
                                }
                        }
                } else if (state == State.Relation) {
                        if (name.equals("relation")) {
                                if (!startNodeOnly)
                                        processor.processRelation(currentRelation);
                                state = State.None;
                                relationCount++;
                                if (relationCount % RELATION_STATUS_UPDATE_THRESHOLD == 0) {
                                        System.out.println(Utils.format(relationCount) + " relations processed...");
                                }
                        }
                }
        }
}