CKAN extension for DCAT-AP Switzerland, templates and different plugins including FTP- and S3-Harvester for opentransportdata.swiss.
- CKAN 2.10+
- ckanext-fluent
- ckanext-harvest
- ckanext-scheming
To install ckanext-switzerland:
-
Activate your CKAN virtual environment, for example:
. /usr/lib/ckan/default/bin/activate
-
Install the ckanext-switzerland Python package into your virtual environment:
pip install ckanext-switzerland
-
Add
switzerland
to theckan.plugins
setting in your CKAN config file (by default the config file is located at/etc/ckan/default/production.ini
). -
Restart CKAN. For example if you've deployed CKAN with Apache on Ubuntu:
sudo service apache2 reload
This extension uses the following config options (.ini file)
# For linking to the associated WordPress site from the header logo
ckanext.switzerland.wp_url
# For managing cookie consent and tracking with OneTrust and Matomo
ckanext.switzerland.cookie_law_url
ckanext.switzerland.cookie_law_id
To install ckanext-switzerland for development, activate your CKAN virtualenv and do:
git clone https://github.com/ogdch/ckanext-switzerland.git
cd ckanext-switzerland
python setup.py develop
pip install -r dev-requirements.txt
pip install -r requirements.txt
We use ckanext-scheming to define our schemas for datasets, groups and organizations. The schemas are defined in the following files:
- ckanext/switzerland/dcat-ap-switzerland_scheming.json (dataset schema)
- ckanext/switzerland/multilingual_group_scheming.json
- ckanext/switzerland/multilingual_organization_scheming.json
NB: If you update the dataset schema to add any fields that are not simple strings, you will also need to update
the OgdchPackagePlugin.before_dataset_index
method to convert the value of those fields to strings, so that SOLR
can consume our datasets and index them properly.
Currently, CKAN-Switzerland support two types of remote storages for harvesting data:
- FTP/SFTP storage
- AWS S3 Buckets
There are two types of configuration: the harvester configuration that comes from the Database,
and the storage configuration, this comes from the .ini
configuration file of CKAN.
Located in the common ini
file of CKAN, and is used to configure accesses to the different storage.
A FTP configuration requires the following properties :
username
: the user name used to connect to the FTPpassword
: the password, in clear, associated to the userkeyfile
: the path of the key file that would be used instead of username/password to connecthost
: the DNS name, or IP Address, of the FTP serverport
: the port on which to connect to the FTP serverlocalpath
: the path on the server where to store temporary files during the harvest processremotedirectory
: the remote directory to consider as root
An example of FTP storage configuration:
ckan.ftp.mainserver.username = TESTUSER
ckan.ftp.mainserver.password = TESTPASS
ckan.ftp.mainserver.keyfile =
ckan.ftp.mainserver.host = ftp-secure.liip.ch
ckan.ftp.mainserver.port = 990
ckan.ftp.mainserver.localpath = /tmp/ftpharvest/tests/
ckan.ftp.mainserver.remotedirectory = /
Where the part of the key located after the ftp is considered as the storage identifier.
For example, all the properties for the FTP server identified as mainserver
will all start with ckan.ftp.mainserver
A S3 configuration requires the following properties :
bucket_name
: the name of the bucket on AWSaccess_key
: the access key provided by AWS to log in to their API (see AWS console)secret_key
: the secret key associated to theaccess_key
(see AWS console)region_name
: the AWS region where the bucket is locatedlocalpath
: the path on the server where to store temporary files during the harvest processremotedirectory
: the remote directory to consider as root
An example of S3 Bucket storage configuration:
ckan.s3.main_bucket.bucket_name = test-bucket
ckan.s3.main_bucket.access_key = test-access-key
ckan.s3.main_bucket.secret_key = test-secret-key
ckan.s3.main_bucket.region_name = eu-central-1
ckan.s3.main_bucket.localpath = /tmp/s3harvest/
ckan.s3.main_bucket.remotedirectory = /a/
Following the same schema, the identifier of this S3 bucket will be main_bucket
.
This configuration is a JSON object, that can be modified in the UI, in the harvester administration.
In this configuration, it is mandatory to specify some information in order for the harvester to connect to the correct remote storage.
In order to use a FTP Harvester, the configuration should look like:
{
"storage_adapter": "FTP",
"ftp_server": "mainserver",
// [...]
}
storage_adapter
allows to specify which harvester type should be used a FTP or a S3 Bucket.
For legacy support reasons, if this property is not defined, the harvester will consider that it uses FTP.
The property is case insensitive.
For a FTP server, the property ftp_server
is mandatory.
If the property is not set, an exception will be raised.
In order to use a S3 Harvester, the configuration should look like
{
"storage_adapter": "S3",
"bucket": "main_bucket",
// [...]
}
For a S3 server, the property bucket
is mandatory.
If the property is not set, an exception will be raised.
The StorageAdapterBase
holds the logic for loading and validating the configuration.
This logic is able to verify that a certain property is present,
read the value, convert the value to the given type (eg: a port number is an int
)
and check some constraints on the value (eg: the port
should be greater than 0).
Each Storage Adapter is responsible to define the configuration properties it needs,
and define the type, the name, the constraints.
This is done through the class ConfigKey
. This class allows to define the name of the configuration property,
its type, if it's a mandatory configuration, a validation function and a custom message.
The harvester will receive its configuration from the Database.
This configuration will be passed to the StorageAdapterFactory
.
This object has for sole responsibility to read the configuration, and decide which Storage Adapter to instantiate.
The two possibilities are S3StorageAdapter
or FTPStorageAdapter
.
Both classes extends the StorageAdapterBase
class.
In this base class, designed as an abstract class (meaning all methods are there,
and the ones that have to be reimplement just throw NotImplementedException
),
are located the method specific to the interactions with the storage itself,
but also some common function to manage the local folder used by the harvesters during the harvest process.
Each implementation (S3StorageAdapter
and FTPStorageAdapter
), are responsible to get their storage configuration,
from the storage identifier received from the harvester configuration.
Each implementation is also unit tested, see respectively TestS3StorageAdapter
and TestFTPStorageAdapter
classes.
The translation files for this ckanext are found in i18n/
:
- ckanext-switzerland.pot
: this is a template file containing all the translatable strings found in the code
- {LANG}/LC_MESSAGES/ckanext-switzerland.po
: one translation file for each of our supported languages apart from
English (German, French and Italian)
- {LANG}/LC_MESSAGES/ckanext-switzerland.mo
: the compiled translations in binary format
If you update the code and add or remove translatable strings, you will have to update the translations too.
The following instructions assume you are working inside a docker container based on the official Docker Compose setup for CKAN.
- Enter the container and go to the ckanext-switzerland directory:
docker compose exec ckan bash cd src_extensions/ckanext-switzerland/
- Update the
.pot
file:python setup.py extract_messages
- Optional If you have copied templates from core CKAN to override them, you might want to copy the translations
from CKAN too:
This adds a lot of translations to our
apt-get install gettext msgcat --use-first i18n/de/LC_MESSAGES/ckanext-switzerland.po ../../src/ckan/ckan/i18n/de/LC_MESSAGES/ckan.po > temp_de.po cp temp_de.po i18n/de/LC_MESSAGES/ckanext-switzerland.po msgcat --use-first i18n/fr/LC_MESSAGES/ckanext-switzerland.po ../../src/ckan/ckan/i18n/fr/LC_MESSAGES/ckan.po > temp_fr.po cp temp_fr.po i18n/fr/LC_MESSAGES/ckanext-switzerland.po msgcat --use-first i18n/it/LC_MESSAGES/ckanext-switzerland.po ../../src/ckan/ckan/i18n/it/LC_MESSAGES/ckan.po > temp_it.po cp temp_it.po i18n/it/LC_MESSAGES/ckanext-switzerland.po rm temp_de.po temp_fr.po temp_it.po
.po
files that we don't want, but they will be removed in the next step. - Update the
.po
files:To update one language at a time, addpython setup.py update_catalog --ignore-obsolete=true --no-fuzzy-matching
-l de
(etc.) to the command. - Check the
.po
files and add any new translations that are needed. - Update the
.mo
files:python setup.py compile_catalog
- Restart the docker container:
docker compose restart ckan