Skip to content

Thuenen-GeoNode-Development/thuenen_atlas

Repository files navigation

Geonode Installation

This document will guide you through the installation of GeoNode, a spatial content management system. The needed components are available as Docker images and will be set up and run via the docker compose tool.

Background

This blueprint is an opnionated GeoNode setup which evolved from several upstream discussions12. The main goal of this blueprint is to have a simplified view on the GeoNode actual setup with less cluttered configuration while preserving flexibility. At the time of writing GeoNode setup is much convoluted at several places so one have to watch out making changes to the defaults (lots of things have side effects). However, the blueprint cannot solve the upstream issues, but tries to narrow the focus on the most important parts.

We tend to establish a better maintainable project setup23 than the official geonode-project offers at the moment. Additionally, we added a development setup using devcontainer configuration for the Thuenen Atlas, which integrates nicely with IDEs like vs-code.

Feel free to test and report any findings like bugs, issues, and even conceptual things. We hope, the setup turns to be helpful for other projects and are eager to further improve the setup based on your requirements. In any case the blueprint may give you a good starting point to create you own setup.

Component Overview

Here is a short overview of the installed components and how they are connected.

GeoNode Architecture

The components are:

  • Django: The actual GeoNode component. It exposes a pyCSW API and the GeoNode API.
  • Celery: Celery forms the asynchronuous task queue of GeoNode.
  • GeoServer: GeoServer is the backend server of GeoNode for sharing geospatial data. It exposes OGC APIs such as WMS, WFS, etc.
  • Nginx: Nginx serves as advanced load balancer, web server and reverse proxy to all GeoNode components.
  • PostgreSQL: GeoNode and GeoServer are using PostgreSQL with the geospatial extension PostGIS as the database.

Thünen Specific Repositories

Thünen Atlas maintains own repositories holding all code changes, fixes, and adjustments which are not (yet) part of GeoNode upstream and related projects as genonde-mapstore-client.

Most important repositories are

Main goal of all forks is, to stay as close to the upstream as possible. Some repositories are maintained in a repository of the German GeoNode user group.

Quick Start

Clone the repository.

git clone --recurse-submodules https://github.com/Thuenen-GeoNode-Development/thuenen_atlas -b thuenen_4.x
cd thuenen_atlas

💡 Note

geonode-mapstore-client is a huge repository. If you want to keep a small footprint, add --shallow-submodules when cloning the repository.

To build and start GeoNode, run:

docker-compose up -d --build 

You can follow the logs: docker-compose logs -f <optional-service-name>. Locate your browser to http://172.18.0.1/ to access the GeoNode UI.

💡 Note

We make GeoNode available under IP 172.18.0.1 so that no components try to communicate via localhost (each container has their own loopback interface).

Configuration

💡 Note

Settings (e.g. geodatabase parameters) are mainly configured in the .env file. To review in-built default settings of an image, run the env command on an image. For example docker run geonode/geoserver env | sort.

For a complete set of available options take the GeoNode Settings documentation as a reference.

The containers get configured during creation via environment variables. The geonode/settings.py settings module takes further configuration of the GeoNode containers (django and celery) and aligns some names with those documented.

Copy the sample.env to .env and make your changes (.env is not versioned). For a quick start taking default values you can run docker compose up -d --env-file=sample.env.

Have a look at the Ways to set environment variables in Compose documentation.

TLS Config

If you want to configure a TLS certificate, you can mount key and cert as pems in the geonode service within the docker-compose.yml file. Uncomment the corresponding lines:

volumes:
   - nginx-confd:/etc/nginx
   - statics:/mnt/volumes/statics
   # Link to a custom certificate here
   #- <path-to-cert>.pem:/geonode-certificates/autoissued/fullchain.pem
   #- <path-to-key>.pem:/geonode-certificates/autoissued/privkey.pem

Volume Configuration

By default compose creates named volumes on its first start (or in case you stopped using the -v flag). The default volume configuration is included from ./compose-volumes_default.yml.

To configure a different volume setup copy ./compose-volumes_default.yml to ./compose-volumes_myconfig.yml. Now, adjust the volume configuration for each volume:

geoserver-data-dir:
  name: ${COMPOSE_PROJECT_NAME}-gsdatadir
  driver_opts:
    device: /mnt/geonode-volume/geoserver_data
    type: none
    o: bind

Make sure to have all volume definitions in /compose-volumes_myconfig.yml.

Define VOLUME=myconfig as an environment variable and verify your setup via docker compose config | less.

💡 Hint:

Bind volumes are not created by Docker during startup. Make sure all directories exist or Docker will fail to mount.

