March 5, 2024

    How to Set up iXGuard with Xcode Cloud

    Guardsquare iXGuard is easy to set up with Xcode Cloud. In this blog I will show you an example of how Xcode Cloud can be set up to produce protected builds. You can easily adapt the example to suit your particular environment.

    Required knowledge: you need to have general knowledge about Xcode Cloud. No special knowledge or expertise is required to get the integration up and running.

    Cloud Build Plan

    To produce a protected application build using Xcode Cloud, you will need to perform some additional steps compared to a standard build.

    1. Download iXGuard and Guardsquare toolchain to the build server.
    2. Produce a bitcode-enabled application build with the help of Guardsquare toolchain.
    3. Protect the application with iXGuard.
    4. Export the resulting protected application artifacts.

    Let’s implement these steps in your Xcode Cloud build plan.

    Preparation: Create an Xcode Cloud workflow

    If you have not done so already, set up a new Xcode Cloud workflow by clicking Integrate → Create Workflow… in Xcode.

    1-diagram-How to Set up iXGuard with Xcode Cloud

    Specify the Archive action in the workflow specification:

    2-diagram-How to Set up iXGuard with Xcode Cloud

    Finally, test your workflow and make sure it produces an unprotected application archive.

    Step 1: Download iXGuard and Guardsquare toolchain to the build server

    The next step is to download all required software to perform the protection. Here’s a general guidance on how to automatically download iXGuard to cloud CI systems. Let’s adapt it for Xcode Cloud.

    First, you will need a keypair to access Guardsquare downloads. Use ssh-keygen to generate a new keypair and add it to the Guardsquare customer portal as described in the article above.

    Let’s get the private key on the Xcode Cloud build machine. To do that, copy the contents of the private key file and assign it to an environment variable named GUARDSQUARE_SECRET. Use the Secret flag to prevent the variable from showing up in the GUI as well as being dumped into the output logs. In this article we will use a keypair that is not protected with a passphrase (specify an empty passphrase in the process of keypair generation).

    3-diagram-How to Set up iXGuard with Xcode Cloud

    Replace the new line characters with \n, so that your private key variable looks like:

    -----BEGIN OPENSSH PRIVATE 
    KEY-----\n000firstline000\n111secondline111\n222thirdline222\n-----END
    OPENSSH PRIVATE KEY-----

    To use this private key on the build machine, we will use Xcode Cloud lifecycle trigger scripts. In the root of your project create a directory named ci_scripts, and place a new shell executable file named ci_post_clone.sh.

    4-diagram-How to Set up iXGuard with Xcode Cloud

    Use the following contents for the ci_post_clone.sh script:

    
    #!/bin/sh
    echo Post-clone script - get iXGuard and install the toolchain
    echo ---------------------------------------------------------
    
    echo Getting Guardsquare CLI
    curl https://downloads.guardsquare.com/cli/latest_macos_amd64 -sL | tar -x
    
    echo Saving Guardsquare access key
    echo "$GUARDSQUARE_SECRET" >./guardsquare.pem
    chmod 600 ./guardsquare.pem
    
    echo Running SSH agent
    eval "$(ssh-agent -s)"
    
    echo Adding the Guardsquare access key to SSH
    ssh-add ./guardsquare.pem
    
    echo Downloading iXGuard
    ./guardsquare --ssh-agent download ixguard -o ixguard.pkg
    
    echo Downloading licenses
    ./guardsquare --ssh-agent download licenses
    
    echo Extracting iXGuard
    rm -rf ./ixguard-package
    pkgutil --expand-full ./ixguard.pkg ./ixguard-package
    
    echo Installing toolchain
    ./ixguard-package/ixguard.pkg/Payload/scripts/create_toolchain -p ./ixguard-package/ixguard.pkg/Payload
    

    This script will use the private key from the environment variable you’ve just defined to download iXGuard and all applicable licenses and install the iXGuard toolchain.

    Do not forget to chmod +x the script and add it to Git.

    Step 2: Produce a bitcode-enabled application with the help of Guardsquare toolchain

    To make a bitcode-enabled build, it is sufficient to define the TOOLCHAINS environment variable as below.

    5-diagram-How to Set up iXGuard with Xcode Cloud

    Step 3: Protect the application archive with iXGuard

    To protect the application in the Xcode Cloud CI flow, you will need to run a few commands at the post-archive step.

    To implement these commands, go to Product → Schemes → Edit Scheme:

    6-diagram-How to Set up iXGuard with Xcode Cloud

    Add a Run Script post-action to the Archive step:7-diagram-How to Set up iXGuard with Xcode Cloud

    Add the following line to your Archive post-action script:

    "$CI_PRIMARY_REPOSITORY_PATH/ci_scripts/ixguard-package/ixguard.pkg/Payload/python/launch_ixguard.py" "$CI_ARCHIVE_PATH" -o="$CI_ARCHIVE_PATH.protected.xcarchive" -config="$CI_PRIMARY_REPOSITORY_PATH/ixguard.yml"

    This command relies on iXGuard configuration to be in the root directory of your project and be named ixguard.yml.

    Note

    Your iXGuard configuration has to be checked into the source code repository. Therefore, only set Xcode Cloud build on private, and never on public repositories.

    Pro tip:

    If you want to run this line only on the CI server, you can use the CI environment variable that is automatically defined by Xcode Cloud.

    Step 4: Export the resulting artifacts

    And the last step of this example workflow is to export the protected application artifacts. Since Xcode Cloud requires a certain archive name in order for that archive to be picked up by the export tasks, I am going to rename the files so that the protected archive substitutes the unprotected one.

    Let’s add two new commands to the post-archive action:

    mv "$CI_ARCHIVE_PATH" "$CI_ARCHIVE_PATH.unprotected.xcarchive"

    mv "$CI_ARCHIVE_PATH.protected.xcarchive" "$CI_ARCHIVE_PATH"

    The complete post-archive script would look like this:

    
    "$CI_PRIMARY_REPOSITORY_PATH/ci_scripts/ixguard-package/ixguard.pkg/Payload/python/launch_ixguard.py" "$CI_ARCHIVE_PATH" -o="$CI_ARCHIVE_PATH.protected.xcarchive" -config="$CI_PRIMARY_REPOSITORY_PATH/ixguard.yml"
    mv "$CI_ARCHIVE_PATH" "$CI_ARCHIVE_PATH.unprotected.xcarchive"
    mv "$CI_ARCHIVE_PATH.protected.xcarchive" "$CI_ARCHIVE_PATH"

    Bonus tip: Logging

    The output of the Archive Post Action script is not logged into the Xcode Cloud build log by default. In order to print the output of certain commands you execute as part of your Archive Post Action script, you can use the following strategy:

    1. In your script, you can add a command that will create a file used to populate with the outputs of certain commands. This file will be put in the CI_WORKSPACE_PATH directory (see second command in the script above).

    # Create a file used for logging
    touch $CI_WORKSPACE_PATH/post-archive.log

    2. For each command you would like to see the output of, you can add:

    YOUR COMMAND | tee "$CI_WORKSPACE_PATH/post-archive.log"

    3. In the root of your project you previously created a directory named ci_scripts. Place a new shell executable file named ci_post_xcodebuild.sh in the ci_scripts directory.

    4. Populate the ci_post_xcodebuild.sh script with a command that will print out the content of the post-archive.log file:

    cat $CI_WORKSPACE_PATH/post-archive.log

    Conclusion

    Making iXGuard work in the Xcode Cloud environment boils down to using Xcode Cloud lifecycle hooks to run Guardsquare tools at the right time.

    You can easily modify this process to incorporate your own logic, such as application security scanning, email notifications, and more.

    Tag(s): iOS , Technical , iXGuard

    Anton Baranenko - Product manager

    Curious about mobile application protection with Guardsquare?

    Download the iXGuard fact sheet to learn more >

    Other posts you might be interested in