Skip to content

Commit

Permalink
add device discovery task
Browse files Browse the repository at this point in the history
  • Loading branch information
MarcelSuleiman committed Nov 30, 2023
1 parent 65d2291 commit 6c2afe1
Show file tree
Hide file tree
Showing 5 changed files with 644 additions and 328 deletions.
124 changes: 124 additions & 0 deletions uniconfig/python/frinx_worker/uniconfig/device_discovery.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,124 @@
from ipaddress import IPv4Address
from ipaddress import IPv6Address

import pydantic
import requests
from frinx.common.conductor_enums import TaskResultStatus
from frinx.common.frinx_rest import UNICONFIG_URL_BASE
from frinx.common.type_aliases import ListStr
from frinx.common.worker.service import ServiceWorkersImpl
from frinx.common.worker.service import WorkerImpl
from frinx.common.worker.task_def import TaskDefinition
from frinx.common.worker.task_def import TaskExecutionProperties
from frinx.common.worker.task_def import TaskInput
from frinx.common.worker.task_def import TaskOutput
from frinx.common.worker.task_result import TaskResult
from frinx_api.uniconfig import OperationsDiscoverPostResponse
from frinx_api.uniconfig.device.discovery.discover import Address
from frinx_api.uniconfig.device.discovery.discover import Input
from frinx_api.uniconfig.device.discovery.discover import TcpPortItem
from frinx_api.uniconfig.device.discovery.discover import UdpPortItem
from frinx_api.uniconfig.rest_api import Discover
from pydantic import Field
from pydantic import IPvAnyAddress
from pydantic import IPvAnyNetwork

from . import class_to_json
from .util import get_list_of_ip_addresses
from .util import parse_ranges


class DeviceDiscoveryWorkers(ServiceWorkersImpl):
class DeviceDiscoveryWorker(WorkerImpl):

class ExecutionProperties(TaskExecutionProperties):
exclude_empty_inputs: bool = False
transform_string_to_json_valid: bool = True

class WorkerDefinition(TaskDefinition):

name: str = 'UNICONFIG_device_discovery'
description: str = 'Verification of reachable devices in a network.'
labels: ListStr = ['BASICS', 'UNICONFIG']

class WorkerInput(TaskInput):
ip: list[Address]
tcp_port: list[TcpPortItem] | None = None
udp_port: list[UdpPortItem] | None = Field(None, max_length=500)

@pydantic.field_validator('ip', mode='before')
def validate_ip(cls, ip: str) -> list[Address]:

ip_list = get_list_of_ip_addresses(ip)
if len(ip_list) == 1:
if '/' in ip_list[0]:
address = Address(network=str(IPvAnyNetwork(ip_list[0])))
else:
address = Address(ip_address=str(IPvAnyAddress(ip_list[0])))
else:
try:
address = Address(
start_ipv4_address=str(IPv4Address(ip_list[0])),
end_ipv4_address=str(IPv4Address(ip_list[1]))
)
except ValueError:
address = Address(
start_ipv6_address=str(IPv6Address(ip_list[0])),
end_ipv6_address=str(IPv6Address(ip_list[1]))
)

return [address]

@pydantic.field_validator('tcp_port', mode='before')
def validate_tcp(cls, tcp_port: str) -> list[TcpPortItem]:
return [TcpPortItem(port=p) for p in parse_ranges(tcp_port.split(','))]

@pydantic.field_validator('udp_port', mode='before')
def validate_udp(cls, udp_port: str) -> list[UdpPortItem]:
return [UdpPortItem(port=p) for p in parse_ranges(udp_port.split(','))]

class WorkerOutput(TaskOutput):
output: OperationsDiscoverPostResponse

def execute(self, worker_input: WorkerInput) -> TaskResult[WorkerOutput]:
if Discover.request is None:
raise Exception(f'Failed to create request {Discover.request}')

if Discover.response is None:
raise Exception(f'Failed to create request {Discover.response}')

template = Input(
address=worker_input.ip,
tcp_port=worker_input.tcp_port or None,
udp_port=worker_input.udp_port or None,
)

response = requests.request(
url=UNICONFIG_URL_BASE + Discover.uri,
method=Discover.method,
data=class_to_json(
Discover.request(
input=template
),
),
)

if response.ok:
status = TaskResultStatus.COMPLETED

return TaskResult(
status=status,
output=self.WorkerOutput(
output=Discover.response(
output=response.json()['output']
)
)
)

status = TaskResultStatus.FAILED
return TaskResult(
status=status,
output=self.WorkerOutput(
output=response.json()
)
)
53 changes: 53 additions & 0 deletions uniconfig/python/frinx_worker/uniconfig/util.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@


def parse_ranges(list_of_strings: list[str]) -> list[int]:
"""Takes list containing integers (str) and ranges (str)
and returns corresponding list of integers.
Example:
input: ["3", "5-6" , "8", "10 - 12"]
output: [3, 5, 6, 8, 10, 11, 12]
"""
output = []
for x in list_of_strings:
try:
number = int(x)
output.append(number)
except ValueError:
try:
range_start, range_stop = map(int, x.split('-'))
except Exception:
# continue
raise ValueError(f"Value '{x}' does not represent an integer.")

output += list(range(range_start, range_stop + 1))
return output


def get_list_of_ip_addresses(ip_addresses: str) -> list[str]:
"""
Creates list of IP addresses.
Args:
ip_addresses:
String containing valid IP subnet e.g.: 172.16.1.0/24
or IP address range e.g.: 172.16.1.10-172.16.1.20
Returns:
Union[list, str]:
List containing first and last IP address of given range
e.g.: ['172.16.1.10', '172.16.1.20']
or '172.16.1.100/32'
"""
ip_addresses = ip_addresses.replace(' ', '')

if '/' in ip_addresses:
return [ip_addresses]

if '-' in ip_addresses:
ip_addresses_list = ip_addresses.split(sep='-', maxsplit=1)
start_ip = ip_addresses_list[0]
end_ip = ip_addresses_list[1]

return [start_ip, end_ip]

else:
return [ip_addresses]
Loading

0 comments on commit 6c2afe1

Please sign in to comment.