The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#) https://grpc.io/
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

180 lines
5.8 KiB

#!/usr/bin/env python3
# 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.
"""Simple Mako renderer.
Just a wrapper around the mako rendering library.
"""
import getopt
import glob
import importlib.util
import os
import pickle
import shutil
import sys
from typing import List
from mako import exceptions
from mako.lookup import TemplateLookup
from mako.runtime import Context
from mako.template import Template
import yaml
PROJECT_ROOT = os.path.join(
os.path.dirname(os.path.abspath(__file__)), "..", ".."
)
# TODO(lidiz) find a better way for plugins to reference each other
sys.path.append(os.path.join(PROJECT_ROOT, "tools", "buildgen", "plugins"))
def out(msg: str) -> None:
print(msg, file=sys.stderr)
def showhelp() -> None:
out(
"mako-renderer.py [-o out] [-m cache] [-P preprocessed_input] [-d dict]"
" [-d dict...] [-t template] [-w preprocessed_output]"
)
def render_template(template: Template, context: Context) -> None:
"""Render the mako template with given context.
Prints an error template to indicate where and what in the template caused
the render failure.
"""
try:
template.render_context(context)
except:
out(exceptions.text_error_template().render())
raise
def main(argv: List[str]) -> None:
got_input = False
module_directory = None
preprocessed_output = None
dictionary = {}
json_dict = {}
got_output = False
output_name = None
got_preprocessed_input = False
output_merged = None
try:
opts, args = getopt.getopt(argv, "hM:m:o:t:P:")
except getopt.GetoptError:
out("Unknown option")
showhelp()
sys.exit(2)
for opt, arg in opts:
if opt == "-h":
out("Displaying showhelp")
showhelp()
sys.exit()
elif opt == "-o":
if got_output:
out("Got more than one output")
showhelp()
sys.exit(3)
got_output = True
output_name = arg
elif opt == "-m":
if module_directory is not None:
out("Got more than one cache directory")
showhelp()
sys.exit(4)
module_directory = arg
elif opt == "-M":
if output_merged is not None:
out("Got more than one output merged path")
showhelp()
sys.exit(5)
output_merged = arg
elif opt == "-P":
assert not got_preprocessed_input
assert json_dict == {}
with open(arg, "rb") as dict_file:
dictionary = pickle.load(dict_file)
got_preprocessed_input = True
cleared_dir = False
for arg in args:
got_input = True
with open(arg) as f:
srcs = list(yaml.safe_load_all(f.read()))
for src in srcs:
if isinstance(src, str):
assert len(srcs) == 1
template = Template(
src,
filename=arg,
module_directory=module_directory,
lookup=TemplateLookup(directories=["."]),
)
with open(output_name, "w") as output_file:
render_template(
template, Context(output_file, **dictionary)
)
else:
# we have optional control data: this template represents
# a directory
if not cleared_dir:
if not os.path.exists(output_name):
pass
elif os.path.isfile(output_name):
os.unlink(output_name)
else:
shutil.rmtree(output_name, ignore_errors=True)
cleared_dir = True
items = []
if "foreach" in src:
for el in dictionary[src["foreach"]]:
if "cond" in src:
args = dict(dictionary)
args["selected"] = el
if not eval(src["cond"], {}, args):
continue
items.append(el)
assert items
else:
items = [None]
for item in items:
args = dict(dictionary)
args["selected"] = item
item_output_name = os.path.join(
output_name, Template(src["output_name"]).render(**args)
)
if not os.path.exists(os.path.dirname(item_output_name)):
os.makedirs(os.path.dirname(item_output_name))
template = Template(
src["template"],
filename=arg,
module_directory=module_directory,
lookup=TemplateLookup(directories=["."]),
)
with open(item_output_name, "w") as output_file:
render_template(template, Context(output_file, **args))
if not got_input and not preprocessed_output:
out("Got nothing to do")
showhelp()
if __name__ == "__main__":
main(sys.argv[1:])