from __future__ import print_function import os import os.path import sys import subprocess import platform # # Script Compatibility: # - Python 2.7.15 # - Python 3.7.0 # SUPPORTED_PYTHON_PLATFORMS = ['Windows', 'Linux', 'Darwin'] # Leave empty to auto-detect version by 'git describe'. FORCE_DISPLAY_SYNCTHING_VERSION = '' GO_VERSION = '1.12.1' GO_EXPECTED_SHASUM_LINUX = '2a3fdabf665496a0db5f41ec6af7a9b15a49fbe71a85a50ca38b1f13a103aeec' GO_EXPECTED_SHASUM_WINDOWS = '2f4849b512fffb2cf2028608aa066cc1b79e730fd146c7b89015797162f08ec5' NDK_VERSION = 'r19c' NDK_EXPECTED_SHASUM_LINUX = 'fd94d0be6017c6acbd193eb95e09cf4b6f61b834' NDK_EXPECTED_SHASUM_WINDOWS = 'c4cd8c0b6e7618ca0a871a5f24102e40c239f6a3' BUILD_TARGETS = [ { 'arch': 'arm', 'goarch': 'arm', 'jni_dir': 'armeabi', 'clang': 'armv7a-linux-androideabi16-clang', 'patch_underaligned_tls': 'yes', }, { 'arch': 'arm64', 'goarch': 'arm64', 'jni_dir': 'arm64-v8a', 'clang': 'aarch64-linux-android21-clang', 'patch_underaligned_tls': 'yes', }, { 'arch': 'x86', 'goarch': '386', 'jni_dir': 'x86', 'clang': 'i686-linux-android16-clang', } ] def fail(message, *args, **kwargs): print((message % args).format(**kwargs)) sys.exit(1) def get_min_sdk(project_dir): with open(os.path.join(project_dir, 'app', 'build.gradle')) as file_handle: for line in file_handle: tokens = list(filter(None, line.split())) if len(tokens) == 2 and tokens[0] == 'minSdkVersion': return int(tokens[1]) fail('Failed to find minSdkVersion') def which(program): import os def is_exe(fpath): return os.path.isfile(fpath) and os.access(fpath, os.X_OK) if (sys.platform == 'win32'): program += ".exe" fpath, fname = os.path.split(program) if fpath: if is_exe(program): return program else: for path in os.environ["PATH"].split(os.pathsep): exe_file = os.path.join(path, program) if is_exe(exe_file): return exe_file return None def change_permissions_recursive(path, mode): import os for root, dirs, files in os.walk(path, topdown=False): for dir in [os.path.join(root,d) for d in dirs]: os.chmod(dir, mode) for file in [os.path.join(root, f) for f in files]: os.chmod(file, mode) def install_git(): import os import zipfile import hashlib if sys.version_info[0] >= 3: from urllib.request import urlretrieve else: from urllib import urlretrieve # Consts. pwd_path = os.path.dirname(os.path.realpath(__file__)) if sys.platform == 'win32': url = 'https://github.com/git-for-windows/git/releases/download/v2.19.0.windows.1/MinGit-2.19.0-64-bit.zip' expected_shasum = '424d24b5fc185a9c5488d7872262464f2facab4f1d4693ea8008196f14a3c19b' zip_fullfn = pwd_path + os.path.sep + 'mingit.zip'; else: print('Portable on-demand git installation is currently not supported on linux.') return None # Download MinGit. url_base_name = os.path.basename(url) if not os.path.isfile(zip_fullfn): print('Downloading MinGit to:', zip_fullfn) zip_fullfn = urlretrieve(url, zip_fullfn)[0] print('Downloaded MinGit to:', zip_fullfn) # Verfiy SHA-256 checksum of downloaded files. with open(zip_fullfn, 'rb') as f: contents = f.read() found_shasum = hashlib.sha256(contents).hexdigest() print("SHA-256:", zip_fullfn, "%s" % found_shasum) if found_shasum != expected_shasum: fail('Error: SHA-256 checksum ' + found_shasum + ' of downloaded file does not match expected checksum ' + expected_shasum) print("[ok] Checksum of", zip_fullfn, "matches expected value.") # Proceed with extraction of the MinGit. if not os.path.isfile(pwd_path + os.path.sep + 'mingit' + os.path.sep + 'LICENSE.txt'): print("Extracting MinGit ...") # This will go to a subfolder "mingit" in the current path. zip = zipfile.ZipFile(zip_fullfn, 'r') zip.extractall(pwd_path + os.path.sep + 'mingit') zip.close() # Add "mingit/cmd" to the PATH. git_bin_path = pwd_path + os.path.sep + 'mingit' + os.path.sep + 'cmd' print('Adding to PATH:', git_bin_path) os.environ["PATH"] += os.pathsep + git_bin_path def install_go(): import os import tarfile import zipfile import hashlib if sys.version_info[0] >= 3: from urllib.request import urlretrieve else: from urllib import urlretrieve # Consts. pwd_path = os.path.dirname(os.path.realpath(__file__)) if sys.platform == 'win32': url = 'https://dl.google.com/go/go' + GO_VERSION + '.windows-amd64.zip' expected_shasum = GO_EXPECTED_SHASUM_WINDOWS tar_gz_fullfn = pwd_path + os.path.sep + 'go.zip'; else: url = 'https://dl.google.com/go/go' + GO_VERSION + '.linux-amd64.tar.gz' expected_shasum = GO_EXPECTED_SHASUM_LINUX tar_gz_fullfn = pwd_path + os.path.sep + 'go.tgz'; # Download prebuilt-go. url_base_name = os.path.basename(url) if not os.path.isfile(tar_gz_fullfn): print('Downloading prebuilt-go to:', tar_gz_fullfn) tar_gz_fullfn = urlretrieve(url, tar_gz_fullfn)[0] print('Downloaded prebuilt-go to:', tar_gz_fullfn) # Verfiy SHA-256 checksum of downloaded files. with open(tar_gz_fullfn, 'rb') as f: contents = f.read() found_shasum = hashlib.sha256(contents).hexdigest() print("SHA-256:", tar_gz_fullfn, "%s" % found_shasum) if found_shasum != expected_shasum: fail('Error: SHA-256 checksum ' + found_shasum + ' of downloaded file does not match expected checksum ' + expected_shasum) print("[ok] Checksum of", tar_gz_fullfn, "matches expected value.") # Proceed with extraction of the prebuilt go. if not os.path.isfile(pwd_path + os.path.sep + 'go' + os.path.sep + 'LICENSE'): print("Extracting prebuilt-go ...") # This will go to a subfolder "go" in the current path. file_name, file_extension = os.path.splitext(url_base_name) if sys.platform == 'win32': zip = zipfile.ZipFile(tar_gz_fullfn, 'r') zip.extractall(pwd_path) zip.close() else: tar = tarfile.open(tar_gz_fullfn) tar.extractall(pwd_path) tar.close() # Add "go/bin" to the PATH. go_bin_path = pwd_path + os.path.sep + 'go' + os.path.sep + 'bin' print('Adding to PATH:', go_bin_path) os.environ["PATH"] += os.pathsep + go_bin_path def install_ndk(): import os import zipfile import hashlib if sys.version_info[0] >= 3: from urllib.request import urlretrieve else: from urllib import urlretrieve # Consts. pwd_path = os.path.dirname(os.path.realpath(__file__)) if sys.platform == 'win32': url = 'https://dl.google.com/android/repository/android-ndk-' + NDK_VERSION + '-windows-x86_64.zip' expected_shasum = NDK_EXPECTED_SHASUM_WINDOWS else: url = 'https://dl.google.com/android/repository/android-ndk-' + NDK_VERSION + '-linux-x86_64.zip' expected_shasum = NDK_EXPECTED_SHASUM_LINUX zip_fullfn = pwd_path + os.path.sep + 'ndk.zip'; # Download NDK. url_base_name = os.path.basename(url) if not os.path.isfile(zip_fullfn): print('Downloading NDK to:', zip_fullfn) zip_fullfn = urlretrieve(url, zip_fullfn)[0] print('Downloaded NDK to:', zip_fullfn) # Verfiy SHA-1 checksum of downloaded files. with open(zip_fullfn, 'rb') as f: contents = f.read() found_shasum = hashlib.sha1(contents).hexdigest() print("SHA-1:", zip_fullfn, "%s" % found_shasum) if found_shasum != expected_shasum: fail('Error: SHA-256 checksum ' + found_shasum + ' of downloaded file does not match expected checksum ' + expected_shasum) print("[ok] Checksum of", zip_fullfn, "matches expected value.") # Proceed with extraction of the NDK if necessary. ndk_home_path = pwd_path + os.path.sep + 'android-ndk-' + NDK_VERSION if not os.path.isfile(ndk_home_path + os.path.sep + "sysroot" + os.path.sep + "NOTICE"): print("Extracting NDK ...") # This will go to a subfolder "android-ndk-r18" in the current path. file_name, file_extension = os.path.splitext(url_base_name) zip = zipfile.ZipFile(zip_fullfn, 'r') zip.extractall(pwd_path) zip.close() # Linux only - Set executable permission on files. if platform.system() == 'Linux': print("Setting permissions on NDK executables ...") change_permissions_recursive(ndk_home_path, 0o755); # Add "ANDROID_NDK_HOME" environment variable. print('Adding ANDROID_NDK_HOME=\'' + ndk_home_path + '\'') os.environ["ANDROID_NDK_HOME"] = ndk_home_path def artifact_patch_underaligned_tls(artifact_fullfn): import struct with open(artifact_fullfn, 'r+b') as f: f.seek(0) hdr = struct.unpack('16c', f.read(16)) if hdr[0] != b'\x7f' or hdr[1] != b'E' or hdr[2] != b'L' or hdr[3] != b'F': print('artifact_patch_underaligned_tls: Not an ELF file') return None if hdr[4] == b'\x01': # 32 bit code f.seek(28) offset = struct.unpack('