2016-10-19: Updated with notes around macOS Sierra

This is part 4 of my Creating a Build-Once iOS Deployment Pipeline series.

In part 3 we configured our build server agent and added build and test phases. Leaving signing out of out build and test phases kept them simple, but it leaves a lot of things to be solved during deployment.

At a high level, here’s what our deployment lane will look like:

  1. Configure the IPA by modifying it’s Info.plist file.
  2. Call Match to import a valid signing identity and provisioning profile from our certificate repository
  3. Call Resign to resign the repository using the imported signing identity and profile
  4. Deploy the IPA to HockeyApp either to be tested (QA) or to improve crash diagnostics.
  5. Deploy to AppStore (exclusive to our “AppStore” environment)

Approach

Credentials and Secrets

In vast contrast to the build lane, deployment has a substantial number of actions that require some kind of credential or secret. Most can use the keychain, through some support alternative methods of authentication, like SSH or environment variables.

Action Usage Keychain Alternatives
Match Certificate Repository (Git) authentication (list-keychains) SSH/RSA
Match Password for encrypting the private keys stored in the repository (list-keychains) ENV["MATCH_PASSWORD"]
Match Importing signing identities ENV['MATCH_KEYCHAIN_NAME'] N/A
Sigh Accessing signing identities (list-keychains) N/A
HockeyApp HockeyApp API token N/A ENV['FL_HOCKEY_API_TOKEN']
Deliver iTunes password for deployment (default) ENV['DELIVER_PASSWORD']

A few things to note:

  • Match supports an option for a specific keychain, but Sigh does not (Edit: As of fastlane 1.102, resign supports a keychain_path option). There’s also no way to avoid Keychains altogether, as even actions that accept a custom keychain tend to rely on downstream functionality that does not.
  • Actions that use the “default keychain” actually use the first item from keychain search list, that is the result of “security list-keychains” as opposed to “security default-keychain”, so there’s no need to change the default/login keychains in order to customise them.

I’ve highlighted which of the options we went with, but I’ll explain the justification as we go.

Configuration

I won’t be making any strong recommendations on how to store your environment specific configuration, since it depends on how complex it is and how many environments there are.

In our scenario, we had a fair number of environments (4-5 white labelled app versions, 3 deployment targets each), so we stored each profile in a YAML file in our configuration repository that our deployment lane parsed. Inside, we had one section for Info.plist values, one for GoogleServices-Info.plist, and one for environment variables which specified miscellaneous fastlane configuration (like HockeyApp team ids, etc). We then had a batch file that wrapped execution of fastlane, and could invoke it numerous times based on matching profiles. Overkill for most apps, but a maintainable approach for us.

Regardless, it’s important to take the time to think about configuration for your requirements, whether it be hardcoded in the fastfile or loaded from a central database at deploy time.

Versioning

Versioning, like configuration, is largely dependent on your internal tools and processes. How semantic is your versioning scheme? What kind of auto-incrementing does your build/release server support?

The only recommendations I would make, though you are free to ignore, are:

Avoid incrementing the build number in your actual build scripts. It both requires that your CI server commits the changes (and this has write access to your repository) and will get out of sync as various developers run the build locally. If your CI server has an auto incrementing number, pass that in and patch it in during (in a .sh script) or after (via act) the build phase. If you swap build servers, commit a hardcoded “seed” value that you add to the CI build number.

Avoid committing your semantic (ie. release) version to your application repository. Because it’s semantic, the change in version won’t be known until you decide to release, at which time you don’t want to have to start the build pipeline again. Either make use of your CI server’s release versioning, or commit the value to your configuration repository.

Implementation

Both release types need to change our environment-specific plist value, resign the app, and deploy to HockeyApp. The App Store deployment will then also deploy to the App Store using Deliver.

Configuration

Before we resign the IPA, we want to make changes to it. IPAs are only zip files, but running through all of the unzip/plistbuddy/zip shell commands introduces noise to your deploy script and makes the intent less clear.

