HPC Container Build System for Apptainer with OpenFOAM, MPI, and scientific computing focus
Project description
Build apptainer containers for your projects
This is a project to automate building HPC-ready containers (originally for OpenFOAM-based projects)
using apptainer.
[!NOTE] Migration from Ansible: The previous Ansible-based system (
build.yaml) has been replaced with a Python-based "container-as-code" system. Theconfig.yamlformat and container paths remain fully backward compatible so no changes are required to your existing configurations.v1tag still has the ansible-based mechanism if you want it
[!NOTE] Brought to you by SDL Energy Conversion from
in collaboration with
.
- Idea
- Highlighted features
- Quick Instructions
- New Features (in v2)
- Migration from Ansible-based mechanism (v1 -> v2)
Idea
Automated workflows to:
- Build a base framework (eg:
OpenFOAM) container (supporting various forks and versions) to run on HPCs - Build project-specific containers that inherit from a target base container
- OpenMPI is a first-class citizen:
mpirun -n 16 apptainer run container.sif "solver -parallel"should 'just work'.
Highlighted features
- Automated, configuration-only workflows to produce containers that behave similarly across frameworks.
- A JSON Database of container metadata, with full control at the hands of the container maintainer.
- Maintaining definition files for your projects can be done in your own repos.
- Loading your own repositories of base framework container definitions works seamlessly.
Quick Instructions
sudo add-apt-repository -y ppa:apptainer/ppa
sudo apt install -y apptainer
curl -LsSf https://astral.sh/uv/install.sh | sh
uvx hpctainers --config config.yaml
[!TIP] The build system uses Python 3.10+ with
uvfor dependency management. The configuration file and base definitions provided serve as examples to build OpenFOAM containers.
The build command (by default) will:
- Create the following tree in the current working folder:
containers/
├── basic
│ ├── opencfd-openfoam.sif
│ └── ubuntu-24.04-openmpi-4.1.5.sif
└── projects
└── test-master.sif
- Build a basic OpenMPI container
containers/basic/ubuntu-24.04-openmpi-4.1.5.sif, or pull it from ghcr.io if possible - Build a base (OpenCFD) OpenFOAM container
containers/basic/opencfd-openfoam.sif, or pull it from ghcr.io if possible - Build a test project container, to make sure MPI works alright in OpenFOAM containers
Check the docs.md for details on how the configuration file is expected to be structured.
Here is a simplified sequence diagram describing the expected workflow:
sequenceDiagram
actor User
participant Build System
participant Build Cache
participant GHCR
participant Docker Hub
participant Host Env
participant Temp File
participant Local Build
participant MPI Container
participant Framework Container
participant Project Container
User->>Build System: Start build.py with config.yaml
Build System->>Build Cache: Check if MPI Container cached
Build Cache-->>Build System: Not cached or content changed
Build System->>GHCR: Try to pull MPI Container
GHCR-->>Build System: MPI Container not found
Build System->>Docker Hub: Pull Ubuntu image
Docker Hub-->>Build System: Ubuntu image pulled
Build System->>Local Build: Build MPI Container on top of Ubuntu image
Local Build-->>MPI Container: MPI Container created
Build System->>Build Cache: Update cache entry
Build System->>GHCR: Try to pull Framework Container
GHCR-->>Build System: Framework Container not found
Build System->>Local Build: Build Framework Container on top of MPI Container
Local Build-->>Framework Container: Framework Container created
Build System->>Build Cache: Update cache entry
Note over Build System,Host Env: Environment Secrets Injection (Optional)
Build System->>Host Env: Read environment secrets (if specified)
Host Env-->>Build System: Return secret values
Build System->>Temp File: Create temp file with secrets
Temp File-->>Build System: Container-specific path
Build System->>Local Build: Build Project Container with bind-mounted secrets
Local Build->>Project Container: Bind-mount temp file to /tmp/hpctainers_build_env_<name>.sh
Project Container->>Project Container: Source secrets in %post, then remove
Local Build-->>Project Container: Project Container created (secrets removed)
Build System->>Temp File: Clean up temp file from host
Build System->>Build Cache: Update cache entry
Project Container-->>User: Container ready for use (no secrets persisted)
New Features (in v2)
1. Intelligent Build Caching
- Content-based hashing of definition files and build arguments
- Automatic rebuild detection when definitions change
- Cascade rebuilds when base containers change
- Persistent cache stored in
.build-cache/ - Use
--force-rebuildto ignore cache
2. Parallel Builds
- Automatically builds independent containers concurrently
- Respects dependency order (MPI → Framework → Project)
- Configurable worker count:
--parallel-jobs N - Significant speedup for multiple container builds
3. Dependency Graph Visualization
- Generates visual dependency graphs in SVG/DOT format
- Shows build order and parallel groups
- Highlights cached containers
- Use
--graph-onlyto visualize without building
4. Enhanced CLI
# Standard build
uvx hpctainers
# Force rebuild everything
uvx hpctainers --config config.yaml --force-rebuild
# Dry run (show what would be built)
uvx hpctainers --config config.yaml --dry-run
# Generate dependency graph
uvx hpctainers config.yaml --graph-only
# Build with 4 parallel jobs
uvx hpctainers config.yaml --parallel-jobs 4
# Disable registry pulls
uvx hpctainers config.yaml --no-pull
# Verbose output for debugging
uvx hpctainers config.yaml --verbose
5. Better Error Handling
- Per-container log files (same as before)
- Detailed error messages with context
- Graceful failure handling
- Build continues for independent containers after failures
6. Container-as-Code Philosophy
- Pure Python implementation
- Type-safe configuration parsing with Pydantic
- Modular library structure for extensibility
7. Secure Environment Secrets
- Inject host environment variables securely into container builds
- Variables sourced during %post and automatically removed
- No secrets persisted in final container image
- Container-specific temp file paths prevent conflicts
- Works across all three build methods (Python API, Shell, YAML)
- YAML: Use
env_orENV_prefix in build_args - See examples/environment_secrets.md for usage
Migration from Ansible-based mechanism (v1 -> v2)
The config.yaml format is fully backward compatible. Simply replace the build command:
# Old (Ansible)
ansible-playbook build.yaml --extra-vars="original_dir=$PWD" --extra-vars="@config.yaml"
# New (Python)
uvx hpctainers --config config.yaml
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file hpctainers-2.0.0.tar.gz.
File metadata
- Download URL: hpctainers-2.0.0.tar.gz
- Upload date:
- Size: 68.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.18
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
498288553cc9346b6778aef10c928c8940c1eefc76224f69d5d335ac4d4dd4c1
|
|
| MD5 |
cb2f76406199f19f195026d27c99e5b1
|
|
| BLAKE2b-256 |
d4ce9e85dbe0cbad493552c0cbe402a0ce54340b06a2ee9dd801a5b915c7fc6e
|
File details
Details for the file hpctainers-2.0.0-py3-none-any.whl.
File metadata
- Download URL: hpctainers-2.0.0-py3-none-any.whl
- Upload date:
- Size: 89.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.18
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a574e3f6b2fa73f9655c7aec9ccc3b9de42a38596848ef0ad15989bf132632bf
|
|
| MD5 |
c285c1caff2a6b2ba985a05b4287a9bb
|
|
| BLAKE2b-256 |
e444ab87e6dcc1a7c7f2a10284778eea0d45c1466ce2bd1993f1a650b5b61df7
|