Preview your Android & iOS React Native apps on your Github Pull Request

Preview your Android & iOS React Native apps on your Github Pull Request

We are playing since a few weeks with React Native for a Proof Of Concept and wanted to have the same development workflow for mobile apps, as we have for the web.

Here is the workflow we use for web development:

When every step is ok, the PR is merged.

The « Branch step », « PR step » and « Code Review step » are mostly related to our CVS (Github Enterprise) and are not a problem. The « Test step » is related to React Native. We already use Jest and ESLint, but we have to dig more for Integration test (Appium ?).

The « Preview step » is more interesting. It was not the simplest thing to do on our web project, but this is probably one of the most useful feature we have on our stack. Having a staging environment for all open PR allows devs, PO, PM and scrum masters to play with this exact version of the code (on any browser they want), and really see if the bug is fixed, or if the feature correspond to the PO needs. It allows everyone to iterate and make feedbacks before the code lands on the master branch. It’s also a good way to be sure your app build didn’t fail.

So, what we want is to have on each of our React Native PR, a link to preview iOS and Android version of our app in a web browser, refreshed after every commit on the branch.

The goal of this blog post is just to show you, that it is something doable and really useful. If you are interested in, here are some more information, that maybe can help you.

The stack

Concerning the CI, we already use Jenkins, so we will continue to. Beware that for building iOS apps, a CI running on OSX is needed. In our case, we had added a Jenkins slave to our Jenkins pool. If you don’t have CI system internally, you should take a look at Bitrise or CircleCi because they propose OSX CI systems. Our CVS is Github Enterprise, but everything is also possible with Gitlab (or any other CVS). We use Fastlane.tools to automate build and credentials support. (Mostly because it was recommended by some of our iOS developers).

In order to preview iOS and Android app in a web browser, we use the amazing SAAS service Appetize.io (free for 100min/month).

![Appetize]”./appetize.png)

How did we do ?

We had set up an OSX machine with a fresh Jenkins install, and created a job that triggers a build everytime a push is made on a PR, thanks to the “Github Pull Request Builder » Jenkins plugin. There is also a lot of things to configure on this machine (Nodejs, Ruby, xCode …), and i recommend you to do some builds (iOS and Android) manually to be sure everything is ready.

Fastlane is an open-source automation toolset for iOS & Android. It lets you write « lane » to automate a lot of things. We set up a unique Fastlane file at the root of our React Native project directory dealing with Android & iOS lanes.

To suit our needs, we created one lane « deployAppetize » for each platform: it performs the corresponding build, uploads it to Appetize.io via their API, and updates the Github PR Statuses during the process.

I’m not a Ruby programmer, so please, don’t blame me, and feel free to improve the code below if you want (on this Github Gist). This is neither the state of the art, nor a beautiful open source thing, we just share what we did in case it helps someone :-)

Before doing anything, you’ll have to set some variables on Fastlane, so go to the Fastfile file in your fastlane folder:

{% highlight ruby %} #3rd party lib to do some http calls require ‘httparty’

fastlane_version “1.95.0” default_platform :ios

before_all do

put here your token and iOS scheme app

ENV[“GITHUB_TOKEN”] = ”----” ENV[“APPETIZE_TOKEN”] = ”----” ENV[“APP_IOS_SCHEME”] = ”----“

get the last git commit information

ENV[“GIT_COMMIT”] = last_git_commit[:commit_hash]

Use ghprbSourceBranch env variable on CI, git_branch lane elsewhere

if !ENV[“ghprbSourceBranch”] ENV[“ghprbSourceBranch”] = git_branch end

end {% endhighlight %}

Create a private lane to make the POST request to your Github statuses API to avoid DRY:

{% highlight ruby %}

Update git statuses of your commit.

private_lane :githubStatusUpdate do |options|