To avoid this, I’ve created a fastlane action called “act”, which takes the hassle out of modifying list files inside IPAs. It also supports replacing the icon with an .iconset, but we won’t be doing that here to keep this simple.

To install it, run:

bundle exec fastlane add_plugin act
bundle exec fastlane install_plugins

Then you can configure an ipa in your fastfile, use:

act(
  ipa: path_to_ipa,
  plist_values: {
    ":DemoEnvironmentValue" => "QA"
  }
)

You may need to call act a number of times, as some configuration is stored in different plist files (eg. GoogleServices-Info.plist)

With our ipa configured, we need to sign it so that we can release it.

Match

Match has a number of hurdles to overcome, since it needs to connect to your certificates repository, the developer portal, and a local keychain.

For your certificates repository, I’d recommend using ssh as it avoids having to introduce another keychain. Regardless of where you want to store your private key, in your configuration repository or on the build server, you should also make sure your known_hosts has your git server in it to avoid the deploying locking up waiting for user input. If you need it, you can override the private key and known_hosts location by setting this environment variable:

# Git 2.3+
export GIT_SSH_COMMAND="ssh -i path/to/id_rsa_git -o UserKnownHostsFile=path/to/known_hosts"

Next, match is going to validate the certificate against the developer portal. These can, and probably should, be set using the DELIVER_USERNAME and DELIVER_PASSWORD environment variables, since that can likely be configured securely on your CI and avoids introducing a keychain.

Match will then import the signing identity into the first keychain in the search list. Attempting to keep that keychain around would only introduce “source of truth” issues (not to mention the security implications), so I’ve written a Gem named spare_keys to help. Install it either globally or to your Gemfile and then use it like this:

SpareKeys.temp_keychain(true) { |keychain_path|
  # our temp kecyhain, stored at keychain_path,
  # will be used by match/resign within this block.
  # Everything will be restored afterwards.
)

Match will also require a password that it uses to decrypt all the files in the repository. Your CI can set this using MATCH_PASSWORD.

With all of this in place, our call to Match looks like this:

SpareKeys.temp_keychain(true) { |keychain_path|
  match(
    type: "adhoc" # or "appstore",
    git_url: "[email protected]:path/to/certificates-repository.git",
    app_identifier: "com.richardszalay.deploydemo",
    keychain_name: keychain_path,
    readonly: true
  )
}

Gotcha #8: You can ignore this error in your logs: “There are no local code signing identities found”. We just created a fresh keychain, so its obviously empty.

I personally run match here with the readonly flag, which prevents it from attempting to create new signing identities and provisioning profiles. Instead, I create a second lane called update_certs that is run manually when required. Whether you should do this too depends entirely on how ok you are with your deployment processes having side effects of this kind.

Gotcha #9: Match will create a provisioning if one doesn’t exist in the repository, and will do so by calling Sigh with the force flag. This will automatically add all devices to it. If your AdHoc release service can’t be solely relied upon to restrict access between owners of registered devices, you might need to handle this yourself (or put your support behind fastlane/#5667).

Resign

Now that we have a valid signing identity and provisioning profile, the call to resign should be pretty basic.

Gotcha #10: resign accepts provisioning profile options using a hash that’s not immediately compatible with the environment variables set by match. To work around this, I use this utility function:

# Creates a hash, suitable for resign's provisioning_profile option, from
# the environment variable set by match
def match_result_provisioning_profile(app_identifier, match_type)
  sigh_profile_key = "sigh_#{app_identifier}_#{match_type}"

  return {
      app_identifier => File.join(FastlaneCore::ProvisioningProfile.profiles_path, "#{ENV[sigh_profile_key]}.mobileprovision")
  }
end

Gotcha #11 macOS Sierra made a change that complicates this story. It introduced the concept of key partition lists to the keychain, configured using the (as-yet undocumented) security set-key-partition-list command. More information can be found at this openradar bug report: http://openradar.appspot.com/28524119. If the correct key partition lists aren’t applied to imported identities, the user will still be prompted to allow access to the keychain even if it was unlocked. To makes things worse, the command requires the keychain password, even if it’s already unlocked. To work around this, spare_keys 1.1.1 now provides the password to the keychain as a second block argument and I’ve uploaded the snippet below to make use of it.

