mirror of
				https://github.com/checktheroads/hyperglass
				synced 2024-05-11 05:55:08 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			90 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			90 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """Execute validated & constructed query on device.
 | |
| 
 | |
| Accepts input from front end application, validates the input and
 | |
| returns errors if input is invalid. Passes validated parameters to
 | |
| construct.py, which is used to build & run the Netmiko connections or
 | |
| hyperglass-frr API calls, returns the output back to the front end.
 | |
| """
 | |
| 
 | |
| # Standard Library
 | |
| import signal
 | |
| from typing import TYPE_CHECKING, Any, Dict, Union, Callable
 | |
| 
 | |
| # Project
 | |
| from hyperglass.log import log
 | |
| from hyperglass.state import use_state
 | |
| from hyperglass.exceptions.public import DeviceTimeout, ResponseEmpty
 | |
| 
 | |
| if TYPE_CHECKING:
 | |
|     from hyperglass.models.api import Query
 | |
|     from .drivers import Connection
 | |
|     from hyperglass.models.data import OutputDataModel
 | |
| 
 | |
| # Local
 | |
| from .drivers import AgentConnection, NetmikoConnection, ScrapliConnection
 | |
| 
 | |
| 
 | |
| def map_driver(driver_name: str) -> "Connection":
 | |
|     """Get the correct driver class based on the driver name."""
 | |
| 
 | |
|     if driver_name == "scrapli":
 | |
|         return ScrapliConnection
 | |
| 
 | |
|     elif driver_name == "hyperglass_agent":
 | |
|         return AgentConnection
 | |
| 
 | |
|     return NetmikoConnection
 | |
| 
 | |
| 
 | |
| def handle_timeout(**exc_args: Any) -> Callable:
 | |
|     """Return a function signal can use to raise a timeout exception."""
 | |
| 
 | |
|     def handler(*args: Any, **kwargs: Any) -> None:
 | |
|         raise DeviceTimeout(**exc_args)
 | |
| 
 | |
|     return handler
 | |
| 
 | |
| 
 | |
| async def execute(query: "Query") -> Union["OutputDataModel", str]:
 | |
|     """Initiate query validation and execution."""
 | |
|     params = use_state("params")
 | |
|     output = params.messages.general
 | |
| 
 | |
|     log.debug("Received query {!r}", query)
 | |
|     log.debug("Matched device config: {!r}", query.device)
 | |
| 
 | |
|     mapped_driver = map_driver(query.device.driver)
 | |
|     driver: "Connection" = mapped_driver(query.device, query)
 | |
| 
 | |
|     signal.signal(
 | |
|         signal.SIGALRM,
 | |
|         handle_timeout(error=TimeoutError("Connection timed out"), device=query.device),
 | |
|     )
 | |
|     signal.alarm(params.request_timeout - 1)
 | |
| 
 | |
|     if query.device.proxy:
 | |
|         proxy = driver.setup_proxy()
 | |
|         with proxy() as tunnel:
 | |
|             response = await driver.collect(tunnel.local_bind_host, tunnel.local_bind_port)
 | |
|     else:
 | |
|         response = await driver.collect()
 | |
| 
 | |
|     output = await driver.response(response)
 | |
| 
 | |
|     if isinstance(output, str):
 | |
|         # If the output is a string (not structured) and is empty,
 | |
|         # produce an error.
 | |
|         if output == "" or output == "\n":
 | |
|             raise ResponseEmpty(query=query)
 | |
| 
 | |
|     elif isinstance(output, Dict):
 | |
|         # If the output an empty dict, responses have data, produce an
 | |
|         # error.
 | |
|         if not output:
 | |
|             raise ResponseEmpty(query=query)
 | |
| 
 | |
|     log.debug("Output for query {!r}:\n{!r}", query, output)
 | |
|     signal.alarm(0)
 | |
| 
 | |
|     return output
 |