Skip to main content

Custom Script

You can use Custom Script steps for additional functionalities in your builds. Appcircle will run the commands in your custom scripts and perform the specified actions. These scripts will be run on the runner and you can use any functionality of the build environment as you need.

Prerequisites

There are no prerequisites required before using the Custom Script step.

tip

Note that you can put the Custom Script component anywhere you want in the workflow. This step is used to add different capabilities to the existing workflow.

Screenshot

Input Variables

This step contains some input variable(s). It needs these variable(s) to work. The table below gives explanation for this variable(s).

Screenshot
Sensitive Variables

If you need to use sensitive variable in your script, please do not use these sensitive variables such as Username, Password, API Key, or Personal Access Key directly within the step.

We recommend using Environment Variables groups for such sensitive variables.

Variable NameDescriptionStatus
ExecuteYou can run your script as Bash or Ruby with two different language environments in the Execute With input value.Required
ScriptWith the Script input variable, you can add the script you want to run and run it directly in the selected language. If you leave this input blank, it will proceed to the next step without taking any action.Optional
caution

Note that the Script area works according to the selected language variable. If you want to run a script in any language, make sure that you select the language correctly.


To access the source code of this component, please use the following link:

Preview of GitHub - appcircleio/appcircle-custom-script-component


FAQ

How to change Java version

Appcircle currently has OpenJDK 17 (default), OpenJDK 8, OpenJDK 11 and OpenJDK 21.

Android Build step uses OpenJDK 17 as default JDK version.

To switch JDK versions, you can now use the dedicated Select Java Version component, so there is no need to use Custom Script for this task. For further details on this component, refer to the documentation:

However, if you prefer to update or improve it manually on Custom Script, the source code is available here:

How to install a new package to the build machine?

You can use the compatible package managers to install packages.

For the macOS build machines for iOS builds, _brew _is a commonly used package manager with commands like brew install maven

For the Linux (Debian) build machines for Android builds, apt-get can be used for 3rd party packages such as apt-get -y install maven

How to change the package name/application ID dynamically?

With custom scripts, you can edit the Info.plist and the build.gradle files.

iOS sample for Info.plist
cd $AC_REPOSITORY_DIR/Your-Target-Folder
/usr/libexec/PlistBuddy -c "Set :CFBundleIdentifier io.myapp" "./Info.plist"

How to access a file in the repository directory?

For each step in the workflow, you can view the input and output variables in the step configuration.

The repository directory is an output of the Git Clone step and its patch can be accessed with the $AC_REPOSITORY_PATH environment variable by any step added after the Git Clone step. An example is as follows:

cd $AC_REPOSITORY_DIR
cat README

How to a add a file as a downloadable build artifact?

You can add any file to the output directory that contain the build artifacts using the $AC_OUTPUT_DIR environment variable. An example is as follows:

cd $AC_REPOSITORY_DIR/app/build/reports/
mv lint-results* $AC_OUTPUT_DIR/

How to break pipeline on low test coverage

This document provides a sample custom script written in Ruby that can be integrated into your CI/CD pipeline to enforce a minimum test coverage threshold. The script is designed to break the pipeline if the covered test result falls below a specified percentage.

danger

Please note that this custom script must be placed after the Test Reports step in the workflow.

require 'json'

def env_has_key(key)
!ENV[key].nil? && ENV[key] != '' ? ENV[key] : abort("Missing #{key}.")
end

output_dir = env_has_key('AC_OUTPUT_DIR')

def read_json_file(test_result_file_path)
JSON.parse(File.read(test_result_file_path))
end

def extract_line_coverage(json_data)
json_data['coverage']['lineCoverage']
end

begin
test_result_file_path = "#{output_dir}/test_results.json"
json_data = read_json_file(test_result_file_path)
line_coverage = extract_line_coverage(json_data)

puts "Current Line Coverage: % #{line_coverage * 100}"

min_coverage = 2.0
puts "Minimum coverage percentage: #{min_coverage}"

if (line_coverage * 100) < min_coverage
puts "Coverage is #{line_coverage} and below minimum coverage percentage given #{min_coverage}. \nExiting."
exit (1)
else
puts "Coverage is above the threshold. It is clear."
end

rescue StandardError => e
puts "An error occurred: #{e.message}"
end
info

Please feel free to edit the following variables according to your own requirements:

  • test_result_file_path: The file path of the test result file from which to retrieve the covered percentage value.
  • min_coverage: The minimum percentage required for the pipeline to continue without breaking.

How to use environment variables along with the sudo command?

Both user-created and Appcircle-reserved environment variables can be used within a custom script with any required command. But, by default, commands that are triggered with sudo will not reach them since the user scope is changed.

In order to use all user environment variables with sudo, you should add the -E argument to the sudo command. The -E (preserve environment) option indicates to the security policy that the user wishes to preserve their existing environment variables.

For instance, you can check the list of environment variables in the build pipeline using the command below.

sudo -E printenv

How can I send a custom email?

Appcircle provides a ready-to-use email structure in the Testing Distribution and Publish modules. This structure varies across the two modules. If desired, the user can customize this structure by using the Custom Script below to send their own custom email.

