-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathdocs.html
194 lines (194 loc) · 19.5 KB
/
docs.html
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
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<meta http-equiv="Content-Style-Type" content="text/css" />
<meta name="generator" content="pandoc" />
<title></title>
<style type="text/css">code{white-space: pre;}</style>
</head>
<body>
<h1 id="speedregions">SpeedRegions</h1>
<p>SpeedRegions is a work-in-progress utility library designed for use with <a href="https://github.com/graphhopper/graphhopper">Graphhopper</a> which allows geographic regions with different speed profiles (e.g. a city or a country) to be defined. SpeedRegions uses several types of text files containing data in <a href="https://en.wikipedia.org/wiki/JSON">JSON</a> format as its input. The most important text file type is an <em>UncompiledSpeedRules</em> file. Here's an example file:</p>
<pre><code>{
"rules" : [ {
"multiplier" : 0.7,
"matchRule" : {
"flagEncoders" : [ "car" ],
"regionTypes" : [ "valleta" ]
}
} ],
"geoJson" : {
"type" : "FeatureCollection",
"features" : [ {
"type" : "Feature",
"properties" : {
"regiontype" : "valleta"
},
"geometry" : {
"type" : "Polygon",
"coordinates" : [ [ [ 14.457384575, 35.89565677 ], [ 14.45653875, 35.899281932 ], [ 14.458598686, 35.902341063 ], [ 14.460328621, 35.903588517 ], [ 14.462642955, 35.903662185 ], [ 14.463669377, 35.903031668 ], [ 14.466885473, 35.903436003 ], [ 14.474620438, 35.900815895 ], [ 14.47560616, 35.902646356 ], [ 14.478262151, 35.903359408 ], [ 14.48170059, 35.903080892 ], [ 14.482825259, 35.905273952 ], [ 14.484661024, 35.906453246 ], [ 14.482064857, 35.909298792 ], [ 14.48261423, 35.911169136 ], [ 14.484530453, 35.912303757 ], [ 14.492025801, 35.914038195 ], [ 14.492099908, 35.916840502 ], [ 14.494218522, 35.918159473 ], [ 14.496863088, 35.917848707 ], [ 14.504244527, 35.914234066 ], [ 14.505821489, 35.912450829 ], [ 14.50531176, 35.910296543 ], [ 14.503015911, 35.90910952 ], [ 14.504048385, 35.908181328 ], [ 14.503548915, 35.905793336 ], [ 14.500927327, 35.904628272 ], [ 14.495495, 35.904402622 ], [ 14.494694111, 35.903789943 ], [ 14.498669763, 35.902729607 ], [ 14.499548669, 35.900796901 ], [ 14.498545044, 35.898904423 ], [ 14.497541163, 35.898554493 ], [ 14.502196889, 35.896830726 ], [ 14.504298724, 35.898354127 ], [ 14.506708899, 35.898981673 ], [ 14.508739395, 35.900704787 ], [ 14.512138017, 35.901597455 ], [ 14.513991283, 35.903362667 ], [ 14.516214595, 35.903592849 ], [ 14.520334468, 35.902897613 ], [ 14.522506008, 35.90169914 ], [ 14.522995309, 35.899607856 ], [ 14.521515746, 35.89784879 ], [ 14.5195844, 35.897552275 ], [ 14.519862388, 35.894473296 ], [ 14.518251389, 35.893098567 ], [ 14.515437882, 35.89195893 ], [ 14.514770674, 35.890967969 ], [ 14.515983301, 35.889085031 ], [ 14.516154084, 35.88682177 ], [ 14.515175118, 35.884706714 ], [ 14.516435314, 35.88291993 ], [ 14.516802508, 35.88092412 ], [ 14.515407889, 35.879252265 ], [ 14.51298533, 35.878784166 ], [ 14.50765095, 35.879475078 ], [ 14.505195661, 35.878075059 ], [ 14.502811432, 35.878618985 ], [ 14.500843023, 35.879815169 ], [ 14.498347497, 35.878257795 ], [ 14.49584433, 35.878623366 ], [ 14.494338579, 35.880284308 ], [ 14.494603199, 35.881985166 ], [ 14.484047356, 35.882519382 ], [ 14.481711218, 35.883442279 ], [ 14.481073397, 35.884899684 ], [ 14.478373636, 35.884188849 ], [ 14.470820536, 35.88460609 ], [ 14.468694102, 35.885361006 ], [ 14.46782897, 35.887694007 ], [ 14.463597861, 35.886755088 ], [ 14.461721805, 35.887686648 ], [ 14.458288577, 35.891024366 ], [ 14.457494404, 35.892720693 ], [ 14.457384575, 35.89565677 ] ] ]
}
} ]
}
}</code></pre>
<p>This file has two sections - (1) the array called <em>rules</em>, which contains a single speed rule and (2) the object geoJson, which must always be a <a href="http://geojson.org/geojson-spec.html#feature-collection-objects">geoJson feature collection</a>. This feature collection contains a single feature which defines a region around the city of Valleta, in Malta. The single speed rule applies to the Graphhopper car flag encoder (i.e. the speed profile for cars) and slows down speeds by 30% (multiplier = 0.7) within Valleta. The <em>regionType</em> field links the Valleta geoJson feature with the speed rule - so different speed rules can be applied to the same geographic region (for example, different speed rules for a truck or motorbike).</p>
<h2 id="compiled-vs-uncompiled-files">Compiled vs uncompiled files</h2>
<p>An uncompiled file is called 'uncompiled' because the SpeedRegions library still needs to build a spatial lookup (a kind of map) which stores the geographic regions in a data structure optimised for querying the region a road sits within. Building this spatial lookup can be slow and take a similar amount of time to building the graph - e.g. a couple of minutes just for the United Kingdom dependent on accuracy. As speed rules will change often (e.g. make a road type a little faster or slower) but the geographic regions won't change as often, we support pre-compiling this spatial tree before using it in Graphhopper. You can therefore compile the spatial tree once and then quickly modify the speed rules in the compiled file without having to recompile the tree.</p>
<h2 id="types-of-json-text-file">Types of JSON text file</h2>
<p>When building the Graphhopper graph, you can give SpeedRegions one of two different types of JSON text files:</p>
<ol style="list-style-type: decimal">
<li><strong>UncompiledSpeedRules</strong>. A JSON text file containing the speed rules and GeoJSON feature collection.</li>
<li><strong>CompiledSpeedRules</strong>. A JSON text file containing the speed rules and the built spatial tree.</li>
</ol>
<p>To help manipulate these files, we have a command line tools project. The command line tools loads and saves the following JSON file types:</p>
<ol style="list-style-type: decimal">
<li><strong>FeatureCollection</strong>. A GeoJSON text file containing a single feature collection object.</li>
<li><strong>CompiledTree</strong>. A JSON text file the containing built spatial tree.</li>
</ol>
<p>You can use the command line tools to build the <em>CompiledTree</em> file from a <em>FeatureCollection</em> file. You build this tree once (and once only), then place it into a <em>CompiledSpeedRules</em> file. You can then tweak the speed rules in the <em>CompiledSpeedRules</em> as you like, without rebuilding the tree again. Alternatively if you don't mind the extra time taken to rebuild the spatial tree when you're building the Graphhopper graph, you can just use the <em>UncompiledSpeedRules</em> file as input without bothering with the other file types.</p>
<p>This is an example <em>CompiledSpeedRules</em> file:</p>
<pre><code>{
"rules" : [ {
"speedsByRoadType" : { },
"multiplier" : 0.7,
"speedUnit" : "KM_PER_HOUR",
"matchRule" : {
"flagEncoders" : [ "car" ],
"regionTypes" : [ "valleta" ]
}
} ],
"tree" : {
"bounds" : {
"minLng" : 14.4140625,
"maxLng" : 14.58984375,
"minLat" : 35.859375,
"maxLat" : 35.947265625
},
"regionType" : "valleta",
"assignedPriority" : 1,
"children" : [ ]
}
}</code></pre>
<p>In this case the tree is very small - it contains only one node. Generally speaking the tree will be a very large multi-level data structure with thousands or more nodes.</p>
<p>This is an example <em>FeatureCollection</em> file - it is pure geoJson and corresponds to the <em>geoJson</em> field in the <em>UncompiledSpeedRules</em> file:</p>
<pre><code>{
"type" : "FeatureCollection",
"features" : [ {
"type" : "Feature",
"properties" : {
"regiontype" : "valleta"
},
"geometry" : {
"type" : "Polygon",
"coordinates" : [ [ [ 14.457384575, 35.89565677 ], [ 14.45653875, 35.899281932 ], [ 14.458598686, 35.902341063 ], [ 14.460328621, 35.903588517 ], [ 14.462642955, 35.903662185 ], [ 14.463669377, 35.903031668 ], [ 14.466885473, 35.903436003 ], [ 14.474620438, 35.900815895 ], [ 14.47560616, 35.902646356 ], [ 14.478262151, 35.903359408 ], [ 14.48170059, 35.903080892 ], [ 14.482825259, 35.905273952 ], [ 14.484661024, 35.906453246 ], [ 14.482064857, 35.909298792 ], [ 14.48261423, 35.911169136 ], [ 14.484530453, 35.912303757 ], [ 14.492025801, 35.914038195 ], [ 14.492099908, 35.916840502 ], [ 14.494218522, 35.918159473 ], [ 14.496863088, 35.917848707 ], [ 14.504244527, 35.914234066 ], [ 14.505821489, 35.912450829 ], [ 14.50531176, 35.910296543 ], [ 14.503015911, 35.90910952 ], [ 14.504048385, 35.908181328 ], [ 14.503548915, 35.905793336 ], [ 14.500927327, 35.904628272 ], [ 14.495495, 35.904402622 ], [ 14.494694111, 35.903789943 ], [ 14.498669763, 35.902729607 ], [ 14.499548669, 35.900796901 ], [ 14.498545044, 35.898904423 ], [ 14.497541163, 35.898554493 ], [ 14.502196889, 35.896830726 ], [ 14.504298724, 35.898354127 ], [ 14.506708899, 35.898981673 ], [ 14.508739395, 35.900704787 ], [ 14.512138017, 35.901597455 ], [ 14.513991283, 35.903362667 ], [ 14.516214595, 35.903592849 ], [ 14.520334468, 35.902897613 ], [ 14.522506008, 35.90169914 ], [ 14.522995309, 35.899607856 ], [ 14.521515746, 35.89784879 ], [ 14.5195844, 35.897552275 ], [ 14.519862388, 35.894473296 ], [ 14.518251389, 35.893098567 ], [ 14.515437882, 35.89195893 ], [ 14.514770674, 35.890967969 ], [ 14.515983301, 35.889085031 ], [ 14.516154084, 35.88682177 ], [ 14.515175118, 35.884706714 ], [ 14.516435314, 35.88291993 ], [ 14.516802508, 35.88092412 ], [ 14.515407889, 35.879252265 ], [ 14.51298533, 35.878784166 ], [ 14.50765095, 35.879475078 ], [ 14.505195661, 35.878075059 ], [ 14.502811432, 35.878618985 ], [ 14.500843023, 35.879815169 ], [ 14.498347497, 35.878257795 ], [ 14.49584433, 35.878623366 ], [ 14.494338579, 35.880284308 ], [ 14.494603199, 35.881985166 ], [ 14.484047356, 35.882519382 ], [ 14.481711218, 35.883442279 ], [ 14.481073397, 35.884899684 ], [ 14.478373636, 35.884188849 ], [ 14.470820536, 35.88460609 ], [ 14.468694102, 35.885361006 ], [ 14.46782897, 35.887694007 ], [ 14.463597861, 35.886755088 ], [ 14.461721805, 35.887686648 ], [ 14.458288577, 35.891024366 ], [ 14.457494404, 35.892720693 ], [ 14.457384575, 35.89565677 ] ] ]
}
} ]
}</code></pre>
<p>This is our example <em>CompiledTree</em> file, containing only a single node; it corresponds to the <em>tree</em> field in the <em>CompiledSpeedRules</em> file:</p>
<pre><code>{
"bounds" : {
"minLng" : 14.4140625,
"maxLng" : 14.58984375,
"minLat" : 35.859375,
"maxLat" : 35.947265625
},
"regionType" : "valleta",
"assignedPriority" : 1,
"children" : [ ]
}</code></pre>
<h2 id="configuration-of-speed-rules-data">Configuration of speed rules data</h2>
<p>You can have as many speed rules as you like in the rules array. Each speed rule can define a speed for a road type and a speed multiplier which is applied to a road type when either (a) no speed is specified in the rule for the road type or (b) the OSM <em>maxspeed</em> tag was present on the road edge and was used instead of the default speed for the road type. The speeds for road types can be set in MILES_PER_HOUR or KM_PER_HOUR depending on the value of the <em>speedUnit</em> field.</p>
<p>Speed rules can have a parent-child hierarchy, where you specify a default set of speeds for an area, for example a country, and give the rule containing these speeds an id. Child rules can inherit the default speeds and multiplier from their parent rule if you set their parentId field, and then override the speeds as needed. A child rule multiplies its parent's speeds by its <em>multiplier</em> field value.</p>
<p>The following <em>CompiledSpeedRules</em> example sets up speeds in MILES_PER_HOUR for different types of roads in the UK, following the OpenStreetMap <a href="http://wiki.openstreetmap.org/wiki/Key:highway">highway tag conventions</a>. A rule is setup with id 'TypicalUKSpeeds' to apply to the regionType 'UK', which would correspond to a region covering the whole UK. Different rules for regionTypes London and Birmingham are then setup with a parentId of TypicalUKSpeeds and a multiplier of between 0.6-0.8. The rules for London and Birmingham will therefore use the speeds setup in their parent rule record (the UK record), but reduced by between 60% and 80%.</p>
<p>To specify speed in kilometres instead set the <em>speedUnit</em> field to KM_PER_HOUR.</p>
<pre><code>{
"rules" : [
{
"parentId" : "TypicalUKSpeeds",
"multiplier" : 0.6,
"matchRule" : {
"flagEncoders" : [ "car" ],
"regionTypes" : [ "InnerLondon" ]
}
}
,{
"parentId" : "TypicalUKSpeeds",
"multiplier" : 0.8,
"matchRule" : {
"flagEncoders" : [ "car" ],
"regionTypes" : [ "OuterLondon" ]
}
}
,{
"parentId" : "TypicalUKSpeeds",
"multiplier" : 0.8,
"matchRule" : {
"flagEncoders" : [ "car" ],
"regionTypes" : [ "Birmingham" ]
}
}
,{
"id" : "TypicalUKSpeeds",
"speedsByRoadType" : {
"living_street" : 3.1,
"motorroad" : 55.9,
"motorway" : 62.1,
"motorway_link" : 43.5,
"primary" : 40.4,
"primary_link" : 37.3,
"residential" : 18.6,
"road" : 12.4,
"secondary" : 37.3,
"secondary_link" : 31.1,
"service" : 12.4,
"tertiary" : 31.1,
"tertiary_link" : 24.9,
"track" : 9.3,
"trunk" : 43.5,
"trunk_link" : 40.4,
"unclassified" : 18.6
},
"multiplier" : 1.0,
"speedUnit" : "MILES_PER_HOUR",
"matchRule" : {
"flagEncoders" : [ "car" ],
"regionTypes" : [ "UK" ]
}
}
],
"tree" : {
....
}
}</code></pre>
<h2 id="configuration-of-feature-collection-data">Configuration of feature collection data</h2>
<p>A featureCollection should only contain <a href="http://geojson.org/geojson-spec.html#polygon">Polygon</a> or <a href="http://geojson.org/geojson-spec.html#multipolygon">MultiPolygon</a> geometry types. Each feature should have a <strong>regionType</strong> property which links to the speed rules. RegionType could refer to a geographic area - e.g. London - or a type of geographic area (e.g. 'big city'). Multiple features can therefore have the same regionType, there is no requirement for it to be a unique value per feature.</p>
<p>The order of a feature in the featureCollection is used as its priority when assigning a geographic area (technically a leaf node in the spatial tree) to a regionType. If for example, you have a featureCollection containing polygons boundaries for United Kingdom, Central London and Outer London, your first feature should be the smallest or innermost feature - Central London - followed by Outer London (which Central London sits within) and then the United Kingdom. The order within the collection is therefore used to model overlapping polygons where one polygon sits within another.</p>
<h2 id="building-a-graphhopper-graph-with-speed-regions">Building a Graphhopper graph with speed regions</h2>
<p>The integration with Graphhopper is currently experimental and available for car speed profile only. Full integration within the Graphhopper project is planned. See the projects com.opendoorlogistics.speedregions.experimental.gh0.5 and com.opendoorlogistics.speedregions.experimental.ghlatest to build a car profile graph using speed regions with Graphhopper 0.5 and Graphhopper latest release (version 0.7 as of 28/9/2016).</p>
<p>To build a Graphhopper graph using speed regions, build the projects using Maven and run from the command line as shown <a href="http://www.opendoorlogistics.com/tutorials/tutorial-vi-advance-configuration/building-road-network-graphs/">here</a> but with one of two sets of additional command line arguments:</p>
<ul>
<li><p>Using a <em>CompiledSpeedRules</em> file. Add the argument:</p>
<pre><code>speedregions.compiled=your_compiled_speed_rules_filename</code></pre></li>
<li><p>Using an <em>UncompiledSpeedRules</em> file. Add the two arguments:</p>
<pre><code>speedregions.uncompiled=your_uncompiled_speed_rules_filename
speedregions.tolerance=tolerance_in_metres</code></pre></li>
</ul>
<h3 id="choosing-the-speed-regions-tolerance-value">Choosing the speed regions tolerance value</h3>
<p>The tolerance measures how accurate the spatial tree will be. Technically speaking, the spatial tree divides the world up into rectangles, assigning each rectangle to a single region. The tolerance is the minimum side length of these rectangles; so no rectangle will be created with a side length less than <em>tolerance_in_metres</em>.</p>
<p>Road edges whose centre is within <em>tolerance_in_metres</em> of a region boundary may therefore be placed in the wrong region. Generally speaking, the effect of placing an edge already near a region boundary within the wrong region will be small, so tolerance can be a relatively high value. The tolerance value should however be signficantly smaller than the approximate width or height of your regions, or your regions will not be represented properly or not even used. The building of the spatial tree (i.e. the compilation step) takes a lot longer the smaller the tolerance is and the compiled files will be a lot larger.</p>
<p>We recommend using a starting value of 500m and only reducing this if needed.</p>
<h2 id="visualising-regions-and-the-tree-in-odl-studio">Visualising regions and the tree in ODL Studio</h2>
<p>ODL Studio can be used for visualising the tree (TODO DOCUMENT THIS).</p>
<div class="figure">
<img src="http://www.opendoorlogistics.com/wp-content/uploads/speedregions/London-route-comparison.png" alt="Speed regions shown for London built with 100m tolerance, and a route superimposed with and without the regions." />
<p class="caption">Speed regions shown for London built with 100m tolerance, and a route superimposed with and without the regions.</p>
</div>
<h2 id="future-developments">Future developments</h2>
<ul>
<li><p>Full integration into Graphhopper project.</p></li>
<li><p>Binary format for compiled tree - the large size of the compile tree in the CompiledSpeedRules file can limit the practicallity of speed regions to national level. The file size could be reduced dramatically with a small amount of future development work (for example, the compressed file size of the compiled JSON is just ~3% of the uncompressed).</p></li>
<li><p>Integrate speeds based on <em>trackType</em> tag, currently we ignore these.</p></li>
</ul>
</body>
</html>