| 
									
										
										
										
											2019-12-29 23:57:39 -07:00
										 |  |  | """Validate router configuration variables.""" | 
					
						
							| 
									
										
										
										
											2019-09-11 01:35:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-03 02:35:11 -07:00
										 |  |  | # Standard Library | 
					
						
							| 
									
										
										
										
											2020-03-21 01:44:38 -07:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2019-10-09 03:10:52 -07:00
										 |  |  | import re | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  | from typing import Any, Set, Dict, List, Tuple, Union, Optional | 
					
						
							| 
									
										
										
										
											2020-03-21 01:44:38 -07:00
										 |  |  | from pathlib import Path | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  | from ipaddress import IPv4Address, IPv6Address | 
					
						
							| 
									
										
										
										
											2019-10-04 16:54:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-03 02:35:11 -07:00
										 |  |  | # Third Party | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  | from pydantic import StrictInt, StrictStr, StrictBool, validator, root_validator | 
					
						
							| 
									
										
										
										
											2019-09-11 01:35:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-03 02:35:11 -07:00
										 |  |  | # Project | 
					
						
							| 
									
										
										
										
											2020-04-14 10:24:20 -07:00
										 |  |  | from hyperglass.log import log | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  | from hyperglass.util import ( | 
					
						
							|  |  |  |     get_driver, | 
					
						
							|  |  |  |     get_fmt_keys, | 
					
						
							|  |  |  |     resolve_hostname, | 
					
						
							|  |  |  |     validate_device_type, | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2020-05-30 08:48:15 -07:00
										 |  |  | from hyperglass.constants import SCRAPE_HELPERS, SUPPORTED_STRUCTURED_OUTPUT | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  | from hyperglass.exceptions.private import ConfigError, UnsupportedDevice | 
					
						
							| 
									
										
										
										
											2021-05-29 21:26:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-11 13:14:07 -07:00
										 |  |  | # Local | 
					
						
							|  |  |  | from .ssl import Ssl | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  | from ..main import HyperglassModel, HyperglassModelWithId | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  | from ..util import check_legacy_fields | 
					
						
							| 
									
										
										
										
											2020-10-11 13:14:07 -07:00
										 |  |  | from .proxy import Proxy | 
					
						
							| 
									
										
										
										
											2021-09-10 01:18:38 -07:00
										 |  |  | from .params import Params | 
					
						
							| 
									
										
										
										
											2021-04-23 23:10:03 -07:00
										 |  |  | from ..fields import SupportedDriver | 
					
						
							| 
									
										
										
										
											2020-10-11 13:14:07 -07:00
										 |  |  | from .network import Network | 
					
						
							|  |  |  | from .credential import Credential | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  | from ..commands.generic import Directive | 
					
						
							| 
									
										
										
										
											2019-09-13 00:36:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-10 00:43:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  | class Device(HyperglassModelWithId, extra="allow"): | 
					
						
							| 
									
										
										
										
											2019-12-29 23:57:39 -07:00
										 |  |  |     """Validation model for per-router config in devices.yaml.""" | 
					
						
							| 
									
										
										
										
											2019-09-11 01:35:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |     id: StrictStr | 
					
						
							| 
									
										
										
										
											2019-12-31 18:29:43 -07:00
										 |  |  |     name: StrictStr | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |     address: Union[IPv4Address, IPv6Address, StrictStr] | 
					
						
							| 
									
										
										
										
											2019-10-09 03:10:52 -07:00
										 |  |  |     network: Network | 
					
						
							|  |  |  |     credential: Credential | 
					
						
							| 
									
										
										
										
											2020-01-05 00:36:12 -07:00
										 |  |  |     proxy: Optional[Proxy] | 
					
						
							| 
									
										
										
										
											2021-02-10 00:43:40 -07:00
										 |  |  |     display_name: Optional[StrictStr] | 
					
						
							| 
									
										
										
										
											2021-02-06 00:19:29 -07:00
										 |  |  |     port: StrictInt = 22 | 
					
						
							| 
									
										
										
										
											2020-01-05 00:36:12 -07:00
										 |  |  |     ssl: Optional[Ssl] | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  |     type: StrictStr | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |     commands: List[Directive] | 
					
						
							| 
									
										
										
										
											2020-05-02 23:05:17 -07:00
										 |  |  |     structured_output: Optional[StrictBool] | 
					
						
							| 
									
										
										
										
											2021-04-23 23:10:03 -07:00
										 |  |  |     driver: Optional[SupportedDriver] | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |     attrs: Dict[str, str] = {} | 
					
						
							| 
									
										
										
										
											2020-05-02 23:05:17 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-10 00:43:40 -07:00
										 |  |  |     def __init__(self, **kwargs) -> None: | 
					
						
							|  |  |  |         """Set the device ID.""" | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  |         kwargs = check_legacy_fields("Device", **kwargs) | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |         _id, values = self._generate_id(kwargs) | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |         super().__init__(id=_id, **values) | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |         self._validate_directive_attrs() | 
					
						
							| 
									
										
										
										
											2021-02-10 00:43:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def _target(self): | 
					
						
							|  |  |  |         return str(self.address) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _generate_id(values: Dict) -> Tuple[str, Dict]: | 
					
						
							|  |  |  |         """Generate device id & handle legacy display_name field.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def generate_id(name: str) -> str: | 
					
						
							|  |  |  |             scrubbed = re.sub(r"[^A-Za-z0-9\_\-\s]", "", name) | 
					
						
							|  |  |  |             return "_".join(scrubbed.split()).lower() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         name = values.pop("name", None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if name is None: | 
					
						
							|  |  |  |             raise ValueError("name is required.") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         legacy_display_name = values.pop("display_name", None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if legacy_display_name is not None: | 
					
						
							| 
									
										
										
										
											2021-09-12 15:09:24 -07:00
										 |  |  |             log.warning("The 'display_name' field is deprecated. Use the 'name' field instead.") | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |             device_id = generate_id(legacy_display_name) | 
					
						
							|  |  |  |             display_name = legacy_display_name | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             device_id = generate_id(name) | 
					
						
							|  |  |  |             display_name = name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return device_id, {"name": display_name, "display_name": None, **values} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-11 13:56:20 -07:00
										 |  |  |     def export_api(self) -> Dict[str, Any]: | 
					
						
							|  |  |  |         """Export API-facing device fields.""" | 
					
						
							|  |  |  |         return { | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |             "id": self.id, | 
					
						
							| 
									
										
										
										
											2021-09-11 13:56:20 -07:00
										 |  |  |             "name": self.name, | 
					
						
							|  |  |  |             "network": self.network.display_name, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-11 00:47:01 -07:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def directive_commands(self) -> List[str]: | 
					
						
							|  |  |  |         """Get all commands associated with the device.""" | 
					
						
							|  |  |  |         return [ | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |             command | 
					
						
							|  |  |  |             for directive in self.commands | 
					
						
							|  |  |  |             for rule in directive.rules | 
					
						
							|  |  |  |             for command in rule.commands | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def directive_ids(self) -> List[str]: | 
					
						
							|  |  |  |         """Get all directive IDs associated with the device.""" | 
					
						
							|  |  |  |         return [directive.id for directive in self.commands] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def has_directives(self, *directive_ids: str) -> bool: | 
					
						
							|  |  |  |         """Determine if a directive is used on this device.""" | 
					
						
							|  |  |  |         for directive_id in directive_ids: | 
					
						
							|  |  |  |             if directive_id in self.directive_ids: | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-11 00:47:01 -07:00
										 |  |  |     def _validate_directive_attrs(self) -> None: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |         # Set of all keys except for built-in key `target`. | 
					
						
							|  |  |  |         keys = { | 
					
						
							|  |  |  |             key | 
					
						
							| 
									
										
										
										
											2021-09-11 00:47:01 -07:00
										 |  |  |             for group in [get_fmt_keys(command) for command in self.directive_commands] | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |             for key in group | 
					
						
							|  |  |  |             if key != "target" | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         attrs = {k: v for k, v in self.attrs.items() if k in keys} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Verify all keys in associated commands contain values in device's `attrs`. | 
					
						
							|  |  |  |         for key in keys: | 
					
						
							|  |  |  |             if key not in attrs: | 
					
						
							|  |  |  |                 raise ConfigError( | 
					
						
							|  |  |  |                     "Device '{d}' has a command that references attribute '{a}', but '{a}' is missing from device attributes", | 
					
						
							|  |  |  |                     d=self.name, | 
					
						
							|  |  |  |                     a=key, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |     @validator("address") | 
					
						
							|  |  |  |     def validate_address(cls, value, values): | 
					
						
							|  |  |  |         """Ensure a hostname is resolvable.""" | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |         if not isinstance(value, (IPv4Address, IPv6Address)): | 
					
						
							|  |  |  |             if not any(resolve_hostname(value)): | 
					
						
							|  |  |  |                 raise ConfigError( | 
					
						
							|  |  |  |                     "Device '{d}' has an address of '{a}', which is not resolvable.", | 
					
						
							|  |  |  |                     d=values["name"], | 
					
						
							|  |  |  |                     a=value, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |         return value | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-02 23:05:17 -07:00
										 |  |  |     @validator("structured_output", pre=True, always=True) | 
					
						
							| 
									
										
										
										
											2021-04-23 23:10:03 -07:00
										 |  |  |     def validate_structured_output(cls, value: bool, values: Dict) -> bool: | 
					
						
							|  |  |  |         """Validate structured output is supported on the device & set a default.""" | 
					
						
							| 
									
										
										
										
											2020-05-02 23:05:17 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 02:38:08 -07:00
										 |  |  |         if value is True: | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |             if values["type"] not in SUPPORTED_STRUCTURED_OUTPUT: | 
					
						
							| 
									
										
										
										
											2021-09-13 02:38:08 -07:00
										 |  |  |                 raise ConfigError( | 
					
						
							|  |  |  |                     "The 'structured_output' field is set to 'true' on device '{d}' with " | 
					
						
							|  |  |  |                     + "NOS '{n}', which does not support structured output", | 
					
						
							|  |  |  |                     d=values["name"], | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |                     n=values["type"], | 
					
						
							| 
									
										
										
										
											2021-09-13 02:38:08 -07:00
										 |  |  |                 ) | 
					
						
							|  |  |  |             return value | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |         elif value is None and values["type"] in SUPPORTED_STRUCTURED_OUTPUT: | 
					
						
							| 
									
										
										
										
											2020-05-02 23:05:17 -07:00
										 |  |  |             value = True | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             value = False | 
					
						
							|  |  |  |         return value | 
					
						
							| 
									
										
										
										
											2019-09-11 01:35:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-21 01:44:38 -07:00
										 |  |  |     @validator("ssl") | 
					
						
							|  |  |  |     def validate_ssl(cls, value, values): | 
					
						
							| 
									
										
										
										
											2021-09-07 22:58:39 -07:00
										 |  |  |         """Set default cert file location if undefined.""" | 
					
						
							| 
									
										
										
										
											2020-03-21 01:44:38 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if value is not None: | 
					
						
							|  |  |  |             if value.enable and value.cert is None: | 
					
						
							|  |  |  |                 app_path = Path(os.environ["hyperglass_directory"]) | 
					
						
							|  |  |  |                 cert_file = app_path / "certs" / f'{values["name"]}.pem' | 
					
						
							|  |  |  |                 if not cert_file.exists(): | 
					
						
							|  |  |  |                     log.warning("No certificate found for device {d}", d=values["name"]) | 
					
						
							|  |  |  |                     cert_file.touch() | 
					
						
							|  |  |  |                 value.cert = cert_file | 
					
						
							|  |  |  |         return value | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  |     @root_validator(pre=True) | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |     def validate_device_commands(cls, values: Dict) -> Dict: | 
					
						
							|  |  |  |         """Validate & rewrite device type, set default commands.""" | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 14:11:07 -07:00
										 |  |  |         _type = values.get("type") | 
					
						
							|  |  |  |         if _type is None: | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |             # Ensure device type is defined. | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  |             raise ValueError( | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |                 f"Device {values['name']} is missing a 'type' (Network Operating System) property." | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |         if _type in SCRAPE_HELPERS.keys(): | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  |             # Rewrite NOS to helper value if needed. | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |             _type = SCRAPE_HELPERS[_type] | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |         # Verify device type is supported by hyperglass. | 
					
						
							|  |  |  |         supported, _ = validate_device_type(_type) | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  |         if not supported: | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |             raise UnsupportedDevice(_type) | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |         values["type"] = _type | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         commands = values.get("commands") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if commands is None: | 
					
						
							|  |  |  |             # If no commands are defined, set commands to the NOS. | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |             inferred = values["type"] | 
					
						
							| 
									
										
										
										
											2020-11-02 23:08:07 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # If the _telnet prefix is added, remove it from the command | 
					
						
							|  |  |  |             # profile so the commands are the same regardless of | 
					
						
							|  |  |  |             # protocol. | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  |             if "_telnet" in inferred: | 
					
						
							|  |  |  |                 inferred = inferred.replace("_telnet", "") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-29 21:26:03 -07:00
										 |  |  |             values["commands"] = [inferred] | 
					
						
							| 
									
										
										
										
											2021-01-16 12:47:38 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return values | 
					
						
							| 
									
										
										
										
											2019-09-11 01:35:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 23:10:03 -07:00
										 |  |  |     @validator("driver") | 
					
						
							|  |  |  |     def validate_driver(cls, value: Optional[str], values: Dict) -> Dict: | 
					
						
							|  |  |  |         """Set the correct driver and override if supported.""" | 
					
						
							| 
									
										
										
										
											2021-09-13 10:00:21 -07:00
										 |  |  |         return get_driver(values["type"], value) | 
					
						
							| 
									
										
										
										
											2021-04-23 23:10:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 00:36:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-10 23:13:27 -07:00
										 |  |  | class Devices(HyperglassModel, extra="allow"): | 
					
						
							| 
									
										
										
										
											2019-12-29 23:57:39 -07:00
										 |  |  |     """Validation model for device configurations.""" | 
					
						
							| 
									
										
										
										
											2019-09-11 01:35:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |     ids: List[StrictStr] = [] | 
					
						
							| 
									
										
										
										
											2019-12-31 18:29:43 -07:00
										 |  |  |     hostnames: List[StrictStr] = [] | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |     objects: List[Device] = [] | 
					
						
							| 
									
										
										
										
											2019-10-09 03:10:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |     def __init__(self, input_params: List[Dict]) -> None: | 
					
						
							| 
									
										
										
										
											2019-12-31 18:29:43 -07:00
										 |  |  |         """Import loaded YAML, initialize per-network definitions.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Remove unsupported characters from device names, dynamically | 
					
						
							|  |  |  |         set attributes for the devices class. Builds lists of common | 
					
						
							|  |  |  |         attributes for easy access in other modules. | 
					
						
							| 
									
										
										
										
											2019-09-11 01:35:31 -07:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |         objects = set() | 
					
						
							|  |  |  |         hostnames = set() | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |         ids = set() | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         init_kwargs = {} | 
					
						
							| 
									
										
										
										
											2019-10-09 03:10:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for definition in input_params: | 
					
						
							|  |  |  |             # Validate each router config against Router() model/schema | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |             device = Device(**definition) | 
					
						
							| 
									
										
										
										
											2019-10-09 03:10:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-30 08:48:15 -07:00
										 |  |  |             # Add router-level attributes (assumed to be unique) to | 
					
						
							|  |  |  |             # class lists, e.g. so all hostnames can be accessed as a | 
					
						
							|  |  |  |             # list with `devices.hostnames`, same for all router | 
					
						
							|  |  |  |             # classes, for when iteration over all routers is required. | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |             hostnames.add(device.name) | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |             ids.add(device.id) | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |             objects.add(device) | 
					
						
							| 
									
										
										
										
											2019-10-09 03:10:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-30 08:48:15 -07:00
										 |  |  |         # Convert the de-duplicated sets to a standard list, add lists | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |         # as class attributes. Sort router list by router name attribute | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |         init_kwargs["ids"] = list(ids) | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |         init_kwargs["hostnames"] = list(hostnames) | 
					
						
							| 
									
										
										
										
											2021-02-10 00:43:40 -07:00
										 |  |  |         init_kwargs["objects"] = sorted(objects, key=lambda x: x.name) | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         super().__init__(**init_kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __getitem__(self, accessor: str) -> Device: | 
					
						
							|  |  |  |         """Get a device by its name.""" | 
					
						
							|  |  |  |         for device in self.objects: | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |             if device.id == accessor: | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  |                 return device | 
					
						
							| 
									
										
										
										
											2021-05-29 21:48:59 -07:00
										 |  |  |             elif device.name == accessor: | 
					
						
							|  |  |  |                 return device | 
					
						
							| 
									
										
										
										
											2020-07-30 01:30:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         raise AttributeError(f"No device named '{accessor}'") | 
					
						
							| 
									
										
										
										
											2021-09-10 01:18:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-11 13:56:20 -07:00
										 |  |  |     def export_api(self) -> List[Dict[str, Any]]: | 
					
						
							|  |  |  |         """Export API-facing device fields.""" | 
					
						
							|  |  |  |         return [d.export_api() for d in self.objects] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-10 01:18:38 -07:00
										 |  |  |     def networks(self, params: Params) -> List[Dict[str, Any]]: | 
					
						
							|  |  |  |         """Group devices by network.""" | 
					
						
							|  |  |  |         names = {device.network.display_name for device in self.objects} | 
					
						
							|  |  |  |         return [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "display_name": name, | 
					
						
							|  |  |  |                 "locations": [ | 
					
						
							|  |  |  |                     { | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  |                         "id": device.id, | 
					
						
							| 
									
										
										
										
											2021-09-10 01:18:38 -07:00
										 |  |  |                         "name": device.name, | 
					
						
							|  |  |  |                         "network": device.network.display_name, | 
					
						
							|  |  |  |                         "directives": [c.frontend(params) for c in device.commands], | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     for device in self.objects | 
					
						
							| 
									
										
										
										
											2021-09-10 23:13:27 -07:00
										 |  |  |                     if device.network.display_name == name | 
					
						
							| 
									
										
										
										
											2021-09-10 01:18:38 -07:00
										 |  |  |                 ], | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for name in names | 
					
						
							|  |  |  |         ] | 
					
						
							| 
									
										
										
										
											2021-09-12 18:27:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def directive_plugins(self) -> Dict[Path, Tuple[StrictStr]]: | 
					
						
							|  |  |  |         """Get a mapping of plugin paths to associated directive IDs.""" | 
					
						
							|  |  |  |         result: Dict[Path, Set[StrictStr]] = {} | 
					
						
							|  |  |  |         # Unique set of all directives. | 
					
						
							|  |  |  |         directives = {directive for device in self.objects for directive in device.commands} | 
					
						
							|  |  |  |         # Unique set of all plugin file names. | 
					
						
							|  |  |  |         plugin_names = {plugin for directive in directives for plugin in directive.plugins} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for directive in directives: | 
					
						
							|  |  |  |             # Convert each plugin file name to a `Path` object. | 
					
						
							|  |  |  |             for plugin in (Path(p) for p in directive.plugins if p in plugin_names): | 
					
						
							|  |  |  |                 if plugin not in result: | 
					
						
							|  |  |  |                     result[plugin] = set() | 
					
						
							|  |  |  |                 result[plugin].add(directive.id) | 
					
						
							|  |  |  |         # Convert the directive set to a tuple. | 
					
						
							|  |  |  |         return {k: tuple(v) for k, v in result.items()} |