If everything looks good start up the services.

Development

As listed above, thuenen_atlas comprises GeoNode components which include customizations. To get an overview, you'd simply git diff <my_branch>..<upstream_branch> [-- [^]spec] or compare it via GitHub:

DevContainers

If you are using vs-code as an IDE you can start the devcontainer setup.

Make sure, you have installed the Dev Containers extension. Build and open the devcontainer and wait until everything has started. After downloading and building the images, the terminal prints when containers are starting:

... # cut for brevity

[188981 ms] Start: Run: docker-compose --project-name geonode -f /home/ridoo/data/coding/projects/thuenen-atlas/thuenen_atlas/docker-compose.yml -f /home/ridoo/data/coding/projects/thuenen-atlas/thuenen_atlas/.devcontainer/docker-compose.yml -f /home/ridoo/.config/Code - Insiders/User/globalStorage/ms-vscode-remote.remote-containers/data/docker-compose/docker-compose.devcontainer.build-1694764280181.yml -f /home/ridoo/.config/Code - Insiders/User/globalStorage/ms-vscode-remote.remote-containers/data/docker-compose/docker-compose.devcontainer.containerFeatures-1694764463670.yml up -d
[+] Running 16/16
 ✔ Network geonode_default          Created                                0.2s 
 ✔ Volume "geonode-gsdatadir"       Create...                              0.0s 
 ✔ Volume "geonode-backup-restore"  C...                                   0.0s 
 ✔ Volume "geonode-statics"         Created                                0.0s 
 ✔ Volume "geonode-dbbackups"       Create...                              0.0s 
 ✔ Volume "geonode-dbdata"          Created                                0.0s 
 ✔ Volume "geonode-data"            Created                                0.0s 
 ✔ Volume "geonode-rabbitmq"        Created                                0.0s 
 ✔ Volume "geonode-tmp"             Created                                0.0s 
 ✔ Container gsconf4geonode         Started                                0.1s 
 ✔ Container nginx4geonode          Started                                0.1s 
 ✔ Container db4geonode             Started                                0.1s 
 ✔ Container rabbitmq4geonode       Starte...                              0.1s 
 ✔ Container geoserver4geonode      Creat...                               0.1s 
 ✔ Container django4geonode         Created                                0.0s 
 ✔ Container celery4geonode         Created                                0.0s

When seeing the above output, you can watch logging via docker-compose logs -f.

The devcontainer setup does not start GeoNode automatically. Once the container is ready, you can press F5 to start debugging GeoNode. Locate your browser to http://172.18.0.1:8001.

💡 Note

In the devcontainer setup, GeoNode is not available via nginx! Starting GeoNode in devcontainer actually runs python manage.py runserver which starts a lightweight development web server based on WSGI. However, GeoServer is routed via nginx and available from http://172.18.0.1/geoserver.

Develop GeoNode UI

The GeoNode UI is written in ReactJS and uses components from the MapStore2 Web framework. As GeoNode is a Django project, integration of the UI works as a django app. This django app is implemented by the geonode-mapstore-client project. At the moment, Thünen Atlas also provides an own fork for UI adjustments.

The UI is a single page application and (mostly) communicates with GeoNode via API. There are exceptions where (still) Django templates are used.

To start developing the UI, you can follow the official development guide.

In devcontainer setup, the vs-code debug launcher opens port 8001 for HTTP connections. Configure that port in your ./geonode_mapstore_client/client/.env file:

DEV_SERVER_PROTOCOL=http
DEV_SERVER_HOSTNAME=localhost
DEV_TARGET_GEONODE_HOST=172.18.0.1:8001

Then build and start the client:

cd ./geonode_mapstore_client/client
npm install
npm start

Make sure to use a compatible node version (e.g. via nvm).

💡 Note

Do not commit compiled JavaScript!

To make code review easier and support branch maintenance, we let GitHub build the client and create a PR dedicated to the compiled JavaScript. After merging the PR you can pull the latest changes again to sync with the remote repo. This is necessary to not clutter any merge with thousands of changes in the static/mapstore/dist folder.

Once an automated build PR has been merged, the geonode-mapstore-client submodule should be updated to point to a the new version. To do this run the command

git submodule update --remote

Alternatively, you may want to point to a specific submodule version:

cd docker/geonode/geonode-mapstore-client
git checkout <commit-id>

Working with Git Submodules

We leverage Git submodules to tie dependent modules laying in different repositories. For the Thünen Atlas project these are (for the time of writing this):

You can get an overview of all submodules by running:

git submodule status

