Skip to content

liip/ckanext-switzerland

Repository files navigation

ckanext-switzerland

CKAN extension for DCAT-AP Switzerland, templates and different plugins including FTP- and S3-Harvester for opentransportdata.swiss.

Requirements

  • CKAN 2.10+
  • ckanext-fluent
  • ckanext-harvest
  • ckanext-scheming

Installation

To install ckanext-switzerland:

  1. Activate your CKAN virtual environment, for example:

    . /usr/lib/ckan/default/bin/activate
    
  2. Install the ckanext-switzerland Python package into your virtual environment:

    pip install ckanext-switzerland
    
  3. Add switzerland to the ckan.plugins setting in your CKAN config file (by default the config file is located at /etc/ckan/default/production.ini).

  4. Restart CKAN. For example if you've deployed CKAN with Apache on Ubuntu:

    sudo service apache2 reload
    

Config Settings

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

Development Installation

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

Schema

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.

FTP- and S3-Harvester

Currently, CKAN-Switzerland support two types of remote storages for harvesting data:

  • FTP/SFTP storage
  • AWS S3 Buckets

Configurations

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.

The storage configuration

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 FTP
  • password : the password, in clear, associated to the user
  • keyfile : the path of the key file that would be used instead of username/password to connect
  • host : the DNS name, or IP Address, of the FTP server
  • port : the port on which to connect to the FTP server
  • localpath : the path on the server where to store temporary files during the harvest process
  • remotedirectory : 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 AWS
  • access_key : the access key provided by AWS to log in to their API (see AWS console)
  • secret_key : the secret key associated to the access_key (see AWS console)
  • region_name : the AWS region where the bucket is located
  • localpath : the path on the server where to store temporary files during the harvest process
  • remotedirectory : 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.

The harvester configuration

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.

Validation

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.

How it works

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.

Updating the translations

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.

  1. Enter the container and go to the ckanext-switzerland directory:
    docker compose exec ckan bash
    cd src_extensions/ckanext-switzerland/
  2. Update the .pot file:
    python setup.py extract_messages
  3. Optional If you have copied templates from core CKAN to override them, you might want to copy the translations from CKAN too:
    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
    This adds a lot of translations to our .po files that we don't want, but they will be removed in the next step.
  4. Update the .po files:
    python setup.py update_catalog --ignore-obsolete=true --no-fuzzy-matching
    To update one language at a time, add -l de (etc.) to the command.
  5. Check the .po files and add any new translations that are needed.
  6. Update the .mo files:
    python setup.py compile_catalog
  7. Restart the docker container:
    docker compose restart ckan