With that last quibble out of the way, we can call resign within our temp keychain scope:

SpareKeys.temp_keychain(true) { |keychain_path, keychain_password|
  match_type = "adhoc" # or "appstore"
  app_identifier = "com.richardszalay.deploydemo"

  match(
    type: match_type,
    git_url: "[email protected]:path/to/certificates-repository.git",
    app_identifier: app_identifier,
    keychain_name: keychain_path,
    readonly: true
  )

  `security set-key-partition-list -S apple-tool:,apple: -k "#{keychain_password}" #{keychain_path}` if requires_key_partition_list

  resign(
    ipa: "fastlane/build/DeploymentPipeline.ipa",
    provisioning_profile: match_result_provisioning_profile(app_identifier, match_type)
  )
}

requires_key_partition_list can be defined at the bottom of your fastfile as:

def requires_key_partition_list()

  osVersion = `sysctl -n kern.osrelease`

  majorOsVersion = Integer(osVersion.split('.')[0])

  return majorOsVersion >= 16 # Sierra

end

And there we have it: a signed IPA, configured to a specific environment, in a fraction of the time it would have taken to rebuild it. You’ll need to offset the time it took to read this wordy blog series, of course, but I’m sure you’ll become net positive eventually.

HockeyApp

From here on out, it’s stock fastlane all the way. We will be uploading to hockeyapp for both QA and App Store deployments, since it provides crash analytics for the latter.

We’re not going to auto release here, as we don’t have the release notes in our configuration repository, so the release will need to be completed via the HockeyApp dashboard. However, if your workflow supports it, the HockeyApp fastlane action supports a myriad of options for you to choose from.

The only thing that needs to be set is FL_HOCKEY_API_TOKEN. The HockeyApp app will be automatically be selected based on the app identifier, though you can override it with FL_HOCKEY_PUBLIC_IDENTIFIER if need be.

hockey(
  ipa: options[:ipa]
)

When deploying to the App Store, you can also supply upload_dsym_only: true (or via FL_HOCKEY_UPLOAD_DSYM_ONLY) to avoid the redundant upload of the IPA.

AppStore

Finally will be releasing to the app store. It’s at this point where you can take advantage of a significant amount of fastlane’s features, such as automatically taking/uploading screenshots and changelog information. We’ll be disabling that here to keep things simple, however.

Deliver will automatically make use of the DELIVER_USERNAME and DELIVER_PASSWORD environment variables we set for match, so there’s nothing more to do here.

deliver(
  ipa: options[:ipa],

  skip_screenshots: true,
  skip_metadata: true,

  submit_for_review: false,
  automatic_release: false
)

Gotcha #12: if you remember from part one, iTunes have different team ids from the developer portal. If the account you are using to authenticate with has access to multiple iTunes teams, you’ll have to specify which by setting FASTLANE_ITC_TEAM_ID or FASTLANE_ITC_TEAM_NAME. You can get the team id by running the release locally - you’ll be prompted to select the correct team id.

Putting it all together

Here is the fully assembled Fastfile, though I’ve taken the liberty of refactoring it to avoid deduplication between release types. I’ve also included bash scripts to simulate what’s needed from the build server.

Closing Thoughts

Having worked with this model for a few weeks, I’m quite happy with it. Given we have 3 “brands” of the app going to three environments each, it’s already reduced the total pipeline time from 115 minutes to 31, and releasing a client-facing “beta” after an internal “alpha” takes 3 minutes (from 25). Granted not all applications have this complexity, but hopefully my research is useful to someone.

Moving forward, I would like to investigate using the xcarchive rather than the IPA in an attempt to make use of the new XCode 7 features like app slimming and bitcode. Once fastlane/#5824 and fastlane-plugin-act/#1 have been resolved, I’ll create a new post.