-
Notifications
You must be signed in to change notification settings - Fork 86
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add a way to mutate JSON within manifests #85
base: main
Are you sure you want to change the base?
Conversation
This is what I have that is working (patch against this PR): manifest: json_config:
- file: "config/zcl/zcl_config.zap"
jq: |
( .endpointTypes[].clusters[].attributes[] | select(.name == "model identifier"))
.defaultValue = "some model"
- file: "config/zcl/zcl_config.zap"
jq: |
( .endpointTypes[].clusters[].attributes[] | select(.name == "manufacturer name"))
.defaultValue = "some manufacturer" |
My 2 cents on this:
zcl_config_zap = build_template_path / "config/zcl/zcl_config.zap"
if zcl_config_zap.exists():
date_code = datetime.today().strftime("%Y-%m-%d")
default_zap_changes = [
# Set software date code to day of build
f'(.endpointTypes[].clusters[].attributes[] | select(.name == "date code")).defaultValue = "{date_code}"',
# Set software build ID to SDK version
f'(.endpointTypes[].clusters[].attributes[] | select(.name == "sw build id")).defaultValue = "{sdk_version.split(":", 1)[1]}"',
# Set path
f'(.package[] | select(.type == "zcl-properties")).path = "{sdk}/app/zcl/zcl-zap.json"',
# Set path
f'(.package[] | select(.type == "gen-templates-json")).path = "{sdk}/protocol/zigbee/app/framework/gen-template/gen-templates.json"',
]
for zap_change in default_zap_changes:
LOGGER.info("Default ZAP change: %s", zap_change)
result = subprocess.run(
[
"jq",
zap_change,
zcl_config_zap,
],
capture_output=True,
)
if result.returncode != 0:
LOGGER.error("jq stderr: %s\n%s", result.returncode, result.stderr)
sys.exit(1)
with open(zcl_config_zap, "wb") as f:
f.write(result.stdout)
# JSON config
for json_config in manifest.get("json_config", []):
json_path = build_template_path / json_config["file"]
if not json_path.exists():
raise ValueError(f"[{json_path}] does not exist")
device_spe_change = json_config["jq"]
LOGGER.info("Device-specific change in %s: %s", json_path, device_spe_change)
result = subprocess.run(
[
"jq",
device_spe_change,
json_path,
],
capture_output=True,
)
if result.returncode != 0:
LOGGER.error("jq stderr: %s\n%s", result.returncode, result.stderr)
sys.exit(1)
with open(json_path, "wb") as f:
f.write(result.stdout) Note: This is tested and working on a Sonoff Dongle-E with router build. https://github.com/Nerivec/silabs-firmware-builder/actions/runs/11570015214/job/32204929046 |
`slc-cli` does not play well in a venv, it seems
Since |
Suggestion for cleaner manifests (can get pretty crowded if you want to keep the project generic enough): DEFAULT_JSON_CONFIG = [
# Fix a few paths by default
{
"file": "config/zcl/zcl_config.zap",
"jq": '(.package[] | select(.type == "zcl-properties")).path = "template:{sdk}/app/zcl/zcl-zap.json"',
"skip_if_missing": True,
},
{
"file": "config/zcl/zcl_config.zap",
"jq": '(.package[] | select(.type == "gen-templates-json")).path = "template:{sdk}/protocol/zigbee/app/framework/gen-template/gen-templates.json"',
"skip_if_missing": True,
},
{
"jq_func": 'zap_set_cluster_attribute("date code", "template:{today}", "all", 0, True)',
},
{
"jq_func": 'zap_set_cluster_attribute("sw build id", "template:{sdk_version}", "all", 0, True)',
},
]
def zap_select_endpoint_type(endpoint_type_id: int | str) -> str:
return (
".endpointTypes[]"
if endpoint_type_id == "all"
else f".endpointTypes[] | select(.id == {endpoint_type_id})"
)
def zap_select_cluster(cluster_code: int | str) -> str:
return (
".clusters[]"
if cluster_code == "all"
else f".clusters[] | select(.code == {cluster_code})"
)
def zap_delete_cluster(
cluster_name: str,
endpoint_type_id: int | str = "all",
skip_if_missing: bool = False,
) -> list[dict[str, typing.Any]]:
return [
{
"file": "config/zcl/zcl_config.zap",
"jq": f'del({zap_select_endpoint_type(endpoint_type_id)}.clusters[] | select(.name == "{cluster_name}"))',
"skip_if_missing": skip_if_missing,
}
]
def zap_set_cluster_attribute(
attribute_name: str,
default_value: typing.Any,
endpoint_type_id: int | str = "all",
cluster_code: int | str = "all",
skip_if_missing: bool = False,
) -> list[dict[str, typing.Any]]:
return [
{
"file": "config/zcl/zcl_config.zap",
"jq": f'({zap_select_endpoint_type(endpoint_type_id)}{zap_select_cluster(cluster_code)}.attributes[] | select(.name == "{attribute_name}")).defaultValue = "{default_value}"',
"skip_if_missing": skip_if_missing,
}
] # Template variables
value_template_env = {
"git_repo_hash": get_git_commit_id(repo=pathlib.Path(__file__).parent.parent),
"manifest_name": args.manifest.stem,
"now": datetime.now(timezone.utc),
"today": date.today(),
"sdk": sdk,
"sdk_version": sdk_version,
}
LOGGER.info("Using templating env:")
LOGGER.info(str(value_template_env))
for json_config in DEFAULT_JSON_CONFIG + manifest.get("json_config", []):
for json_config_item in (
eval(expand_template(json_config["jq_func"], value_template_env))
if json_config.get("jq_func", None)
else [json_config]
):
json_path = build_template_path / json_config_item["file"]
if (
json_config_item.get("skip_if_missing", False)
and not json_path.exists()
):
continue
jq_arg = expand_template(json_config_item["jq"], value_template_env)
LOGGER.info(f"Patching {json_path} with {jq_arg}")
result = subprocess.run(
[
"jq",
jq_arg,
json_path,
],
capture_output=True,
)
if result.returncode != 0:
LOGGER.error("jq stderr: %s\n%s", result.returncode, result.stderr)
sys.exit(1)
with open(json_path, "wb") as f:
f.write(result.stdout) A very basic manifest already goes from: json_config:
- file: "config/zcl/zcl_config.zap"
jq: |
(.endpointTypes[].clusters[].attributes[] | select(.name == "model identifier")).defaultValue = "DONGLE-E_R"
- file: "config/zcl/zcl_config.zap"
jq: |
(.endpointTypes[].clusters[].attributes[] | select(.name == "manufacturer name")).defaultValue = "SONOFF"
# remove OTA cluster
- file: "config/zcl/zcl_config.zap"
jq: |
del(.endpointTypes[].clusters[] | select(.name == "Over the Air Bootloading")) to: json_config:
- jq_func: 'zap_set_cluster_attribute("model identifier", "DONGLE-E_R", "all", 0)'
- jq_func: 'zap_set_cluster_attribute("manufacturer name", "SONOFF", "all", 0)'
- jq_func: 'zap_delete_cluster("Over the Air Bootloading", 1)' More functions can be added as needed, but I think these two are a good start. Note: this is definitely a rough draft, I'm sure there are a lot of optimizations to be done... It works though 😬 |
My idea with using json_config:
- file: "config/zcl/zcl_config.zap"
jq: (.endpointTypes[].clusters[].attributes[] | select(.name == "model identifier")).defaultValue = $value
args:
value: "DONGLE-E_R"
- file: "config/zcl/zcl_config.zap"
jq: (.endpointTypes[].clusters[].attributes[] | select(.name == "manufacturer name")).defaultValue = $value
args:
value: "SONOFF"
# Set software date code to day of build
- file: "config/zcl/zcl_config.zap"
jq: (.endpointTypes[].clusters[].attributes[] | select(.name == "date code")).defaultValue = $value
args:
value: template:{now:%Y%m%d%H%M%S}
# Set software build ID to SDK version
- file: "config/zcl/zcl_config.zap"
jq: (.endpointTypes[].clusters[].attributes[] | select(.name == "sw build id")).defaultValue = $value
args:
value: template:{sdk_version}
# remove OTA cluster
- file: "config/zcl/zcl_config.zap"
jq: del(.endpointTypes[].clusters[] | select(.name == "Over the Air Bootloading"))
# remove Occupancy cluster
- file: "config/zcl/zcl_config.zap"
jq: del(.endpointTypes[].clusters[] | select(.name == "Occupancy Sensing")) I'm not a fan of including any Python code in the YAML and zap_config:
file: "config/zcl/zcl_config.zap"
cluster_defaults:
"model identifier": "DONGLE-E_R"
"manufacturer name": "SONOFF"
"date code": template:{now:%Y%m%d%H%M%S}
remove_clusters:
- "Over the Air Bootloading"
- "Occupancy Sensing" |
The There might be a better option with the actual zap (since it's in the container anyway) to do all this (https://github.com/project-chip/zap)? I haven't looked into what it can do/what's required/etc.. |
I personally don't mind the
The ZAP tool seems very much focussed around the GUI editor, there is support for templates for adding custom clusters and ensuring they are editible in the GUI. I dont think there is anyway to modify values from CLI commands, although documentation is pretty limited, and I didnt dig too deeply into the source code. |
Both would be kept, zap_config:
endpoint_types:
- name: "Centralized"
clusters:
- name: "Basic"
attribute_defaults:
"model identifier": DONGLE-E_R
"manufacturer name": SONOFF
"date code": template:{now:%Y%m%d%H%M%S}
"sw build id": template:{sdk_version}
remove:
- "Over the Air Bootloading" zap_config = manifest.get("zap_config", None)
zap_json_config: list[dict[str, typing.Any]] = []
if zap_config:
for endpoint_type in zap_config.get("endpoint_types", []):
for cluster in endpoint_type.get("clusters", []):
for to_remove in cluster.get("remove", []):
zap_json_config += zap_delete_cluster(
to_remove, endpoint_type["name"]
)
for attribute_name, default_value in cluster.get(
"attribute_defaults", {}
).items():
zap_json_config += zap_set_cluster_attribute(
attribute_name,
expand_template(default_value, value_template_env),
endpoint_type["name"],
cluster["name"],
)
# JSON config
for json_config in (
DEFAULT_JSON_CONFIG + manifest.get("json_config", []) + zap_json_config
): (with some adjusted versions of the utils from previous post, using |
In the context of a Zigbee router that is probably true, but lets say I now want to build a Thread Matter device, the requirements there are sure to differ from Zigbee use case. |
Yes, but it's just a matter (/joke) of properly defining the format of the |
This PR has been sitting for a while. In terms of feature use, are there any router firmwares or other device types being built that aren't a copy/paste of one another? I'm deciding between the DSL and just using |
I went with zero yaml-specific involvement for zigbee_router in my fork, everything is derived from existing information. Just easier, avoids mistakes (like forgetting to update the ZCL build date, etc.), and avoids editing many variables every time you want to try a new build internally (can easily differentiate by metadata...). Not sure how that fares for other devices (non adapter-turned-router scenario), but it also allows plain jq, so can still customize at will... |
See #84.
Example: