Self Hosted Expo Update Server is a ready to use batteries included Expo updates compliant server to manage updates that you can host yourself in the cloud and have full control and visibility on the update cycle, including rollbacks!
I love the ability to push over-the-air updates with expo, it is a fantastic feature, but with great power comes great responsibility. The console-only interface can be tricky, the risk of making mistakes is high (especially on ejected app with incompatible binaries), if you want to roll back you really need to know what you are doing, and a single mistake can have potentially devastating impact.
I have already made a simple helper library that I use in my expo projects to simplify the update setup on the mobile side, check out expo-custom-updater
This is my attempt to simplify my own life when dealing with updates on the server side, and hopefully it can be useful to you too!
Features:
- Manage multiple Expo Apps
- Manage multiple Versions and Release Channels
- Send expo updates securely to the server and decide later when / how to release to users
- Roll back to a previous update
- Get insight on how many client app downloaded the update, see your changes being released in realtime
- Get a ton of info on the update, including git branch, commit, package.json and app.json information
- Assisted app configuration with self-signed certificate generator.
- All from a simple Web interface
Monitor client updates in realtime
A lot of useful information on every update
Details on dependencies to avoid incompatible updates
Roll Back to a previous update
If you have Docker installed you clone this project you and play around by running yarn
in the root folder and then running yarn dev:run
to start the Docker/development docker compose. Web credentials are admin/devserver (admin password is set in the docker-compose file).
If you use Docker you can find a production-ready docker-compose files under the Docker folder, just copy Docker/production on your server, set your secrets / credentials and you are ready to go. The two docker images are public and ready to go. Explanation for the Environment settings is in the docker-compose file. For production reverse proxy with Apache take a look at README DOCKER.md
Otherwise you can build from code, there's a NodeJS API server under the API folder and a React / Vite web project under the Web foder.
Use the web interface to add your application, just enter the expo slug name
You can download the ready to use publish script or create your own, the script logic is simple:
- use
expo export
to generate an update - Add app.json and package.json to the build folder
- Zip the build folder
- use curl to push the zip file to the server
In order to validate the update expo needs a certificate.pem inside your app and a private key on the update server. Use the SERVER CONFIGURATION section to generate a new self-signed key, make sure to SAVE, then downlaod both key and back them up!
It is necessary to configure your app.json with the provided keys from the APP CONFIGURATION section. Make sure to have a runtime version specified.
If you are on an Ejected project you can run expo prebuild
to autogenerate the code in android-manifest.xml and Expo.plist, otherwise use the provided settings in the APP CONFIGURATION with the autogenerated code.
Updates don't work in dev / expo app, you need to build a new app to test the system. You can use the provided examples (Managed and Ejected) in the relative folders. The provided example app have a button to request and download an update if present, and another button to reload the app. It is expected that an app will automatically do this operation on start, check out expo-custom-updater
IMPORTANT: During development you may get certificate issues when loading the JS bundle from your computer, there are two options:
On EXPO >= 49: you can now start the development server specifying your private signing key using the following command:
npx expo start --private-key-path (path to the private key)
(Thanks @SMhdAsadi )
Alternatively you may allow unsigned manifests by editing Expo.plist and AndroidManifest.xml:
AndroidManifest.xml
<meta-data android:name="expo.modules.updates.CODE_SIGNING_ALLOW_UNSIGNED_MANIFESTS" android:value="true"/>
Expo.plist
<key>EXUpdatesConfigCodeSigningAllowUnsignedManifests</key>
<true/>
Remember to set those back to false before staging / production build!.
Once the app pings the update server you will see an update appearing in the homepage, and after an update / download / reload cycle the updated app will reflect in the dashboar AFTER it asks for an update again.
This part should be the simplest: call the provided script with appropiate parameters and see your update pop in the web UI. Release, rollback or delete the updates and see the clients updating in the homepage.
Configure the throttle value in docker-compose Environment in order to slow down updates (in case you have thousands of users) to avoid overloads. Default throttle is no more than one dashboard update every 5 seconds.
You will find an example app in the Example Managed folder, to start tweaking it up you need:
- Copy App public certificate from server to /code-signing/certificate.pem
- Teweak app.json and app.config.js, specially the updates section to point to the right server
- Create an Android or iOS build using the respective scripts in package.json
- Install and test the app
- Make some changes in the app code in App.js
- Publish an update with
scripts/expo-publish-selfhosted.sh staging ./ExampleManaged abc123def456 http://localhost:3000
- Test the update within the app
You can see tailored console output using ADB logcat like this:
- View only RN and Expo updates output
adb logcat "*:S" ReactNative:V ReactNativeJS:V FirebaseMessaging:V dev.expo.updates:V
- Clear logs
adb logcat -c
Feel free to clone, costomize and send back PRs!
Have fun!