Core functions for setting results, logging, registering secrets and exporting variables across actions.
from actions_toolkit import core
Action inputs can be read with get_input
which returns a string
or get_boolean_input
which parses a boolean based on the yaml 1.2 specification. If required
set to be false, the input should have a default value in action.yml
.
Outputs can be set with set_output
which makes them available to be mapped into inputs of other actions to ensure they are decoupled.
from actions_toolkit import core
my_input = core.get_input('input_name', required=True)
my_second_input = core.get_input('input_name', required=True, trim_whitespace=False)
my_boolean_input = core.get_boolean_input('boolean_input_name', required=True)
my_multiline_input = core.get_multiline_input('multiline_input_name', required=True)
core.set_output('output_key', 'output_val')
Since each step runs in a separate process, you can use export_variable
to add it to this step and future steps environment blocks.
from actions_toolkit import core
core.export_variable('env_var', 'val')
Setting a secret registers the secret with the runner to ensure it is masked in logs.
from actions_toolkit import core
core.set_secret('my_password')
To make a tool's path available in the path for the remainder of the job (without altering the machine or containers state), use add_path
. The runner will prepend the path given to the jobs PATH.
from actions_toolkit import core
core.add_path('/path/to/my_tool')
You should use this library to set the failing exit code for your action. If status is not set and the script runs to completion, that will lead to a success.
from actions_toolkit import core
try:
# Do stuff
pass
except Exception as e:
# set_failed logs the message and sets a failing exit code
core.set_failed(f'Action failed with error {str(e)}')
Note that set_neutral
is not yet implemented in actions V2 but equivalent functionality is being planned.
Finally, this library provides some utilities for logging. Note that debug logging is hidden from the logs by default. This behavior can be toggled by enabling the Step Debug Logs.
from actions_toolkit import core
my_input = core.get_input('input')
try:
core.debug('Inside try block')
if not my_input:
core.warning('my_input was not set')
if core.is_debug():
# curl -v https://github.com
pass
else:
# curl https://github.com
pass
# Do stuff
core.info('Output to the actions build log')
core.notice('This is a message that will also emit an annotation')
except Exception as e:
core.error(f'Error {str(e)}, action may still succeed though')
This library can also wrap chunks of output in foldable groups.
import asyncio
from actions_toolkit import core
# Manually wrap output
core.start_group('Do some function')
# do_some_function()
core.end_group()
async def do_some_http_request():
print('hello world')
await asyncio.sleep(2)
asyncio.run(core.group('Do something async', do_some_http_request))
This library has 3 methods that will produce annotations.
from actions_toolkit import core
core.error('This is a bad error, action may still succeed though.')
core.warning('Something went wrong, but it\'s not bad enough to fail the build.')
core.notice('Something happened that you might want to know about.')
These will surface to the UI in the Actions page and on Pull Requests. They look something like this:
These annotations can also be attached to particular lines and columns of your source files to show exactly where a problem is occuring.
These options are:
class AnnotationProperties:
def __init__(self, title: str = None, file: str = None,
start_line: int = None, end_line: int = None,
start_column: int = None, end_column: int = None):
# A title for the annotation.
self.title = title
# The path of the file for which the annotation should be created.
self.file = file
# The start line for the annotation.
self.start_line = start_line
# The end line for the annotation. Defaults to `start_line` when `start_line` is provided.
self.end_line = end_line
# The start column for the annotation. Cannot be sent when `start_line` and `end_line` are different values.
self.start_column = start_column
# The end column for the annotation. Cannot be sent when `start_line` and `end_line` are different values.
# Defaults to `start_column` when `start_column` is provided.
self.end_column = end_column
Colored output is supported in the Action logs via standard ANSI escape codes. 3/4 bit, 8 bit and 24 bit colors are all supported.
Foreground colors:
from actions_toolkit import core
# 3/4 bit
core.info('\u001b[35mThis foreground will be magenta')
# 8 bit
core.info('\u001b[38;5;6mThis foreground will be cyan')
# 24 bit
core.info('\u001b[38;2;255;0;0mThis foreground will be bright red')
Background colors:
from actions_toolkit import core
# 3/4 bit
core.info('\u001b[43mThis background will be yellow')
# 8 bit
core.info('\u001b[48;5;6mThis background will be cyan')
# 24 bit
core.info('\u001b[48;2;255;0;0mThis background will be bright red')
Special styles:
from actions_toolkit import core
core.info('\u001b[1mBold text')
core.info('\u001b[3mItalic text')
core.info('\u001b[4mUnderlined text')
ANSI escape codes can be combined with one another:
from actions_toolkit import core
core.info('\u001b[31;46mRed foreground with a cyan background and \u001b[1mbold text at the end')
Note: Escape codes reset at the start of each line
from actions_toolkit import core
core.info('\u001b[35mThis foreground will be magenta')
core.info('This foreground will reset to the default')
Manually typing escape codes can be a little difficult, but you can use third party modules such as colored.
from actions_toolkit import core
from colored import fg, attr
core.info(f'{fg("#abcdef")} Hello world!{attr("reset")}')
You can use this library to save state and get state for sharing information between a given wrapper action:
In action's main.py
:
from actions_toolkit import core
core.save_state('pid_to_kill', 12345)
In action's cleanup.py
:
import os, signal
from actions_toolkit import core
pid = core.get_state('pid_to_kill')
os.kill(int(pid), signal.SIGTERM)
You can use these methods to interact with the GitHub OIDC provider and get a JWT ID token which would help to get access token from third party cloud providers.
- Method Name:
getIDToken()
- Inputs:
audience : optional
- Outputs: A JWT ID Token
In action's main.py
:
from actions_toolkit import core
def get_id_token_action():
audience = core.get_input('audience', required=False)
id_token1 = core.get_id_token() # ID Token with default audience
id_token2 = core.get_id_token(audience) # ID token with custom audience
# this id_token can be used to get access token from third party cloud providers
get_id_token_action()
In action's actions.yml
:
name: "GetIDToken"
description: "Get ID token from Github OIDC provider"
inputs:
audience:
description: "Audience for which the ID token is intended for"
required: false
outputs:
id_token1:
description: "ID token obtained from OIDC provider"
id_token2:
description: "ID token obtained from OIDC provider"
runs:
using: "docker"
image: "Dockerfile"
You can use these methods to manipulate file paths across operating systems.
The to_posix_path
function converts input paths to Posix-style (Linux) paths.
The to_win32_path
function converts input paths to Windows-style paths. These
functions work independently of the underlying runner operating system.
from actions_toolkit import core
core.to_posix_path('\\foo\\bar') # => /foo/bar
core.to_win32_path('/foo/bar') # => \foo\bar
The to_platform_path
function converts input paths to the expected value on the runner's operating system.
from actions_toolkit import core
# On a Windows runner.
core.to_platform_path('/foo/bar') # => \foo\bar
# On a Linux runner.
core.to_platform_path('\\foo\\bar') # => /foo/bar