Skip to main content

Bitrise CI template for a React native app

Here is a conceptual monopipeline for building and deploying a React native app to the app stores in Bitrise. The described pipeline serves us well, and I discuss the benefits and tradeoffs in the afterword.


Setting up .env and release method

The first step is for the CI machinery to overwrite or add relevant environment variables based on the branch/tag. You can already at this point determine if this pipeline will release the app to the stores or just run through the build.

Run linters and tests, check test coverage

For TS/JS, these can be just npm scripts in the Bitrise npm step. For native code, you can use the predefined steps or run some Gradle tasks. Prefer one-liners: avoid writing complicated initialization commands. npm run lint, npm run test:coverage...

Set release versions

You could simply use the git tag here for the version. You could also use automatic semver and tools such as commitizen to version the build. For private builds, X.Y.BUILD_NUMBER is enough. There is a Bitrise step for both iOS and Android.

Run iOS build

Run and create an archive, use the export method determined by the first step(s).

Conditional step, deploy to the app store.

The previous CI steps should result in a flag (ENV) to determine if we should deploy or not, and should we deploy all the way to production or just to Testflight.

Download Android upload signing keystore. 

The keystore should be stored in some secure location. You can use the Bitrise predefined step for this and upload the keystore to Bitrise.

Build Android

Similar to the iOS step

Conditional step, deploy to Play store

Yes, we can end up in a situation where the iOS app is in the app store, and the Android app does not build and does not end up in the store. In case of fire, do manual steps :/

Run codecov / deepcode.ai / whatever

Some extra static analysis for your code

Upload and trigger Device farm tests

Can be done for both platforms or only one. Since the tests can take a while, this step is not blocking. Use some other tool to check the test results from the Device farm. If you release from this CI pipeline to production, this pipeline setup does not work since you would need to cancel the release if this step fails.

Send a slack message.

It can contain the release version, test/static analysis result links, and a download URL. You could send it only if the build fails to avoid spam.


The benefit of having only one pipeline is simplicity; we can see what is happening at one glance. We don't need to move artifacts around, and we guarantee that the configs and versions are the same for both apps. The downside is that in situations where we want to treat them separately, for example, release a fix for only for Android, we need to do some tricks to run only the iOS part of the pipeline.  Automating the release to production with screenshots and release notes would also require some shuffling of the steps. You would, in that case, want to run the device tests before the store upload.

Comments

Popular posts from this blog

I'm not a passionate developer

A family friend of mine is an airlane pilot. A dream job for most, right? As a child, I certainly thought so. Now that I can have grown-up talks with him, I have discovered a more accurate description of his profession. He says that the truth about the job is that it is boring. To me, that is not that surprising. Airplanes are cool and all, but when you are in the middle of the Atlantic sitting next to the colleague you have been talking to past five years, how stimulating can that be? When he says the job is boring, it is not a bad kind of boring. It is a very specific boring. The "boring" you would want as a passenger. Uneventful.  Yet, he loves his job. According to him, an experienced pilot is most pleased when each and every tiny thing in the flight plan - goes according to plan. Passengers in the cabin of an expert pilot sit in the comfort of not even noticing who is flying. As someone employed in a field where being boring is not exactly in high demand, this sounds pro

Extracting object properties from an IFC file with IfcOpenShell

Besides the object geometry information, IFC files may contain properties for the IFC objects. The properties can be, for example, some predefined dimension information such as an object volume or a choice of material. Some of the properties are predefined in the IFC standards, but custom ones can be added. IFC files can be massive and resource-intensive to process, so in some cases, it helps to separate the object properties from the geometry data. IfcOpenShell  is a toolset for processing IFC files. It is written mostly in C++ but also provides a Python interface. To read an IFC file >>> ifc_file = ifcopenshell.open("model.ifc") Fetch all objects of type IfcSlab >>> slab = ifc_file.by_type("IfcSlab")[1] Get the list of properties >>> slab.IsDefinedBy (#145075=IfcRelDefinesByType('2_fok0__fAcBZmMlQcYwie',#1,$,$,(#27,#59),#145074), #145140=IfcRelDefinesByProperties('3U2LyORgXC2f_hWf6I16C1',#1,$,$,(#27,#59),#145141), #145142

Hubristic developer

Almost half of any Finnish generation goes through a shared experience: the conscript army. An integral part of that experience is learning military slang, a set way people in the army talk. The stories told with said jargon often spread outside of the barracks. It is not uncommon to hear strangers bonding together over beers reminiscing and feeling nostalgic about freezing cold nights spent in tents. There is a similar phenomenon detectable among us coders. To be part of the coder tribe, there is at least one type of story that one must master. That is - of course - ranting about legacy codebases. "Can you believe how much of a mess the previous coders left? Hear, hear!" There is no better way to onboard a new team member than to blame some previous B-team for all the murky parts of the system at hand. This can be seen as harmless, a subject for a good  meme . Rarely do we hold real grudges against "the legacy folk" and can be the best of friends in a social gather