Export tool from PostGIS to CityJSON
MIT license
PostgreSQL
- I'm testing against Postgres 10 + PostGIS 2.5, Postgres 15 + PostGIS 3.3
The software does not have an installer but only an executable which you can run directly in the command line. Head over to the latest release and download the executable for your platform.
Extract the archive (in Windows you'll need 7zip) and run the executable from the command line.
Requires Python 3.6+
The project is alpha, please install directly from GitHub with pip:
$ pip install -U --force-reinstall git+/~https://github.com/cityjson/cjio_dbexport@master
Also install the development requirements from requirements_dev.txt
Call the cjdb tool from the command line and pass it the configuration file.
$ cjdb --help Usage: cjdb [OPTIONS] CONFIGURATION COMMAND [ARGS]... Export tool from PostGIS to CityJSON. CONFIGURATION is the YAML configuration file. Options: --version Show the version and exit. --log [DEBUG|INFO|WARNING|ERROR|CRITICAL] Set the logging level in the log file 'cjdb.log'. --help Show this message and exit. Commands: export Export the whole database into a CityJSON file. export_bbox Export the objects within a 2D Bounding Box into a CityJSON... export_extent Export the objects within the given polygon into a CityJSON... export_tiles Export the objects within the given tiles into a CityJSON... index Create a tile index for the specified extent.
This tool uses a YAML-based configuration file to managing the database connections and to declare what to export.
- The block
geometries
declares the Level of Detail (LoD) in and geometry type in of the CityObjects. The geometry type is one of the allowed CityJSON geometry types. The LoD can be either an integer (following the CityGML standards), or a number following the improved LoDs by TU Delft. - The block
database
specifies the database connection parameters. The password can be empty if it is stored a in a.pgpass
file. - The block
tile_index
specifies the location of the tile index for using with theexport_tiles
command. - The block
cityobject_type
maps the database tables to CityObject types.
Each key in cityobject_type
is a 1st-level or 2nd-level CityObject , and it contains a sequence of mappings. Each of these mappings refer to a single table, thus you can collect CityObjects from several tables into a single CityObject type. The table
is exported as one record per CityObject.
The mapping of the fields of the table to CityObjects is done as:
pk
: the primary keygeometry
: the geometry fieldcityobject_id
: the field that is used as CityObject ID
By default all columns, excluding the three above, are added as Attributes to the CityObject. If you want to exclude certain fields, specify the filed names in a string array in exclude
. In the example below, the fields xml
and _clipped
are excluded from the export.
geometries: lod: 1.2 type: MultiSurface database: dbname: db3dnl host: localhost port: 5432 user: some_user password: some_password tile_index: schema: tile_index table: tile_index_1 srid: 7415 field: pk: id geometry: geom cityobject_type: Building: - schema: public table: building field: pk: ogc_fid geometry: lod12: name: wkb_geometry type: Solid lod13: name: wkb_geometry_lod13 cityobject_id: identificatie exclude: ["xml", "_clipped"] WaterBody: - schema: public table: waterdeel_vlak field: pk: ogc_fid geometry: wkb_geometry cityobject_id: identificatie exclude: ["xml", "_clipped"] LandUse: - schema: public table: onbegroeidterreindeel_vlak field: pk: ogc_fid geometry: wkb_geometry cityobject_id: identificatie exclude: ["xml"] - schema: public table: ondersteunendwaterdeel_vlak field: pk: ogc_fid geometry: wkb_geometry cityobject_id: identificatie exclude: ["xml"]
You can provide a bounding box (minx miny maxx maxy) to limit the extent of the export.
$ cjdb config.yml export_bbox 123.4 545.5 678.8 987.8 path/to/output.json
To export an irregular extent, provide a single Polygon in a GeoJSON file.
$ cjdb config.yml export_extent polygon.geojson path/to/output.json
To export a set of tiles into a separate CityJSON file each, provide their
tile IDs. The command below will export the tiles ci1
, ci2
, gb4
into the given directory. If you want to merge the tiles into a single file,
provide
the --merge
option to export_tiles
. If you want to export all the
tiles from the tile index, then pass all
as the tile ID.
$ cjdb config.yml export_tiles ci1 ci2 gb4 path/to/directory
The lod
parameter in the YAML configuration file declares the LoD value
that each CityObject will get in the output file. However, in case you have
objects with multiple geometric representations (multiple LoD), you can
choose to export the each LoD into the same file or write a separate file
for each LoD.
For instance we have a table that stores building models and each building has a geometry in LoD0 and LoD1.3. Note that this is the case of single table with multiple geometry columns. In this case we can declare the mapping of the geometry column as here below.
cityobject_type: Building: - schema: public table: building field: pk: ogc_fid geometry: lod0: name: geom_lod0 lod13: name: geom_lod13
Notice that,
geometry
becomes a mapping of mappings,- the keys in
geometry
follow the convention oflod<value>
, where<value>
is the level of detail, - the
lod<value>
keys are mappings of the geometry column with the corresponding LoD, where thename
key points to the name of the column.
For example if you want to export the LoD0 and LoD1.3 (see yaml above) but write each LoD into a separate file,
then you need to run the export process twice. Once for each LoD,
by keeping only lod0.name.geom_lod0
or lod13.name.geom_lod13
respectively for the
desired LoD.
The global (for the whole file) setting for both the LoD and the output
geometry type is in the geometries
block in the configuration file. By default each geometry will get the global LoD and geometry type on export.
If you want to export a table to a different geometry type than that of the global setting, then you need to declare it under the corresponding LoD-key in the geometry
mapping of the table.
geometries: lod: 0 type: MultiSurface cityobject_type: Building: - schema: public table: building field: pk: ogc_fid geometry: lod0: name: geom_lod0 lod13: name: geom_lod13 type: Solid
Furthermore, it is possible to assign different a different LoD per object. In this case the LoD name (e.g. 1.2
or 2
) is expected to be stored in column of the CityObject table. In the example below each CityObject will get the LoD that is stored in the lod_column
column of the building
table.
cityobject_type: Building: - schema: public table: building field: pk: ogc_fid geometry: geometry_column lod: lod_column
When the LoD is declared on multiple levels for a CityObject (e.g. on global, per column or per object) then the lower-level declaration overrules the higher-level one. For instance the per-column declaration overrules the global.
In order to export LoD2 models with semantic surfaces, the semantics need to be in the same table as the geometry. The semantics need to be stored as an array of integers where,
- the array has the same length and order as the surfaces of the geometry,
- the integers in the array represent the semantics.
This is almost the same as the values
member of a Semantic Surface object, with some distinctions,
- Null values are not supported in the array,
- there is no distinction in the array-nesting for the different geometry types.
An example of such a semantics array would be implemented as '{0,0,2,2,2,2,2,2,2,2,1,1}'::int2[]
in PostgreSQL.
Note that the semantics array above is 12 elements long, thus the first element points to the first surface of the geometry boundary (the MultiPolygonZ), while the 12th element points to the 12th surface of the geometry boundary.
The integer mapping of the semantic values are declared at the root level of the configuration file.
Additionally, in the semantics
member of the table fields you need to declare the name of the column in which the semantic arrays are stored.
It is not possible to export multiple LoDs into the same CityJSON file when exporting semantics too.
semantics_mapping: 0: "GroundSurface" 1: "RoofSurface" 2: "WallSurface" cityobject_type: Building: - schema: public table: lod2_sample field: pk: ogc_fid geometry: lod22: name: wkb_geometry type: Solid semantics: labels cityobject_id: id_column
If you want to export the CityObjects as CityJSONFeatures,
you can do so with the export_tiles --features
command. This command will create a
single metadata.city.json
file at the root of the output directory, and a
subdirectory for each tile, containing the CityJSONFeatures (.city.jsonl
) of that
tile.
The metadata.city.json
contains the main CityJSON object with the Transform and CRS
properties that are required for parsing the CityJSONFeatures.
Note
The transform parameters are currently hardcoded to the the location of the Zwaartepunt bij Putten in the Netherlands. If your are not exporting Dutch data, open an issue and I'll add an option for setting the transform parameters.
While the default export_tiles
command writes a CityObject into each tile that
intersects it, the --features
option creates a 1-to-1 mapping of CityObject-tile.
Which means that each CityObject is assigned to only one tile.
In order to make this feasible, the input geometries need to have a spatial index on
their centroids.
You can index the geometry centroids on your own, with a query like this:
CREATE INDEX my index ON my table USING gist (ST_Centroid(geometry column))
Alternatively, use the index --centroid
command as described below.
If you have a database of a large area, you probably want to export it piece-by-piece, in tiles. This requires a tile index, which is a rectangular grid of polygons that fully covers your area, and each polygon has a unique ID.
The index
command can help you create such a tile index. It requires a
polygonal extent of your area as GeoJSON file and the width and height
of the tiles you want to create. The units for the tile size are same as the
unit of the CRS in the database.
$ cjdb config.yml index netherlands.json 1000 1000
The command above will,
1. create rectangular polygons (tiles) of 1000m by 1000m for the extent of the polygon that is netherlands.json,
2. sort the tiles in Morton-order and create unique IDs for them accordingly,
3. upload the tile index into the relation that is declared in
config.yml
under the tile_index
node.
When using the export_tiles
command with the --features
option to export
CityJSONFeatures, a spatial index (eg. GIST) is required on the feature geometry
centroids. The index
command can create this for you if you pass the --centroid
options, creating an index on each input table as
USING gist (ST_Centroid(geometry column))
.
Note that this will add an additional index to your data tables.
- No appearances
- The geometry is expected to be a
MULTIPOLYGON
ofPOLYGON Z
in PostGIS - CRS is hardcoded to 7415
- When exporting CityJSONFeatures, the Transform parameters are hardcoded to the Zwaartepunt bij Putten in the Netherlands
Mapping of the 3DNL tables to CityJSON CityObjects:
3dnl table | CityObject type |
---|---|
begroeidterreindeel_vlak | PlantCover |
building | Building |
kunstwerkdeel_vlak | OtherConstruction |
onbegroeidterreindeel_vlak | LandUse |
ondersteunendwaterdeel_vlak | LandUse |
ondersteunendwegdeel_vlak | LandUse |
overbruggingsdeel_vlak | Bridge |
overigbouwwerk | LandUse |
pand | LandUse |
scheiding_vlak | OtherConstruction |
waterdeel_vlak | WaterBody |
wegdeel_vlak | Road |
This package was created with Cookiecutter and the audreyr/cookiecutter-pypackage project template.