You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-android/scripts/build-and-release.py

289 lines
12 KiB
Python

#!/usr/bin/env python3
import subprocess
import json
import os
import sys
import shutil
import re
import tomllib
from dataclasses import dataclass
import tempfile
import base64
import string
import glob
# Number of versions to keep in the fdroid repo. Will remove all the older versions.
KEEP_FDROID_VERSIONS = 4
@dataclass
class BuildResult:
max_version_code: int
version_name: str
apk_paths: list[str]
bundle_path: str
package_id: str
@dataclass
class BuildCredentials:
keystore_b64: str
keystore_password: str
key_alias: str
key_password: str
def __init__(self, credentials: dict):
self.keystore_b64 = credentials['keystore'].strip()
self.keystore_password = credentials['keystore_password']
self.key_alias = credentials['key_alias']
self.key_password = credentials['key_password']
def build_releases(project_root: str, flavor: str, credentials_property_prefix: str, credentials: BuildCredentials, huawei: bool=False) -> BuildResult:
(keystore_fd, keystore_file) = tempfile.mkstemp(prefix='keystore_', suffix='.jks', dir=build_dir)
try:
with os.fdopen(keystore_fd, 'wb') as f:
f.write(base64.b64decode(credentials.keystore_b64))
gradle_commands = f"""./gradlew \
-P{credentials_property_prefix}_STORE_FILE='{keystore_file}'\
-P{credentials_property_prefix}_STORE_PASSWORD='{credentials.keystore_password}' \
-P{credentials_property_prefix}_KEY_ALIAS='{credentials.key_alias}' \
-P{credentials_property_prefix}_KEY_PASSWORD='{credentials.key_password}'"""
if huawei:
gradle_commands += ' -Phuawei '
subprocess.run(f"""{gradle_commands} \
assemble{flavor.capitalize()}Release \
bundle{flavor.capitalize()}Release --stacktrace""", shell=True, check=True, cwd=project_root)
apk_output_dir = os.path.join(project_root, f'app/build/outputs/apk/{flavor}/release')
with open(os.path.join(apk_output_dir, 'output-metadata.json')) as f:
play_outputs = json.load(f)
apks = [os.path.join(apk_output_dir, f['outputFile']) for f in play_outputs['elements']]
max_version_code = max(map(lambda element: element['versionCode'], play_outputs['elements']))
package_id = play_outputs['applicationId']
version_name = play_outputs['elements'][0]['versionName']
print('Max version code is: ', max_version_code)
return BuildResult(max_version_code=max_version_code,
apk_paths=apks,
package_id=package_id,
version_name=version_name,
bundle_path=os.path.join(project_root, f'app/build/outputs/bundle/{flavor}Release/session-{version_name}-{flavor}-release.aab'))
finally:
print(f'Cleaning up keystore file: {keystore_file}')
os.remove(keystore_file)
project_root = os.path.dirname(sys.path[0])
build_dir = os.path.join(project_root, 'build')
credentials_file_path = os.path.join(project_root, 'release-creds.toml')
fdroid_repo_path = os.path.join(build_dir, 'fdroidrepo')
def detect_android_sdk() -> str:
sdk_dir = os.environ.get('ANDROID_HOME')
if sdk_dir is None:
with open(os.path.join(project_root, 'local.properties')) as f:
matched = next(re.finditer(r'^sdk.dir=(.+?)$', f.read(), re.MULTILINE), None)
sdk_dir = matched.group(1) if matched else None
if sdk_dir is None or not os.path.isdir(sdk_dir):
raise Exception('Android SDK not found. Please set ANDROID_HOME or add sdk.dir to local.properties')
return sdk_dir
def update_fdroid(build: BuildResult, fdroid_workspace: str, creds: BuildCredentials):
# Check if there's a git repo at the fdroid repo path by running git status
try:
subprocess.check_call(f'git -C {fdroid_repo_path} status', shell=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
subprocess.check_call(f'git fetch --depth=1', shell=True, cwd=fdroid_workspace)
print(f'Found fdroid git repo at {fdroid_repo_path}')
except subprocess.CalledProcessError:
print(f'No fdroid git repo found at {fdroid_repo_path}. Cloning using gh.')
subprocess.run(f'gh repo clone session-foundation/session-fdroid {fdroid_repo_path} -- -b master --depth=1', shell=True, check=True)
# Create a branch for the release
print(f'Creating a branch for the fdroid release: {build.version_name}')
try:
branch_name = f'release/{build.version_name}'
# Clean and switch to master before doing anything
subprocess.check_call(f'git reset --hard HEAD && git checkout master', shell=True, cwd=fdroid_workspace, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
# Delete the existing local branch regardlessly
subprocess.run(f'git branch -D {branch_name}', check=False, shell=True, cwd=fdroid_workspace)
# Check if the remote branch already exists, or we need to create a new one
try:
subprocess.check_call(f'git ls-remote --exit-code origin refs/heads/{branch_name}', shell=True, cwd=fdroid_workspace, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
print(f'Branch {branch_name} already exists. Checking out...')
subprocess.check_call(f'git checkout {branch_name}', shell=True, cwd=fdroid_workspace)
except subprocess.CalledProcessError:
print(f'Branch {branch_name} not found. Creating a new branch.')
subprocess.check_call(f'git checkout -b {branch_name} origin/master', shell=True, cwd=fdroid_workspace)
except subprocess.CalledProcessError:
print(f'Failed to create a branch for the release. ')
sys.exit(1)
# Copy the apks to the fdroid repo
for apk in build.apk_paths:
if apk.endswith('-universal.apk'):
print('Skipping universal apk:', apk)
continue
dst = os.path.join(fdroid_workspace, 'repo/' + os.path.basename(apk))
print('Copying', apk, 'to', dst)
shutil.copy(apk, dst)
# Make sure there are only last three versions of APKs
all_apk_versions_and_ctime = [(re.search(r'session-(.+?)-', os.path.basename(name)).group(1), os.path.getmtime(name))
for name in glob.glob(os.path.join(fdroid_workspace, 'repo/session-*-arm64-v8a.apk'))]
# Sort by ctime DESC
all_apk_versions_and_ctime.sort(key=lambda x: x[0], reverse=True)
# Remove all but the last three versions
for version, _ in all_apk_versions_and_ctime[KEEP_FDROID_VERSIONS:]:
for apk in glob.glob(os.path.join(fdroid_workspace, f'repo/session-{version}-*.apk')):
print('Removing old apk:', apk)
os.remove(apk)
# Update the metadata file
metadata_file = os.path.join(fdroid_workspace, f'metadata/{build.package_id}.yml')
with open(f'{metadata_file}.tpl', 'r') as template_file:
metadata_template = string.Template(template_file.read())
metadata_contents = metadata_template.substitute({
'currentVersionCode': build.max_version_code,
})
with open(metadata_file, 'w') as file:
file.write(metadata_contents)
[keystore_fd, keystore_path] = tempfile.mkstemp(prefix='fdroid_keystore_', suffix='.p12', dir=build_dir)
config_file_path = os.path.join(fdroid_workspace, 'config.yml')
try:
android_sdk = detect_android_sdk()
with os.fdopen(keystore_fd, 'wb') as f:
f.write(base64.b64decode(creds.keystore_b64))
# Read the config template and create a config file
with open(f'{config_file_path}.tpl') as config_template_file:
config_template = string.Template(config_template_file.read())
with open(config_file_path, 'w') as f:
f.write(config_template.substitute({
'keystore_file': keystore_path,
'keystore_pass': creds.keystore_password,
'repo_keyalias': creds.key_alias,
'key_pass': creds.key_password,
'android_sdk': android_sdk
}))
# Run fdroid update
print("Running fdroid update...")
environs = os.environ.copy()
subprocess.run('fdroid update', shell=True, check=True, cwd=fdroid_workspace, env=environs)
finally:
print(f'Cleaning up...')
if os.path.exists(metadata_file):
os.remove(metadata_file)
if os.path.exists(keystore_path):
os.remove(keystore_path)
if os.path.exists(config_file_path):
os.remove(config_file_path)
# Commit the changes
print('Committing the changes...')
subprocess.run(f'git add . && git commit -am "Prepare for release {build.version_name}"', shell=True, check=True, cwd=fdroid_workspace)
# Create Pull Request for releases
print('Creating a pull request...')
subprocess.run(f'''\
gh pr create --base master \
--title "Release {build.version_name}" \
-R session-foundation/session-fdroid \
--body "This is an automated release preparation for Release {build.version_name}. Human beings are still required to approve and merge this PR."\
''', shell=True, check=True, cwd=fdroid_workspace)
# Make sure gh command is available
if shutil.which('gh') is None:
print('`gh` command not found. It is required to automate fdroid releases. Please install it from https://cli.github.com/', file=sys.stderr)
sys.exit(1)
# Make sure fdroid command is available
if shutil.which('fdroid') is None:
print('`fdroid` command not found. It is required to automate fdroid releases. Please install it from https://f-droid.org/', file=sys.stderr)
sys.exit(1)
# Make sure credentials file exists
if not os.path.isfile(credentials_file_path):
print(f'Credentials file not found at {credentials_file_path}. You should ask the project maintainer for the file.', file=sys.stderr)
sys.exit(1)
with open(credentials_file_path, 'rb') as f:
credentials = tomllib.load(f)
# Make sure build folder exists
if not os.path.isdir(build_dir):
os.makedirs(build_dir)
print("Building play releases...")
play_build_result = build_releases(
project_root=project_root,
flavor='play',
credentials=BuildCredentials(credentials['build']['play']),
credentials_property_prefix='SESSION'
)
print("Updating fdroid repo...")
update_fdroid(build=play_build_result, creds=BuildCredentials(credentials['fdroid']), fdroid_workspace=os.path.join(fdroid_repo_path, 'fdroid'))
print("Building huawei releases...")
huawei_build_result = build_releases(
project_root=project_root,
flavor='huawei',
credentials=BuildCredentials(credentials['build']['huawei']),
credentials_property_prefix='SESSION_HUAWEI',
huawei=True
)
# If the a github release draft exists, upload the apks to the release
try:
release_info = json.loads(subprocess.check_output(f'gh release view --json isDraft {play_build_result.version_name}', shell=True, cwd=project_root))
if release_info['isDraft'] == True:
print(f'Uploading build artifact to the release {play_build_result.version_name} draft...')
files_to_upload = [*play_build_result.apk_paths,
play_build_result.bundle_path,
*huawei_build_result.apk_paths]
upload_commands = ['gh', 'release', 'upload', play_build_result.version_name, '--clobber', *files_to_upload]
subprocess.run(upload_commands, shell=False, cwd=project_root, check=True)
print('Successfully uploaded these files to the draft release: ')
for file in files_to_upload:
print(file)
else:
print(f'Release {play_build_result.version_name} not a draft. Skipping upload of apks to the release.')
except subprocess.CalledProcessError:
print(f'{play_build_result.version_name} has not had a release draft created. Skipping upload of apks to the release.')
print('\n=====================')
print('Build result: ')
print('Play:')
for apk in play_build_result.apk_paths:
print(f'\t{apk}')
print(f'\t{play_build_result.bundle_path}')
print('Huawei:')
for apk in huawei_build_result.apk_paths:
print(f'\t{apk}')
print('=====================')