-
Notifications
You must be signed in to change notification settings - Fork 2.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[codegen] Add ability to define CustomGenerator plugin. (#24861)
Adds the ability to run the codegen.py tool with an externally defined custom generator. Provides an example that generates a protobuf representation of the Matter schema and instructions on how to run that example. Example usage as provided in README: 1. Create a directory for your python generator module, for example "matter_idl_plugin". 2. Add an `__init__.py` under "matter_idl_plugin" implementing a subclass of `CodeGenerator` named `CustomGenerator`. 3. Have `CustomGenerator` load jinja templates, also under the "matter_idl_plugin" subdirectory. 4. Execute the `codegen.py` script passing the path to the parent directory of "matter_idl_plugin" via `--generator custom:<plugin_path>:<plugin_module_name>` argument. ``` # From top-of-tree in this example ./scripts/codegen.py --generator custom:./scripts/py_matter_idl/examples:matter_idl_plugin ./src/controller/data_model/controller-clusters.matter ```
- Loading branch information
Showing
15 changed files
with
645 additions
and
28 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1164,6 +1164,8 @@ saveAs | |
sbin | ||
scalability | ||
scalable | ||
schema | ||
schemas | ||
scm | ||
sco | ||
scp | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
## Creating a custom matter_idl generator | ||
|
||
The matter_idl tool can be used to generate arbitrary code based on the Matter | ||
data model schemas. To create a custom generator that lives outside of the | ||
Matter SDK tree, follow the design pattern of | ||
scripts/py_matter_idl/examples/matter_idl_plugin: | ||
|
||
1. Create a directory for your python generator module, for example | ||
"matter_idl_plugin". | ||
2. Add an `__init__.py` under "matter_idl_plugin" implementing a subclass of | ||
`CodeGenerator` named `CustomGenerator`. | ||
3. Have `CustomGenerator` load jinja templates, also under the | ||
"matter_idl_plugin" subdirectory. | ||
4. Execute the `codegen.py` script passing the path to the parent directory of | ||
"matter_idl_plugin" via | ||
`--generator custom:<plugin_path>:<plugin_module_name>` argument. | ||
|
||
``` | ||
# From top-of-tree in this example | ||
./scripts/codegen.py --generator custom:./scripts/py_matter_idl/examples:matter_idl_plugin ./src/controller/data_model/controller-clusters.matter | ||
``` |
240 changes: 240 additions & 0 deletions
240
scripts/py_matter_idl/examples/matter_idl_plugin/__init__.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,240 @@ | ||
# Copyright (c) 2023 Project CHIP 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 os | ||
from typing import List | ||
|
||
import jinja2 | ||
from matter_idl.generators import CodeGenerator, GeneratorStorage | ||
from matter_idl.matter_idl_types import Attribute, Cluster, ClusterSide, Command, Field, Idl | ||
|
||
|
||
def toUpperSnakeCase(s): | ||
""" Convert camelCaseString to UPPER_SNAKE_CASE with proper handling of acronyms and numerals. """ | ||
lastLower = False | ||
letters = [] | ||
for c in s: | ||
if c.isupper(): | ||
if lastLower: | ||
letters.append('_') | ||
letters.append(c) | ||
lastLower = False | ||
else: | ||
letters.append(c.upper()) | ||
lastLower = True | ||
|
||
return ''.join(letters) | ||
|
||
|
||
def toLowerSnakeCase(s): | ||
""" Convert camelCaseString to lower_snake_case with proper handling of acronyms and numerals. """ | ||
return toUpperSnakeCase(s).lower() | ||
|
||
|
||
def toUpperAcronym(s): | ||
""" Remove lower case letters and numbers from the given string""" | ||
return ''.join([i for i in s if i.isupper() or i.isnumeric()]).upper() | ||
|
||
|
||
def toEnumEntryName(enumEntry, enumName): | ||
""" Create enum entry name by prepending that acronym of the enum name and converting to upper snake case """ | ||
prefix = toUpperAcronym(enumName) | ||
if (enumEntry[0] == 'k'): | ||
enumEntry = enumEntry[1:] | ||
return prefix + '_' + toUpperSnakeCase(enumEntry) | ||
|
||
|
||
def toProtobufType(zapType: str) -> str: | ||
""" Convert zap type to protobuf type """ | ||
u32Types = ["uint32", "enum8", "enum16", "enum32", "bitmap8", "bitmap16", "bitmap32", "cluster_id", "attrib_id", "event_id", "command_id", | ||
"endpoint_no", "group_id", "devtype_id", "fabric_idx", "vendor_id", "status_code", "faulttype", "levelcontroloptions", "percent100ths", "percent"] | ||
u64Types = ["uint64", "enum64", "bitmap64", "node_id", "fabric_id", "int40u", "int48u", "int56u", "int64u"] | ||
i32Types = ["int32", "int8s", "int16s", "int24s", "int32s"] | ||
i64Types = ["int64", "int40s", "int48s", "int56s", "int64s"] | ||
floatTypes = ["float", "double"] | ||
stringTypes = ["char_string", "long_char_string"] | ||
bytesTypes = ["octet_string", "long_octet_string"] | ||
|
||
zapTypeLower = zapType.lower() | ||
if zapTypeLower in u32Types: | ||
return "uint32" | ||
if zapTypeLower in u64Types: | ||
return "uint64" | ||
if zapTypeLower in i32Types: | ||
return "int32" | ||
if zapTypeLower in i64Types: | ||
return "int64" | ||
if zapTypeLower in floatTypes: | ||
return "float" | ||
if zapTypeLower == "double": | ||
return "double" | ||
if zapTypeLower == "boolean": | ||
return "bool" | ||
if zapTypeLower in stringTypes: | ||
return "string" | ||
if zapTypeLower in bytesTypes: | ||
return "bytes" | ||
|
||
# If no match, return the original type name for the Struct, Enum, or Bitmap. | ||
return zapType | ||
|
||
|
||
# Enum for encoding the type information into protobuf field tag for stateless translation. | ||
# These values encoded to the upper range of the protobuf field tag. | ||
class EncodingDataType: | ||
UINT = 1 | ||
INT = 2 | ||
BOOL = 3 | ||
CHAR_STRING = 4 | ||
OCT_STRING = 5 | ||
STRUCT = 6 | ||
FLOAT = 7 | ||
DOUBLE = 8 | ||
|
||
@staticmethod | ||
def fromType(protobufType: str): | ||
if protobufType == "uint32": | ||
return EncodingDataType.UINT | ||
if protobufType == "uint64": | ||
return EncodingDataType.UINT | ||
if protobufType == "int32": | ||
return EncodingDataType.INT | ||
if protobufType == "int64": | ||
return EncodingDataType.INT | ||
if protobufType == "bool": | ||
return EncodingDataType.BOOL | ||
if protobufType == "string": | ||
return EncodingDataType.CHAR_STRING | ||
if protobufType == "bytes": | ||
return EncodingDataType.OCT_STRING | ||
if protobufType == "float": | ||
return EncodingDataType.FLOAT | ||
if protobufType == "double": | ||
return EncodingDataType.DOUBLE | ||
|
||
# If not a primitive type, it is a named type; assume it is a Struct. | ||
# NOTE: the actual type may be an Enum or Bitmap. | ||
return EncodingDataType.STRUCT | ||
|
||
|
||
def commandArgs(command: Command, cluster: Cluster): | ||
"""Return the list of fields for the command request for the given command and cluster.""" | ||
for struct in cluster.structs: | ||
if struct.name == command.input_param: | ||
return struct.fields | ||
|
||
# If the command has no input parameters, just return an empty list. | ||
return [] | ||
|
||
|
||
def commandResponseArgs(command: Command, cluster: Cluster): | ||
"""Return the list of fields for the command response for the given command and cluster.""" | ||
for struct in cluster.structs: | ||
if struct.name == command.output_param: | ||
return struct.fields | ||
|
||
return [] | ||
|
||
|
||
def toEncodedTag(tag, typeNum: EncodingDataType): | ||
""" Return the final encoded tag from the given field number and field encoded data type. | ||
The Matter field type information is encoded into the upper range of the protobuf field | ||
tag for stateless translation to Matter TLV. """ | ||
tag = (int(typeNum) << 19) | int(tag) | ||
return tag | ||
|
||
|
||
def toProtobufFullType(field: Field): | ||
"""Return the full protobuf type for the given field, including repeated and optional specifiers.""" | ||
prefix = "" | ||
protobufType = toProtobufType(field.data_type.name) | ||
if field.is_list: | ||
prefix = "repeated " + prefix | ||
elif field.is_optional: | ||
prefix = "optional " + prefix | ||
return prefix + protobufType | ||
|
||
|
||
def toFieldTag(field: Field): | ||
protobufType = toProtobufType(field.data_type.name) | ||
typeNum = EncodingDataType.fromType(protobufType) | ||
tag = toEncodedTag(field.code, typeNum) | ||
return tag | ||
|
||
|
||
def toFieldComment(field: Field): | ||
protobufType = toProtobufType(field.data_type.name) | ||
typeNum = EncodingDataType.fromType(protobufType) | ||
tagComment = "/** %s Type: %d IsList: %d FieldId: %d */" % ( | ||
field.data_type.name, typeNum, field.is_list, field.code) | ||
return tagComment | ||
|
||
|
||
class CustomGenerator(CodeGenerator): | ||
""" | ||
Example of a custom generator. Outputs protobuf representation of Matter clusters. | ||
""" | ||
|
||
def __init__(self, storage: GeneratorStorage, idl: Idl, **kargs): | ||
""" | ||
Inintialization is specific for java generation and will add | ||
filters as required by the java .jinja templates to function. | ||
""" | ||
super().__init__(storage, idl) | ||
|
||
# Override the template path to use local templates within this plugin directory | ||
self.jinja_env = jinja2.Environment( | ||
loader=jinja2.FileSystemLoader( | ||
searchpath=os.path.dirname(__file__)), | ||
keep_trailing_newline=True) | ||
|
||
# String helpers | ||
self.jinja_env.filters['toLowerSnakeCase'] = toLowerSnakeCase | ||
self.jinja_env.filters['toUpperSnakeCase'] = toUpperSnakeCase | ||
|
||
# Type helpers | ||
self.jinja_env.filters['toEnumEntryName'] = toEnumEntryName | ||
self.jinja_env.filters['toProtobufType'] = toProtobufType | ||
self.jinja_env.filters['toEncodedTag'] = toEncodedTag | ||
|
||
# Tag helpers | ||
self.jinja_env.filters['toFieldTag'] = toFieldTag | ||
self.jinja_env.filters['toProtobufFullType'] = toProtobufFullType | ||
self.jinja_env.filters['toFieldComment'] = toFieldComment | ||
|
||
# Command helpers | ||
self.jinja_env.filters['commandArgs'] = commandArgs | ||
self.jinja_env.filters['commandResponseArgs'] = commandResponseArgs | ||
|
||
def internal_render_all(self): | ||
""" | ||
Renders the given custom template to the given output filename. | ||
""" | ||
|
||
# Every cluster has its own impl, to avoid | ||
# very large compilations (running out of RAM) | ||
for cluster in self.idl.clusters: | ||
if cluster.side != ClusterSide.CLIENT: | ||
continue | ||
|
||
filename = "proto/%s_cluster.proto" % toLowerSnakeCase(cluster.name) | ||
|
||
# Header containing a macro to initialize all cluster plugins | ||
self.internal_render_one_output( | ||
template_path="./matter_cluster_proto.jinja", | ||
output_file_name=filename, | ||
vars={ | ||
'cluster': cluster, | ||
} | ||
) |
Oops, something went wrong.