If you have not initiated the submodules when cloning the repository the directories appear empty. You have to initiate the submodules before you can build and run Thünen Atlas.

git submodule update --remote --init

You would have to include the --recursively flag as geonode-mapstore-client includes MapStore2 as submodule as well.

Git keeps track of a submodule by its commit hash. In case you want to develop from within the submodule directory switch to a dedicated branch before committing. If there are remote changes update the parent project in your local working copy:

git submodule update --remote

You can find a lot of good readings on Git submodules, either in the official Git Submodule reference or by Atlassian.

Working with Feature Branches

During development we follow the goal to contribute features to the upstream projects. This means, we have to keep feature relevant commits in a dedicated branch which later can be used to create pull requests. Those feature branches get a thuenen_<feature name> suffix, whereas <feature name> should be a good name describing or referencing the feature.

The following workflow proposes how to develop feature branches based on a dedicated commit/tag and how to merge them onto the thuenen development branch.

  1. Create a feature branch from dedicated commit: git checkout -b thuenen_<feature name> <commit-id>. Do not use a commit-id from a thuenen branch as it already contains other commits. For GeoNode for example, you can use the tag 4.1.3 or its commit hash.
  2. Make your commits on that feature branch.
  3. To merge the feature, git switch to main development branch and do a git merge --no-ff thuenen_<feature name>. Do not rebase the feature branch onto the main development branch! This would smudge the dedicated commit you created the feature branch from.

Testing

To run the tests a running GeoNode instance is needed. All test data is stored in separate databases which are prefixed by TEST_.

Before starting tests you have to prepare test databases. Once devcontainer setup is started, just run the command:

docker-compose exec db sh /docker-entrypoint-initdb.d/prepare-testdb.sh

Verify if test_geonode and test_geonode_data is available.

docker-compose exec -e PGPASSWORD=postgres db psql -U postgres -c \\l

Afterwards the necessary test databases and users have been created. Now, you can start the tests by running:

python manage.py test -v 3 --keepdb [ module to test ]

Docker Images

All images are based on pre-built base images. This ensures more performant builds regarding to time and size. Docker provides concepts to extend and adjust the setup, for example via volume mounts, or building atop of those images.

💡 Base Images

The Thünen Atlas GeoNode instance contains unmerged changes and customized features. Therefore it uses a project specific image which is built from a project specific fork of GeoNode. The Docker images are built from the Thuenen-GeoNode-Development/geonode fork. This way, we can track upstream changes (to stay close to the upstream) and keep all customization and upstream fixes in our repository.

The project prepares Dockerfiles for each component to allow a well-defined extension structure, e.g.:

docker
├── geonode                         # Extension point for GeoNode
│   ├── Dockerfile                  # builds thuenen-atlas/geonode_django on top of geonode/geonode
│   ├── geonode-mapstore-client     # Submodule for the GeoNode UI
│   └── requirements.txt            # Further packages to install
├── geoserver                       # Extension point for GeoServer
│   └── Dockerfile                  # Uses geonode/geoserver
├── geoserver_data                  # Extension point for GeoServer data dir
│   └── Dockerfile                  # Uses geonode/geonode_data (may change when tagging becomes unstable as well)
├── nginx                           # Extension point for Nginx
│   └── Dockerfile                  # Uses geonode/geonode-nginx
└── postgresql                      # Extension point for Postgres
    └── Dockerfile                  # Uses geonode/postgis (may change when tagging becomes unstable as well)

Start and Run

Docker-Compose Basics

Run docker compose up -d to start all geonode components. Review all started components by executing docker compose ps. You can follow logs via docker compose logs -f and optionally pass a service to only follow a service's log.

Stop all components via docker compose down, and pass a -v flag to clean up all volumes (CAUTION: removes all persisted data).

For more features and available commands, docker compose --help, or read the docker compose CLI documentation.

Add a Service Unit

When running GeoNode on a systemd-based Linux, you may want to add a service unit:

/etc/systemd/system/geonode.service

[Unit]
Description=GeoNode Docker Installation

[Service]
Type=oneshot
ExecStart=docker compose up -d /path/to/workingcopy

[Install]
WantedBy=multi-user.target

Then reload the systemd daemon:

systemctl daemon-reload

And enable GeoNode start on each boot:

systemctl enable geonode.service

Check the service status:

systemctl status geonode.service

Footnotes

  1. https://github.com/GeoNode/geonode-project/issues/471

  2. https://github.com/GeoNode/geonode-project/discussions/460 2

  3. https://lists.osgeo.org/pipermail/geonode-devel/2023-August/003335.html