This repository includes tools related to the "VSC" interface/service description language.
Primarily, it allows reading a service description and generating various types of output. It uses the Jinja2 templating language ((alt. link)) for most output definitions.
Work in progress! This is the usage pattern:
usage: vsc_generator.py <input-yaml-file (path)> <output-template-file (name only, not path)>
For the moment, try this:
git clone https://github.com/GENIVI/vehicle_service_catalog
python model/vsc_generator.py vehicle_service_catalog/comfort-service.yml simple_overview.tpl
This example exercises the parser to create the AST out of a YAML file from the Vehicle Service Specification and then prints out an overview using the template.
^^^ Edit the code if you want to try out other things. This will soon be more flexible of course.
Looking at the jinja2 template shows how to traverse it directly by referencing each object's public member variables (see template simple_overview.tpl).
The project uses pytest to define unit tests. In the tests directory is a simple starting point. More can be added.
To run tests, just run pytest but make sure it is from within the tests directory:
cd tests
pytest -v
A simple generator (with only one template) can be done like this:
- Import the vsc_generator.py and vsc_parser.py modules
- Get the Service description file (YAML), for example from command line argument
- Get the Abstract Syntax Tree representation by calling
vsc_parser.get_ast_from_file(service_desc_file)
- Call
gen()
function in generator module, and pass the AST, and the name of a single template. - If needed, add any of your own custom logic (see also advanced usage for more information)
Unless you need to add more logic, generating one input file with a single template is basically already available if vsc_generator.py is called as a main program:
python model/vsc_generator.py <service-description.yml> <jinja-template-name>
NOTE: Due to how jinja loads templates (without adding a custom loader, which has not been done), the first argument is the path to the YAML file, but the second argument is only the name of the template (which must be in templates/ dir. Pointing to the full path of a template file in a different location is not implemented in vsc_generator.py
An advanced generator (with several templates) can be done like this:
-
Import the vsc_generator.py and vsc_parser.py modules
-
Get the needed Service description file(s) (YAML), for example from command line argument
-
For each file, get the Abstract Syntax Tree representation by calling
vsc_parser.get_ast_from_file(service_desc_file)
-
Write templates according to (some, not all) node types. You can call
gen(node)
orgen(node, <Template>)
from within a template - see details below. -
Set the default_templates member variable in generator to configure which templates to use (see section below).
-
Implement the
gen()
function, normally by delegating directly intovsc_generator.gen()
, but you can put your own logic here if needed. -
Call the
gen()
function with the topmost node. Store result in appropriate file. -
Continue calling the
gen()
function with another node, and another template for another file (for example, generating .c and .h header files from the same input) -
... implement any other required logic in the generator code, and/or with the advanced control options available in jinja2 templates.
-
NOTE! Remember to inject any globals (e.g. functions) into the jinja environment if these are referred by the templates. See vsc_generator.py for example. You can also pass data into templates through other variables (see jinja2 documentation).
This setting is a dictionary mapping the AST node types to the default template to use for that node type. It can be overridden in a call to gen() but for a particular code generation purpose it is better to set this up.
generator.default_templates = {
'AST' : 'AST-mystuff-toplevel.txt',
'Service' : 'Service-mystuff.c',
'Type' : 'Type-mystuff-smart_type_mapper.c',
'Method' : 'Method-mystuff.c'
}
NOTE: It is not required to specify a separate template for every type. Very often, nodes can be generated directly from the parent template type. For simple cases, a single top-level template might suffice for the whole generation.
The gen() function has a short name to be convenient in templates.
You can call the gen() function from within a template to delegate the required work for generating a certain node type within a larger generation. (See Template example). Thus the templates can effectively "call" each other, without the need for template-include or template-inheritance features, but those features in jinja2 are still available if they are needed.
gen() can be called with just the node (evaluating the passed parameter type at run-time) or by explicitly stating a template:
- Providing the node reference only.
def gen(node : AST)
example use:
gen(node, Method)
gen(anothernode, Type)
This variant will dynamically determine the node type (a subclass of AST) and generate using the predetermined template for that node type.
- Providing the node and a specific template. The default template for the node type is not used. The specified template is used instead.
def gen(node : AST, templatename : str)
example use:
gen(node, 'My-alternative-method-template.tpl')
Templates must be stored in the templates/ directory.
Templates should be named using this convention:
<node-type>-<target-format>-<variant>.<suffix>
- node-type must be the exact name of the corresponding class in the AST, e.g. Service, Method, Type, etc. A template that consumes the top-level root node would use the node type AST.
- target-format is a free form name for whatever it is you are generating
- variant is optional and just to indicate there might be multiple ways to generate the same thing for the same target. Basically, use your own judgement to name the template with useful information here.
- suffix should try to mimic the normal suffix for the file format that is being generated. For example, if the templates aims to generate a HTML document, name the template with .html. If it is generating a programming language, use an appropriate suffix for the file, etc.
Examples:
Datatypes-my_docs-simple.html
Datatypes-my_docs-advanced.html
Method-c++-without-comments.cpp
Method-c++-with-comments.cpp
Service-nodejs.js
Service-rust.rs
-
The standard functions in vsc_generator.py will pass in only a single node of AST type to the template generation framework. The node will be of a particular type, depending on what type of template is being rendered (or more correctly, what was specified when the generation function was called -- either a top level function like render_ast_with_template_file() or a call to gen() with the node in question as parameter.)
-
When using the provided generator convenience functions, the node that is "passed to the template" (roughly speaking) will always be named item
-
Since the class members are public, they can be walked directly through dot-notation in code that is embedded in the template. For example, if the passed item is a Service the template can get to the namespaces list by just referencing it:
item.namespaces
Dot notation can be chained as needed:
item.namespaces[1].methods[0].description
- the first method name in the
second namespace.
...but of course it is more likely to use loop constructs to iterate over
lists than to address specific indexes like that:
{% for i in item.namespaces %}
this is each namespace name: {{ i.name }}
{% endfor %}
jinja2 is a very capable templating language. Advanced generators can of course make use of any features in python or jinja2 to create an advanced generator. Any features that might be applicable in more than one place would however be best generalized and introduced into the vsc_generator.py helper modules, for better reuse.
This templates expects that the "item" passed in is the Service object. It calls the gen() function from within the template to delegate work to a separate template for Methods.
Service-mygen.c
// General file header information
// ...
// Service name: {{ item.name }}
{% for i in s.namespaces %}
// namespace: {{ i.name }}
// {{ i.description }}
{% for x in i.methods %}
{{ gen(Method, x) }} <- delegate work to a separate template for Methods
{% endfor %}
{% endfor %}
Please refer to GitHub tickets (Feel free to make a proposal or ask a question)
Please refer to GitHub tickets with the label "bug"