Important
This repository is no longer maintained.
Job's Done is a tool heavily inspired by Travis, and works in the same way
in that configuring a .jobs_done.yaml
file in your repository's root to create and trigger Continuous Integration jobs.
Example of a .jobs_done.yaml
file:
matrix:
platform:
- "win64"
- "redhat64"
platform-win64:build_batch_commands:
- |
python -m venv .env3 || goto eof
call .env3\Scripts\activate || goto eof
pytest --junitxml=tests-{platform}.xml
platform-redhat64:build_shell_commands:
- |
python3 -m venv .env3
source .env3/bin/activate
pytest --junitxml=tests-{platform}.xml
junit_patterns:
- "tests.*.xml"
Considering this file is in the root of repository myproject
and was pushed to branch feat-71
, this will generate two Jenkins jobs:
myproject-feat-71-win64
myproject-feat-71-linux64
Jobs done can be executed in the command-line.
To use it, from the repository's folder that you want to create jobs for, execute:
$ jobs_done jenkins --username USER https://example.com/jenkins
This will create/update existing jobs.
Below are the possible installation options.
-
Create a virtual environment using Python 3.10 and activate it:
$ python -m venv .env $ .env\Scripts\activate # Windows $ source .env/bin/activate # Linux
-
Install jobs_done10:
$ pip install jobs_done10
-
Clone the repository:
git clone git@github.com:ESSS/jobs_done10.git cd jobs_done10
-
Create a virtual environment using Python 3.10 and activate it:
$ python -m venv .env $ .env\Scripts\activate # Windows $ source .env/bin/activate # Linux
-
Install dependencies and the project in editable mode:
$ pip install -r requirements.txt -e .
The dependencies are declared in setup.py, and we use pip-tools to manage versions.
Use this command to upgrade all dependencies to the latest versions in requirements.txt
:
python -m piptools compile --extra=dev --upgrade
jobs done includes a flask
server in jobs_done10.server
which can be deployed using Docker.
It provides two end-points for PUSH
and GET
:
/stash
: tailored to receive the push event from BitBucket Server (formerly known as Stash)./github
: tailored to receive the push event from GitHub. Important: the webhook must be configured with content-type ofapplication/json
and a secret.
A GET
to either ends points will return the JobsDone version, useful to check the installed version and
that the end-point is correct.
Posting to /
is the same as posting to /stash
for backwards compatibility (might be removed in the future).
Configuration is done by having a .env
file (cortesy of python-dotenv)
in the root of this repository with the following variables:
JD_JENKINS_URL=https://example.com/jenkins
JD_JENKINS_USERNAME=jenkins-user
JD_JENKINS_PASSWORD=some password
JD_STASH_URL=https://example.com/stash
JD_STASH_USERNAME=stash-user
JD_STASH_PASSWORD=some password
JD_GH_TOKEN=github-user-personal-access-token
JD_GH_WEBHOOK_SECRET=webhook-secret
JD_EMAIL_USER=mail-sender@example.com
JD_EMAIL_FROM=JobsDone Bot <mail-sender@example.com>
JD_EMAIL_PASSWORD=email password
JD_EMAIL_SERVER=smtp.example.com
JD_EMAIL_PORT=587
Clone the repository and checkout the tag:
$ git clone https://github.com/ESSS/jobs_done10.git
$ cd jobs_done10
$ git checkout <VERSION>
Build a docker image:
$ docker build . --tag jobsdone:<VERSION> --build-arg SETUPTOOLS_SCM_PRETEND_VERSION=<VERSION>
$ docker run --publish 5000:5000 jobsdone:<VERSION>
This is an example of a Job's Done file, and what you might expect of its contents.
build_batch_commands:
- "echo MESSAGE: Hello, world!"
description_regex: "MESSAGE\\:(.*)"
Adding this file to a repository hooked into our CI system will create a single job that when executed run a Windows batch command, and later on catches the message echoed and sets that as the build description.
This is an example of a simple application with tests:
build_batch_commands:
- "pytest --junitxml=pytest_results.xml"
junit_patterns:
- "pytest_results.xml"
This jobs runs pytest in the repository and outputs test results to a file. We also configure the job to look for that file, and present test results to us at then end of the build.
The same application as above, but now running on multiple platforms.
platform-win64:build_batch_commands:
- "pytest --junitxml=pytest_results-{platform}.xml"
platform-redhat64:build_shell_commands:
- "pytest --junitxml=pytest_results-{platform}.xml"
junit_patterns:
- "pytest_results.*.xml"
matrix:
platform:
- "win64"
- "redhat64"
Here we add a matrix section to define variations of this job. In this case, we have the platform variable,
with two possible values, win64
and redhat64
.
One job will be created for each possible combination in the matrix (only two jobs in this case).
Since we can't run batch commands in linux, we add another builder section, build_shell_commands
. Using some flags
before defining sections we can choose which one will be available in each job.
Values from the matrix can also be used as variables, in this case, {platform}
will be replaced by the platform used
in that job (win64
or redhat64
).
Branch patterns are used to filter which branches will produce jobs. This list of regular expressions (using Python syntax), ensures that a branch will only produce jobs if at least one of the regular expressions matches the name of the branch.
Here's an example that filter only master
and feature branches:
branch_patterns:
- "master"
- "fb-*"
If this section is not defined in the file, all branches will produce jobs.
As shown in the examples above, the job matrix can be used to create multiple variations of a job. One job for each combination of entries in this matrix is created.
matrix:
mode:
- "app"
- "cases"
platform:
- "win64"
- "linux64"
In this case 4 jobs will be generated:
app-win64
app-linux64
cases-win64
cases-linux64
Note that you can use any variables you need, Job's done has no idea what mode
or platform
means.
There's an exclude
clause which can be used to remove particular entries from the matrix:
matrix:
mode:
- "app"
- "cases"
platform:
- "win64"
- "linux64"
mode-cases:platform-win.*:exclude: "yes"
This will exclude all cases jobs from windows.
Variables defined in the job matrix can be used to replace strings in the job file.
On top of matrix variables, there are a few special string templates that can be used in any job:
name
- Name of the repositorybranch
- Branch being built
matrix:
platform:
- "win64"
- "linux64"
platform-win.*:build_batch_commands:
- "echo Building project {name} in branch {branch} on platform {platform}"
Note that we use Python's format syntax, so if you need an actual {
or }
use double braces: {{
, }}
.
Variables defined in your job matrix can also be used to control some of the contents in your job file. A common example here is using different builder for windows (bash) and linux (shell).
This is done by adding a prefix to sections of the YAML file, with the variable name and value necessary to use it:
platform-win.*:build_batch_commands:
- "dir ."
platform-linux.*:build_shell_commands:
- "ls -la ."
matrix:
platform:
- "win32"
- "linux64"
Matrix variables can also define aliases, useful to reduce duplication when using such flags. To add aliases, simply use commas to separate additional names for matrix values:
platform-windows:build_batch_commands:
- "dir ."
platform-linux:build_shell_commands:
- "ls -la ."
matrix:
platform:
- "win32,windows"
- "win64,windows"
- "linux64,linux"
On top of that you can use a special variable branch
that's always available, and points to your branch:
branch-master:build_batch_commands:
- "echo Build"
branch-deploy:build_batch_commands:
- "echo Build + Deploy"
Condition values can use Python regex syntax for extra flexibility:
branch-master:build_batch_commands:
- "echo Build"
branch-fb.*:build_batch_commands:
- "echo Feature branch!"
branch-rb.*:build_batch_commands:
- "echo Release branch!"
Create a virtual environment and install it in development mode:
$ python -m virtualenv .env36
$ source .env36/bin/activate
$ pip install -e .
Run tests:
$ pytest src
Open a PR updating the CHANGELOG and after it passes, push a tag to the repository.
This will automatically publish it to PyPI and to the configured Docker Registry in the organization.
Additional repositories to be checked out in this job.
The repository where this .jobs_done file is included by default.
Requires Multiple SCMs Plugin and Git Plugin.
Uses same options as git
.
additional_repositories:
- git:
url: "https://project.git"
branch: "{branch}"
Job authentication token required to triggers builds remotely.
auth_token: "my_token"
List of boosttest file patterns to look for test results.
Requires the xUnit Plugin.
boosttest_patterns:
- "*.xml"
List of regexes used to match branch names. Only branches that match one of these will create jobs.
branch_patterns:
- "master"
- "fb-*"
List of Windows batch commands used to build the job. If errorcode is not 0 after any command, the build fails.
build_batch_commands:
- "pytest src"
- "echo Finished"
List of shell commands used to build the job. If errorcode is not 0 after any command, the build fails.
build_shell_commands:
- "pytest src"
- "echo Finished"
List of python commands used to build the job.
Requires Python Plugin.
build_python_commands:
- "print(5)"
Enable support for ANSI escape sequences, including color, to Console Output.
Requires AnsiColor Plugin.
Accepted values:
xterm
(default)vga
css
gnome-terminal
console_color: "css"
Enables code coverage report.
Require Cobertura Plugin.
Options:
report_pattern
: mandatory, pattern where XML coverage files are searched. These XML files are usually in Cobertura format, which is also format by pytest-cov XML output (because pytest-cov uses coverage library).healthy
: optional, specifies desired method, line and conditional metric. Any omitted metric defaults to80
.unhealthy
: optional, specifies desired method, line and conditional metric. Any omitted metric defaults to0
. Builds below these thresholds are marked as unhealthy.failing
: optional, specifies desired method, line and conditional metric. Any omitted metric defaults to0
. Builds below these thresholds are marked as failed.
coverage:
report_pattern: "**/build/coverage/*.xml"
healthy:
method: 100
line: 100
conditional: 90
unhealthy:
method: 95
line: 95
conditional: 85
failing:
method: 90
line: 90
conditional: 80
Schedules to build to run periodically.
cron: |
# Everyday at 22pm
* 22 * * *
Defines a custom workspace directory for the job. To maintain the same base directory as the default workspace directories prefix it with "workspace/"
.
custom_workspace: "workspace/devspace-user"
Regular expression for searching job output for a description. If a match is found, the contents of the first group will be set as the description.
Requires Description Setter Plugin.
description_regex: "OUTPUT: (.*)"
Configures the display name of the job.
display_name: "{branch} {name}"
Sends emails for failed builds.
email_notification: "email1@example.com email2@example.com"
# or
email_notification:
recipients: "email1@example.com email2@example.com"
notify_every_build: true
notify_individuals: true
Excludes a job from the matrix.
platform-linux64:exclude: "yes"
Additional git options for the main project.
Requires Git Plugin.
Options available here are shared with additional_repositories
.
git:
branch: master
clean_checkout: false
lfs: true
recursive_submodules: true
reference: "/home/reference.git"
shallow_clone: 50
tags: true
target_dir: "main_application"
timeout: 30
url: "ssh://server/somerepo.git"
Note: by default, tags
is false
because we noticed that fetching tags takes quite some time and
we don't use tags for anything. If tags are needed, set tags
to true
.
List of jsunit file patterns to look for test results.
Requires JSUnit Plugin.
jsunit_patterns:
- "*.xml"
List of junit file patterns to look for test results.
Requires xUnit Plugin.
junit_patterns:
- "*.xml"
Configures the label expression of the job.
The label-expression is used to determine which workers can run the job.
label_expression: "{platform}"
Configures variations of a job.
matrix:
python:
- "27"
- "36"
Notifies GitHub of job build status.
Currently it does not support any parameters, using default values.
Requires GitHub Plugin.
notify_github:
Notifies a Stash instance of job build status.
When no parameters are given, uses configurations set in the Jenkins instance.
Requires StashNotifier Plugin.
notify_stash:
url: "example.com/stash"
username: "user"
password: "pass"
# Using default from Jenkins
notify_stash:
Job parameters for Jenkins jobs.
Currently, only choice
and string
are implemented.
parameters:
- choice:
name: "PARAM_BIRD"
choices:
- "African"
- "European"
description: "Area where the bird is from"
- string:
name: "PARAM_VERSION"
default: "dev"
description: "App version"
Schedules to periodically poll SCM for changes, and trigger builds.
scm_poll: |
# Everyday at 22pm
* 22 * * *
Configure notification with slack.
- Configure your Jenkins integration on Slack
- Obtain the token
- Configure your job to notify slack using this option.
slack:
team: esss
channel: dev
token: XXX
url: https://example.com/jenkins
Job timeout in minutes.
timeout: 60
Show timestamps on the left side of the console output.
Requires the Timestamper Plugin.
timestamps:
Trigger other jobs after the current job finishes. Parameters are optional.
trigger_jobs:
names:
- myrepo-{branch}-synthetic-{platform}
condition: SUCCESS # can be one of: SUCCESS, UNSTABLE, FAILED, ALWAYS. Defaults to SUCCESS.
parameters: # optional
- PARAM1=VALUE1
- PARAM2=VALUE2
Configures parsing of warnings and static analysis in a CI job.
Requires Warnings Plugin.
warnings:
console:
- parser: Clang (LLCM based)
- parser: PyLint
file:
- parser: CppLint
file_pattern: *.cpplint
- parser: CodeAnalysis
file_pattern: *.codeanalysis