The following Bash script is set to use a Gmail SMTP Server. For more information, please visit Gmail SMTP Server documentation.


# Set SMTP server
CS_HOST="smtp.gmail.com"
CS_PORT="587"
CS_ACCOUNT="gmail"

# Make sure to replace CS_EMAIL, CS_USERNAME, and CS_PASSWORD with your account details
CS_EMAIL="your-email-address@gmail.com"
CS_USERNAME="your-email-address@gmail.com"
CS_PASSWORD="your-email-password"

# Set email details
CS_EMAIL_SUBJECT="Test Email Subject"
CS_EMAIL_TO="recipient-address@mail.com,recipient-2-address@mail.com"
# This part will be used for visualization
CS_EMAIL_FROM="Sender Name <sender-address@gmail.com>"
CS_EMAIL_BODY="This is the body of the sent email."
# Set TLS and SSL usage
CS_USE_TLS="True"
CS_USE_SSL="False"

# Detect operating system
os=""
if uname -a | grep -iq "darwin"; then
os="darwin"
elif uname -a | grep -iq "linux"; then
os="linux"
fi

# Create the .msmtprc file with appropriate permissions
cat <<EOF > ~/.msmtprc
defaults
auth on
tls on
EOF

# Check if OS is supported and install necessary packages
if [ "$os" == "darwin" ]; then
if ! command -v brew > /dev/null 2>&1; then
echo "Can't find brew installation; make brew command visible or install homebrew and try again."
exit 1
fi
brew install mailutils
brew install msmtp
echo "set sendmail=/usr/local/bin/msmtp" | sudo tee -a /etc/mail.rc
{ echo -n "tls_fingerprint " && msmtp --serverinfo --tls --tls-certcheck=off --host=$CS_HOST --port=$CS_PORT | egrep -o "([0-9A-Za-z]{2}:){31}[0-9A-Za-z]{2}"; } >> ~/.msmtprc
elif [ "$os" == "linux" ]; then
if ! command -v apt > /dev/null 2>&1; then
echo "Apt is not installed; install apt and try again."
exit 1
fi
apt-get update
apt-get install -y mailutils msmtp msmtp-mta
echo "tls_trust_file /etc/ssl/certs/ca-certificates.crt" >> ~/.msmtprc
else
echo "Unsupported OS: $os. Darwin or Linux was expected."
exit 1
fi

cat <<EOF >> ~/.msmtprc
logfile ~/.msmtp.log
account $CS_ACCOUNT
host $CS_HOST
port $CS_PORT
from $CS_EMAIL
user $CS_USERNAME
password $CS_PASSWORD
account default: $CS_ACCOUNT
EOF

# Restrict permissions for the .msmtprc file to avoid security issues
chmod 600 ~/.msmtprc

# Cleanup .msmtprc whether script ends with success or failure
function cleanup {
cat /dev/null > ~/.msmtprc
}
trap cleanup EXIT

# Send email
echo "From: $CS_EMAIL_FROM
To: $CS_EMAIL_TO
Subject: $CS_EMAIL_SUBJECT
$CS_EMAIL_BODY" | msmtp --debug --from=$CS_EMAIL -t $CS_EMAIL_TO

Script Execution

This script is written in Bash. When running with Custom Script, you need to set the Execute With parameter as Bash. For more information, please visit the Custom Script Input Variables documentation section.

Input Variables

When using your own SMTP server credentials for the three variables below, using Environment Variables is strongly suggested since this prevents sensitive information, such as passwords, from being exposed to unauthorized individuals.

For more detailed information, please refer to the Environment Variables documentation.

  • $CS_EMAIL: SMTP Server email address.
  • $CS_USERNAME: Sender email address.
  • $CS_PASSWORD: Sender email address password.

Otherwise, to send an email, you need to have some information, such as the email subject, sender email, and recipient email. You can use these parameters to:

  • $CS_EMAIL_SUBJECT: Subject of sending email
  • $CS_EMAIL_TO: Recipient email address.
  • $CS_EMAIL_FROM: Sender email address.
  • $CS_EMAIL_BODY: Content of sending email.
Recipient Email Address

If you want to send an email to multiple email addresses instead of a single email address, in the $CS_EMAIL_TO parameter, it will be enough to write all the addresses to send an email separated by commas. For example, $CS_EMAIL_TO=example@email.com,example2@email.com

Sensitive Informations

Since the variables mentioned above, which need to be provided by the user, contain sensitive information like passwords, please use Environment Variables for these types of values.

To do this, comment out or remove the sensitive variables such as $CS_EMAIL, $CS_USERNAME, and $CS_PASSWORD defined at the top of the script, and add them as environment variables instead.

For other variables that need to be defined, you can also make use of environment variables.

Self-hosted Runners

If you're using self-hosted runners and they're not starting to build pipelines with clean states as in Appcircle Cloud, this custom script can be insecure to use since the consecutive pipelines are using the same build environment, and it might have last sent email sensitive configuration in some cases.

