# This Source Code Form is subject to the terms of the Mozilla Public # License, v. 2.0. If a copy of the MPL was not distributed with this # file, You can obtain one at http://mozilla.org/MPL/2.0/. """ Transform the beetmover task into an actual task description. """ from __future__ import absolute_import, print_function, unicode_literals from six import text_type from taskgraph.loader.multi_dep import schema from taskgraph.transforms.base import TransformSequence from taskgraph.transforms.beetmover import craft_release_properties from taskgraph.util.attributes import copy_attributes_from_dependent_job from taskgraph.util.partials import (get_balrog_platform_name, get_partials_artifacts_from_params, get_partials_info_from_params) from taskgraph.util.scriptworker import (generate_beetmover_artifact_map, generate_beetmover_upstream_artifacts, generate_beetmover_partials_artifact_map, get_beetmover_bucket_scope, get_beetmover_action_scope) from taskgraph.util.taskcluster import get_artifact_prefix from taskgraph.util.treeherder import replace_group, inherit_treeherder_from_dep from taskgraph.transforms.task import task_description_schema from voluptuous import Required, Optional import logging logger = logging.getLogger(__name__) beetmover_description_schema = schema.extend({ # depname is used in taskref's to identify the taskID of the unsigned things Required('depname', default='build'): text_type, # unique label to describe this beetmover task, defaults to {dep.label}-beetmover Required('label'): text_type, # treeherder is allowed here to override any defaults we use for beetmover. See # taskcluster/taskgraph/transforms/task.py for the schema details, and the # below transforms for defaults of various values. Optional('treeherder'): task_description_schema['treeherder'], Optional('attributes'): task_description_schema['attributes'], # locale is passed only for l10n beetmoving Optional('locale'): text_type, Required('shipping-phase'): task_description_schema['shipping-phase'], # Optional until we fix asan (run_on_projects?) Optional('shipping-product'): task_description_schema['shipping-product'], }) transforms = TransformSequence() transforms.add_validate(beetmover_description_schema) @transforms.add def make_task_description(config, jobs): for job in jobs: dep_job = job['primary-dependency'] attributes = dep_job.attributes treeherder = inherit_treeherder_from_dep(job, dep_job) upstream_symbol = dep_job.task['extra']['treeherder']['symbol'] if 'build' in job['dependent-tasks']: upstream_symbol = job['dependent-tasks']['build'].task['extra']['treeherder']['symbol'] treeherder.setdefault( 'symbol', replace_group(upstream_symbol, 'BMR') ) label = job['label'] description = ( "Beetmover submission for locale '{locale}' for build '" "{build_platform}/{build_type}'".format( locale=attributes.get('locale', 'en-US'), build_platform=attributes.get('build_platform'), build_type=attributes.get('build_type') ) ) upstream_deps = job['dependent-tasks'] signing_name = "build-signing" build_name = "build" repackage_name = "repackage" repackage_signing_name = "repackage-signing" msi_signing_name = "repackage-signing-msi" mar_signing_name = "mar-signing" if job.get('locale'): signing_name = "shippable-l10n-signing" build_name = "shippable-l10n" repackage_name = "repackage-l10n" repackage_signing_name = "repackage-signing-l10n" mar_signing_name = "mar-signing-l10n" dependencies = { "build": upstream_deps[build_name], "repackage": upstream_deps[repackage_name], "signing": upstream_deps[signing_name], "mar-signing": upstream_deps[mar_signing_name], } if 'partials-signing' in upstream_deps: dependencies['partials-signing'] = upstream_deps['partials-signing'] if msi_signing_name in upstream_deps: dependencies[msi_signing_name] = upstream_deps[msi_signing_name] if repackage_signing_name in upstream_deps: dependencies["repackage-signing"] = upstream_deps[repackage_signing_name] attributes = copy_attributes_from_dependent_job(dep_job) attributes.update(job.get('attributes', {})) if job.get('locale'): attributes['locale'] = job['locale'] bucket_scope = get_beetmover_bucket_scope(config) action_scope = get_beetmover_action_scope(config) task = { 'label': label, 'description': description, 'worker-type': 'beetmover', 'scopes': [bucket_scope, action_scope], 'dependencies': dependencies, 'attributes': attributes, 'run-on-projects': dep_job.attributes.get('run_on_projects'), 'treeherder': treeherder, 'shipping-phase': job['shipping-phase'], 'shipping-product': job.get('shipping-product'), } yield task def generate_partials_upstream_artifacts(job, artifacts, platform, locale=None): artifact_prefix = get_artifact_prefix(job) if locale and locale != 'en-US': artifact_prefix = '{}/{}'.format(artifact_prefix, locale) upstream_artifacts = [{ 'taskId': {'task-reference': ''}, 'taskType': 'signing', 'paths': ["{}/{}".format(artifact_prefix, path) for path, _ in artifacts], 'locale': locale or 'en-US', }] return upstream_artifacts @transforms.add def make_task_worker(config, jobs): for job in jobs: locale = job["attributes"].get("locale") platform = job["attributes"]["build_platform"] worker = { 'implementation': 'beetmover', 'release-properties': craft_release_properties(config, job), 'upstream-artifacts': generate_beetmover_upstream_artifacts( config, job, platform, locale ), 'artifact-map': generate_beetmover_artifact_map( config, job, platform=platform, locale=locale ) } if locale: worker["locale"] = locale job["worker"] = worker yield job @transforms.add def strip_unwanted_langpacks_from_worker(config, jobs): """ Strips out langpacks where we didn't sign them. This explicitly deletes langpacks from upstream artifacts and from artifact-maps. Due to limitations in declarative artifacts, doing this was our easiest way right now. """ ALWAYS_OK_PLATFORMS = {'linux64-shippable', 'linux64-devedition'} OSX_OK_PLATFORMS = {'macosx64-shippable', 'macosx64-devedition'} for job in jobs: platform = job['attributes'].get('build_platform') if platform in ALWAYS_OK_PLATFORMS: # No need to strip anything yield job continue for map in job['worker'].get('artifact-map', [])[:]: if not any([path.endswith('target.langpack.xpi') for path in map['paths']]): continue if map['locale'] == 'ja-JP-mac': # This locale should only exist on mac assert platform in OSX_OK_PLATFORMS continue # map[paths] is being modified while iterating, so we need to resolve the # ".keys()" iterator up front by throwing it into a list. for path in list(map['paths'].keys()): if path.endswith('target.langpack.xpi'): del map['paths'][path] if map['paths'] == {}: job['worker']['artifact-map'].remove(map) for artifact in job['worker'].get('upstream-artifacts', []): if not any([path.endswith('target.langpack.xpi') for path in artifact['paths']]): continue if artifact['locale'] == 'ja-JP-mac': # This locale should only exist on mac assert platform in OSX_OK_PLATFORMS continue artifact['paths'] = [path for path in artifact['paths'] if not path.endswith('target.langpack.xpi')] if artifact['paths'] == []: job['worker']['upstream-artifacts'].remove(artifact) yield job @transforms.add def make_partials_artifacts(config, jobs): for job in jobs: locale = job["attributes"].get("locale") if not locale: locale = 'en-US' platform = job["attributes"]["build_platform"] if 'partials-signing' not in job['dependencies']: yield job continue balrog_platform = get_balrog_platform_name(platform) artifacts = get_partials_artifacts_from_params(config.params.get('release_history'), balrog_platform, locale) upstream_artifacts = generate_partials_upstream_artifacts( job, artifacts, balrog_platform, locale ) job['worker']['upstream-artifacts'].extend(upstream_artifacts) extra = list() partials_info = get_partials_info_from_params( config.params.get('release_history'), balrog_platform, locale) job['worker']['artifact-map'].extend( generate_beetmover_partials_artifact_map( config, job, partials_info, platform=platform, locale=locale)) for artifact in partials_info: artifact_extra = { 'locale': locale, 'artifact_name': artifact, 'buildid': partials_info[artifact]['buildid'], 'platform': balrog_platform, } for rel_attr in ('previousBuildNumber', 'previousVersion'): if partials_info[artifact].get(rel_attr): artifact_extra[rel_attr] = partials_info[artifact][rel_attr] extra.append(artifact_extra) job.setdefault('extra', {}) job['extra']['partials'] = extra yield job @transforms.add def convert_deps(config, jobs): for job in jobs: job['dependencies'] = { name: dep_job.label for name, dep_job in job['dependencies'].items() } yield job