response = HTTParty.post( “https://<yourgithubenterprisedomain.tld>/api/v3/repos///statuses/#{ENV[“GIT_COMMIT”]}?access_token=#{ENV[“GITHUB_TOKEN”]}”, :body => { :context => options[:context], :state => options[:state], :description => options[:description], :target_url => options[:url] }.to_json, :headers => { ‘Content-Type’ => ‘application/json’ } ) end {% endhighlight %}

Appetize allows you to create different apps. We want one app per PR, and update the corresponding app when a new commit is made on a PR. For that, we keep track of the branch name by storing it in the « notes » field of the app on Appetize.io.

So, here’s a private lane to get back the public key of the corresponding app on Appetize.io, to update the good one if it already exists.

{% highlight ruby %}

get the publicKey of the appetizeApp corresponding to your git branch

private_lane :getAppetizePublicKey do |options| publicKey = ""

response = HTTParty.get(“https://#{ENV[“APPETIZE_TOKEN”]}@api.appetize.io/v1/apps”) json = JSON.parse(response.body)

Find branch name in notes

json[“data”].each do |value| if value[“note”] == ENV[“ghprbSourceBranch”] && value[“platform”] == options[:platform] publicKey = value[“publicKey”] end end

publicKey end {% endhighlight %}

Now, we have everything ready to do the deployAppetize lane for iOS :

{% highlight ruby %} platform :ios do

desc “Deployment iOS lane”

lane :deployAppetize do

  githubStatusUpdate(
    context: 'Appetize iOS',
    state: 'pending',
    url: "https://appetize.io/dashboard",
    description: 'iOS build in progress'
  )

  Dir.chdir "../ios" do
    tmp_path = "/tmp/fastlane_build"

    #seems not possible to use gym to do the simulator release ?
    xcodebuild_configs = {
      configuration: "Release",
      sdk: "iphonesimulator",
      derivedDataPath: tmp_path,
      xcargs: "CONFIGURATION_BUILD_DIR=" + tmp_path,
      scheme: "#{ENV["APP_IOS_SCHEME"]}"
    }

    Actions::XcodebuildAction.run(xcodebuild_configs)

    app_path = Dir[File.join(tmp_path, "**", "*.app")].last

    zipped_bundle = Actions::ZipAction.run(path: app_path, output_path: File.join(tmp_path, "Result.zip"))

    Actions::AppetizeAction.run(
      path: zipped_bundle,
      api_token: "#{ENV["APPETIZE_TOKEN"]}",
      platform: "ios",
      note: "#{ENV["ghprbSourceBranch"]}",
      public_key: getAppetizePublicKey({platform: "ios"})
    )

    FileUtils.rm_rf(tmp_path)

  end

  githubStatusUpdate(
    context: 'Appetize iOS',
    state: 'success',
    url: "#{lane_context[SharedValues::APPETIZE_APP_URL]}",
    description: 'iOS build succeed'
  )
end

error do |lane, exception|
  case lane
    when /deployAppetize/
      githubStatusUpdate(
        context: 'Appetize iOS',
        state: 'failure',
        url: "https://appetize.io/dashboard",
        description: 'iOS build failed'
      )
    end
  end

end {% endhighlight %}

For Android, it’s almost the same things, except we have to do some small business logic to find the apk generated by Gradle, with this private lane :

{% highlight ruby %}

find the path of the last apk build

private_lane :getLastAPKPath do apk_search_path = File.join(’../android/’, ‘app’, ‘build’, ‘outputs’, ‘apk’, ‘.apk’) new_apks = Dir[apk_search_path].reject { |path| path =~ /^.-unaligned.apk$/i} new_apks = new_apks.map { |path| File.expand_path(path)} last_apk_path = new_apks.sort_by(&File.method(:mtime)).last

last_apk_path end {% endhighlight %}

And now you should be able to also deploy to Appetize.io on Android :

{% highlight ruby %} platform :android do

desc “Deployment Android lane”

lane :deployAppetize do

  githubStatusUpdate(
    context: 'Appetize Android',
    state: 'pending',
    url: "https://appetize.io/dashboard",
    description: 'Android build in progress'
  )

  gradle(
    task: "assemble",
    build_type: "Release",
    project_dir: "android/"
  )

  Actions::AppetizeAction.run(
    path: getLastAPKPath,
    api_token: "#{ENV["APPETIZE_TOKEN"]}",
    platform: "android",
    note: "#{ENV["ghprbSourceBranch"]}",
    public_key: getAppetizePublicKey({platform: "android"})
  )

  githubStatusUpdate(
    context: 'Appetize Android',
    state: 'success',
    url: "#{lane_context[SharedValues::APPETIZE_APP_URL]}",
    description: 'Android build succeed'
  )
end

error do |lane, exception|
  case lane
    when /deployAppetize/
      githubStatusUpdate(
        context: 'Appetize Android',
        state: 'failure',
        url: "https://appetize.io/dashboard",
        description: 'Android build failed'
      )
  end

end {% endhighlight %}

It’s over. You just have to add those commands to your CI to do the job :

npm install
Fastlane ios deployAppetize
Fastlane android deployAppetize

You have now two new checks on each PR with a link to the iOS or Android instance on Appetize.io.

![Github Pull Request with preview url]”./githubpr.png)

The complete Fastfile on a Github Gist : FastFile

Conclusion

At M6web, we are glad to see the whole React Native promise taking a concrete shape: the developer experience is the same for both mobile & web development, even about tooling. We are continuing to play with it and we’ll certainly keep posting articles here, stay tuned !

P.S.: You could look at the Fabric Blog post on the device grid for Fabric but with Danger commenting on the PR instead of Github Statuses, and iOS only.

P.S.2: You could also look at Reploy.io, which try to improve this workflow with extra features and a more cleaner UX than Appetize.io, but it is “alpha” for now.