In order to send email, the script persists SMTP configuration to the ~/.msmtprc file. Although the script has a cleanup mechanism that works for success or failure cases, it cannot work on all conditions. For example, the "cancel build" case should be considered. In this case, the ~/.msmtprc might not be cleaned, and it can be read in the next pipeline.

If you have configured runners similar to the Appcircle Cloud pools and they are wiped off after a build is executed to provide a fresh environment for every build, then there is nothing to consider; it's safe.

Username and Password for Google SMTP Users

When you want to send an email with your Gmail account using Google's SMTP server, you must first authenticate to the Google SMTP server. For this process, you need to enter your App Password in the password field.

In order to generate this password, 2FA authentication must be turned on in your Google account. You can generate and retrieve this password from the App Passwords section under Google Account management. For detailed information about App Passwords, please visit the Google App Password documentation.

Protocols and SMTP Host

This script uses the TLS protocol for SMTP server usage. Since the Gmail SMTP server is used in the script, the required protocols are pulled from Google's SMTP server using the $CS_HOST parameter. If you are using your own SMTP server, don't forget to change the $CS_HOST value here.

On the other hand, to change TLS or SSL usage, you can change the protocol by setting the $CS_USE_TLS or $CS_USE_SSL parameters in the script to true or false. Note that you need to change the $CS_PORT parameter when using SSL and TLS.

For more information about protocols, please visit the Google's TLS and SSL documentation.

Sender Email and Spoofing

To use an SMTP server, this script first installs the necessary certificates, then authenticates to the server with the required credentials and sends the prepared email content to the recipient's email address. In order to change the sender's email address, the SMTP server must allow it by providing the necessary permissions. Otherwise, SMTP servers will send the email using the authenticated email address to prevent spoofing (impersonating someone else).

How can I send the Appcircle build log to another platform?

To send your build log to another platform using an API, you can access the log in your workflow through $AC_LOGFILE.

Here is an example using Dropbox's file-upload API. Replace it with the appropriate API for your platform. If you are going to use DropBox, you can follow the document below to obtain the access token:

danger

Ensure sensitive data, like access tokens, are defined as private environment variables. Learn more:

current_datetime=$(date +"%Y-%m-%d-%H-%M")
dropbox_log_filename="/home/appcircle-logs/ac-log-$current_datetime.txt"

curl -X POST https://content.dropboxapi.com/2/files/upload \
--header "Authorization: Bearer $DROPBOX_ACCESS_TOKEN" \
--header "Dropbox-API-Arg: {\"autorename\":false,\"mode\":\"add\",\"mute\":false,\"path\":\"$dropbox_log_filename\",\"strict_conflict\":false}" \
--header "Content-Type: application/octet-stream" \
--data-binary @"$AC_LOGFILE"

This script generates a timestamped log file (e.g., ac-log-2024-10-01-14-55.txt) in the /home/appcircle-logs in Dropbox.

Screenshot
caution

Ensure that the Custom Script step runs after the Export Build Artifacts step to capture the full log.

How can I print the status of workflow steps with detailed information?

If you want to track or share the status of your workflow or individual steps during a build, you can use the following environment variables:

  • $AC_BUILD_STATUS: Displays the running status of the workflow so far.
  • $AC_BUILD_STEPS_STATUS: Displays detailed information about each executed step.
caution

The runner version must be 1.8.0 or later to use the above two environment variables.

However, the output of $AC_BUILD_STEPS_STATUS is in raw JSON format, which may not be easy to read directly. To make it more readable, you can use the following Ruby script to format and print the information in a user-friendly way:

require 'json'

puts "AC_BUILD_STATUS: #{ENV['AC_BUILD_STATUS']}"
# Read the environment variable
json_data = ENV['AC_BUILD_STEPS_STATUS']

begin
# Parse and beautify the JSON
parsed_data = JSON.parse(json_data)
pretty_json = JSON.pretty_generate(parsed_data)

# Output the formatted JSON
puts "AC_BUILD_STEPS_STATUS:"
puts pretty_json
rescue JSON::ParserError => e
puts "Failed to parse JSON: #{e.message}"
end
info

The script above is written in Ruby. To execute it, select Ruby as the Execute with option in the Custom Script step.

warning

To ensure your script works even if one of the steps in the workflow fails (and you want to capture the failed status as well), enable the "Always run this step even if the previous steps fail" option.

Screenshot

If you add a Custom Script step with the code above after the Git Clone step in your workflow, the script will generate an output similar to this:

AC_BUILD_STATUS: Success
AC_BUILD_STEPS_STATUS:
[
{
"StepName": "Activate SSH Private Key",
"BuildStatus": "Success",
"Duration": 0.133102,
"StartDate": "2024-12-30T16:48:47.919386Z",
"FinishDate": "2024-12-30T16:48:48.052488Z"
},
{
"StepName": "Git Clone",
"BuildStatus": "Success",
"Duration": 1.90708,
"StartDate": "2024-12-30T16:48:48.186532Z",
"FinishDate": "2024-12-30T16:48:50.093612Z"
}
]
info

Steps that are disabled in the workflow will not appear in the above output.

Simply include this script in your workflow to better understand and monitor the status of your workflow steps.