Skip to content

A packer + ansible project to build and deploy an Amazon Machine Image and Docker Container for use by the various THREDDS projects at Unidata for running automated tests.

License

Notifications You must be signed in to change notification settings

Unidata/thredds-test-environment

THREDDS Test Environment

A packer + ansible project to build Docker images for use by the various THREDDS projects at Unidata for running automated tests. The Docker images are used as the basis for Jenkins build nodes as well as a custom GitHub action for testing the various THREDDS projects. The image is based off the latest available image for ubuntu 20.04. The docker images produced by packer are called "thredds-test-environment" (for use by Jenkins) of thredds-test-action (for use by the GitHub action).

Requirements

If this is your first time running the build, you will need to initialize packer using:

packer init .

While this project's packer configuration relies heavily on Ansible, you do not need to have Ansible installed locally. We utilize the ansible-local provisioner of Packer, which means Ansible is run on the remote/guest machine, and not on the build machine (the machine running packer). This does mean that part of the Packer build process includes installing Ansible on the remote/guest machine (see packer/provisioners/scripts/bootstrap-common.sh), which does add some time to the total build process (approximately two minutes). However, Since Ansible does not support the use of Windows systems as a control node, the ability create the THREDDS test environment images across all major platforms outweighs the extra cost in time.

Building the images

To generate the golden Docker images, start off by validating the packer configuration by first moving into the packer/ directory and then running:

packer validate .

Once validated, you may run the build using:

packer build --only=<type> thredds-test-env.pkr.hcl

<type> will one or more (separated by commas) of the following:

  • docker.docker-jenkins: Provision a Docker container and generate and tag a local Docker image (docker.unidata.ucar.edu/thredds-test-environment:latest).
  • docker.docker-export: Provision a Docker container and generate a local Docker image as a file (image.tar).
  • docker.docker-github-action: Provision a Docker container for use with GitHub Actions and publish to the GitHub Package Repository (runs using manual workflow_dispatch trigger on github actions).
  • docker.docker-github-action-nexus: Same as docker-github-action, but tags for publishing to the Unidata Nexus Repository.

Typically, we would run the following to update the Jenkins and Github Action Docker images (hosted on Nexus) at the same time:

packer build --only=docker.docker-jenkins,docker.docker-github-action-nexus thredds-test-env.pkr.hcl

The Docker image builds takes about 1 hour to create. Packer will run the builders in parallel, so the total time to create the thredds-test-environment images is around an hour.

If using docker-jenkins, then once the image is built you can test out the environment by using:

docker run -i -t --rm docker.unidata.ucar.edu/thredds-test-environment:latest

Note that images are not pushed as part of this build process. Pushes can be done via the normal docker mechanisms, e.g. docker image push docker.unidata.ucar.edu/thredds-test-environment:latest.

Project layout

Inside the packer directory is a file called thredds-test-env.pkr.hcl. This contains the packer configuration for the builders (docker builder), the provisioners (shell scripts, ansible playbooks), and the post-processors (tagging the docker image).

The provisioners directory contains the provisioner configurations used by packer. There are three types of provisioners used by the packer configuration: ansible, file, and scripts.

The files/ directory contains Docker entrypoint scripts:

  • entrypoint.sh: drives the GitHub Action.
  • jenkins-agent.sh: connects a container to the jenkins control .

The scripts/ directory contains the following shell scripts:

  • bootstrap-common.sh: Install and configure ansible (common to both builders)
  • cleanup.sh: Runs after the ansible provisioner and cleans up the apt cache, as well as some general build environment things.

The ansible/ directory contains the ansible playbooks that are used to configure the testing environment. The ansible/ directory is laid out as follows:

  • site.yml: The main ansible playbook that references all other playbooks used by the build
  • roles/: Playbooks organized by common tasks

