# Copyright 2015 gRPC authors. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. import argparse import glob import multiprocessing import os import pickle import shutil import sys import tempfile from typing import Dict, List, Union import _utils import yaml PROJECT_ROOT = os.path.join(os.path.dirname(os.path.abspath(__file__)), "..", "..") os.chdir(PROJECT_ROOT) # TODO(lidiz) find a better way for plugins to reference each other sys.path.append(os.path.join(PROJECT_ROOT, 'tools', 'buildgen', 'plugins')) # from tools.run_tests.python_utils import jobset jobset = _utils.import_python_module( os.path.join(PROJECT_ROOT, 'tools', 'run_tests', 'python_utils', 'jobset.py')) PREPROCESSED_BUILD = '.preprocessed_build' test = {} if os.environ.get('TEST', 'false') == 'true' else None assert sys.argv[1:], 'run generate_projects.sh instead of this directly' parser = argparse.ArgumentParser() parser.add_argument('build_files', nargs='+', default=[], help="build files describing build specs") parser.add_argument('--templates', nargs='+', default=[], help="mako template files to render") parser.add_argument('--output_merged', '-m', default='', type=str, help="merge intermediate results to a file") parser.add_argument('--jobs', '-j', default=multiprocessing.cpu_count(), type=int, help="maximum parallel jobs") parser.add_argument('--base', default='.', type=str, help="base path for generated files") args = parser.parse_args() def preprocess_build_files() -> _utils.Bunch: """Merges build yaml into a one dictionary then pass it to plugins.""" build_spec = dict() for build_file in args.build_files: with open(build_file, 'r') as f: _utils.merge_json(build_spec, yaml.safe_load(f.read())) # Executes plugins. Plugins update the build spec in-place. for py_file in sorted(glob.glob('tools/buildgen/plugins/*.py')): plugin = _utils.import_python_module(py_file) plugin.mako_plugin(build_spec) if args.output_merged: with open(args.output_merged, 'w') as f: f.write(yaml.dump(build_spec)) # Makes build_spec sort of immutable and dot-accessible return _utils.to_bunch(build_spec) def generate_template_render_jobs(templates: List[str]) -> List[jobset.JobSpec]: """Generate JobSpecs for each one of the template rendering work.""" jobs = [] base_cmd = [sys.executable, 'tools/buildgen/_mako_renderer.py'] for template in sorted(templates, reverse=True): root, f = os.path.split(template) if os.path.splitext(f)[1] == '.template': out_dir = args.base + root[len('templates'):] out = os.path.join(out_dir, os.path.splitext(f)[0]) if not os.path.exists(out_dir): os.makedirs(out_dir) cmd = base_cmd[:] cmd.append('-P') cmd.append(PREPROCESSED_BUILD) cmd.append('-o') if test is None: cmd.append(out) else: tf = tempfile.mkstemp() test[out] = tf[1] os.close(tf[0]) cmd.append(test[out]) cmd.append(args.base + '/' + root + '/' + f) jobs.append(jobset.JobSpec(cmd, shortname=out, timeout_seconds=None)) return jobs def main() -> None: templates = args.templates if not templates: for root, _, files in os.walk('templates'): for f in files: templates.append(os.path.join(root, f)) build_spec = preprocess_build_files() with open(PREPROCESSED_BUILD, 'wb') as f: pickle.dump(build_spec, f) err_cnt, _ = jobset.run(generate_template_render_jobs(templates), maxjobs=args.jobs) if err_cnt != 0: print('ERROR: %s error(s) found while generating projects.' % err_cnt, file=sys.stderr) sys.exit(1) if test is not None: for s, g in test.items(): if os.path.isfile(g): assert 0 == os.system('diff %s %s' % (s, g)), s os.unlink(g) else: assert 0 == os.system('diff -r %s %s' % (s, g)), s shutil.rmtree(g, ignore_errors=True) if __name__ == "__main__": main()