Skip to content

Commit

Permalink
[codegen] Add ability to define CustomGenerator plugin. (#24861)
Browse files Browse the repository at this point in the history
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
turon authored and pull[bot] committed Sep 18, 2023
1 parent 940a8a0 commit 12d7fa6
Show file tree
Hide file tree
Showing 15 changed files with 645 additions and 28 deletions.
2 changes: 2 additions & 0 deletions .github/.wordlist.txt
Original file line number Diff line number Diff line change
Expand Up @@ -1164,6 +1164,8 @@ saveAs
sbin
scalability
scalable
schema
schemas
scm
sco
scp
Expand Down
26 changes: 20 additions & 6 deletions scripts/codegen.py
Original file line number Diff line number Diff line change
Expand Up @@ -70,8 +70,10 @@ def write_new_data(self, relative_path: str, content: str):
@click.option(
'--generator',
default='JAVA',
type=click.Choice(GENERATORS.keys(), case_sensitive=False),
help='What code generator to run')
help='What code generator to run. The choices are: '+'|'.join(GENERATORS.keys())+'. ' +
'When using custom, provide the plugin path using `--generator custom:<path_to_plugin>:<plugin_module_name>` syntax. ' +
'For example, `--generator custom:./my_plugin:my_plugin_module` will load `./my_plugin/my_plugin_module/__init.py__` ' +
'that defines a subclass of CodeGenerator named CustomGenerator.')
@click.option(
'--output-dir',
type=click.Path(exists=False),
Expand Down Expand Up @@ -110,16 +112,28 @@ def main(log_level, generator, output_dir, dry_run, name_only, expected_outputs,
datefmt='%Y-%m-%d %H:%M:%S'
)

logging.info("Parsing idl from %s" % idl_path)
idl_tree = CreateParser().parse(open(idl_path, "rt").read())

if name_only:
storage = ListGeneratedFilesStorage()
else:
storage = FileSystemGeneratorStorage(output_dir)

logging.info("Parsing idl from %s" % idl_path)
idl_tree = CreateParser().parse(open(idl_path, "rt").read())

plugin_module = None
if generator.startswith('custom'):
# check that the plugin path is provided
if ':' not in generator:
logging.fatal("Custom generator plugin path not provided. Use --generator custom:<path_to_plugin>")
sys.exit(1)
custom_params = generator.split(':')
(generator, plugin_path, plugin_module) = custom_params
logging.info("Using CustomGenerator at plugin path %s.%s" % (plugin_path, plugin_module))
sys.path.append(plugin_path)
generator = 'CUSTOM'

logging.info("Running code generator %s" % generator)
generator = CodeGenerator.FromString(generator).Create(storage, idl=idl_tree)
generator = CodeGenerator.FromString(generator).Create(storage, idl=idl_tree, plugin_module=plugin_module)
generator.render(dry_run)

if expected_outputs:
Expand Down
21 changes: 21 additions & 0 deletions scripts/py_matter_idl/examples/README.md
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 scripts/py_matter_idl/examples/matter_idl_plugin/__init__.py
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,
}
)
Loading

0 comments on commit 12d7fa6

Please sign in to comment.