D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
modules
/
Filename :
napalm_formula.py
back
Copy
""" NAPALM Formula helpers ====================== .. versionadded:: 2019.2.0 This is an Execution Module providing helpers for various NAPALM formulas, e.g., napalm-interfaces-formula, napalm-bgp-formula, napalm-ntp-formula etc., meant to provide various helper functions to make the templates more readable. """ import copy import fnmatch import logging import salt.utils.dictupdate # Import salt modules import salt.utils.napalm from salt.defaults import DEFAULT_TARGET_DELIM from salt.utils.data import traverse_dict_and_list as _traverse_dict_and_list __proxyenabled__ = ["*"] __virtualname__ = "napalm_formula" log = logging.getLogger(__name__) def __virtual__(): """ Available only on NAPALM Minions. """ return salt.utils.napalm.virtual(__opts__, __virtualname__, __file__) def _container_path(model, key=None, container=None, delim=DEFAULT_TARGET_DELIM): """ Generate all the possible paths within an OpenConfig-like object. This function returns a generator. """ if not key: key = "" if not container: container = "config" for model_key, model_value in model.items(): if key: key_depth = "{prev_key}{delim}{cur_key}".format( prev_key=key, delim=delim, cur_key=model_key ) else: key_depth = model_key if model_key == container: yield key_depth else: yield from _container_path( model_value, key=key_depth, container=container, delim=delim ) def container_path(model, key=None, container=None, delim=DEFAULT_TARGET_DELIM): """ Return the list of all the possible paths in a container, down to the ``config`` container. This function can be used to verify that the ``model`` is a Python object correctly structured and respecting the OpenConfig hierarchy. model The OpenConfig-structured object to inspect. delim: ``:`` The key delimiter. In particular cases, it is indicated to use ``//`` as ``:`` might be already used in various cases, e.g., IPv6 addresses, interface name (e.g., Juniper QFX series), etc. CLI Example: .. code-block:: bash salt '*' napalm_formula.container_path "{'interfaces': {'interface': {'Ethernet1': {'config': {'name': 'Ethernet1'}}}}}" The example above would return a list with the following element: ``interfaces:interface:Ethernet1:config`` which is the only possible path in that hierarchy. Other output examples: .. code-block:: text - interfaces:interface:Ethernet1:config - interfaces:interface:Ethernet1:subinterfaces:subinterface:0:config - interfaces:interface:Ethernet2:config """ return list(_container_path(model)) def setval(key, val, dict_=None, delim=DEFAULT_TARGET_DELIM): """ Set a value under the dictionary hierarchy identified under the key. The target 'foo/bar/baz' returns the dictionary hierarchy {'foo': {'bar': {'baz': {}}}}. .. note:: Currently this doesn't work with integers, i.e. cannot build lists dynamically. CLI Example: .. code-block:: bash salt '*' formula.setval foo:baz:bar True """ if not dict_: dict_ = {} prev_hier = dict_ dict_hier = key.split(delim) for each in dict_hier[:-1]: if each not in prev_hier: prev_hier[each] = {} prev_hier = prev_hier[each] prev_hier[dict_hier[-1]] = copy.deepcopy(val) return dict_ def traverse(data, key, default=None, delimiter=DEFAULT_TARGET_DELIM): """ Traverse a dict or list using a colon-delimited (or otherwise delimited, using the ``delimiter`` param) target string. The target ``foo:bar:0`` will return ``data['foo']['bar'][0]`` if this value exists, and will otherwise return the dict in the default argument. Function will automatically determine the target type. The target ``foo:bar:0`` will return data['foo']['bar'][0] if data like ``{'foo':{'bar':['baz']}}`` , if data like ``{'foo':{'bar':{'0':'baz'}}}`` then ``return data['foo']['bar']['0']`` CLI Example: .. code-block:: bash salt '*' napalm_formula.traverse "{'foo': {'bar': {'baz': True}}}" foo:baz:bar """ return _traverse_dict_and_list(data, key, default=default, delimiter=delimiter) def dictupdate(dest, upd, recursive_update=True, merge_lists=False): """ Recursive version of the default dict.update Merges upd recursively into dest If recursive_update=False, will use the classic dict.update, or fall back on a manual merge (helpful for non-dict types like ``FunctionWrapper``). If ``merge_lists=True``, will aggregate list object types instead of replace. The list in ``upd`` is added to the list in ``dest``, so the resulting list is ``dest[key] + upd[key]``. This behaviour is only activated when ``recursive_update=True``. By default ``merge_lists=False``. """ return salt.utils.dictupdate.update( dest, upd, recursive_update=recursive_update, merge_lists=merge_lists ) def defaults(model, defaults_, delim="//", flipped_merge=False): """ Apply the defaults to a Python dictionary having the structure as described in the OpenConfig standards. model The OpenConfig model to apply the defaults to. defaults The dictionary of defaults. This argument must equally be structured with respect to the OpenConfig standards. For ease of use, the keys of these support glob matching, therefore we don't have to provide the defaults for each entity but only for the entity type. See an example below. delim: ``//`` The key delimiter to use. Generally, ``//`` should cover all the possible cases, and you don't need to override this value. flipped_merge: ``False`` Whether should merge the model into the defaults, or the defaults into the model. Default: ``False`` (merge the model into the defaults, i.e., any defaults would be overridden by the values from the ``model``). CLI Example: .. code-block:: bash salt '*' napalm_formula.defaults "{'interfaces': {'interface': {'Ethernet1': {'config': {'name': 'Ethernet1'}}}}}" "{'interfaces': {'interface': {'*': {'config': {'enabled': True}}}}}" As one can notice in the example above, the ``*`` corresponds to the interface name, therefore, the defaults will be applied on all the interfaces. """ merged = {} log.debug("Applying the defaults:") log.debug(defaults_) log.debug("openconfig like dictionary:") log.debug(model) for model_path in _container_path(model, delim=delim): for default_path in _container_path(defaults_, delim=delim): log.debug("Comparing %s to %s", model_path, default_path) if not fnmatch.fnmatch(model_path, default_path) or not len( model_path.split(delim) ) == len(default_path.split(delim)): continue log.debug("%s matches %s", model_path, default_path) # If there's a match, it will build the dictionary from the top devault_val = _traverse_dict_and_list( defaults_, default_path, delimiter=delim ) merged = setval(model_path, devault_val, dict_=merged, delim=delim) log.debug("Complete default dictionary") log.debug(merged) log.debug("Merging with the model") log.debug(model) if flipped_merge: return salt.utils.dictupdate.update(model, merged) return salt.utils.dictupdate.update(merged, model) def render_field(dictionary, field, prepend=None, append=None, quotes=False, **opts): """ Render a field found under the ``field`` level of the hierarchy in the ``dictionary`` object. This is useful to render a field in a Jinja template without worrying that the hierarchy might not exist. For example if we do the following in Jinja: ``{{ interfaces.interface.Ethernet5.config.description }}`` for the following object: ``{'interfaces': {'interface': {'Ethernet1': {'config': {'enabled': True}}}}}`` it would error, as the ``Ethernet5`` key does not exist. With this helper, we can skip this and avoid existence checks. This must be however used with care. dictionary The dictionary to traverse. field The key name or part to traverse in the ``dictionary``. prepend: ``None`` The text to prepend in front of the text. Usually, we need to have the name of the field too when generating the configuration. append: ``None`` Text to append at the end. quotes: ``False`` Whether should wrap the text around quotes. CLI Example: .. code-block:: bash salt '*' napalm_formula.render_field "{'enabled': True}" enabled # This would return the value of the ``enabled`` leaf key salt '*' napalm_formula.render_field "{'enabled': True}" description # This would not error Jinja usage example: .. code-block:: jinja {%- set config = {'enabled': True, 'description': 'Interface description'} %} {{ salt.napalm_formula.render_field(config, 'description', quotes=True) }} The example above would be rendered on Arista / Cisco as: .. code-block:: text description "Interface description" While on Junos (the semicolon is important to be added, otherwise the configuration won't be accepted by Junos): .. code-block:: text description "Interface description"; """ value = traverse(dictionary, field) if value is None: return "" if prepend is None: prepend = field.replace("_", "-") if append is None: if __grains__["os"] in ("junos",): append = ";" else: append = "" if quotes: value = '"{value}"'.format(value=value) return "{prepend} {value}{append}".format( prepend=prepend, value=value, append=append ) def render_fields(dictionary, *fields, **opts): """ This function works similarly to :mod:`render_field <salt.modules.napalm_formula.render_field>` but for a list of fields from the same dictionary, rendering, indenting and distributing them on separate lines. dictionary The dictionary to traverse. fields A list of field names or paths in the dictionary. indent: ``0`` The indentation to use, prepended to the rendered field. separator: ``\\n`` The separator to use between fields. CLI Example: .. code-block:: bash salt '*' napalm_formula.render_fields "{'mtu': 68, 'description': 'Interface description'}" mtu description Jinja usage example: .. code-block:: jinja {%- set config={'mtu': 68, 'description': 'Interface description'} %} {{ salt.napalm_formula.render_fields(config, 'mtu', 'description', quotes=True) }} The Jinja example above would generate the following configuration: .. code-block:: text mtu "68" description "Interface description" """ results = [] for field in fields: res = render_field(dictionary, field, **opts) if res: results.append(res) if "indent" not in opts: opts["indent"] = 0 if "separator" not in opts: opts["separator"] = "\n{ind}".format(ind=" " * opts["indent"]) return opts["separator"].join(results)