We use the following roles when provisioning our images:

  • cleanup: General cleanup related tasks, such as remove the temporary build directory and running ldconfig
  • general-packages: Install general packages needed for the build environment using the OS package manager.
  • gradle-builds-cache-bootstrap: Pull in and build netCDF-Java to populate the gradle cache for user ubuntu.
  • init: Initialize the build environment by ensuring the temporary ansible build directory exists.
  • libnetcdf-and-deps: Configure, build, and install zlib, HDF5, and netCDF-C.
  • maven: Obtain and install the Apache Maven software project management and comprehension tool.
  • miniconda: Obtain and install the Anaconda miniconda python distribution.
  • security:
    • Add the ubuntu user.
    • Add a default maven-settings.xml file configured to publish to the Unidata artifacts server.
    • Configure ssh (uses modified version of a task from Jeff Geerling's ansible-role-security project - see packer/provisioners/ansible/roles/security/README.md).
    • Configure a system wide bash environment.
  • temurin: Obtain and install LTS versions of Temurin.
  • test-data-mount-prep: Prepare the environment to mount the thredds-test-data datasets when available (currently used on Jenkins worker nodes).
  • zulu: Obtain and install LTS versions of Zulu.

We also use a role from Ansible Galaxy to setup a Ruby environment (geerlingguy.ruby).

THREDDS Test Environment Highlights

netCDF-C

  • location: /usr/thredds-test-environment
  • version: 4.8.1
  • dependencies (same location):
    • zlib version: 1.2.11
    • hdf5 version: 1.12.1

miniconda

  • location: /usr/thredds-test-environment/miniconda3
  • version: Miniconda3-latest-Linux-x86_64

maven:

  • location: /usr/thredds-test-environment/mvn
  • version: 3.6.3

Java:

  • Temurin (latest version available from adoptium.net)

    • 8 (/usr/thredds-test-environment/temurin8)
    • 11 (/usr/thredds-test-environment/temurin11)
    • 17 (/usr/thredds-test-environment/temurin17)
    • 21 (/usr/thredds-test-environment/temurin21)
  • Zulu (latest version available from azul.com)

    • 8 (/usr/thredds-test-environment/zulu8)
    • 11 (/usr/thredds-test-environment/zulu11)
    • 17 (/usr/thredds-test-environment/zulu17)
    • 21 (/usr/thredds-test-environment/zulu21)

Ruby

Bash functions:

  • select-java <vendor> <version> (where version is 8, 11, 17, or 21, and vendor is temurin or zulu)
  • activate-conda
  • get_pw <key>

Latest version available via the OS Package Manager

  • sed
  • dos2unix
  • git
  • fonts-dejavu
  • fontconfig
  • openssh-server

Example Timings

Docker Image

    docker.docker-jenkins: Friday 22 November 2024  21:12:32 +0000 (0:00:02.871)       0:20:20.107 *******
    docker.docker-jenkins: ===============================================================================
    docker.docker-jenkins: Wait for the HDF5 async test task to complete. ------------------------ 362.21s
    docker.docker-jenkins: libnetcdf-and-deps : Install hdf5. ------------------------------------ 272.84s
    docker.docker-jenkins: libnetcdf-and-deps : Configure netCDF-c. ------------------------------ 128.28s
    docker.docker-jenkins: libnetcdf-and-deps : Configure hdf5. ---------------------------------- 114.44s
    docker.docker-jenkins: zulu : Fetch latest Zulu Java builds. ---------------------------------- 54.52s
    docker.docker-jenkins: temurin : Fetch latest Temurin Java builds. ---------------------------- 46.20s
    docker.docker-jenkins: zulu : Unpack Zulu Java Installations. --------------------------------- 35.03s
    docker.docker-jenkins: libnetcdf-and-deps : Install netCDF-c. --------------------------------- 28.84s
    docker.docker-jenkins: general-packages : Install os managed tools. --------------------------- 28.32s
    docker.docker-jenkins: temurin : Unpack Temurin Java Installations. --------------------------- 26.71s
    docker.docker-jenkins: general-packages : Install os managed tools. --------------------------- 24.31s
    docker.docker-jenkins: geerlingguy.ruby : Install ruby and other required dependencies. ------- 12.96s
    docker.docker-jenkins: general-packages : Install os managed tools. --------------------------- 12.77s
    docker.docker-jenkins: miniconda : Download and unpack miniconda. ------------------------------ 9.31s
    docker.docker-jenkins: libnetcdf-and-deps : Install zlib. -------------------------------------- 5.46s
    docker.docker-jenkins: libnetcdf-and-deps : Configure zlib. ------------------------------------ 4.13s
    docker.docker-jenkins: security : Update SSH configuration to be more secure. ------------------ 3.78s
    docker.docker-jenkins: libnetcdf-and-deps : Download and unpack hdf5. -------------------------- 3.59s
    docker.docker-jenkins: zulu : Read versions of installed Zulu. --------------------------------- 3.34s
    docker.docker-jenkins: cleanup : Remove packages that are not needed in final environment. ----- 3.07s

About

A packer + ansible project to build and deploy an Amazon Machine Image and Docker Container for use by the various THREDDS projects at Unidata for running automated tests.

Topics

Resources

License

Code of conduct

Security policy

Stars

Watchers

Forks

Contributors 4

  •  
  •  
  •  
  •