D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
proc
/
self
/
root
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
proxy
/
Filename :
junos.py
back
Copy
""" Interface with a Junos device via proxy-minion. To connect to a junos device \ via junos proxy, specify the host information in the pillar in '/srv/pillar/details.sls' .. code-block:: yaml proxy: proxytype: junos host: <ip or dns name of host> username: <username> port: 830 password: <secret> In '/srv/pillar/top.sls' map the device details with the proxy name. .. code-block:: yaml base: 'vmx': - details After storing the device information in the pillar, configure the proxy \ in '/etc/salt/proxy' .. code-block:: yaml master: <ip or hostname of salt-master> Run the salt proxy via the following command: .. code-block:: bash salt-proxy --proxyid=vmx """ import logging try: HAS_JUNOS = True import jnpr.junos import jnpr.junos.utils import jnpr.junos.utils.config import jnpr.junos.utils.sw from jnpr.junos.exception import ( ConnectAuthError, ConnectClosedError, ConnectError, ConnectRefusedError, ConnectTimeoutError, ProbeError, RpcError, RpcTimeoutError, ) from ncclient.operations.errors import TimeoutExpiredError from ncclient.transport.third_party.junos.ioproc import IOProc except ImportError: HAS_JUNOS = False __proxyenabled__ = ["junos"] thisproxy = {} log = logging.getLogger(__name__) class RebootActive: """ Class to get static variable, to indicate when a reboot/shutdown is being processed and the keep_alive should not probe the connection since it interferes with the shutdown process. """ reboot_shutdown = False def __init__(self, **kwargs): pass # Define the module's virtual name __virtualname__ = "junos" def __virtual__(): """ Only return if all the modules are available """ if not HAS_JUNOS: return ( False, "Missing dependency: The junos proxy minion requires the 'jnpr' Python" " module.", ) return __virtualname__ def init(opts): """ Open the connection to the Junos device, login, and bind to the Resource class """ opts["multiprocessing"] = False log.debug("Opening connection to junos") args = {"host": opts["proxy"]["host"]} optional_args = [ "user", "username", "password", "passwd", "port", "gather_facts", "mode", "baud", "attempts", "auto_probe", "ssh_private_key_file", "ssh_config", "normalize", "huge_tree", ] if "username" in opts["proxy"].keys(): opts["proxy"]["user"] = opts["proxy"].pop("username") proxy_keys = opts["proxy"].keys() for arg in optional_args: if arg in proxy_keys: args[arg] = opts["proxy"][arg] log.debug("Args: %s", args) thisproxy["conn"] = jnpr.junos.Device(**args) try: thisproxy["conn"].open() except ( ProbeError, ConnectAuthError, ConnectRefusedError, ConnectTimeoutError, ConnectError, ) as ex: log.error("%s : not able to initiate connection to the device", ex) thisproxy["initialized"] = False return if "timeout" in proxy_keys: timeout = int(opts["proxy"]["timeout"]) try: thisproxy["conn"].timeout = timeout except Exception as ex: # pylint: disable=broad-except log.error("Not able to set timeout due to: %s", str(ex)) else: log.debug("RPC timeout set to %d seconds", timeout) try: thisproxy["conn"].bind(cu=jnpr.junos.utils.config.Config) except Exception as ex: # pylint: disable=broad-except log.error("Bind failed with Config class due to: %s", ex) try: thisproxy["conn"].bind(sw=jnpr.junos.utils.sw.SW) except Exception as ex: # pylint: disable=broad-except log.error("Bind failed with SW class due to: %s", ex) thisproxy["initialized"] = True def initialized(): return thisproxy.get("initialized", False) def conn(): return thisproxy["conn"] def reboot_active(): RebootActive.reboot_shutdown = True def reboot_clear(): RebootActive.reboot_shutdown = False def get_reboot_active(): return RebootActive.reboot_shutdown def alive(opts): """ Validate and return the connection status with the remote device. .. versionadded:: 2018.3.0 """ dev = conn() # check if SessionListener sets a TransportError if there is a RpcTimeoutError thisproxy["conn"].connected = ping() local_connected = dev.connected if not local_connected: __salt__["event.fire_master"]( {}, "junos/proxy/{}/stop".format(opts["proxy"]["host"]) ) return local_connected def ping(): """ Ping? Pong! """ dev = conn() # Check that the underlying netconf connection still exists. if dev._conn is None: return False # call rpc only if ncclient queue is empty. If not empty that means other # rpc call is going on. if hasattr(dev._conn, "_session"): if ( dev._conn._session._transport is not None and dev._conn._session._transport.is_active() ) or ( dev._conn._session._transport is None and isinstance(dev._conn._session, IOProc) ): # there is no on going rpc call. buffer tell can be 1 as it stores # remaining char after "]]>]]>" which can be a new line char if dev._conn._session._buffer.tell() <= 1 and dev._conn._session._q.empty(): return _rpc_file_list(dev) else: log.debug("skipped ping() call as proxy already getting data") return True else: # ssh connection is lost return False else: # other connection modes, like telnet return _rpc_file_list(dev) def _rpc_file_list(dev): try: dev.rpc.file_list(path="/dev/null", dev_timeout=5) return True except (RpcTimeoutError, ConnectClosedError): try: dev.close() return False except (RpcError, ConnectError, TimeoutExpiredError): return False except AttributeError as ex: if "'NoneType' object has no attribute 'timeout'" in str(ex): return False def proxytype(): """ Returns the name of this proxy """ return "junos" def get_serialized_facts(): facts = dict(thisproxy["conn"].facts) if "version_info" in facts: facts["version_info"] = dict(facts["version_info"]) # For backward compatibility. 'junos_info' is present # only of in newer versions of facts. if "junos_info" in facts: for re in facts["junos_info"]: facts["junos_info"][re]["object"] = dict(facts["junos_info"][re]["object"]) return facts def shutdown(opts): """ This is called when the proxy-minion is exiting to make sure the connection to the device is closed cleanly. """ log.debug("Proxy module %s shutting down!!", opts["id"]) try: thisproxy["conn"].close() except Exception: # pylint: disable=broad-except pass