Setting Up a System
This tutorial will walk you through setting up NSP for a simple mock system so you can play around with it.
Important
You will need to have ansible installed. This can be in a python environment (venv
or conda
). Just run
pip install ansible
.
Note
This tutorial is not comprehensive in the details of each role. If you want a detailed explanation of how each role works see Ansible Roles. This tutorial is designed to give you working configuration files that you can explore and use the detailed docs to clarify if you have questions.
Setup Configuration Repository
Create a folder and initialize an empty git repository.
$ mkdir nsp_tutorial && cd nsp_tutorial
$ git init --initial-branch=main
Initialized empty Git repository in /home/software/nsp_docs/.git/
Next add the NSP repository as a submodule in the roles directory.
Note
You can create you own ansible.cfg
but we provide one with good defaults in the nsp repository which we
suggest you link in.
$ git submodule add https://github.com/olcf/nccs-software-provisioning.git roles
Cloning into '/home/software/nsp_tutorial/roles'...
Username for 'https://github.com': REDACTED
Password for 'https://REDACTED@github.com':
remote: Enumerating objects: 115, done.
remote: Counting objects: 100% (115/115), done.
remote: Compressing objects: 100% (77/77), done.
remote: Total 115 (delta 17), reused 110 (delta 12), pack-reused 0 (from 0)
Receiving objects: 100% (115/115), 33.95 KiB | 331.00 KiB/s, done.
Resolving deltas: 100% (17/17), done.
$ ln -s roles/ansible.cfg ansible.cfg
$ git add .gitmodules roles/ ansible.cfg
$ git commit -m "Adding NSP submodule."
[main (root-commit) 5310052] Adding NSP submodule.
2 files changed, 4 insertions(+)
create mode 100644 .gitmodules
create mode 160000 roles
Add a New System
Create a directory with the name of your system. For this tutorial we use the name moria
.
$ mkdir moria
Next create a new playbook in the directory you just created for your system.
moria/playbook.yaml
- name: Deploy Moria
hosts: localhost
vars:
NSP_system_name: moria
NSP_install_root: "{{ ['/tmp', NSP_system_name] | path_join }}"
NSP_help_email: example@example.com
NSP_site_name: MySiteName
roles: [ ]
To validate your setup run.
$ ansible-playbook moria/playbook.yaml
PLAY [Deploy Moria] ************************************************************************************************
TASK [Gathering Facts] *********************************************************************************************
ok: [localhost]
PLAY RECAP *********************************************************************************************************
localhost : ok=1 changed=0 unreachable=0 failed=0 skipped=0 rescued=0 ignored=0
Init Scripts
For most of our systems, we maintain a set of init scripts that are sourced when a user logs in. These scripts allow us
to fine tune Lmod, set up convenience variables and more. As a first step for our new system we will add the
init role to moria/playbook.yaml
moria/playbook.yaml
- name: Deploy Moria
hosts: localhost
vars:
NSP_system_name: moria
NSP_install_root: "{{ ['/tmp', NSP_system_name] | path_join }}"
NSP_help_email: example@example.com
NSP_site_name: MySiteName
roles:
- role: init
Various roles in NSP will add to the init scripts but if you have extra content, that you want added to the init
scripts, you can do so by creating a profile
and/or cshrc
file in <system>/init
. For this tutorial we
will add some content to our init scripts that prints a welcome message.
moria/init/profile
echo "Welcome to {{ NSP_system_name }}!!!"
moria/init/cshrc
echo "Welcome to {{ NSP_system_name }}!!!"
Let’s run our playbook now and see what happens.
$ ansible-playbook moria/playbook.yaml
If you look, you will see that NSP has deployed our init scripts to /tmp/moria/init
.
tree /tmp/moria
/tmp/moria/
└── init
├── cshrc
└── profile
/tmp/moria/init/profile
#!/usr/bin/env bash
##
#| WARNING! This file is managed by Ansible.
#| Do NOT make manual changes to this file.
#| Please email example@example.com to request a change.
#
#| Info:
#| Role: init
#| Template: profile.j2
#| User: software
##
# BEGIN INIT MANAGED
echo "Welcome to moria!!!"
# END INIT MANAGED
/tmp/moria/init/cshrc
#!/usr/bin/env csh
##
#| WARNING! This file is managed by Ansible.
#| Do NOT make manual changes to this file.
#| Please email example@example.com to request a change.
#
#| Info:
#| Role: init
#| Template: cshrc.j2
#| User: software
##
# BEGIN INIT MANAGED
echo "Welcome to moria!!!"
# END INIT MANAGED
Bootstrap Lmod
The next step in setting up a system is installing Lmod. We will have to add a little more configuration to lmod
when we get to setting up the spack role but for now let’s add the lmod role to
moria/playbook.yaml
, create our rc.lua
template (this is where we set default modules etc.), create our
admin.list
template (this is where we set messages for modules) and run our playbook.
moria/playbook.yaml
- name: Deploy Moria
hosts: localhost
vars:
NSP_system_name: moria
NSP_install_root: "{{ ['/tmp', NSP_system_name] | path_join }}"
NSP_help_email: example@example.com
NSP_site_name: MySiteName
roles:
- role: init
- role: lmod
vars:
NSP_LMOD_version: 8.7.31
NSP_LMOD_install_type: internal
After we run our playbook again we can source the init script and see our new software stack!
$ ansible-playbook moria/playbook.yaml
...
$ source /tmp/moria/init/profile
$ module avail
------------------------------------------------- [ Base Modules ] -------------------------------------------------
DefApps (L)
Where:
L: Module is loaded
If the avail list is too long consider trying:
"module --default avail" or "ml -d av" to just list the default modules.
"module overview" or "ml ov" to display the number of modules for each name.
Use "module spider" to find all possible modules and extensions.
Use "module keyword key1 key2 ..." to search for all possible modules matching any of the "keys".
If you were to look at the init scripts you would see that they now have an additional section that was added by lmod.
/tmp/moria/init/profile
#!/usr/bin/env bash
##
#| WARNING! This file is managed by Ansible.
#| Do NOT make manual changes to this file.
#| Please email example@example.com to request a change.
#
#| Info:
#| Role: init
#| Template: profile.j2
#| User: software
##
# BEGIN INIT MANAGED
echo "Welcome to moria!!!"
# END INIT MANAGED
# BEGIN LMOD MANAGED
type module > /dev/null 2>&1
if [ "$?" -eq 0 ]; then
clearLmod -q > /dev/null 2>&1
unset LMOD_MODULEPATH_INIT
fi
export LMOD_SYSTEM_NAME=moria
export LMOD_SYSTEM_DEFAULT_MODULES=DefApps
export LMOD_PACKAGE_PATH=/tmp/moria/lmod/etc
export LMOD_AVAIL_STYLE=nsp-pretty:system
export LMOD_MODULERCFILE=/tmp/moria/lmod/etc/rc.lua
export LMOD_ADMIN_FILE=/tmp/moria/lmod/etc/admin.list
export LMOD_RC=/tmp/moria/lmod/etc/lmodrc.lua
source /tmp/moria/lmod/lmod/init/profile
module --initial_load --no_redirect restore
# END LMOD MANAGED
Adding Software
We can now add a variety of software through different roles. For our purposes we will add
one version of miniforge3
, gcc
and llvm
each. Run the playbook and observe where they are installed and
where their module files are placed.
Note
The gcc
and llvm
builds can take some time so be patient.
moria/playbook.yaml
- name: Deploy Moria
hosts: localhost
vars:
NSP_system_name: moria
NSP_install_root: "{{ ['/sw', NSP_system_name] | path_join }}"
NSP_help_email: example@example.com
NSP_site_name: MySiteName
roles:
- role: init
- role: lmod
vars:
NSP_LMOD_install_type: internal
NSP_LMOD_version: 8.7.37
- role: miniforge3
vars:
NSP_MINIFORGE3_version: 24.11.3
- role: gcc
vars:
NSP_GCC_version: 14.2.0
- role: llvm
vars:
NSP_LLVM_version: 19.1.0
# if your system architecture is not x86_64 you will need to set `NSP_LLVM_targets`
$ source /tmp/moria/init/profile
$ module avail
------------------------------------------------- [ Base Modules ] -------------------------------------------------
DefApps (L) gcc/14.2.0 llvm/19.1.0 miniforge3/24.11.3-0
Where:
L: Module is loaded
If the avail list is too long consider trying:
"module --default avail" or "ml -d av" to just list the default modules.
"module overview" or "ml ov" to display the number of modules for each name.
Use "module spider" to find all possible modules and extensions.
Use "module keyword key1 key2 ..." to search for all possible modules matching any of the "keys".
$ tree -L 2 /tmp/moria/
/sw/moria/
├── gcc
│ └── 14.2.0
├── init
│ ├── cshrc
│ └── profile
├── llvm
│ └── 19.1.0
├── lmod
│ ├── 8.7.37
│ ├── bootstrap
│ ├── cache
│ ├── etc
│ └── lmod -> 8.7.37
├── miniforge3
│ └── 24.11.3-0
└── modules
├── DefApps.lua
├── gcc
├── llvm
└── miniforge3
Spack
We are going to set up spack for the system gcc (mine is 13.3.0 but yours may be different) and the gcc and llvm versions
that we built above. We will have a Core
set of modules built by the system gcc and then a software stack
built on gcc 13.3.0 and llvm 18.1.6.
We name our spack environments according to the following schema core<year>.<month>
and sw<year>.<month>
(in this tutorial we will use core25.02
and sw25.02
). Create the following files:
moria/spack/environments/mirrors.yaml.j2
{{ ansible_managed | comment(beginning="##", end="##", decoration="#", prefix_count=0, postfix_count=0) }}
mirrors:
facility_builds: {{ [NSP_SPACK_root, "envs/bmirrors"] | path_join }}
source_mirror: {{ [NSP_SPACK_root, "envs/smirrors"] | path_join }}
moria/spack/environments/compilers.yaml.j2
{{ ansible_managed | comment(beginning="##", end="##", decoration="#", prefix_count=0, postfix_count=0) }}
compilers:
- compiler:
spec: gcc@13.3.0
paths:
cc: /usr/bin/gcc
cxx: /usr/bin/g++
f77: /usr/bin/gfortran
fc: /usr/bin/gfortran
operating_system: ubuntu24.04
modules: [ ]
# GCC compiler
- compiler:
spec: gcc@{{ gcc.version }}
paths:
cc: {{ gcc.path }}/bin/gcc
cxx: {{ gcc.path }}/bin/g++
f77: {{ gcc.path }}/bin/gfortran
fc: {{ gcc.path }}/bin/gfortran
operating_system: ubuntu24.04
modules:
- gcc/{{ gcc.version }}
# LLVM compiler
{% set gcc_v_list = gcc.version.split(".") %}
- compiler:
spec: clang@{{ llvm.version }}-gfortran{{ gcc_v_list[0] }}
paths:
cc: {{ llvm.path }}/bin/clang
cxx: {{ llvm.path }}/bin/clang++
f77: {{ gcc.path }}/bin/gfortran
fc: {{ gcc.path }}/bin/gfortran
operating_system: ubuntu24.04
modules:
- llvm/{{ llvm.version }}
moria/spack/environments/packages.yaml.j2
{{ ansible_managed | comment(beginning="##", end="##", decoration="#", prefix_count=0, postfix_count=0) }}
packages:
all:
buildable: true
providers:
blas: [openblas]
lapack: [openblas]
mpi: [openmpi]
moria/spack/environments/core25.02/spack.yaml.j2
{{ ansible_managed | comment(beginning="##", end="##", decoration="#", prefix_count=0, postfix_count=0) }}
# OLCF {{ NSP_system_name }} {{ _SPACK_environment_name }} Spack Environment
spack:
view: false
include:
- concretizer.yaml
- mirrors.yaml
- modules.yaml
- compilers.yaml
- config.yaml
- packages.yaml
modules:
default:
lmod:
core_compilers: [ gcc@13.3.0 ]
projections:
# core
'%gcc@13.3.0': '25.02/{name}/{version}'
# -------------------------------------------------------------------
# Specs Definitions
# -------------------------------------------------------------------
definitions:
- core_compiler:
- '%gcc@13.3.0'
- core_25.02:
- matrix:
- - cmake
- tmux
- wget
- - $core_compiler
specs:
- $core_25.02
moria/spack/environments/sw25.02/spack.yaml.j2
{{ ansible_managed | comment(beginning="##", end="##", decoration="#", prefix_count=0, postfix_count=0) }}
# OLCF {{ NSP_system_name }} {{ _SPACK_environment_name }} Spack Environment
spack:
view: false
include:
- concretizer.yaml
- mirrors.yaml
- modules.yaml
- compilers.yaml
- config.yaml
- packages.yaml
# -------------------------------------------------------------------
# Specs Definitions
# -------------------------------------------------------------------
definitions:
- gcc_compilers:
- '%gcc@{{ gcc.version }}'
- llvm_compilers:
- '%clang@{{ llvm.version }}'
- all_compilers:
- $gcc_compilers
- $llvm_compilers
- sw-25.02:
- boost ~mpi
- boost +mpi
- openmpi
# -------------------------------------------------------------------
# Final Spec Matrices
# -------------------------------------------------------------------
- sw_cpu:
- matrix:
- - $sw-25.02
- - $all_compilers
specs:
- $sw_cpu
moria/spack/environments/sw25.02/variables.yaml
# GCC
gcc:
version: 14.2.0
path: '{{ NSP_install_root }}/gcc/14.2.0'
# LLVM
llvm:
version: 19.1.0
path: '{{ NSP_install_root }}/llvm/19.1.0'
moria/spack/environments/core25.02/variables.yaml
# this should be a symlink to `moria/spack/environments/sw25.02/variables.yaml`
Finally we will need to add the spack role to our playbook.
moria/playbook.yaml
- name: Deploy Moria
hosts: localhost
vars:
NSP_system_name: moria
NSP_install_root: "{{ ['/sw', NSP_system_name] | path_join }}"
NSP_help_email: example@example.com
NSP_site_name: MySiteName
roles:
- role: init
- role: lmod
vars:
NSP_LMOD_install_type: internal
NSP_LMOD_version: 8.7.37
- role: miniforge3
vars:
NSP_MINIFORGE3_version: 24.11.3
- role: gcc
vars:
NSP_GCC_version: 14.2.0
- role: llvm
vars:
NSP_LLVM_version: 19.1.0
# if your system architecture is not x86_64 you will need to set `NSP_LLVM_targets`
- role: spack
vars:
NSP_SPACK_versions:
v0.23.1:
git_reference: 2bfcc69
NSP_SPACK_environments:
core25.02:
spack_version: v0.23.1
shared_templates:
- mirrors
sw25.02:
spack_version: v0.23.1
shared_templates:
- mirrors
After running the playbook explore /tmp/moria/spack/configs
. To install our software via spack run the following.
cd /sw/moria/spack/configs
source spacktivate # choose the core25.02 env
spack concretize
spack install
source spacktivate # choose the sw25.02 env
spack concretize
spack install
All of our software should now be installed to /tmp/moria/spack/envs
; however, none of it shows up for
module avail
yet, but all of the modules are in /tmp/moria/spack/modules
.
Lmod Hook & Core
The final part of our Lmod configuration is setting up the custom hook. This will make the software that we built with
our sw25.02
spack environment visible. Also, because we have a versioned core environment, we will create a module
file to add the Core
(a.k.a core25.02
) environment that we installed.
moria/files/Core/25.02.lua.j2
{{ ansible_managed | comment(beginning="--[[", end="]]--", decoration="", prefix_count=0, postfix_count=0) }}
help("Add path for Core 25.02 modules to MODULEPATH")
prepend_path{"MODULEPATH","{{ NSP_SPACK_root }}/modules/Core/25.02", priority=10}
Modify the playbook to include configuration for our hook and the files role which is where we are keeping
the Core
module files. Also add some modules to lmod’s NSP_LMOD_default_modules.
erebor.yaml
- name: Deploy Moria
hosts: localhost
vars:
NSP_system_name: moria
NSP_install_root: "{{ ['/sw', NSP_system_name] | path_join }}"
NSP_help_email: example@example.com
NSP_site_name: MySiteName
roles:
- role: init
- role: lmod
vars:
NSP_LMOD_install_type: internal
NSP_LMOD_version: 8.7.37
NSP_LMOD_default_modules:
- Core/25.02
- gcc/14.2.0
NSP_LMOD_hierarchy:
compiler:
members: [ gcc, llvm ]
paths:
- {path: '|compiler.name|-|compiler.version|', weight: 20}
- {path: '|mpi.name|-|mpi.version|/|compiler.name|-|compiler.version|', weight: 30}
level: 0
mpi:
members: [ openmpi ]
paths:
- {path: '|mpi.name|-|mpi.version|/|compiler.name|-|compiler.version|', weight: 30}
level: 1
- role: miniforge3
vars:
NSP_MINIFORGE3_version: 24.11.3
- role: gcc
vars:
NSP_GCC_version: 14.2.0
- role: llvm
vars:
NSP_LLVM_version: 19.1.0
# if your system architecture is not x86_64 you will need to set `NSP_LLVM_targets`
- role: spack
vars:
NSP_SPACK_versions:
v0.23.1:
git_reference: 2bfcc69
NSP_SPACK_environments:
core25.02:
spack_version: v0.23.1
shared_templates:
- mirrors
sw25.02:
spack_version: v0.23.1
shared_templates:
- mirrors
- role: files
vars:
NSP_FILES_inventory:
- src: Core
dest: "{{ [NSP_module_root, 'Core'] | path_join }}"
After running the playbook again our example software stack is complete! Run the playbook to generate new init scripts, source the init scripts and test out the new stack.
$ ansible-playbook moria/playbook.yaml
...
$ source /tmp/moria/init/profile
$ module load openmpi boost
$ module avail
------------------------------------------ [ gcc/14.2.0, openmpi/5.0.5 ] -------------------------------------------
boost/1.86.0-mpi
-------------------------------------------------- [ gcc/14.2.0 ] --------------------------------------------------
boost/1.86.0 (D) openmpi/5.0.5 (L)
-------------------------------------------------- [ Core/25.02 ] --------------------------------------------------
cmake/3.30.5 tmux/3.4 wget/1.24.5
------------------------------------------------- [ Base Modules ] -------------------------------------------------
Core/25.02 (L) DefApps (L) gcc/14.2.0 (L) llvm/19.1.0 miniforge3/24.11.3-0
Where:
L: Module is loaded
D: Default Module
...
$ ml load llvm
Lmod is automatically replacing "gcc/14.2.0" with "llvm/19.1.0".
Due to MODULEPATH changes, the following have been reloaded:
1) openmpi/5.0.5
$ module avail
------------------------------------------ [ llvm/19.1.0, openmpi/5.0.5 ] ------------------------------------------
boost/1.86.0-mpi
------------------------------------------------- [ llvm/19.1.0 ] --------------------------------------------------
boost/1.86.0 (D) openmpi/5.0.5 (L)
-------------------------------------------------- [ Core/25.02 ] --------------------------------------------------
cmake/3.30.5 tmux/3.4 wget/1.24.5
------------------------------------------------- [ Base Modules ] -------------------------------------------------
Core/25.02 (L) DefApps (L) gcc/14.2.0 llvm/19.1.0 (L) miniforge3/24.11.3-0
...