|  | @@ -0,0 +1,225 @@
 | 
	
		
			
				|  |  | +#!/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 argparse
 | 
	
		
			
				|  |  | +import atexit
 | 
	
		
			
				|  |  | +import dockerjob
 | 
	
		
			
				|  |  | +import itertools
 | 
	
		
			
				|  |  | +import jobset
 | 
	
		
			
				|  |  | +import json
 | 
	
		
			
				|  |  | +import multiprocessing
 | 
	
		
			
				|  |  | +import os
 | 
	
		
			
				|  |  | +import re
 | 
	
		
			
				|  |  | +import subprocess
 | 
	
		
			
				|  |  | +import sys
 | 
	
		
			
				|  |  | +import time
 | 
	
		
			
				|  |  | +import uuid
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +# Docker doesn't clean up after itself, so we do it on exit.
 | 
	
		
			
				|  |  | +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_jobspec(name, cmdline, environ=None, shell=False,
 | 
	
		
			
				|  |  | +                   flake_retries=0, timeout_retries=0):
 | 
	
		
			
				|  |  | +  """Creates jobspec."""
 | 
	
		
			
				|  |  | +  test_job = 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 test_job
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +def 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 TestDockerArtifact:
 | 
	
		
			
				|  |  | +  """Demo artifact that shows that docker-based artifacts work"""
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def __init__(self):
 | 
	
		
			
				|  |  | +    self.name = 'test_docker_artifact'
 | 
	
		
			
				|  |  | +    self.labels = ['docker', 'linux']
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def pre_build_jobspecs(self):
 | 
	
		
			
				|  |  | +    return []
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def build_jobspec(self):
 | 
	
		
			
				|  |  | +    return create_jobspec(self.name, ['sleep', '5'])
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def __str__(self):
 | 
	
		
			
				|  |  | +    return self.name
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +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,
 | 
	
		
			
				|  |  | +                            ['vsprojects\\build_vs2013.bat',
 | 
	
		
			
				|  |  | +                             'vsprojects\\grpc_csharp_ext.sln',
 | 
	
		
			
				|  |  | +                             '/p:Configuration=Release',
 | 
	
		
			
				|  |  | +                             '/p:PlatformToolset=v120',
 | 
	
		
			
				|  |  | +                             '/p:Platform=%s' % msbuild_platform],
 | 
	
		
			
				|  |  | +                            shell=True)
 | 
	
		
			
				|  |  | +    else:
 | 
	
		
			
				|  |  | +      environ = {'CONFIG': 'opt'}
 | 
	
		
			
				|  |  | +      if self.platform == 'macos':
 | 
	
		
			
				|  |  | +        environ.update(macos_arch_env(self.arch))
 | 
	
		
			
				|  |  | +      return create_jobspec(self.name,
 | 
	
		
			
				|  |  | +                            ['make', 'grpc_csharp_ext'],
 | 
	
		
			
				|  |  | +                            environ=environ)
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def __str__(self):
 | 
	
		
			
				|  |  | +    return self.name
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +_ARTIFACTS = [
 | 
	
		
			
				|  |  | +    TestDockerArtifact(),
 | 
	
		
			
				|  |  | +    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 build
 | 
	
		
			
				|  |  | +artifacts = []
 | 
	
		
			
				|  |  | +for label in args.build:
 | 
	
		
			
				|  |  | +  artifacts += _BUILD_MAP[label]
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +# Among target selected by -b, filter out those that don't match the filter
 | 
	
		
			
				|  |  | +artifacts = [a for a in artifacts if all(f in a.labels for f in args.filter)]
 | 
	
		
			
				|  |  | +artifacts = sorted(set(artifacts))
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +# Execute pre-build phase
 | 
	
		
			
				|  |  | +prebuild_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)
 |