Tidy up unused source
[bus.git] / origin-src / osm.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
#!/usr/bin/python
 
# The code in this module was gratuitously stolen from 
# graphserver (http://graphserver.sourceforge.net/)
# Copyright (c) 2007, Brandon Martin-Anderson
 
import xml.sax
import copy
import sys
from math import *
 
class Node:
    def __init__(self, id, lon, lat):
        self.id = id
        self.lon = lon
        self.lat = lat
        self.tags = {}
 
class Way:
    def __init__(self, id, osm):
        self.osm = osm
        self.id = id
        self.nds = []
        self.tags = {}
 
    def split(self, dividers):
        # slice the node-array using this nifty recursive function
        def slice_array(ar, dividers):
            for i in range(1,len(ar)-1):
                if dividers[ar[i]]>1:
                    #print "slice at %s"%ar[i]
                    left = ar[:i+1]
                    right = ar[i:]
 
                    rightsliced = slice_array(right, dividers)
 
                    return [left]+rightsliced
            return [ar]
 
        slices = slice_array(self.nds, dividers)
 
        # create a way object for each node-array slice
        ret = []
        i=0
        for slice in slices:
            littleway = copy.copy( self )
            littleway.id += "-%d"%i
            littleway.nds = slice
            ret.append( littleway )
            i += 1
 
        return ret
 
    def get_projected_points(self, reprojection_func=lambda x,y:(x,y)):
        """nodedir is a dictionary of nodeid->node objects. If reprojection_func is None, returns unprojected points"""
        ret = []
 
        for nodeid in self.nds:
            node = self.osm.nodes[ nodeid ]
            ret.append( reprojection_func(node.lon,node.lat) )
 
        return ret
 
    def to_canonical(self, srid, reprojection_func=None):
        """Returns canonical string for this geometry"""
 
        return "SRID=%d;LINESTRING(%s)"%(srid, ",".join( ["%f %f"%(x,y) for x,y in self.get_projected_points()] ) )
 
    @property
    def fromv(self):
        return self.nds[0]
 
    @property
    def tov(self):
        return self.nds[-1]
 
class OSM:
 
    def __init__(self, filename_or_stream):
        """ File can be either a filename or stream/file object."""
        nodes = {}
        ways = {}
 
        superself = self
 
        class OSMHandler(xml.sax.ContentHandler):
            @classmethod
            def setDocumentLocator(self,loc):
                pass
 
            @classmethod
            def startDocument(self):
                pass
 
            @classmethod
            def endDocument(self):
                pass
 
            @classmethod
            def startElement(self, name, attrs):
                if name=='node':
                    if (int(attrs['id']) % 1000) == 0:
                        print "Parsing node %s" % attrs['id']
                    self.currElem = Node(attrs['id'], float(attrs['lon']), float(attrs['lat']))
                elif name=='way':
                    if (int(attrs['id']) % 1000) == 0:
                        print "Parsing way %s" % attrs['id']
                    self.currElem = Way(attrs['id'], superself)
                elif name=='tag':
                    pass
                    #self.currElem.tags[attrs['k']] = attrs['v']
                elif name=='nd':
                    self.currElem.nds.append( attrs['ref'] )
 
            @classmethod
            def endElement(self,name):
                if name=='node':
                    nodes[self.currElem.id] = self.currElem
                elif name=='way':
                    ways[self.currElem.id] = self.currElem
 
            @classmethod
            def characters(self, chars):
                pass
 
        xml.sax.parse(filename_or_stream, OSMHandler)
 
        self.nodes = nodes
        self.ways = ways
 
        #count times each node is used
        node_histogram = dict.fromkeys( self.nodes.keys(), 0 )
        print "Counting and pruning ways"
        for way in self.ways.values():
            #if a way has only one node, delete it out of the osm collection
            #similarly if it's not a road
            if len(way.nds) < 2:# or not way.tags.get('highway') or way.tags['highway'] == 'footway':  
                del self.ways[way.id]
            else:
                for node in way.nds:
                    # toss out any ways that don't have all nodes on map
                    if not self.nodes.get(node) and self.ways.get(way.id):
                        del self.ways[way.id]
                    elif self.ways.get(way.id):
                        node_histogram[node] += 1
 
        # delete nodes that don't appear in ways
        for node in self.nodes.values():
            if node_histogram[node.id] == 0:
                del self.nodes[node.id]
 
        #use that histogram to split all ways, replacing the member set of ways
        print "Splitting ways"
        new_ways = {}
        for id, way in self.ways.iteritems():
            split_ways = way.split(node_histogram)
            for split_way in split_ways:
                new_ways[split_way.id] = split_way
        self.ways = new_ways
 
    @property
    def connecting_nodes(self):
        """List of nodes that are the endpoint of one or more ways"""
 
        ret = {}
        for way in self.ways.values():
            ret[way.fromv] = self.nodes[way.fromv]
            ret[way.tov] = self.nodes[way.tov]
 
        return ret