| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 | #!/usr/bin/env python# Copyright 2016, Google Inc.# All rights reserved.## Redistribution and use in source and binary forms, with or without# modification, are permitted provided that the following conditions are# met:##     * Redistributions of source code must retain the above copyright# notice, this list of conditions and the following disclaimer.#     * Redistributions in binary form must reproduce the above# copyright notice, this list of conditions and the following disclaimer# in the documentation and/or other materials provided with the# distribution.#     * Neither the name of Google Inc. nor the names of its# contributors may be used to endorse or promote products derived from# this software without specific prior written permission.## THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."""Builds gRPC distribution artifacts."""import argparseimport atexitimport dockerjobimport itertoolsimport jobsetimport jsonimport multiprocessingimport osimport reimport subprocessimport sysimport timeimport uuid# Docker doesn't clean up after itself, so we do it on exit.if jobset.platform_string() == 'linux':  atexit.register(lambda: subprocess.call(['stty', 'echo']))ROOT = os.path.abspath(os.path.join(os.path.dirname(sys.argv[0]), '../..'))os.chdir(ROOT)def create_docker_jobspec(name, dockerfile_dir, shell_command, environ={},                   flake_retries=0, timeout_retries=0):  """Creates jobspec for a task running under docker."""  environ = environ.copy()  environ['RUN_COMMAND'] = shell_command  #docker_args = ['-v', '%s/artifacts:/var/local/jenkins/grpc/artifacts' % ROOT]  docker_args=[]  for k,v in environ.iteritems():    docker_args += ['-e', '%s=%s' % (k, v)]  docker_env = {'DOCKERFILE_DIR': dockerfile_dir,                'DOCKER_RUN_SCRIPT': 'tools/jenkins/docker_run.sh',                'OUTPUT_DIR': 'artifacts'}  jobspec = jobset.JobSpec(          cmdline=['tools/jenkins/build_and_run_docker.sh'] + docker_args,          environ=docker_env,          shortname='build_artifact.%s' % (name),          timeout_seconds=30*60,          flake_retries=flake_retries,          timeout_retries=timeout_retries)  return jobspecdef create_jobspec(name, cmdline, environ=None, shell=False,                   flake_retries=0, timeout_retries=0):  """Creates jobspec."""  jobspec = jobset.JobSpec(          cmdline=cmdline,          environ=environ,          shortname='build_artifact.%s' % (name),          timeout_seconds=5*60,          flake_retries=flake_retries,          timeout_retries=timeout_retries,          shell=shell)  return jobspecdef macos_arch_env(arch):  """Returns environ specifying -arch arguments for make."""  if arch == 'x86':    arch_arg = '-arch i386'  elif arch == 'x64':    arch_arg = '-arch x86_64'  else:    raise Exception('Unsupported arch')  return {'CFLAGS': arch_arg, 'LDFLAGS': arch_arg}class CSharpExtArtifact:  """Builds C# native extension library"""  def __init__(self, platform, arch):    self.name = 'csharp_ext_%s_%s' % (platform, arch)    self.platform = platform    self.arch = arch    self.labels = ['csharp', platform, arch]  def pre_build_jobspecs(self):    if self.platform == 'windows':      return [create_jobspec('prebuild_%s' % self.name,                             ['tools\\run_tests\\pre_build_c.bat'],                             shell=True,                             flake_retries=5,                             timeout_retries=2)]    else:      return []  def build_jobspec(self):    if self.platform == 'windows':      msbuild_platform = 'Win32' if self.arch == 'x86' else self.arch      return create_jobspec(self.name,                            ['tools\\run_tests\\build_artifact_csharp.bat',                             'vsprojects\\grpc_csharp_ext.sln',                             '/p:Configuration=Release',                             '/p:PlatformToolset=v120',                             '/p:Platform=%s' % msbuild_platform],                            shell=True)    else:      environ = {'CONFIG': 'opt',                 'EMBED_OPENSSL': 'true',                 'EMBED_ZLIB': 'true'}      if self.platform == 'linux':        return create_docker_jobspec(self.name,            'tools/dockerfile/grpc_artifact_linux_%s' % self.arch,            'tools/run_tests/build_artifact_csharp.sh')      else:        environ.update(macos_arch_env(self.arch))        return create_jobspec(self.name,                              ['tools/run_tests/build_artifact_csharp.sh'],                              environ=environ)  def __str__(self):    return self.name_ARTIFACTS = [    CSharpExtArtifact('linux', 'x86'),    CSharpExtArtifact('linux', 'x64'),    CSharpExtArtifact('macos', 'x86'),    CSharpExtArtifact('macos', 'x64'),    CSharpExtArtifact('windows', 'x86'),    CSharpExtArtifact('windows', 'x64')]def _create_build_map():  """Maps artifact names and labels to list of artifacts to be built."""  artifact_build_map = dict([(artifact.name, [artifact])                             for artifact in _ARTIFACTS])  if len(_ARTIFACTS) > len(artifact_build_map.keys()):    raise Exception('Artifact names need to be unique')  label_build_map = {}  label_build_map['all'] = [a for a in _ARTIFACTS]  # to build all artifacts  for artifact in _ARTIFACTS:    for label in artifact.labels:      if label in label_build_map:        label_build_map[label].append(artifact)      else:        label_build_map[label] = [artifact]  if set(artifact_build_map.keys()).intersection(label_build_map.keys()):    raise Exception('Artifact names need to be distinct from label names')  return dict( artifact_build_map.items() + label_build_map.items())_BUILD_MAP = _create_build_map()argp = argparse.ArgumentParser(description='Builds distribution artifacts.')argp.add_argument('-b', '--build',                  choices=sorted(_BUILD_MAP.keys()),                  nargs='+',                  default=['all'],                  help='Artifact name or artifact label to build.')argp.add_argument('-f', '--filter',                  choices=sorted(_BUILD_MAP.keys()),                  nargs='+',                  default=[],                  help='Filter artifacts to build with AND semantics.')argp.add_argument('-j', '--jobs', default=multiprocessing.cpu_count(), type=int)argp.add_argument('-t', '--travis',                  default=False,                  action='store_const',                  const=True)args = argp.parse_args()# Figure out which artifacts to buildartifacts = []for label in args.build:  artifacts += _BUILD_MAP[label]# Among target selected by -b, filter out those that don't match the filterartifacts = [a for a in artifacts if all(f in a.labels for f in args.filter)]artifacts = sorted(set(artifacts))# Execute pre-build phaseprebuild_jobs = []for artifact in artifacts:  prebuild_jobs += artifact.pre_build_jobspecs()if prebuild_jobs:  num_failures, _ = jobset.run(    prebuild_jobs, newline_on_success=True, maxjobs=args.jobs)  if num_failures != 0:    jobset.message('FAILED', 'Pre-build phase failed.', do_newline=True)    sys.exit(1)build_jobs = []for artifact in artifacts:  build_jobs.append(artifact.build_jobspec())if not build_jobs:  print 'Nothing to build.'  sys.exit(1)jobset.message('START', 'Building artifacts.', do_newline=True)num_failures, _ = jobset.run(    build_jobs, newline_on_success=True, maxjobs=args.jobs)if num_failures == 0:  jobset.message('SUCCESS', 'All artifacts built successfully.',                 do_newline=True)else:  jobset.message('FAILED', 'Failed to build artifacts.',                 do_newline=True)  sys.exit(1)
 |