Source code for hammurabi.rules.json

"""
This module adds JSON file support. JSON module is an extension for text rules
tailor made for .json files. The main difference lies in the way it works.
First, the .json file is parsed, then the modifications are made on the
already parsed file.
"""

from abc import abstractmethod
from pathlib import Path
from typing import Any, Optional

from hammurabi.rules.dictionaries import (
    DictKeyExists,
    DictKeyNotExists,
    DictKeyRenamed,
    DictValueExists,
    DictValueNotExists,
    SinglePathDictParsedRule,
)

try:
    import ujson as json
except ImportError:
    import json  # type: ignore


[docs]class SingleJSONFileRule(SinglePathDictParsedRule): """ Extend :class:`hammurabi.rules.dictionaries.SinglePathDictParsedRule` to handle parsed content manipulations on a single JSON file. """ def __init__( self, name: str, path: Optional[Path] = None, key: str = "", **kwargs ) -> None: super().__init__(name, path, key, loader=json.loads, **kwargs) def _write_dump(self, data: Any, delete: bool = False) -> None: """ Helper function to write the dump into file. :param data: The modified data :type data: :class:``hammurabi.rules.mixins.Any` :param delete: Indicate if the key should be deleted :type delete: bool """ self.param.write_text( json.dumps( self.set_by_selector(self.loaded_data, self.split_key, data, delete) ) )
[docs] @abstractmethod def task(self) -> Path: """ Abstract method representing how a :func:`hammurabi.rules.base.Rule.task` must be parameterized. Any difference in the parameters will result in pylint/mypy errors. For more details please check :func:`hammurabi.rules.base.Rule.task`. """
[docs]class JSONKeyExists(DictKeyExists, SingleJSONFileRule): """ Ensure that the given key exists. If needed, the rule will create a key with the given name, and optionally the specified value. In case the value is set, the value will be assigned to the key. If no value is set, the key will be created with an empty value. Example usage: >>> from pathlib import Path >>> from hammurabi import Law, Pillar, JSONKeyExists >>> >>> example_law = Law( >>> name="Name of the law", >>> description="Well detailed description what this law does.", >>> rules=( >>> JSONKeyExists( >>> name="Ensure service descriptor has stack", >>> path=Path("./service.json"), >>> key="stack", >>> value="my-awesome-stack", >>> ), >>> ) >>> ) >>> >>> pillar = Pillar() >>> pillar.register(example_law) .. warning:: Compared to :mod:`hammurabi.rules.text.LineExists`, this rule is NOT able to add a key before or after a target. """
[docs]class JSONKeyNotExists(DictKeyNotExists, SingleJSONFileRule): """ Ensure that the given key not exists. If needed, the rule will remove a key with the given name, including its value. Example usage: >>> from pathlib import Path >>> from hammurabi import Law, Pillar, JSONKeyNotExists >>> >>> example_law = Law( >>> name="Name of the law", >>> description="Well detailed description what this law does.", >>> rules=( >>> JSONKeyNotExists( >>> name="Ensure outdated_key is removed", >>> path=Path("./service.json"), >>> key="outdated_key", >>> ), >>> ) >>> ) >>> >>> pillar = Pillar() >>> pillar.register(example_law) """
[docs]class JSONKeyRenamed(DictKeyRenamed, SingleJSONFileRule): """ Ensure that the given key is renamed. In case the key can not be found, a ``LookupError`` exception will be raised to stop the execution. The execution must be stopped at this point, because if other rules depending on the rename they will fail otherwise. Example usage: >>> from pathlib import Path >>> from hammurabi import Law, Pillar, JSONKeyRenamed >>> >>> example_law = Law( >>> name="Name of the law", >>> description="Well detailed description what this law does.", >>> rules=( >>> JSONKeyRenamed( >>> name="Ensure service descriptor has dependencies", >>> path=Path("./service.json"), >>> key="development.depends_on", >>> value="dependencies", >>> ), >>> ) >>> ) >>> >>> pillar = Pillar() >>> pillar.register(example_law) """
[docs]class JSONValueExists(DictValueExists, SingleJSONFileRule): """ Ensure that the given key has the expected value(s). In case the key cannot be found, a ``LookupError`` exception will be raised to stop the execution. This rule is special in the way that the value can be almost anything. For more information please read the warning below. Example usage: >>> from pathlib import Path >>> from hammurabi import Law, Pillar, JSONValueExists >>> >>> example_law = Law( >>> name="Name of the law", >>> description="Well detailed description what this law does.", >>> rules=( >>> JSONValueExists( >>> name="Ensure service descriptor has dependencies", >>> path=Path("./service.json"), >>> key="development.dependencies", >>> value=["service1", "service2", "service3"], >>> ), >>> # Or >>> JSONValueExists( >>> name="Add infra alerting to existing alerting components", >>> path=Path("./service.json"), >>> key="development.alerting", >>> value={"infra": "#slack-channel-2"}, >>> ), >>> # Or >>> JSONValueExists( >>> name="Add support info", >>> path=Path("./service.json"), >>> key="development.supported", >>> value=True, >>> ), >>> # Or even >>> JSONValueExists( >>> name="Make sure that no development branch is set", >>> path=Path("./service.json"), >>> key="development.branch", >>> value=None, >>> ), >>> ) >>> ) >>> >>> pillar = Pillar() >>> pillar.register(example_law) .. warning:: Since the value can be anything from ``None`` to a list of lists, and rule piping passes the 1st argument (``path``) to the next rule the ``value`` parameter can not be defined in ``__init__`` before the ``path``. Hence the ``value`` parameter must have a default value. The default value is set to ``None``, which translates to the following: Using the ``JSONValueExists`` rule and not assigning value to ``value`` parameter will set the matching ``key``'s value to `None`` by default in the document. """
[docs]class JSONValueNotExists(DictValueNotExists, SingleJSONFileRule): """ Ensure that the key has no value given. In case the key cannot be found, a ``LookupError`` exception will be raised to stop the execution. Compared to ``hammurabi.rules.json.JSONValueExists``, this rule can only accept simple value for its ``value`` parameter. No ``list``, ``dict``, or ``None`` can be used. Based on the key's value's type if the value contains (or equals for simple types) value provided in the ``value`` parameter the value is: 1. Set to None (if the key's value's type is not a dict or list) 2. Removed from the list (if the key's value's type is a list) 3. Removed from the dict (if the key's value's type is a dict) Example usage: >>> from pathlib import Path >>> from hammurabi import Law, Pillar, JSONValueNotExists >>> >>> example_law = Law( >>> name="Name of the law", >>> description="Well detailed description what this law does.", >>> rules=( >>> JSONValueNotExists( >>> name="Remove decommissioned service from dependencies", >>> path=Path("./service.json"), >>> key="development.dependencies", >>> value="service4", >>> ), >>> ) >>> ) >>> >>> pillar = Pillar() >>> pillar.register(example_law) """