| # 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() |