The workflow testing framework consists mainly of 3 components:
- Jest - testing framework, also used for application tests
- Mock-github - package allowing for creation of local repositories, which can be used to make sure that the workflow tests have access only to these files that they should and that they won't modify the actual repository
- Act-js - JS wrapper around Act. Act is a tool that allows to run GitHub Actions workflows locally, and Act-js allows to configure and run Act from JS code, like Jest tests. It also provides additional tools like mocking workflow steps and retrieving the workflow output a JSON, which allows for comparison of the actual output with expected values
- Install dependencies from
package.json
file withnpm install
- Make sure you have fulfilled the prerequisites for running
Act
- Install
Act
withbrew install act
and follow the documentation on first Act run - Set the environment variable
ACT_BINARY
to the path to yourAct
executable (which act
if you're not sure what the path is) - You should be ready to run the tests now with
npm run workflow-test
- You can pre-generate new mocks/assertions/test files for a given workflow by running
npm run workflow-test:generate <workflow_file>
- To run the workflow tests simply use
npm run workflow-test
- this will run all the tests sequentially, which can take some time
- To run a specific test suite you can use
npm run workflow-test -- -i <path_to_test_file>
- this will run only the test from that specific test file
- To run a specific test or subset of tests use
npm run workflow-test -- -t "<test_name_substring>"
- this will run only the tests having
<test_name_substring>
in their name/description
- You can combine these like
npm run workflow-test -- -i workflow_tests/preDeploy.test.js -t "single specific test"
- You can also use all other options which are normally usable with
jest
Not all workflows can always be tested this way, for example:
- Act and Act-js do not support all the runner types available in GitHub Actions, like
macOS
runners or some specific version ofUbuntu
runners likeubuntu-20.04-64core
. In these cases the job will be omitted entirely and cannot be tested - Testing more complex workflows in their entirety can be extremely time-consuming and cumbersome. It is often optimal to mock most of the steps with expressions printing the input and output conditions
- Due to the way
Act
andAct-js
handle workflow output, not much can be checked in the test. What is available, namely whether the job/step executed or not, whether it was successful or not and what its printed output was, should be enough in most scenarios Act
does not seem to support the conditions set on event parameters when determining whether to run the workflow or not, namely for a workflow defined with:
on:
pull_request:
types: [opened, edited, reopened]
running act pull_request -e event_data.json
with event_data.json
having {"action": "opened"}
will execute the workflow (as expected), running for example act push
will not execute it (as expected), but running act pull_request -e event_data.json
with event_data.json
having for example {"action": "assigned"}
will still execute the workflow even though it should only be executed with action
being opened
, edited
or reopened
. This only applies to running the workflow with Act
, in the GitHub environment it still works as expected
The testing framework file structure within the repository is as follows:
App/
- main application folder.github/
- GitHub Actions folderworkflows/
- workflows folder<workflow_name>.yml
- workflow file...
- other workflow files
...
- other GitHub Actions files
workflow_tests/
- workflow testing folderjest.config.ts
-Jest
configuration fileREADME.md
- this readme fileutils.js
- various utility functions used in multiple tests<workflow_name>.test.js
- test suite file for a GitHub Actions workflow named<workflow_name>
mocks/
- folder with step mock definitions<workflow_name>Mocks.js
- file with step mock definitions for the../<workflow_name>.test.js
suite, or for the<workflow_name>
workflow...
- other step mock definition files
assertions/
- folder with output assertions<workflow_name>Assertions.js
- file with output assertions for the../<workflow_name>.test.js
suite, or for the<workflow_name>
workflow...
- other output assertion files
...
- other test suites
...
- other application files
utils.js
file provides several helper methods to speed up the tests development and maintenance
setUpActParams
allows for initiating the context in which Act will execute the workflow
Parameters:
act
- instance of previously createdAct
object that will be updated with new paramsevent
- the name of the event, this can be any event name used by GitHub Actions, likepull_request
,push
,workflow_dispatch
, etc.event_options
- object with options of the event, allowing for customising it for different scenarios, for examplepush
event can be customised for pushing to different branches with options{head: {ref: '<branch_name>'}}
secrets
- object with secret values provided, like{<SECRET_NAME>: <secret_value>, ...}
github_token
- value of the GitHub token, analogous to providingGITHUB_TOKEN
secret
Returns an updated Act
object instance
Example:
let act = new kieActJs.Act(repoPath, workflowPath);
act = utils.setUpActParams(
act,
'push',
{head: {ref: 'main'}},
{OS_BOTIFY_TOKEN: 'dummy_token', GITHUB_ACTOR: 'Dummy Tester', SLACK_WEBHOOK: 'dummy_slack_webhook', LARGE_SECRET_PASSPHRASE: '3xtr3m3ly_s3cr3t_p455word'},
'dummy_github_token',
);
getMockStep
allows for creating uniform mock step definitions compatible with Act-js
and reduces time required, as well as possibility of errors/typos slipping in while developing tests. More complex behaviours have to be mocked manually
Parameters:
name
- name of the step that must correspond to thename
in the<workflow>.yml
file, otherwise the step cannot be foundmessage
- the message to be printed to default output when mock gets executedjob_id
- an optional id of the job that will be printed in[]
square brackets along themessage
, useful when assessing the output with many steps from many jobsinputs
- a list of input parameters to be printed, useful when checking if the step had been executed with expected inputsin_envs
- a list of input environment variables, to be printed, useful when checking if the step had been executed in expected environmentoutputs
- an object with values which should be printed by the mock to$GITHUB_OUTPUT
, useful for setting the step outputout_envs
- an objects with values of environment variables set by the step in$GITHUB_ENV
, useful for modifying the environment by the mockisSuccessful
- a boolean value indicating whether the step succeeds or not, exits with status0
(if successful) or1
(if not)
Returns an object with step mock definition, ready to be provided to the Act
object instance
Example:
let mockStep = utils.getMockStep(
'Name of the step from <workflow>.yml',
'Message to be printed',
'TEST_JOB',
['INPUT_1', 'INPUT_2'],
['ENV_1', 'ENV_2'],
{output_1: true, output_2: 'Some Result'},
{OUT_ENV: 'ENV_VALUE'},
false,
);
results in
{
name: 'Name of the step from <workflow>.yml',
mockWith: 'echo [MOCK]'
+ ' [TEST_JOB]'
+ ' Message to be printed'
+ ', INPUT_1="{{ inputs.INPUT_1 }}'
+ ', INPUT_2="{{ inputs.INPUT_2 }}'
+ ', ENV_1="{{ env.ENV_1 }}'
+ ', ENV_1="{{ env.ENV_1 }}'
+ '\necho "output_1=true" >> "$GITHUB_OUTPUT"',
+ '\necho "output_2=Some Result" >> "$GITHUB_OUTPUT"',
+ '\necho "OUT_ENV=ENV_VALUE" >> "$GITHUB_ENV"',
+ '\nexit 1',
}
getStepAssertion
allows for creating uniform assertions for output from executed step, compatible with step mocks provided by getMockStep
Parameters:
name
- name of the step, has to correspond to the name from<workflow>.yml
file, and the name in the step mock if applicableisSuccessful
- boolean value for checking if the step should have exited successfullyexpectedOutput
- an output that is expected from the step, compared directly - if provided the subsequent parameters are ignoredjobId
- an optional expected job identifiermessage
- expected message printed by the stepinputs
- expected input values provided to the stepenvs
- expected input environment variables for the step
Returns an object with step expected output definition ready to be provided to expect()
matcher
Example:
utils.getStepAssertion(
'Name of the step from <workflow>.yml',
false,
null,
'TEST_JOB',
'Message to be printed',
[{key: 'INPUT_1', value: true}, {key: 'INPUT_2', value: 'Some value'}],
[{key: 'PLAIN_ENV_VAR', value: 'Value'}, {key: 'SECRET_ENV_VAR', value: '***'}],
)
results in
{
name: 'Name of the step from <workflow>.yml',
status: 1,
output: '[MOCK]'
+ ' [TEST_JOB]'
+ ' Message to be printed'
+ ', INPUT_1=true'
+ ', INPUT_2=Some value'
+ ', PLAIN_ENV_VAR=Value'
+ ', SECRET_ENV_VAR=***',
}
setJobRunners
overwrites the runner types for given jobs, helpful when the runner type in the workflow is not supported by Act
Parameters:
act
- instance of previously createdAct
objectjobs
- object with keys being the IDs of the workflow jobs to be modified and values being the names of runners that should be used for them in the testworkflowPath
- path to the workflow file to be updated, NOTE: this will modify the file, use the one from the local test repo, not fromApp/.github/workflows
!
Returns an Act
object instance
Let's say you have a workflow with a job using macos-12
runner, which is unsupported by Act
- in this case that job will simply be skipped altogether, not allowing you to test it in any way.
iOS:
name: Build and deploy iOS
needs: validateActor
if: ${{ fromJSON(needs.validateActor.outputs.IS_DEPLOYER) }}
runs-on: macos-12
steps:
You can use this method to change the runner to something that is supported, like
act = utils.setJobRunners(
act,
{
iOS: 'ubuntu-latest',
},
workflowPath,
);
Now the test workflow will look as follows, which will allow you to run the job and do at least limited testing
iOS:
name: Build and deploy iOS
needs: validateActor
if: ${{ fromJSON(needs.validateActor.outputs.IS_DEPLOYER) }}
runs-on: ubuntu-latest
steps:
The following is the typical test file content, which will be followed by a detailed breakdown
const path = require('path');
const kieActJs = require('@kie/act-js');
const kieMockGithub = require('@kie/mock-github');
const utils = require('./utils');
const assertions = require('./assertions/<workflow>Assertions');
const mocks = require('./mocks/<workflow>Mocks');
let mockGithub;
const FILES_TO_COPY_INTO_TEST_REPO = [
{
src: path.resolve(__dirname, '..', '.github', 'actions'),
dest: '.github/actions',
},
{
src: path.resolve(__dirname, '..', '.github', 'libs'),
dest: '.github/libs',
},
{
src: path.resolve(__dirname, '..', '.github', 'scripts'),
dest: '.github/scripts',
},
{
src: path.resolve(__dirname, '..', '.github', 'workflows', '<workflow>.yml'),
dest: '.github/workflows/<workflow>.yml',
},
];
beforeEach(async () => {
mockGithub = new kieMockGithub.MockGithub({
repo: {
testWorkflowsRepo: {
files: FILES_TO_COPY_INTO_TEST_REPO,
},
},
});
await mockGithub.setup();
});
afterEach(async () => {
await mockGithub.teardown();
});
describe('test some general behaviour', () => {
test('something happens - test if expected happened next', async () => {
// get path to the local test repo
const repoPath = mockGithub.repo.getPath('testWorkflowsRepo') || '';
// get path to the workflow file under test
const workflowPath = path.join(repoPath, '.github', 'workflows', '<workflow>.yml');
// instantiate Act in the context of the test repo and given workflow file
let act = new kieActJs.Act(repoPath, workflowPath);
// set run parameters
act = utils.setUpActParams(
act,
'<event>',
{head: {ref: '<branch_name>'}},
{'<SECRET_NAME>': '<secret_value'},
'<github_token>',
);
// set up mocks
const testMockSteps = {
'<job_1_name>': [
{
name: '<step_1_1_name>',
mockWith: '<mock_command>',
},
{
name: '<step_1_2_name>',
mockWith: '<mock_command>',
},
],
'<job_2_name>': [
utils.getMockStep('<step_2_1_name>', '<message>'),
utils.getMockStep('<step_2_2_name>', '<message>'),
],
};
// run an event and get the result
const result = await act
.runEvent('<event>', {
workflowFile: path.join(repoPath, '.github', 'workflows'),
mockSteps: testMockSteps,
});
// assert results (some steps can run in parallel to each other so the order is not assured
// therefore we can check which steps have been executed, but not the set job order
assertions.assertSomethingHappened(result);
assertions.assertSomethingDidNotHappen(result, false);
}, timeout);
);
Define which files should be copied into the test repo. In this case we copy actions
, libs
, scripts
folders in their entirety and just the one workflow file we want to test
const FILES_TO_COPY_INTO_TEST_REPO = [
{
src: path.resolve(__dirname, '..', '.github', 'actions'),
dest: '.github/actions',
},
{
src: path.resolve(__dirname, '..', '.github', 'libs'),
dest: '.github/libs',
},
{
src: path.resolve(__dirname, '..', '.github', 'scripts'),
dest: '.github/scripts',
},
{
src: path.resolve(__dirname, '..', '.github', 'workflows', '<workflow>.yml'),
dest: '.github/workflows/<workflow>.yml',
},
];
beforeEach
gets executed before each test. Here we create the local test repository with the files defined in the FILES_TO_COPY_INTO_TEST_REPO
variable. testWorkflowRepo
is the name of the test repo and can be changed to whichever name you choose, just remember to use it later when accessing this repo. Note that we can't use beforeAll()
method, because while mocking steps Act-js
modifies the workflow file copied into the test repo and thus mocking could persist between tests
beforeEach(async () => {
mockGithub = new kieMockGithub.MockGithub({
repo: {
testWorkflowsRepo: {
files: FILES_TO_COPY_INTO_TEST_REPO,
},
},
});
await mockGithub.setup();
});
Similarly, afterEach
gets executed after each test. In this case we remove the test repo after the test finishes
afterEach(async () => {
await mockGithub.teardown();
});
Get path to the local test repo, useful to have it in a variable
const repoPath = mockGithub.repo.getPath('testWorkflowsRepo') || '';
Get path to the workflow under test. Note that it's the path in the test repo
const workflowPath = path.join(repoPath, '.github', 'workflows', '<workflow>.yml');
Instantiate Act
object instance. Here we provide the constructor with the path to the test repo (so that Act
can execute in its context) and the path the workflow file under test (so just the workflow we want to test would be executed)
let act = new kieActJs.Act(repoPath, workflowPath);
Set up initial parameters for Act
. This is where we can set secrets, GitHub token and options for the events (like the name of the branch to which the push has been made, etc.)
act = utils.setUpActParams(
act,
'<event>',
{head: {ref: '<branch_name>'}},
{'<SECRET_NAME>': '<secret_value'},
'<github_token>',
);
Set up step mocks. Here we configure which steps in the workflow should be mocked, and with what behaviour. This takes form of an object with keys corresponding to the names of the jobs in the workflow, and values being mock definitions for specific steps. The steps can be identified either by id
, name
, uses
or run
. Step mock can be defined either by hand (<job_1_name>
) or with the helper method utils.getMockStep()
(<job_2_name>
). Not mocked steps will be executed normally - make sure this will not have unexpected consequences
const testMockSteps = {
'<job_1_name>': [
{
name: '<step_1_1_name>',
mockWith: '<mock_command>',
},
{
name: '<step_1_2_name>',
mockWith: '<mock_command>',
},
],
'<job_2_name>': [
utils.getMockStep('<step_2_1_name>', '<message>'),
utils.getMockStep('<step_2_2_name>', '<message>'),
],
};
Most important part - actually running the event with Act
. This executes the specified <event>
in the context of the local test repo created before and with the workflow under test set up. result
stores the output of Act
execution, which can then be compared to what was expected. Note that the workflowFile
is actually path to workflow folder and not the file itself - Act-js
determines the name of the workflow by itself, and tries to find it in the specified workflowFile
path, so providing the full path to the file will fail
const result = await act
.runEvent('<event>', {
workflowFile: path.join(repoPath, '.github', 'workflows'),
mockSteps: testMockSteps,
});
Assert results are as expected. This can, for example, include using expect()
to check if the steps that should be executed have indeed been executed, steps that shouldn't run have not been executed, compare statuses (which steps succeeded, which failed) and step outputs. Outputs can include additional information, like input values, environmental variables, secrets (although these are usually not accessible and represented by ***
, this can still be useful to check if the value exists or not). Here it's usually done with the helper assertion methods defined in the assertions file. Step assertions can be created manually or with getStepAssertion()
helper method
assertions.assertSomethingHappened(result);
assertions.assertSomethingDidNotHappen(result, false);