Main script that is aimed to be run with scriptworker (but runs perfectly fine as a standalone script). This project is a fork of signingscript. Most of the documentation from signing script applies to this project.
Get the code
First, you need
# create the virtualenv in ./venv3 virtualenv3 venv3 # activate it . venv3/bin/activate git clone https://github.com/mozilla-releng/pushapkscript cd pushapkscript pip install pushapkscript
Then you need to install jarsigner (usually included with JDK).
Add the nightly certificate to the java keystore:
keytool -import -keystore ~/.keystores/mozilla-android -file pushapkscript/data/android-nightly.cer -alias nightly
Note: The keystore location and the certificate alias will be used in the
cp examples/config.example.json config.json # edit it with your favorite text editor
There are many values to edit. Example values should give you a hint about what to provide. If not, please see signingscript's README for more details about allowing URLs, or contact the author for other unclear areas.
directories and file naming
If you aren't running through scriptworker, you need to manually create the directories that
artifact_dir point to. It's better to use new directories for these rather than cluttering and potentially overwriting an existing directory. Once you set up scriptworker, the
artifact_dir will be regularly wiped and recreated.
cp examples/task.example.json /path/to/work_dir # edit it with your favorite text editor
Ordinarily, scriptworker would get the task definition from TaskCluster, and write it to a
task.json in the
work_dir. Since you're initially not going to run through scriptworker, you need to put this file on disk yourself.
The important entries to edit are the:
apks: point to the file(s) to publish to Google Play
dependencies: need to match the
taskIds of the URLs unless you modify the
valid_artifact_*config items as specified above
scopes: the first and only scope,
project:releng:googleplay:*, tells which product in Google Play store should be updated (either aurora, beta, or release)
google_play_track: refers to which Google Play track (either production, beta, or alpha) the APK will be uploaded
(aurora, beta, release) vs (alpha, beta, production)?
Google Play allows a product to have 3 different tracks (
production). Tracks are used by end-users when they want to enroll in a beta-testing program.
However, this feature wasn't out when we started publishing Fennec. This is why Fennec is registred as 3 different product: one for each regular Firefox channel (aurora, beta, release). As a consequence, here's how products/tracks should be used.
|aurora||Firefox Aurora for Developers||
||production is not used to show the product is not aimed at regular users|
Note: For development purpose, aurora on the
alpha track can also be used.
You're ready to run pushapkscript!
CONFIG_FILE is the config json you created above.
This should download the file(s) specified in the payload, check their signatures with jarsigner and publish them to Google Play Store.
running through scriptworker
Follow the scriptworker readme to set up scriptworker, and use
["path/to/pushapkscript", "path/to/script_config.json"] as your
artifact_dir point to the same directories between the scriptworker config and the pushapkscript config!
README.md is the master readme, and
README.rst is generated via
pandoc --from=markdown --to=rst README.md > README.rst
This is purely because
- @escapewindow prefers writing markdown, and
- pypi appears to deal with rst better than markdown.
Frequently asked questions
I'd like to test out changes in pushapkscript...
Do I need to activate chain of trust for local development?
No. Chain of trust is used to securely download artifacts. You can bypass that step by having artifacts already on-disk. Just put the APKs in:
$work_dir/cot/$task_id/public/build/target.apk (each APK has a different task_id). Then, you can run pushapkscript.
Is there a staging instance I can push my code to?
There used to be one, but it's now decommissioned. You can spawn a new instance via puppet. To do so:
- Create a new VM instance. You can ask for a loaner.
- On the puppet master node, set up a user environment.
- Add a new node to moco-nodes.pp. The config example is present in this repo at
- Activate chain of trust by creating the gpg keys and whitelisting them. Otherwise, artifacts won't be downloaded.
- Edit your tasks to point to the dev worker group.
- On your VM, make the slave take the config of your user environment.
"dry_run": true in your task definitions (or don't define it).
I'd like to test out Taskcluster tasks...
Is there an instance which doesn't interact with production data?
Sadly, no. The Google Play documentation doesn't mention any server we can plug to. This means, you will interact with production data. There are ways to mitigate the risk, though.
How can I avoid to publish to actual users?
There are 3 incremental ways to avoid targetting real users (or the entire user base):
"dry_run": true in your task definition.
This will execute every step implemented in pushapkscript, but the last one, which commits the transaction to the Play store.
This allows to publish the same APK several times.
However, there are a few final checks that Google Play does only when the transaction is committed. We have already experienced one: the integrity of l10n stores (descriptions and "what's new" sections) is verified only at this time. We may extrapolate the behavior to: everything that can be done in several calls to Google Play will be checked at commit time.
2. Push to a closed alpha track
At some point, you may want to publish your APK anyway.
You can ask release management to set up a closed alpha testing on the Google Play console (Go to Release management -> App releases -> Manage Alpha) and target users by email address. Then, edit your task definition to contain
Before going further: tear down the alpha track
HttpError 403 when requesting https://www.googleapis.com/androidpublisher/v2/applications/org.mozilla.fennec_aurora/edits/17791185193608549142:commit?alt=json returned "Version 2015491409 of this app can not be downloaded by any devices as they will all receive APKs with higher version codes.
In the case where Release Management isn't connect, there's a temporary workaround:
- Publish the APKs on the alpha track.
- On the Google Play dashboard, promote the APKs to the beta (then the rollout channel). Reuploading the same APKs to the beta track won't be accepted by Google Play, because APKs can only be pushed once.
3. For non-aurora products: Push to the rollout track
If you are confident enough to publish to percentage of our user base, you can use the rollout track. Just edit your task definition to contain:
"google_play_track": "rollout", "rollout_percentage": 10,
if you want to target 10% of the production user base.