diff --git a/INSTALL.md b/INSTALL.md index d5a7c7a1f1..af8d28fac3 100644 --- a/INSTALL.md +++ b/INSTALL.md @@ -5,7 +5,7 @@ You need OpenStreetMap data loaded into a PostGIS database (see below for [depen It's probably easiest to grab an PBF of OSM data from [Mapzen](https://mapzen.com/metro-extracts/) or [geofabrik](http://download.geofabrik.de/). Once you've set up your PostGIS database, import with osm2pgsql: ``` -osm2pgsql -d gis ~/path/to/data.osm.pbf --style openstreetmap-carto.style +osm2pgsql -G --hstore -d gis ~/path/to/data.osm.pbf --style openstreetmap-carto.style --tag-transform-script ~/osm/openstreetmap-carto.lua ``` You can find a more detailed guide to setting up a database and loading data with osm2pgsql at [switch2osm.org](http://switch2osm.org/loading-osm-data/). diff --git a/openstreetmap-carto.lua b/openstreetmap-carto.lua new file mode 100644 index 0000000000..5ddfd9bd57 --- /dev/null +++ b/openstreetmap-carto.lua @@ -0,0 +1,447 @@ +-- For documentation of Lua tag transformations, see: +-- https://github.com/openstreetmap/osm2pgsql/blob/master/docs/lua.md + +-- Custom keys that are defined by this file +custom_keys = { 'z_order', 'osmcarto_z_order' } + +-- Objects with any of the following keys will be treated as polygon +polygon_keys = { 'building', 'landuse', 'amenity', 'harbour', 'historic', 'leisure', + 'man_made', 'military', 'natural', 'office', 'place', 'power', + 'public_transport', 'shop', 'sport', 'tourism', 'waterway', + 'wetland', 'water', 'aeroway', 'abandoned:aeroway', 'abandoned:amenity', + 'abandoned:building', 'abandoned:landuse', 'abandoned:power', 'area:highway' } + +-- Objects with any of the following key/value combinations will be treated as polygon +polygon_values = { + {'highway', 'services'}, + {'junction', 'yes'} + } + +-- Objects with any of the following key/value combinations will be treated as linestring +linestring_values = { + {'leisure', 'track'}, + {'man_made', 'embankment'}, + {'man_made', 'breakwater'}, + {'man_made', 'groyne'}, + {'natural', 'cliff'}, + {'natural', 'tree_row'}, + {'historic', 'citywalls'}, + {'waterway', 'canal'}, + {'waterway', 'derelict_canal'}, + {'waterway', 'ditch'}, + {'waterway', 'drain'}, + {'waterway', 'river'}, + {'waterway', 'stream'}, + {'waterway', 'wadi'}, + {'waterway', 'weir'}, + {'power', 'line'}, + {'power', 'minor_line'} + } + +-- The following keys will be deleted +delete_tags = { + 'note', + 'note:.*', + 'source', + 'source_ref', + 'source:.*', + 'attribution', + 'comment', + 'fixme', + -- Tags generally dropped by editors, not otherwise covered + 'created_by', + 'odbl', + 'odbl:note', + -- Lots of import tags + -- EUROSHA (Various countries) + 'project:eurosha_2012', + -- Corine (CLC) (Europe) + 'CLC:.*', + + -- UrbIS (Brussels, BE) + 'ref:UrbIS', + + -- Geobase (CA) + 'geobase:.*', + -- NHN (CA) + 'accuracy:meters', + 'sub_sea:type', + 'waterway:type', + -- CanVec (CA) + 'canvec:.*', + -- StatsCan (CA) + 'statscan:rbuid', + -- Geobase (CA) + 'geobase:.*', + + -- RUIAN (CZ) + 'ref:ruian:addr', + 'ref:ruian', + 'building:ruian:type', + -- DIBAVOD (CZ) + 'dibavod:id', + -- UIR-ADR (CZ) + 'uir_adr:ADRESA_KOD', + + -- osak (DK) + 'osak:.*', + -- kms (DK) + 'kms:.*', + -- GST (DK) + 'gst:feat_id', + + -- Maa-amet (EE) + 'maaamet:ETAK', + + -- ngbe (ES) + -- See also note:es and source:file above + 'ngbe:.*', + + -- FANTOIR (FR) + 'ref:FR:FANTOIR', + + -- Friuli Venezia Giulia (IT) + 'it:fvg:.*', + + -- KSJ2 (JA) + -- See also note:ja and source_ref above + 'KSJ2:.*', + -- Yahoo/ALPS (JA) + 'yh:.*', + + -- 3dshapes (NL) + '3dshapes:ggmodelk', + -- AND (NL) + 'AND_nosr_r', + + -- OPPDATERIN (NO) + 'OPPDATERIN', + + -- LINZ (NZ) + 'LINZ2OSM:.*', + 'linz2osm:.*', + 'LINZ:.*', + + -- Various imports (PL) + 'addr:city:simc', + 'addr:street:sym_ul', + 'building:usage:pl', + 'building:use:pl', + -- WroclawGIS (PL) + 'WroclawGIS:.*', + -- TERYT (PL) + 'teryt:simc', + + -- RABA (SK) + 'raba:id', + + -- Naptan (UK) + 'naptan:.*', + + -- TIGER (US) + 'tiger:.*', + -- GNIS (US) + 'gnis:.*', + -- DCGIS (Washington DC, US) + 'dcgis:gis_id', + -- National Hydrography Dataset (US) + 'NHD:.*', + 'nhd:.*', + -- Building Identification Number (New York, US) + 'nycdoitt:bin', + -- Chicago Building Inport (US) + 'chicago:building_id', + -- Louisville, Kentucky/Building Outlines Import (US) + 'lojic:bgnum', + -- MassGIS (Massachusetts, US) + 'massgis:way_id', + + -- mvdgis (Montevideo, UY) + 'mvdgis:.*', + + -- misc + 'import', + 'import_uuid', + 'OBJTYPE', + 'SK53_bulk:load' +} + + +-- Array used to specify z_order and osmcarto_z_order per key/value combination. +-- The former is used for backwards compatibility and for uses that use a single +-- database for multiple rendering styles. +-- Each element has the form {key, value, z_order, osmcarto_z_order, is_road}. +-- If is_road=1, the object will be added to planet_osm_roads. +zordering_tags = { + { 'railway', nil, 5, 0, 1}, + { 'boundary', 'administrative', 0, 0, 1}, + { 'bridge', 'yes', 10, 0, 0 }, + { 'bridge', 'true', 10, 0, 0 }, + { 'bridge', 1, 10, 0, 0 }, + { 'tunnel', 'yes', -10, 0, 0}, + { 'tunnel', 'true', -10, 0, 0}, + { 'tunnel', 1, -10, 0, 0}, + {'railway', 'rail', 0, 440, 1}, + {'railway', 'subway', 0, 420, 1}, + {'railway', 'narrow_gauge', 0, 420, 1}, + {'railway', 'light_rail', 0, 420, 1}, + {'railway', 'preserved', 0, 420, 1}, + {'railway', 'funicular', 0, 420, 1}, + {'railway', 'monorail', 0, 420, 1}, + {'railway', 'miniature', 0, 420, 1}, + {'railway', 'turntable', 0, 420, 1}, + {'railway', 'tram', 0, 410, 1}, + {'railway', 'tram-service', 0, 405, 1}, + {'railway', 'disused', 0, 400, 1}, + {'railway', 'construction', 0, 400, 1}, + {'highway', 'motorway', 9, 380, 1}, + {'highway', 'trunk', 8, 370, 1}, + {'highway', 'primary', 7, 360, 1}, + {'highway', 'secondary', 6, 350, 1}, + {'highway', 'tertiary', 4, 340, 0}, + {'highway', 'residential', 3, 330, 0}, + {'highway', 'unclassified', 3, 330, 0}, + {'highway', 'road', 3, 330, 0}, + {'highway', 'living_street', 0, 320, 0}, + {'highway', 'pedestrian', 0, 310, 0}, + {'highway', 'raceway', 0, 300, 0}, + {'highway', 'motorway_link', 9, 240, 1}, + {'highway', 'trunk_link', 6, 230, 1}, + {'highway', 'primary_link', 5, 220, 1}, + {'highway', 'secondary_link', 4, 210, 1}, + {'highway', 'tertiary_link', 3, 200, 0}, + {'highway', 'service', 0, 150, 0}, + {'highway', 'track', 0, 110, 0}, + {'highway', 'path', 0, 100, 0}, + {'highway', 'footway', 0, 100, 0}, + {'highway', 'bridleway', 0, 100, 0}, + {'highway', 'cycleway', 0, 100, 0}, + {'highway', 'steps', 0, 100, 0}, + {'highway', 'platform', 0, 90, 0}, + {'highway', 'minor', 3, 0, 0}, + {'railway', 'platform', 0, 90, 0}, + {'aeroway', 'runway', 0, 60, 0}, + {'aeroway', 'taxiway', 0, 50, 0}, + {'highway', 'construction', 0, 10, 0} +} + +function add_z_order(keyvalues) + -- The default z_order is 0 + z_order = 0 + osmcarto_z_order = 0 + + -- Increase or decrease z_order based on the specific key/value combination as specified in zordering_tags + for i,k in ipairs(zordering_tags) do + -- If the value in zordering_tags is specified, match key and value. Otherwise, match key only. + if ((k[2] and keyvalues[k[1]] == k[2]) or (k[2] == nil and keyvalues[k[1]] ~= nil)) then + -- If the fifth component of the element of zordering_tags is 1, add the object to planet_osm_roads + if (k[5] == 1) then + roads = 1 + end + z_order = z_order + k[3] + osmcarto_z_order = math.max(osmcarto_z_order, k[4]) + end + end + + -- Add z_order as key/value combination + keyvalues["osmcarto_z_order"] = osmcarto_z_order + keyvalues["z_order"] = z_order + + return keyvalues, roads +end + +-- Filtering on nodes, ways, and relations +function filter_tags_generic(keyvalues, numberofkeys) + filter = 0 -- Will object be filtered out? + + -- Delete tags listed in delete_tags + for k, v in pairs (keyvalues) do + match = false + for _, d in ipairs(delete_tags) do + match = match or string.find(k, d) + end + if match then + keyvalues[k] = nil + numberofkeys = numberofkeys - 1 + end + end + + -- Filter out objects with 0 tags + if numberofkeys == 0 then + filter = 1 + return filter, keyvalues + end + + return filter, keyvalues +end + +-- Filtering on nodes +function filter_tags_node (keyvalues, numberofkeys) + return filter_tags_generic(keyvalues, numberofkeys) +end + +-- Filtering on relations +function filter_basic_tags_rel (keyvalues, numberofkeys) + -- Filter out objects that are filtered out by filter_tags_generic + filter, keyvalues = filter_tags_generic(keyvalues, numberofkeys) + if filter == 1 then + return filter, keyvalues + end + + -- Filter out all relations except route, multipolygon and boundary relations + if ((keyvalues["type"] ~= "route") and (keyvalues["type"] ~= "multipolygon") and (keyvalues["type"] ~= "boundary")) then + filter = 1 + return filter, keyvalues + end + + return filter, keyvalues +end + +-- Filtering on ways +function filter_tags_way (keyvalues, numberofkeys) + filter = 0 -- Will object be filtered out? + polygon = 0 -- Will object be treated as polygon? + roads = 0 -- Will object be added to planet_osm_roads? + + -- Filter out objects that are filtered out by filter_tags_generic + filter, keyvalues = filter_tags_generic(keyvalues, numberofkeys) + if filter == 1 then + return filter, keyvalues, polygon, roads + end + + -- Treat objects with a key in polygon_keys as polygon + for i,k in ipairs(polygon_keys) do + if keyvalues[k] then + polygontag = 1 + -- However, if the key/value combination occurs in linestring_values, do not treat the object as polygon + for index,tag in pairs(linestring_values) do + if k == tag[1] and keyvalues[k] == tag[2] then + polygontag = 0 + break + end + end + if polygontag == 1 then + polygon = 1 + break + end + end + end + + -- Treat objects with a key/value combination in polygon_values as polygon + if polygon == 0 then + for index,tag in pairs(polygon_values) do + if keyvalues[tag[1]] == tag[2] then + polygon=1 + break + end + end + end + + -- Treat objects tagged as area=yes, area=1, or area=true as polygon, + -- and treat objects tagged as area=no, area=0, or area=false not as polygon + if ((keyvalues["area"] == "yes") or (keyvalues["area"] == "1") or (keyvalues["area"] == "true")) then + polygon = 1; + elseif ((keyvalues["area"] == "no") or (keyvalues["area"] == "0") or (keyvalues["area"] == "false")) then + polygon = 0; + end + + -- Add z_order key/value combination and determine if the object should also be added to planet_osm_roads + keyvalues, roads = add_z_order(keyvalues) + + return filter, keyvalues, polygon, roads +end + +function filter_tags_relation_member (keyvalues, keyvaluemembers, roles, membercount) + filter = 0 -- Will object be filtered out? + linestring = 0 -- Will object be treated as linestring? + polygon = 0 -- Will object be treated as polygon? + roads = 0 -- Will object be added to planet_osm_roads? + membersuperseded = {} + for i = 1, membercount do + membersuperseded[i] = 0 -- Will member be ignored when handling areas? + end + + type = keyvalues["type"] + + -- Remove type key + keyvalues["type"] = nil + + -- Relations with type=boundary are treated as linestring + if (type == "boundary") then + linestring = 1 + end + -- Relations with type=multipolygon and boundary=* are treated as linestring + if ((type == "multipolygon") and keyvalues["boundary"]) then + linestring = 1 + -- For multipolygons... + elseif (type == "multipolygon") then + -- Treat as polygon + polygon = 1 + haspolygontags = false + -- Count the number of polygon tags + -- First count keys in polygon_keys + for i,k in ipairs(polygon_keys) do + if keyvalues[k] then + polygontag = 1 + -- However, if the key/value combination occurs in linestring_values, do not count the object as polygon + for index,tag in pairs(linestring_values) do + if k == tag[1] and keyvalues[k] == tag[2] then + polygontag = 0 + break + end + end + if polygontag == 1 then + haspolygontags = true + break + end + end + end + -- Treat objects with a key/value combination in polygon_values as polygon + if not haspolygontags then + for index,tag in pairs(polygon_values) do + if keyvalues[tag[1]] == tag[2] then + haspolygontags = true + break + end + end + end + -- Support for old-style multipolygons (1/2): + -- If there are no polygon tags, add tags from all outer elements to the multipolygon itself + if not haspolygontags then + for i = 1,membercount do + if (roles[i] == "outer") then + for k,v in pairs(keyvaluemembers[i]) do + keyvalues[k] = v + end + end + end + end + -- Support for old-style multipolygons (2/2): + -- For any member of the multipolygon, set membersuperseded to 1 (i.e. don't deal with it as area as well), + -- except when the member has a (non-custom) key/value combination that is not also a key/value combination of the multipolygon itself + for i = 1,membercount do + superseded = 1 + for k,v in pairs(keyvaluemembers[i]) do + if ((keyvalues[k] == nil or keyvalues[k] ~= v) and not is_in(k,custom_keys)) then + superseded = 0; + break + end + end + membersuperseded[i] = superseded + end + end + + -- Add z_order key/value combination and determine if the object should also be added to planet_osm_roads + keyvalues, roads = add_z_order(keyvalues) + + return filter, keyvalues, membersuperseded, linestring, polygon, roads +end + +function is_in (needle, haystack) + for index, value in ipairs (haystack) do + if value == needle then + return true + end + end + return false +end diff --git a/openstreetmap-carto.style b/openstreetmap-carto.style index f2322cf322..8d6cb05aff 100644 --- a/openstreetmap-carto.style +++ b/openstreetmap-carto.style @@ -1,5 +1,82 @@ -# This is the .style file for OpenStreetMap Carto, which is currently -# the same as the upstream osm2pgsql style +# This is the osm2pgsql .style file for openstreetmap-carto. +# +# A .style file has 4 columns that define how OSM objects end up in tables in +# the database and what columns are created. It interacts with the command-line +# hstore options. +# +# Columns +# ======= +# +# OsmType: This is either "node", "way" or "node,way" and indicates if this tag +# applies to nodes, ways, or both. +# +# Tag: The tag +# +# DataType: The type of the column to be created. Normally "text" +# +# Flags: Flags that indicate what table the OSM object is moved into. +# +# There are 6 possible flags. These flags are used both to indicate if a column +# should be created, and if ways with the tag are assumed to be areas. The area +# assumptions can be overridden with an area=yes/no tag +# +# polygon - Create a column for this tag, and objects the tag with are areas +# +# linear - Create a column for this tag +# +# nocolumn - Override the above and don't create a column for the tag, but do +# include objects with this tag +# +# phstore - Same as polygon,nocolumn for backward compatibility +# +# delete - Drop this tag completely and don't create a column for it. This also +# prevents the tag from being added to hstore columns +# +# nocache - Deprecated and does nothing +# +# If an object has a tag that indicates it is an area or has area=yes/1, +# osm2pgsql will try to turn it into an area. If it succeeds, it places it in +# the polygon table. If it fails (e.g. not a closed way) it places it in the +# line table. +# +# Nodes are never placed into the polygon or line table and are always placed in +# the point table. +# +# Hstore +# ====== +# +# The options --hstore, --hstore-match-only, and --hstore-all interact with +# the .style file. +# +# With --hstore any tags without a column will be added to the hstore column. +# This will also cause all objects to be kept. +# +# With --hstore-match-only the behavior for tags is the same, but objects are +# only kept if they have a non-NULL value in one of the columns. +# +# With --hstore-all all tags are added to the hstore column unless they appear +# in the style file with a delete flag, causing duplication between the normal +# columns and the hstore column. +# +# Special database columns +# ======================== +# +# There are some special database columns that if present in the .style file +# will be populated by osm2pgsql. +# +# These are +# +# way_area - datatype real. The area of the way, in the units of the projection +# (e.g. square mercator meters). Only applies to areas +# +# osm_user - datatype text +# osm_uid - datatype integer +# osm_version - datatype integer +# osm_changeset - datatype integer +# osm_timestamp - datatype timestamptz(0). +# Used with the --extra-attributes option to include metadata in the database. +# If importing with both --hstore and --extra-attributes the meta-data will +# end up in the tags hstore column regardless of the style file. # OsmType Tag DataType Flags node,way access text linear @@ -10,25 +87,19 @@ node,way admin_level text linear node,way aerialway text linear node,way aeroway text polygon node,way amenity text polygon -node,way area text # hard coded support for area=1/yes => polygon is in osm2pgsql +node,way area text polygon # hard coded support for area=1/yes => polygon is in osm2pgsql node,way barrier text linear -node,way bicycle text -node,way brand text linear +node,way bicycle text linear node,way bridge text linear node,way boundary text linear node,way building text polygon node capital text linear node,way construction text linear node,way covered text linear -node,way culvert text linear -node,way cutting text linear node,way denomination text linear -node,way disused text linear -node ele text linear -node,way embankment text linear +node,way ele text linear node,way foot text linear node,way generator:source text linear -node,way harbour text polygon node,way highway text linear node,way historic text polygon node,way horse text linear @@ -40,114 +111,30 @@ node,way leisure text polygon node,way lock text linear node,way man_made text polygon node,way military text polygon -node,way motorcar text linear node,way name text linear node,way natural text polygon # natural=coastline tags are discarded by a hard coded rule in osm2pgsql -node,way office text polygon node,way oneway text linear node,way operator text linear node,way place text polygon -node poi text node,way population text linear node,way power text polygon node,way power_source text linear -node,way public_transport text polygon node,way railway text linear node,way ref text linear -node,way religion text nocache +node,way religion text linear node,way route text linear node,way service text linear node,way shop text polygon -node,way sport text polygon node,way surface text linear -node,way toll text linear node,way tourism text polygon -node,way tower:type text linear way tracktype text linear node,way tunnel text linear node,way water text polygon node,way waterway text polygon node,way wetland text polygon node,way width text linear -node,way wood text linear -node,way z_order int4 linear # This is calculated during import -way way_area real # This is calculated during import +way way_area real linear # This is calculated during import -# Area tags -# We don't make columns for these tags, but objects with them are areas. -way abandoned:aeroway text phstore -way abandoned:amenity text phstore -way abandoned:building text phstore -way abandoned:landuse text phstore -way abandoned:power text phstore -way area:highway text phstore - -# Deleted tags -# These are tags that are generally regarded as useless for most rendering. -# Most of them are from imports or intended as internal information for mappers -# Some of them are automatically deleted by editors. -# If you want some of them, perhaps for a debugging layer, just delete the lines. - -# These tags are used by mappers to keep track of data. -# They aren't very useful for rendering. -node,way note text delete -node,way note:* text delete -node,way source text delete -node,way source_ref text delete -node,way source:* text delete -node,way attribution text delete -node,way comment text delete -node,way fixme text delete - -# Tags generally dropped by editors, not otherwise covered -node,way created_by text delete -node,way odbl text delete -node,way odbl:note text delete -node,way SK53_bulk:load text delete - -# Lots of import tags -# TIGER (US) -node,way tiger:* text delete - -# NHD (US) -# NHD has been converted every way imaginable -node,way NHD:* text delete -node,way nhd:* text delete - -# GNIS (US) -node,way gnis:* text delete - -# Geobase (CA) -node,way geobase:* text delete -# NHN (CA) -node,way accuracy:meters text delete -node,way sub_sea:type text delete -node,way waterway:type text delete - -# KSJ2 (JA) -# See also note:ja and source_ref above -node,way KSJ2:* text delete -# Yahoo/ALPS (JA) -node,way yh:* text delete - -# osak (DK) -node,way osak:* text delete - -# kms (DK) -node,way kms:* text delete - -# ngbe (ES) -# See also note:es and source:file above -node,way ngbe:* text delete - -# naptan (UK) -node,way naptan:* text delete - -# Corine (CLC) (Europe) -node,way CLC:* text delete - -# misc -node,way 3dshapes:ggmodelk text delete -node,way AND_nosr_r text delete -node,way import text delete -node,way it:fvg:* text delete +# Columns defined in openstreetmap-carto.lua file +way z_order int4 linear +way osmcarto_z_order int4 linear