This is a minimal guide to getting started with PEcAn development under Docker. You can find more information about docker in the pecan documentation.
Docker is the primary software requirement; it handles all of the other software dependencies. This has been tested on Ubuntu 18.04 and above, MacOS Catalina, and Windows 10 with Windows Subsystem for Linux 2.
- Software (installation instructions below):
- Docker version 19
- Docker-compose version 1.26
- Git (optional until you want to make major changes)
- Hardware
- 100 GB storage (minimum 50 GB)
- 16 GB RAM (minimum 8 GB)
We recommend following the the gitflow workflow and working in your own fork of the PEcAn repsitory. See the PEcAn developer guide for further details. In the /scripts
folder there is a script called syncgit.sh that will help with synchronizing your fork with the official repository.
To clone the PEcAn repository:
git clone git@github.com:pecanproject/pecan
cd pecan
# alternatively, if you haven't set up ssh keys with GitHub
# git clone https://github.com/PecanProject/pecan
The use of Docker in PEcAn is described in detail in the PEcAn documentation. This is intended as a quick start.
To install Docker and docker-compose, see the docker documentation:
- Docker Desktop in Mac OSX or Windows
- Docker (e.g. Ubuntu) and docker-compose on your linux operating system.
Note for Linux users: add your user to the docker group. This will prevent you from having to use sudo
to start the docker containers, and makes sure that any file that is written to a mounted volume is owned by you. This can be done using
# for linux users
sudo adduser ${USER} docker`.
To get started with development in docker we need to bring up the docker stack first. In the main pecan folder you will find the docker-compose.yml file that can be used to bring up the pecan stack. There is also the docker-compose.dev.yaml file that adds additional containers, and changes some services to make it easier for development.
By default docker-compose will use the files docker-compose.yml
and docker-compose.override.yml
. We will use the default docker-compose.yml
file from PEcAn. The docker-compose.override.yml
file can be used to configure it for your specific environment, in our case we will use it to setup the docker environment for development. Copy the docker-compose.dev.yml
file to docker-compose.override.yml
to start working with your own override file, i.e. :
For Linux/MacOSX
cp docker-compose.dev.yml docker-compose.override.yml
For Windows
copy docker-compose.dev.yml docker-compose.override.yml
You can now use the command docker-compose
to work with the containers setup for development. The rest of this document assumes you have done this step.
The steps in this section only need to be done the first time you start working with the stack in docker. After this is done you can skip these steps. You can find more detail about the docker commands in the pecan documentation.
- setup .env file
- create folders to hold the data
- load the postgresql database
- load some test data
- copy all R packages (optional but recommended)
- setup for web folder development (optional)
You can copy the docker/env.example
file as .env in your pecan folder. The variables we want to modify are:
For Linux/MacOSX
cp docker/env.example .env
For Windows
copy docker/env.example .env
COMPOSE_PROJECT_NAME
set this to pecan, the prefix for all containersPECAN_VERSION
set this to develop, the docker image we start with
Both of these variables should also be uncommented by removing the # preceding them. At the end you should see the following if you run the following command egrep -v '^(#|$)' .env
. If you have a windows system, you will need to set the variable PWD as well, and for linux you will need to set UID and GID (for rstudio).
For Linux
echo "COMPOSE_PROJECT_NAME=pecan" >> .env
echo "PECAN_VERSION=develop" >> .env
echo "UID=$(id -u)" >> .env
echo "GID=$(id -g)" >> .env
For MacOSX
echo "COMPOSE_PROJECT_NAME=pecan" >> .env
echo "PECAN_VERSION=develop" >> .env
For Windows:
echo "COMPOSE_PROJECT_NAME=pecan" >> .env
echo "PECAN_VERSION=develop" >> .env
echo "PWD=%CD%" >> .env
Once you have setup docker-compose.override.yml
and the .env
files, it is time to pull all docker images that will be used. Doing this will make sure you have the latest version of those images on your local system.
docker-compose pull
The goal of the development is to share the development folder with your container, whilst minimizing the latency. What this will do is setup the folders to allow for your pecan folder to be shared, and keep the rest of the folders managed by docker. Some of this is based on a presentation done during DockerCon 2020. In this talk it is recommended to keep the database on the filesystem managed by docker, as well as any other folders that are not directly modified on the host system (not using the docker managed volumes could lead to a large speed loss when reading/writing to the disk). The docker-compose.override.yml
can be modified to copy all the data to the local filesystem, you will need to comment out the appropriate blocks. If you are sharing more than the pecan home directory you will need to make sure that these folder exist. As from the video, it is recommended to keep these folders outside of the actual pecan folder to allow for better caching capabilities of the docker system.
If you have commented out the volumes in docker-compose.override.yml
you will need to create the folders. Assuming you have not modified the values, you can do this with:
mkdir -p $HOME/volumes/pecan/{lib,pecan,portainer,postgres,rabbitmq,traefik}
The following volumes are specified:
- pecan_home : is the checked out folder of PEcAn. This is shared with the executor and rstudio container allowing you to share and compile PEcAn. (defaults to current folder)
- pecan_web : is the checked out web folder of PEcAn. This is shared with the web container allowing you to share and modify the PEcAn web app. (defaults to web folder in the current folder)
- pecan_lib : holds all the R packages for the specific version of PEcAn and R. This folder will be shared amongst all other containers, and will contain the compiled PEcAn code. (defaults to managed by docker, or $HOME/volumes/pecan/lib)
- pecan this holds all the data, such as workflows and any downloaded data. (defaults to managed by docker, or $HOME/volumes/pecan/pecan)
- traefik holds persisent data for the web proxy, that directs incoming traffic to the correct container. (defaults to managed by docker, or $HOME/volumes/pecan/traefik)
- postgres holds the actual database data. If you want to backup the database, you can stop the postgres container, zip up the folder. (defaults to managed by docker, or $HOME/volumes/pecan/postgres)
- rabbitmq holds persistent information of the message broker (rabbitmq). (defaults to managed by docker, or $HOME/volumes/pecan/rabbitmq)
- portainer if you enabled the portainer service this folder is used to hold persistent data for this service. You will need to enable this service. (defaults to managed by docker, or $HOME/volumes/pecan/portainer)
These folders will hold all the persistent data for each of the respective containers and can grow. For example the postgres database is multiple GB. The pecan folder will hold all data produced by the workflows, including any downloaded data, and can grow to many giga bytes.
First we bring up postgresql (we will start RabbitMQ as well since it takes some time to start):
docker-compose up -d postgres rabbitmq
This will start postgresql and rabbitmq. We need to wait for a few minutes (you can look at the logs using docker-compose logs postgres
) to see if it is ready.
Once the database has finished starting up we will initialize the database. Now you can load the database using the following commands. The first command will make sure we have the latest version of the image, the second command will actually load the information into the database.
docker pull pecan/db
docker run --rm --network pecan_pecan pecan/db
Once that is done we create two users for BETY, first user is the guest user that you can use to login in the BETY interface. The second user is a user with admin rights.
docker-compose run --rm bety user guestuser guestuser "Guest User" guestuser@example.com 4 4
docker-compose run --rm bety user carya illinois "Carya Demo User" carya@example.com 1 1
Once the database is loaded we can add some example data, some of the example runs and runs for the ED model, assume some of this data is available. This can take some time, but all the data needed will be copied to the /data
folder in the pecan containers. As with the database we first pull the latest version of the image, and then execute the image to copy all the data:
docker pull pecan/data:develop
docker run -ti --rm --network pecan_pecan --volume pecan_pecan:/data --env FQDN=docker pecan/data:develop
Linux & Mac
# Change ownership of /data directory in pecan volume to the current user
docker run -ti --rm --network pecan_pecan --volume pecan_pecan:/data pecan/data:develop chown -R "$(id -u).$(id -g)" /data
docker run -ti --user="$(id -u)" --rm --network pecan_pecan --volume pecan_pecan:/data --env FQDN=docker pecan/data:develop
Next copy the R packages from a container to volume pecan_lib
. This is not really needed, but will speed up the process of the first compilation. Later we will put our newly compiled code here as well. This folder is shared with all PEcAn containers, allowing you to compile the code in one place, and have the compiled code available in all other containers. For example modify the code for a model, allows you to compile the code in rstudio container, and see the results in the model container.
You can copy all the data using the following command. This will copy all compiled packages to your local machine.
docker run -ti --rm -v pecan_lib:/rlib pecan/base:develop cp -a /usr/local/lib/R/site-library/. /rlib/
If you want to use the web interface, you will need to:
- Uncomment the web section from the
docker-compose.override.yml
file. This section includes three lines at the top of the file, just under theservices
section. Uncomment the lines that startweb:
,volumes:
, and- pecan_web:
. - Then copy the config.php from the docker/web folder. You can do this using
For Linux/MacOSX
cp docker/web/config.docker.php web/config.php
For Windows
copy docker\web\config.docker.php web\config.php
To begin development we first have to bring up the full PEcAn stack. This assumes you have done once the steps above. You don't need to stop any running containers, you can use the following command to start all containers. At this point you have PEcAn running in docker.
docker-compose up -d
The current folder (most likely your clone of the git repository) is mounted in some containers as /pecan
, and in the case of rstudio also in your home folder as pecan
. You can see which containers exactly in docker-compose.override.yml
.
You can now modify the code on your local machine, or you can use rstudio in the docker stack. Once you made changes to the code you can compile the code either in the terminal of rstudio (cd pecan && make
) or using ./scripts/compile.sh
from your machine (latter is nothing more than a shell script that runs docker-compose exec executor sh -c 'cd /pecan && make'
.
The compiled code is written to /usr/local/lib/R/site-library
which is mapped to volumes/lib
on your machine. This same folder is mounted in many other containers, allowing you to share the same PEcAn modules in all containers. Now if you change a module, and compile all other containers will see and use this new version of your module.
To compile the PEcAn code you can use the make command in either the rstudio container, or in the executor container. The script compile.sh
will run make inside the executor container.
You can submit your workflow either in the executor container or in rstudio container. For example to run the docker.sipnet.xml
workflow located in the tests folder you can use:
docker-compose exec executor bash
# inside the container
cd /pecan/tests
R CMD ../web/workflow.R docker.sipnet.xml
A better way of doing this is developed as part of GSOC, in which case you can leverage of the restful interface defined, or using the new R PEcAn API package.
Following are the main folders inside the pecan repository.
These are the core packages of PEcAn. Most other packages will depend on the packages in this folder.
Each subfolder contains the required pieces to run the model in PEcAn
Contains packages that either do analysis, or download and convert different data products.
The Pecan web application
Each subfolder is its own shiny application.
The PEcAn documentation that is compiled and uploaded to the PEcAn webpage.
Some of the docker build files. The Dockerfiles for each model are placed in the models folder.
Small scripts that are used as part of the development and installation of PEcAn.
If you want to start from scratch and remove all old data, but keep your pecan checked out folder, you can remove the folders where you have written the data (see folders
below). You will also need to remove any of the docker managed volumes. To see all volumes you can do docker volume ls -q -f name=pecan
. If you are sure, you can either remove them one by one, or remove them all at once using the command below. THIS DESTROYS ALL DATA IN DOCKER MANAGED VOLUMES..
docker volume rm $(docker volume ls -q -f name=pecan)
If you changed the docker-compose.override.yml file to point to a location on disk for some of the containers (instead of having them managed by docker) you will need to actually delete the data on your local disk, docker will NOT do this.
If you want to reset the pecan lib folder that is mounted across all machines, for example when there is a new version of PEcAn or a a new version of R, you will need to delete the volume pecan_lib, and repopulate it. To delete the volume use the following command, and then look at "copy R packages" to copy the data again.
docker-compose down
docker rm pecan_lib
(On Mac OSX and Windows files should automatically be owned by the user running the docker-compose commands).
If you use mounted folders, make sure that these folders are writable by the containers. Docker on Linux will try to preserve the file permissions. To do this it might be necessary for the folders to have rw permissions. This can be done by using chmod 777 $HOME/volumes/pecan/{lib,pecan,portainer,postgres,rabbitmq,traefik}
.
This will leverage of NFS to mount the file system in your local docker image, changing the files to owned by the user specified in the export file. Try to limit this to only your PEcAn folder since this will allow anybody on this system to get access to the exported folder as you!
First install nfs server:
apt-get install nfs-kernel-server
Next export your home directory:
echo -e "$PWD\t127.0.0.1(rw,no_subtree_check,all_squash,anonuid=$(id -u),anongid=$(id -g))" | sudo tee -a /etc/exports
And export the filesystem.
sudo exportfs -va
At this point you have exported your home directory, only to your local machine. All files written to that exported filesystem will be owned by you (id -u
) and your primary group (id -g
).
Finally we can modify the docker-compose.override.yml
file to allow for writing files to your PEcAn folder as you:
volumes:
pecan_home:
driver_opts:
type: "nfs"
device: ":${PWD}"
o: "addr=127.0.0.1"