File: ci.md

package info (click to toggle)
libsbml 5.20.5%2Bdfsg-2
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 117,108 kB
  • sloc: cpp: 469,781; xml: 364,270; ansic: 54,078; python: 12,540; makefile: 9,759; sh: 9,245; cs: 8,586; java: 8,151; perl: 6,133; ruby: 4,760; javascript: 1,605; php: 202; csh: 3
file content (202 lines) | stat: -rw-r--r-- 12,588 bytes parent folder | download | duplicates (4)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
# Developer documentation of CI/CD for libSBML
Continuous integration/continuous deployment (CI/CD) for libSMBL is implemented in [GitHub actions](https://docs.github.com/en/actions). 

This file provides a description of the jobs that make up this CI/CD system, as documentation for developers - a description of the artefacts provided to users can be found in [artefacts.md](https://github.com/sbmlteam/libsbml/blob/development/artefacts.md).

The jobs run differ depending on the event (push, PR, nightly build) and are summarised below. The YAML files triggering each CI/CD run can be found under [`./github/workflows/`](https://github.com/sbmlteam/libsbml/tree/development/.github/workflows).

Note that links herein point to the SBML Team GitHub repo default branch, and may differ in branches and forks.
## Nightly builds ([store-artefact.yml](https://github.com/sbmlteam/libsbml/actions/workflows/store-artefact.yml))

### Configurations
Nightly builds are provided for the latest Ubuntu/MacOS and Windows environments, as well as a `pep 513`-compliant build (`manylinux2010`) for backwards compatibility with older CentOS-based linux systems (CentOS 6 and more recent). All nightly builds are available via the [Actions tab of the libSBML Github repository site](https://github.com/sbmlteam/libsbml/actions/workflows/store-artefact.yml). The varying parameters are summarised below, resulting in 8 artefacts being rebuilt and uploaded every 24 hours, at 5 AM GMT.

| parameters | value(s) taken                                                     |
| ---------- | ------------------------------------------------------------------ |
| OS         | `windows-latest`, `macos-latest`, `ubuntu-16.04`, `manylinux2010`* |
| packages   | all, stable                                                        |

<sub>\*Note on ManyLinux: `manylinux2010` runs as a separate job (not as part of the strategy matrix) in the same workflow. This is because it runs inside a container, and not directly in a virtual environment provided by GitHub actions. `manylinux2010` also requires `checkout@v1` and `upload-artefact@v1`. Older ManyLinux is incompatible with GitHub Actions, as they require even older versions of `node`. A future workaround would be to run the ManyLinux job "manually" by using `docker` calls inside a more modern GitHub actions runner, instead of the current solution of using the `container: ` syntax.</sub>

The parameters kept constant in the nightly builds are:

| constant parameters | value               |
| ------------------- | ------------------- |
| bindings            | Java, C#, Python, R |
| XML parser          | libxml2             |
| namespaces          | true                |
| with examples       | true                |
| strict includes     | true                |
| C++ standard        | 98                  |

Note that the nightly builds are run only on the default branch, and only on the SBML team repo (and not on its forks).


## On push ([brief.yml](https://github.com/sbmlteam/libsbml/actions/workflows/brief.yml))
On every push, we run 8 tests configurations. These differ by the following parameters:

| parameters   | value(s) taken                                                    |
| ------------ | ----------------------------------------------------------------- |
| OS           | `windows-latest`, `macos-latest`, `ubuntu-16.04`, `manylinux2010` |
| C++ standard | 98, 20                                                            |

while these build parameters are kept constant:

| constant parameters | value                |
| ------------------- | -------------------- |
| bindings            | Java, C#, Python, R* |
| XML parser          | libxml2              |
| namespaces          | true                 |
| with examples       | true                 |
| strict includes     | true                 |
| packages            | all                  |

<sub>\* R builds only run for Ubuntu, to provide feedback more quickly.</sub>

## On PR ([extensive.yml](https://github.com/sbmlteam/libsbml/actions/workflows/extensive.yml))
Testing is more extensive when a full (non-draft) PR is opened.

This step includes 24 different configurations, 8 per OS (Windows, Mac, Ubuntu). No `manylinux` build is currently run on PR.

This is the stage where the two non-default XML parsers (`xerces`, `expat`) are also tested. As the XML parsers are largely independent of the rest of the codebase, they are only tested in one configuration/OS - with all packages, but without language bindings.

| parameters | value(s) taken                                   |
| ---------- | ------------------------------------------------ |
| OS         | `windows-latest`, `macos-latest`, `ubuntu-16.04` |
| packages   | all, stable, none                                |
| namespaces | true, false                                      |
| XML parser | libxml2, expat*, xerces*                         |

<sub>\* only with all packages and no bindings, for all OS.</sub>

| constant parameters | value               |
| ------------------- | ------------------- |
| bindings            | Java, C#, Python, R |
| C++ standard        | 98                  |
| with examples       | true                |
| strict includes     | true                |


## Notes on the CI/CD system

### Understanding the structure of the YAML files
The GitHub Actions `cmake` template was used as a basis. [The GitHub Actions introduction page](https://docs.github.com/en/actions/learn-github-actions/introduction-to-github-actions) defines the fundamental terminology needed to understand the YAML files: job, step, action. Understanding [a strategy matrix](https://docs.github.com/en/actions/learn-github-actions/managing-complex-workflows#using-a-build-matrix) is also necessary.

For libSBML, the YAML files are generally structured into:
- define strategy matrix
- checkout the repository
- dependency installation and system setup
- configure with `cmake`
- build
- test with `ctest`

The nightly build additionally uploads some artefacts at the end.

### Dependency installation
Dependency installation varies across operating systems, and this is implemented via conditions on the strategy matrix entry, e.g.
```yaml
      - name: Do something Windows-specific
        if: matrix.platform == 'windows-latest'
        run: # some windows-specific commands
```
For convenience, we set up the `ninja` generator for all runs using the [`seanmiddleditch/gha-setup-ninja`](https://github.com/marketplace/actions/install-ninja-build-tool) action.

On Unix-based systems, missing dependences are typically installed via standard package managers (`brew`, `apt-get`, `yum`). In `manylinux2010`, recent enough versions of `cmake` (via `pip`) and `swig` (compile from source) are installed in a non-standard way.

Windows builds require a precompiled dependencies folder, which is downloaded from SourceForge and cached, as well as a "manual" swig set-up by the CI. Additionally, we use the [`ilammy/msvc-dev-cmd`](https://github.com/marketplace/actions/enable-developer-command-prompt) action to ensure `msbuild` remains accessible to `cmake` (it's not by default because we use the `ninja` generator).

### CMake configuration
All runs use the `ninja` generator to configure the `cmake` build. This has the advantage of automatically parallelising the build under the hood.

### ccache
We further use `ccache` to take advantage of several runs repeatedly calling the same compilation command (currently on MacOS and Ubuntu only). Essentially, [`ccache` caches compilation outputs and creates a unique hash for each of them](https://ccache.dev/manual/4.3.html#_how_ccache_works). If there is a cache hit, the compilation is not run, but is replaced by the compilation output.
Windows builds are compiled using the MSVC static runtime (`-DWITH_STATIC_RUNTIME`). In our setup, this means that we use [GitHub actions `cache` action](https://docs.github.com/en/actions/guides/caching-dependencies-to-speed-up-workflows) to cache the `.ccache` folder (where the hashes and compilation outputs are stored) with an OS-specific key, meaning runs can take advantage of the `ccache` information from previous CI runs on the same OS to speed up compilation.

### namespaces and strict includes
All artefacts are compiled with namespaces and "strict includes" enabled. Using strict includes preserves backwards compatibility with previous routine practice to `#include <SBMLTypes.h>` to access the entirety of SBML, which was achieved via indirect includes (i.e. `SBMLTypes` would call further `#include`s). This provides a simple access to `libSBML`, but leads to longer include chains, and therefore unnecessary re-compilations, when partially re-building.

For quicker re-builds, not using strict includes accelerates compilation through constructs like
```
#ifndef LIBSBML_USE_STRICT_INCLUDES
#include not_totally_necessary.h
#endif
```
but you need to be sure that you're including everything you need directly.

### Python

Unix-based builds use the `PYTHON_USE_DYNAMIC_LOOKUP` option, which allows libSBML binaries to dynamically link to available Python libraries, instead of being tied to the Python libraries of the OS the binaries were compiled on.

### R package and bindings
Unix-based systems additionally build a binary R package in two steps: first the compiliation is configured to skip building the R binaries and create a source package instead (`-DWITH_CREATE_R_SOURCE=ON -DWITH_SKIP_R_BINARY=ON`). The binary R package is the created from the source package using the `R CMD INSTALL`.See the documentation for [creating R source packages and binary packages](https://cran.r-project.org/doc/manuals/r-release/R-exts.html#Building-binary-packages).

# Using the CI system to make a libSBML release
This section of the documentation is work-in-progress. Knowing the differences will help us develop a process to release libSBML more automatically in the future.

## Differences to last manual release
Up to libSBML 5.19.0, releases were done manually &mdash; this took a lot of time. These can be found on [Sourceforge](https://sourceforge.net/projects/sbml/files/libsbml/5.19.0/) and comprised: 

### Structure of Sourceforge libSBML release 5.19.0
- stable
    - MATLAB interface
        - MATLAB binaries for all OS (zip + tar)
    - R interface
        - R Source Package (tar)
    - Windows 
        - R installer for 32/64 bit Windows (zip)
        - 64-bit
            - Windows installer with MATLAB (exe)
            - Windows installer without MATLAB (exe)
            - Python
                - python interface installers for py2.7 and py3.6-3.9 (exe)
        - 32-bit
            - [analogous to 64-bit]
    - MacOS
        - installer (dmg, tar)
    - Linux
        - 64-bit
            - Ubuntu binaries (tar)
            - CentOS binaries (tar)
            - Debian installer (deb)
            - RPM installer (rpm)
        - 32-bit
            - [analogous to 64-bit]
    - core libSMBL source code (zip+tar)
    - full libSBML source code (zip+tar)
    - individual accepted level-3 packages source code (zip+tar)
    - libSBML documentation (zip+tar)
- experimental
    - source
        - R interface
            - R source package (tar)
        - libSBML source code (zip, tar)
        - experimental packages source code (spatial, req, dyn, arrays) (zip)
    - binaries
        - Windows
            - R interface
                - R interface windows installer (zip)
            - python
                - python interface installers (32+64-bit and py2.7+3.6-3.9) (exe)
            - installer 64-bit (zip)
            - installer 32-bit (zip)
        - MacOS
            - binaries (tar)
        - Linux
            - binaries (32+64 bit, for centOS and Ubuntu) (tar)
            - installers (32+64 bit, for Debian- and RPM-based) (deb, rpm)

Each folder also contains a `ReadMe.md` describing the contents of the folder.
All builds contain bindings for C/C++/C#(Mono)/Python/Java/Perl and most of the Ruby.

### Summary of differences of manual releases to nightly builds
The latest versions of many of the artefact provided there are now available through the nightly build of the CI system. Differences are documented below. The nightly build does **not** currently contain
- Installers (dmg, msi, rpm, deb), 
- Ruby bindings
- Perl bindings
- mechanisms to provide libSBML-python via conda or pyPI
- MATLAB interface
- precompiled dependencies for Windows
- tar archives
- ReadMe files
- 32-bit versions
- python interface installers