2011-04-11 19:06:28 +02:00
==============================================
ckanext-spatial - Geo related plugins for CKAN
==============================================
This extension contains plugins that add geospatial capabilities to CKAN.
2011-09-30 12:33:36 +02:00
The following plugins are currently available:
2012-02-21 13:00:47 +01:00
* Spatial model for CKAN datasets and automatic geo-indexing (`spatial_metadata` )
* Spatial search integration and API call (`spatial_query` ).
* Map widget integrated on the search form (`spatial_query_widget` ).
2011-10-03 14:39:18 +02:00
* Map widget showing a dataset extent (`dataset_extent_map` ).
2011-09-30 12:33:36 +02:00
* A Web Map Service (WMS) previewer (`wms_preview` ).
2011-04-11 19:06:28 +02:00
2012-04-13 11:36:06 +02:00
**Note:** All plugins except the WMS previewer require the `spatial_metadata` plugin.
2012-02-21 13:00:47 +01:00
2011-04-11 19:06:28 +02:00
Dependencies
============
2011-11-30 18:11:29 +01:00
You will need CKAN installed. The present module should be installed at least
2011-04-11 19:06:28 +02:00
with `setup.py develop` if not installed in the normal way with
`setup.py install` or using pip or easy_install.
2011-09-29 16:40:49 +02:00
The extension uses the GeoAlchemy_ and Shapely_ libraries. You can install them
via `pip install -r pip-requirements.txt` from the extension directory.
.. _GeoAlchemy: http://www.geoalchemy.org
.. _Shapely: https://github.com/sgillies/shapely
2011-04-11 19:06:28 +02:00
If you want to use the spatial search API, you will need PostGIS installed
and enable the spatial features of your PostgreSQL database. See the
"Setting up PostGIS" section for details.
2012-01-20 16:03:35 +01:00
Note that Shapely requires libgeos to be installed. If you installed PostGIS on
the same machine you already got it, but if PostGIS is located on another server
you will need to install GEOS::
sudo apt-get install libgeos-c1
2011-04-11 19:06:28 +02:00
Configuration
=============
2011-11-30 18:11:29 +01:00
You will first need to have have PostGIS installed and configured in your
database (see the "Setting up PostGIS" section for details)
Once this is done, you need to create the necessary DB tables running the
following command (with your python env activated)::
2011-04-13 13:05:59 +02:00
paster spatial initdb [srid] --config=../ckan/development.ini
2011-11-30 18:11:29 +01:00
You can define the SRID of the geometry column. Default is 4326. If you
are not familiar with projections, we recommend to use the default value.
2011-04-13 13:05:59 +02:00
2012-04-13 11:36:06 +02:00
Check the Troubleshooting_ section if you get errors at this stage.
2011-04-13 13:05:59 +02:00
2011-09-30 12:33:36 +02:00
Plugins are configured as follows in the CKAN ini file (Add only the ones you
are interested in)::
2011-04-11 19:06:28 +02:00
2012-02-21 13:00:47 +01:00
ckan.plugins = spatial_metadata spatial_query spatial_query_widget dataset_extent_map wms_preview
2011-04-11 19:06:28 +02:00
2012-02-21 13:00:47 +01:00
When enabling the spatial metadata, you can define the projection
2011-11-30 18:11:29 +01:00
in which extents are stored in the database with the following option. Use
the EPSG code as an integer (e.g 4326, 4258, 27700, etc). It defaults to
2011-09-29 16:40:49 +02:00
4326::
2011-11-30 18:11:29 +01:00
2011-09-29 16:40:49 +02:00
ckan.spatial.srid = 4326
2011-04-11 19:06:28 +02:00
2012-02-21 12:05:13 +01:00
If you want to define a default map extent for the different map widgets,
(e.g. if you are running a national instace of CKAN) you can do so adding
this configuration option::
ckan.spatial.default_map_extent=<minx>,<miny>,<maxx>,<maxy>
Coordinates must be in latitude/longitude, e.g.::
ckan.spatial.default_map_extent=-6.88,49.74,0.50,59.2
2012-04-13 11:36:06 +02:00
Troubleshooting
===============
Here are some common problems you may find when installing or using the
extension:
* When initializing the spatial tables::
LINE 1: SELECT AddGeometryColumn('package_extent','the_geom', E'4326...
^
HINT: No function matches the given name and argument types. You might need to add explicit type casts.
"SELECT AddGeometryColumn('package_extent','the_geom', %s, 'GEOMETRY', 2)" ('4326',)
PostGIS was not installed correctly. Please check the "Setting up PostGIS" section.
::
sqlalchemy.exc.ProgrammingError: (ProgrammingError) permission denied for relation spatial_ref_sys
The user accessing the ckan database needs to be owner (or have permissions) of the geometry_columns and spatial_ref_sys tables.
* When performing a spatial query::
InvalidRequestError: SQL expression, column, or mapped entity expected - got '<class 'ckanext.spatial.model.PackageExtent'>'
The spatial model has not been loaded. You probably forgot to add the `spatial_metadata` plugin to your ini configuration file.
::
InternalError: (InternalError) Operation on two geometries with different SRIDs
The spatial reference system of the database geometry column and the one used by CKAN differ. Remember, if you are using a different spatial reference system from the default one (WGS 84 lat/lon, EPSG:4326), you must define it in the configuration file as follows::
ckan.spatial.srid = 4258
2011-10-17 00:40:19 +02:00
Tests
=====
2011-04-11 19:06:28 +02:00
2011-10-17 00:40:19 +02:00
Please note that the tests currently only work with Postgres. You must use the
test-core.ini located in the extension directory to run them. Most of the time
you should run something like::
nosetests --ckan --with-pylons=test-core.ini ckanext/spatial/tests
2011-04-13 13:05:59 +02:00
2011-04-11 19:06:28 +02:00
Command line interface
======================
2011-11-30 18:11:29 +01:00
The following operations can be run from the command line using the
2011-04-13 13:05:59 +02:00
`` paster spatial `` command::
2011-11-30 18:11:29 +01:00
2011-04-13 13:05:59 +02:00
initdb [srid]
- Creates the necessary tables. You must have PostGIS installed
and configured in the database.
2011-09-29 16:40:49 +02:00
You can privide the SRID of the geometry column. Default is 4326.
2011-11-30 18:11:29 +01:00
extents
2011-10-03 14:39:18 +02:00
- creates or updates the extent geometry column for datasets with
2011-09-29 16:40:49 +02:00
an extent defined in the 'spatial' extra.
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
The commands should be run from the ckanext-spatial directory and expect
2011-11-30 18:11:29 +01:00
a development.ini file to be present. Most of the time you will specify
2011-04-11 19:06:28 +02:00
the config explicitly though::
2012-04-13 11:36:06 +02:00
paster spatial extents --config=../ckan/development.ini
2011-04-11 19:06:28 +02:00
2011-09-30 12:33:36 +02:00
Spatial Query
=============
To enable the spatial query you need to add the `spatial_query` plugin to your
2012-02-21 13:00:47 +01:00
ini file (See `Configuration`_ ). This plugin requires the `spatial_metadata`
plugin.
2011-04-11 19:06:28 +02:00
The extension adds the following call to the CKAN search API, which returns
2011-10-03 14:39:18 +02:00
datasets with an extent that intersects with the bounding box provided::
2011-04-11 19:06:28 +02:00
2011-10-03 14:39:18 +02:00
/api/2/search/dataset/geo?bbox={minx,miny,maxx,maxy}[&crs={srid}]
2011-04-11 19:06:28 +02:00
If the bounding box coordinates are not in the same projection as the one
defined in the database, a CRS must be provided, in one of the following
forms:
2011-09-29 16:40:49 +02:00
- urn:ogc:def:crs:EPSG::4326
- EPSG:4326
- 4326
2012-02-21 13:00:47 +01:00
As of CKAN 1.6, you can integrate your spatial query in the full CKAN
search, via the web interface (see the `Spatial Query Widget`_ ) or
via the `action API`__ , e.g.::
POST http://localhost:5000/api/action/package_search
{
"q": "Pollution",
"extras": {
"ext_bbox": "-7.535093,49.208494,3.890688,57.372349"
}
}
__ http://docs.ckan.org/en/latest/apiv3.html
2011-10-03 14:39:18 +02:00
Geo-Indexing your datasets
2011-09-30 13:15:53 +02:00
--------------------------
2011-09-29 16:40:49 +02:00
2011-10-03 14:39:18 +02:00
In order to make a dataset queryable by location, an special extra must
2011-09-29 16:40:49 +02:00
be defined, with its key named 'spatial'. The value must be a valid GeoJSON_
geometry, for example::
{"type":"Polygon","coordinates":[[[2.05827, 49.8625],[2.05827, 55.7447], [-6.41736, 55.7447], [-6.41736, 49.8625], [2.05827, 49.8625]]]}
or::
2011-11-30 18:11:29 +01:00
{ "type": "Point", "coordinates": [-3.145,53.078] }
2011-09-29 16:40:49 +02:00
.. _GeoJSON: http://geojson.org
2011-10-03 14:39:18 +02:00
Every time a dataset is created, updated or deleted, the extension will synchronize
2011-09-29 16:40:49 +02:00
the information stored in the extra with the geometry table.
2011-04-11 19:06:28 +02:00
2012-02-21 13:00:47 +01:00
Spatial Query Widget
====================
**Note** : this plugin requires CKAN 1.6 or higher.
To enable the search map widget you need to add the `spatial_query_widget` plugin to your
ini file (See `Configuration`_ ). You also need to load both the `spatial_metadata`
and the `spatial_query` plugins.
When the plugin is enabled, a map widget will be shown in the dataset search form,
where users can refine their searchs drawing an area of interest.
2011-09-30 12:33:36 +02:00
Dataset Map Widget
==================
2011-09-30 13:15:53 +02:00
To enable the dataset map you need to add the `dataset_map` plugin to your
2012-02-21 13:00:47 +01:00
ini file (See `Configuration`_ ). You need to load the `spatial_metadata` plugin also.
2011-09-30 12:33:36 +02:00
When the plugin is enabled, if datasets contain a 'spatial' extra like the one
2011-09-30 13:15:53 +02:00
described in the previous section, a map will be shown on the dataset details page.
WMS Previewer
=============
To enable the WMS previewer you need to add the `wms_preview` plugin to your
2012-02-21 13:00:47 +01:00
ini file (See `Configuration`_ ).
2011-09-30 13:15:53 +02:00
Please note that this is an experimental plugin and may be unstable.
When the plugin is enabled, if datasets contain a resource that has 'WMS' format,
a 'View available WMS layers' link will be displayed on the dataset details page.
It forwards to a simple map viewer that will attempt to load the remote service
layers, based on the GetCapabilities response.
2011-09-30 12:33:36 +02:00
2011-04-11 19:06:28 +02:00
Setting up PostGIS
==================
2012-02-21 13:00:47 +01:00
PostGIS Configuration
---------------------
2011-04-11 19:06:28 +02:00
* Install PostGIS::
sudo apt-get install postgresql-8.4-postgis
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
* Create a new PostgreSQL database::
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
sudo -u postgres createdb [database]
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
(If you just want to spatially enable an exisiting database, you can
ignore this point, but it's a good idea to create a template to
make easier to create new databases)
* Many of the PostGIS functions are written in the PL/pgSQL language,
so we need to enable it in our database::
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
sudo -u postgres createlang plpgsql [database]
* Run the following commands. The first one will create the necessary
tables and functions in the database, and the second will populate
the spatial reference table::
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
sudo -u postgres psql -d [database] -f /usr/share/postgresql/8.4/contrib/postgis-1.5/postgis.sql
2011-11-30 18:11:29 +01:00
sudo -u postgres psql -d [database] -f /usr/share/postgresql/8.4/contrib/postgis-1.5/spatial_ref_sys.sql
**Note** : depending on your distribution and PostGIS version, the
scripts may be located on a slightly different location, e.g.::
/usr/share/postgresql/8.4/contrib/postgis.sql
2011-04-11 19:06:28 +02:00
* Execute the following command to see if PostGIS was properly
installed::
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
sudo -u postgres psql -d [database] -c "SELECT postgis_full_version()"
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
You should get something like::
2011-11-30 18:11:29 +01:00
postgis_full_version
2011-04-11 19:06:28 +02:00
------------------------------------------------------------------------------------------------------
POSTGIS="1.5.2" GEOS="3.2.2-CAPI-1.6.2" PROJ="Rel. 4.7.1, 23 September 2009" LIBXML="2.7.7" USE_STATS
(1 row)
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
Also, if you log into the database, you should see two tables,
`` geometry_columns `` and `` spatial_ref_sys `` (and probably a view
called `` geography_columns `` ).
Note: This commands will create the two tables owned by the postgres
user. You probably should make owner the user that will access the
database from ckan::
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
ALTER TABLE spatial_ref_sys OWNER TO [your_user];
ALTER TABLE geometry_columns OWNER TO [your_user];
More information on PostGIS installation can be found here:
http://postgis.refractions.net/docs/ch02.html#PGInstall
2011-11-30 18:11:29 +01:00
Migrating to an existing PostGIS database
-----------------------------------------
If you are loading a database dump to an existing PostGIS database, you may
find errors like ::
ERROR: type "spheroid" already exists
This means that the PostGIS functions are installed, but you may need to
create the necessary tables anyway. You can force psql to ignore these
errors and continue the transaction with the ON_ERROR_ROLLBACK=on::
sudo -u postgres psql -d [database] -f /usr/share/postgresql/8.4/contrib/postgis-1.5/postgis.sql -v ON_ERROR_ROLLBACK=on
You will still need to populate the spatial_ref_sys table and change the
tables permissions. Refer to the previous section for details on how to do
it.
2011-04-11 19:06:28 +02:00
Setting up a spatial table
--------------------------
2011-04-13 13:05:59 +02:00
**Note:** If you run the `` initdb `` command, the table was already created for
2011-09-29 16:40:49 +02:00
you. This section just describes what's going on for those who want to know
2011-04-13 13:05:59 +02:00
more.
2011-04-11 19:06:28 +02:00
To be able to store geometries and perform spatial operations, PostGIS
needs to work with geometry fields. Geometry fields should always be
added via the `` AddGeometryColumn `` function::
CREATE TABLE package_extent(
package_id text PRIMARY KEY
);
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
ALTER TABLE package_extent OWNER TO [your_user];
2011-11-30 18:11:29 +01:00
2011-09-29 16:40:49 +02:00
SELECT AddGeometryColumn('package_extent','the_geom', 4326, 'POLYGON', 2);
2011-11-30 18:11:29 +01:00
2011-04-11 19:06:28 +02:00
This will add a geometry column in the `` package_extent `` table called
2011-11-30 18:11:29 +01:00
`` the_geom `` , with the spatial reference system EPSG:4326. The stored
2011-09-29 16:40:49 +02:00
geometries will be polygons, with 2 dimensions (The actual table on CKAN
uses the GEOMETRY type to support multiple geometry types).
2011-04-11 19:06:28 +02:00
Have a look a the table definition, and see how PostGIS has created
three constraints to ensure that the geometries follow the parameters
defined in the geometry column creation::
# \d package_extent
Table "public.package_extent"
2011-11-30 18:11:29 +01:00
Column | Type | Modifiers
2011-04-11 19:06:28 +02:00
------------+----------+-----------
package_id | text | not null
2011-11-30 18:11:29 +01:00
the_geom | geometry |
2011-04-11 19:06:28 +02:00
Indexes:
"package_extent_pkey" PRIMARY KEY, btree (package_id)
Check constraints:
"enforce_dims_the_geom" CHECK (st_ndims(the_geom) = 2)
"enforce_geotype_the_geom" CHECK (geometrytype(the_geom) = 'POLYGON'::text OR the_geom IS NULL)
2011-09-29 16:40:49 +02:00
"enforce_srid_the_geom" CHECK (st_srid(the_geom) = 4326)