| #!/usr/bin/env python3 |
| # Copyright (c) 2019, the R8 project authors. Please see the AUTHORS file |
| # for details. All rights reserved. Use of this source code is governed by a |
| # BSD-style license that can be found in the LICENSE file. |
| |
| # This script is designed to run on a buildbot to build from the source |
| # of https://github.com/google/desugar_jdk_libs and publish to the |
| # r8-release Cloud Storage Bucket. |
| # |
| # These files are uploaded: |
| # |
| # raw/desugar_jdk_libs/<VERSION>/desugar_jdk_libs.jar |
| # raw/desugar_jdk_libs/<VERSION>/desugar_jdk_libs.zip |
| # raw/com/android/tools/desugar_jdk_libs/<VERSION>/desugar_jdk_libs-<VERSION>.jar |
| # |
| # The first two are the raw jar file and the maven compatible zip file. The |
| # third is the raw jar file placed and named so that the URL |
| # https://storage.googleapis.com/r8-releases/raw can be treated as a maven |
| # repository to fetch the artifact com.android.tools:desugar_jdk_libs:1.0.0 |
| |
| import archive |
| import defines |
| import git_utils |
| import gradle |
| import hashlib |
| import jdk |
| import optparse |
| import os |
| import re |
| import shutil |
| import subprocess |
| import sys |
| import utils |
| import zipfile |
| |
| VERSION_FILE_JDK8 = 'VERSION.txt' |
| VERSION_FILE_JDK11_LEGACY = 'VERSION_JDK11_LEGACY.txt' |
| VERSION_FILE_JDK11_MINIMAL = 'VERSION_JDK11_MINIMAL.txt' |
| VERSION_FILE_JDK11 = 'VERSION_JDK11.txt' |
| VERSION_FILE_JDK11_NIO = 'VERSION_JDK11_NIO.txt' |
| |
| VERSION_MAP = { |
| 'jdk8': VERSION_FILE_JDK8, |
| 'jdk11_legacy': VERSION_FILE_JDK11_LEGACY, |
| 'jdk11_minimal': VERSION_FILE_JDK11_MINIMAL, |
| 'jdk11': VERSION_FILE_JDK11, |
| 'jdk11_nio': VERSION_FILE_JDK11_NIO |
| } |
| |
| GITHUB_REPRO = 'desugar_jdk_libs' |
| |
| BASE_LIBRARY_NAME = 'desugar_jdk_libs' |
| |
| LIBRARY_NAME_MAP = { |
| 'jdk8': BASE_LIBRARY_NAME, |
| 'jdk11_legacy': BASE_LIBRARY_NAME, |
| 'jdk11_minimal': BASE_LIBRARY_NAME + '_minimal', |
| 'jdk11': BASE_LIBRARY_NAME, |
| 'jdk11_nio': BASE_LIBRARY_NAME + '_nio' |
| } |
| |
| MAVEN_RELEASE_TARGET_MAP = { |
| 'jdk8': 'maven_release', |
| 'jdk11_legacy': 'maven_release_jdk11_legacy', |
| 'jdk11_minimal': 'maven_release_jdk11_minimal', |
| 'jdk11': 'maven_release_jdk11', |
| 'jdk11_nio': 'maven_release_jdk11_nio' |
| } |
| |
| MAVEN_RELEASE_ZIP = { |
| 'jdk8': BASE_LIBRARY_NAME + '.zip', |
| 'jdk11_legacy': BASE_LIBRARY_NAME + '_jdk11_legacy.zip', |
| 'jdk11_minimal': BASE_LIBRARY_NAME + '_jdk11_minimal.zip', |
| 'jdk11': BASE_LIBRARY_NAME + '_jdk11.zip', |
| 'jdk11_nio': BASE_LIBRARY_NAME + '_jdk11_nio.zip' |
| } |
| |
| DESUGAR_JDK_LIBS_HASH_FILE = os.path.join( |
| defines.THIRD_PARTY, 'openjdk', 'desugar_jdk_libs_11', 'desugar_jdk_libs_hash') |
| |
| |
| def ParseOptions(argv): |
| result = optparse.OptionParser() |
| result.add_option('--variant', |
| help="Variant(s) to build", |
| metavar=('<variants(s)>'), |
| choices=['jdk8', 'jdk11_legacy', 'jdk11_minimal', 'jdk11', 'jdk11_nio'], |
| default=[], |
| action='append') |
| result.add_option('--dry-run', '--dry_run', |
| help='Running on bot, use third_party dependency.', |
| default=False, |
| action='store_true') |
| result.add_option('--dry-run-output', '--dry_run_output', |
| help='Output directory for dry run.', |
| type="string", action="store") |
| result.add_option('--github-account', '--github_account', |
| help='GitHub account to clone from.', |
| default="google", |
| type="string", action="store") |
| result.add_option('--build_only', '--build-only', |
| help='Build desugared library without archiving.', |
| type="string", action="store") |
| (options, args) = result.parse_args(argv) |
| return (options, args) |
| |
| |
| def GetVersion(version_file_name): |
| with open(version_file_name, 'r') as version_file: |
| lines = [line.strip() for line in version_file.readlines()] |
| lines = [line for line in lines if not line.startswith('#')] |
| if len(lines) != 1: |
| raise Exception('Version file ' |
| + version_file + ' is expected to have exactly one line') |
| version = lines[0].strip() |
| utils.check_basic_semver_version( |
| version, 'in version file ' + version_file_name, allowPrerelease = True) |
| return version |
| |
| |
| def Upload(options, file_name, storage_path, destination, is_main): |
| print('Uploading %s to %s' % (file_name, destination)) |
| if options.dry_run: |
| if options.dry_run_output: |
| dry_run_destination = \ |
| os.path.join(options.dry_run_output, os.path.basename(file_name)) |
| print('Dry run, not actually uploading. Copying to ' |
| + dry_run_destination) |
| shutil.copyfile(file_name, dry_run_destination) |
| else: |
| print('Dry run, not actually uploading') |
| else: |
| utils.upload_file_to_cloud_storage(file_name, destination) |
| print('File available at: %s' % |
| destination.replace('gs://', 'https://storage.googleapis.com/', 1)) |
| |
| def CloneDesugaredLibrary(github_account, checkout_dir, desugar_jdk_libs_hash): |
| git_utils.GitClone( |
| 'https://github.com/' |
| + github_account + '/' + GITHUB_REPRO, checkout_dir) |
| git_utils.GitCheckout(desugar_jdk_libs_hash, checkout_dir) |
| |
| def GetJavaEnv(androidHomeTemp): |
| java_env = dict(os.environ, JAVA_HOME = jdk.GetJdk11Home()) |
| java_env['PATH'] = java_env['PATH'] + os.pathsep + os.path.join(jdk.GetJdk11Home(), 'bin') |
| java_env['GRADLE_OPTS'] = '-Xmx1g' |
| java_env['ANDROID_HOME'] = androidHomeTemp |
| return java_env |
| |
| def setUpFakeAndroidHome(androidHomeTemp): |
| # Bazel will check if 30 is present then extract android.jar from 32. |
| # We copy android.jar from third_party to mimic repository structure. |
| subpath = os.path.join(androidHomeTemp, "platforms") |
| cmd = ["mkdir", subpath] |
| subprocess.check_call(cmd) |
| subpath30 = os.path.join(subpath, "android-30") |
| cmd = ["mkdir", subpath30] |
| subprocess.check_call(cmd) |
| subpath = os.path.join(subpath, "android-32") |
| cmd = ["mkdir", subpath] |
| subprocess.check_call(cmd) |
| dest = os.path.join(subpath, "android.jar") |
| sha = os.path.join(utils.THIRD_PARTY, "android_jar", "lib-v32.tar.gz.sha1") |
| utils.DownloadFromGoogleCloudStorage(sha) |
| src = os.path.join(utils.THIRD_PARTY, "android_jar", "lib-v32", "android.jar") |
| cmd = ["cp", src, dest] |
| subprocess.check_call(cmd) |
| |
| def BuildDesugaredLibrary(checkout_dir, variant, version = None): |
| if not variant in MAVEN_RELEASE_TARGET_MAP: |
| raise Exception('Variant ' + variant + ' is not supported') |
| if variant != 'jdk8' and variant != 'jdk11_legacy' and version is None: |
| raise Exception('Variant ' + variant + ' require version for undesugaring') |
| if variant != 'jdk8': |
| # Hack to workaround b/256723819. |
| os.remove( |
| os.path.join( |
| checkout_dir, |
| "jdk11", |
| "src", |
| "java.base", |
| "share", |
| "classes", |
| "java", |
| "time", |
| "format", |
| "DesugarDateTimeFormatterBuilder.java")) |
| with utils.ChangedWorkingDirectory(checkout_dir): |
| with utils.TempDir() as androidHomeTemp: |
| setUpFakeAndroidHome(androidHomeTemp) |
| javaEnv = GetJavaEnv(androidHomeTemp) |
| bazel = os.path.join(utils.BAZEL_TOOL, 'lib', 'bazel', 'bin', 'bazel') |
| cmd = [ |
| bazel, |
| '--bazelrc=/dev/null', |
| 'build', |
| '--spawn_strategy=local', |
| '--verbose_failures', |
| MAVEN_RELEASE_TARGET_MAP[variant]] |
| utils.PrintCmd(cmd) |
| subprocess.check_call(cmd, env=javaEnv) |
| cmd = [bazel, 'shutdown'] |
| utils.PrintCmd(cmd) |
| subprocess.check_call(cmd, env=javaEnv) |
| |
| # Locate the library jar and the maven zip with the jar from the |
| # bazel build. |
| if variant == 'jdk8': |
| library_jar = os.path.join( |
| checkout_dir, 'bazel-bin', 'src', 'share', 'classes', 'java', 'libjava.jar') |
| else: |
| # All JDK11 variants use the same library code. |
| library_jar = os.path.join( |
| checkout_dir, 'bazel-bin', 'jdk11', 'src', 'd8_java_base_selected_with_addon.jar') |
| maven_zip = os.path.join( |
| checkout_dir, |
| 'bazel-bin', |
| MAVEN_RELEASE_ZIP[variant]) |
| |
| if variant != 'jdk8' and variant != 'jdk11_legacy': |
| # The undesugaring is temporary... |
| undesugared_maven_zip = os.path.join(checkout_dir, 'undesugared_maven') |
| Undesugar(variant, maven_zip, version, undesugared_maven_zip) |
| undesugared_maven_zip = os.path.join(checkout_dir, 'undesugared_maven.zip') |
| return (library_jar, undesugared_maven_zip) |
| else: |
| return (library_jar, maven_zip) |
| |
| def hash_for(file, hash): |
| with open(file, 'rb') as f: |
| while True: |
| # Read chunks of 1MB |
| chunk = f.read(2 ** 20) |
| if not chunk: |
| break |
| hash.update(chunk) |
| return hash.hexdigest() |
| |
| def write_md5_for(file): |
| hexdigest = hash_for(file, hashlib.md5()) |
| with (open(file + '.md5', 'w')) as file: |
| file.write(hexdigest) |
| |
| def write_sha1_for(file): |
| hexdigest = hash_for(file, hashlib.sha1()) |
| with (open(file + '.sha1', 'w')) as file: |
| file.write(hexdigest) |
| |
| def Undesugar(variant, maven_zip, version, undesugared_maven_zip): |
| gradle.RunGradle(['testJar', 'repackageTestDeps', '-Pno_internal']) |
| with utils.TempDir() as tmp: |
| with zipfile.ZipFile(maven_zip, 'r') as zip_ref: |
| zip_ref.extractall(tmp) |
| desugar_jdk_libs_jar = os.path.join( |
| tmp, |
| 'com', |
| 'android', |
| 'tools', |
| LIBRARY_NAME_MAP[variant], |
| version, |
| '%s-%s.jar' % (LIBRARY_NAME_MAP[variant], version)) |
| print(desugar_jdk_libs_jar) |
| undesugared_jar = os.path.join(tmp, 'undesugared.jar') |
| buildLibs = os.path.join(defines.REPO_ROOT, 'build', 'libs') |
| cmd = [jdk.GetJavaExecutable(), |
| '-cp', |
| '%s:%s:%s' % (os.path.join(buildLibs, 'r8_with_deps.jar'), os.path.join(buildLibs, 'r8tests.jar'), os.path.join(buildLibs, 'test_deps_all.jar')), |
| 'com.android.tools.r8.desugar.desugaredlibrary.jdk11.DesugaredLibraryJDK11Undesugarer', |
| desugar_jdk_libs_jar, |
| undesugared_jar] |
| print(cmd) |
| try: |
| output = subprocess.check_output(cmd, stderr = subprocess.STDOUT).decode('utf-8') |
| except subprocess.CalledProcessError as e: |
| print(e) |
| print(e.output) |
| raise e |
| print(output) |
| # Copy the undesugared jar into place and update the checksums. |
| shutil.copyfile(undesugared_jar, desugar_jdk_libs_jar) |
| write_md5_for(desugar_jdk_libs_jar) |
| write_sha1_for(desugar_jdk_libs_jar) |
| shutil.make_archive(undesugared_maven_zip, 'zip', tmp) |
| print(undesugared_maven_zip) |
| output = subprocess.check_output(['ls', '-l', os.path.dirname(undesugared_maven_zip)], stderr = subprocess.STDOUT).decode('utf-8') |
| print(output) |
| |
| def MustBeExistingDirectory(path): |
| if (not os.path.exists(path) or not os.path.isdir(path)): |
| raise Exception(path + ' does not exist or is not a directory') |
| |
| def BuildAndUpload(options, variant): |
| desugar_jdk_libs_hash = '' |
| with open(DESUGAR_JDK_LIBS_HASH_FILE, 'r') as input_hash: |
| desugar_jdk_libs_hash = input_hash.readline() |
| if options.build_only: |
| with utils.TempDir() as checkout_dir: |
| CloneDesugaredLibrary(options.github_account, checkout_dir, desugar_jdk_libs_hash) |
| (library_jar, maven_zip) = BuildDesugaredLibrary(checkout_dir, variant, desugar_jdk_libs_hash) |
| shutil.copyfile( |
| library_jar, |
| os.path.join(options.build_only, os.path.basename(library_jar))) |
| shutil.copyfile( |
| maven_zip, |
| os.path.join(options.build_only, os.path.basename(maven_zip))) |
| return |
| |
| # Only handling versioned desugar_jdk_libs. |
| is_main = False |
| |
| with utils.TempDir() as checkout_dir: |
| CloneDesugaredLibrary(options.github_account, checkout_dir, desugar_jdk_libs_hash) |
| version = GetVersion(os.path.join(checkout_dir, VERSION_MAP[variant])) |
| |
| destination = archive.GetVersionDestination( |
| 'gs://', LIBRARY_NAME_MAP[variant] + '/' + version, is_main) |
| if utils.cloud_storage_exists(destination) and not options.dry_run: |
| raise Exception( |
| 'Target archive directory %s already exists' % destination) |
| |
| (library_jar, maven_zip) = BuildDesugaredLibrary(checkout_dir, variant, version) |
| |
| storage_path = LIBRARY_NAME_MAP[variant] + '/' + version |
| # Upload the jar file with the library. |
| destination = archive.GetUploadDestination( |
| storage_path, LIBRARY_NAME_MAP[variant] + '.jar', is_main) |
| Upload(options, library_jar, storage_path, destination, is_main) |
| |
| # Upload the maven zip file with the library. |
| destination = archive.GetUploadDestination( |
| storage_path, MAVEN_RELEASE_ZIP[variant], is_main) |
| Upload(options, maven_zip, storage_path, destination, is_main) |
| |
| # Upload the jar file for accessing GCS as a maven repro. |
| maven_destination = archive.GetUploadDestination( |
| utils.get_maven_path(LIBRARY_NAME_MAP[variant], version), |
| '%s-%s.jar' % (LIBRARY_NAME_MAP[variant], version), |
| is_main) |
| if options.dry_run: |
| print('Dry run, not actually creating maven repo') |
| else: |
| utils.upload_file_to_cloud_storage(library_jar, maven_destination) |
| print('Maven repo root available at: %s' % archive.GetMavenUrl(is_main)) |
| |
| def Main(argv): |
| (options, args) = ParseOptions(argv) |
| if (len(args) > 0): |
| raise Exception('Unsupported arguments') |
| if not utils.is_bot() and not (options.dry_run or options.build_only): |
| raise Exception('You are not a bot, don\'t archive builds. ' |
| + 'Use --dry-run or --build-only to test locally') |
| if options.dry_run_output: |
| MustBeExistingDirectory(options.dry_run_output) |
| if options.build_only: |
| MustBeExistingDirectory(options.build_only) |
| if utils.is_bot(): |
| archive.SetRLimitToMax() |
| |
| # Make sure bazel is extracted in third_party. |
| utils.DownloadFromGoogleCloudStorage(utils.BAZEL_SHA_FILE) |
| utils.DownloadFromGoogleCloudStorage(utils.JAVA8_SHA_FILE) |
| utils.DownloadFromGoogleCloudStorage(utils.JAVA11_SHA_FILE) |
| utils.DownloadFromGoogleCloudStorage(utils.DESUGAR_JDK_LIBS_11_SHA_FILE) |
| |
| for v in options.variant: |
| BuildAndUpload(options, v) |
| |
| if __name__ == '__main__': |
| sys.exit(Main(sys.argv[1:])) |