Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Remove the ObjectAliasTarget model #49

Merged
merged 12 commits into from
Nov 2, 2024
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ Details of the data flow specifications, displaying all the IP addresses, IP ran

### Application

![All the data flows mapped to one application](docs/media/readme-dataflow-details.png)
![All the data flows mapped to one application](docs/media/tuto-application-details.png)
The application allows you to group all the related data flows.

### Device tab views
Expand Down
18 changes: 6 additions & 12 deletions docs/data-model.md
Original file line number Diff line number Diff line change
Expand Up @@ -42,13 +42,6 @@ It was decided not to use the native **Service** object:
To ease maintenance, the IP Addresses, IP Ranges and Prefixes are grouped in Object Aliases. These can be seen as reusable groups of addresses that can be a source or a destination to one or several data flows.


## Plugin limitations

The plugin cannot modify the native NetBox objects and add ForeignKey relations. However, it requires a Many to Many relationship between the Object Aliases (or the source/destination) and different types of objects (IP Address, IP Range and Prefix) and Generic Many to Many relationships are not supported by Django (nor a good idea in general).

To circumvent this limitation, the plugin makes use of a proxy object, Object Alias Target, that represents one-to-one a NetBox's native IP Address, IP Range or Prefix. These proxy objects are automatically created when needed and destroyed when the linked object is deleted, and the user should not have to worry about them. However, some usecases (like creating object aliases via the API) may need to be aware of them.


## Data Model

The following sections explain the different objects created by the plugin.
Expand Down Expand Up @@ -82,19 +75,20 @@ Examples of roles:

Data Flows should have a source, a destination, a protocol, source ports and destination ports. Only the protocol is mandatory.

By convention, if the list of source ports or destination ports is empty, this means "Any" port is accepted (for transport protocols with ports). The interface will display `Any`. API and exports will return an empty list.

**Data Flow Groups** form a forest of groups. They can also be assigned to an Application. Data Flow Groups can be enabled and disabled and inherit the status of their parent. Disabled Data Flow Groups disable all the Data Flows contained within.

### Object Alias

**Object Aliases** are a group of references to other NetBox objects. Object Aliases are used as sources and destinations of Data Flows and corresponds to the groups or aliases used in firewall configuration.

Internally, Object Aliases contain **Object Alias Targets**, because Django cannot create ManyToMany relationships to generic objects. Object Alias Targets are not exposed in the interface and should be transparent for the user.

Object Aliases can contain:
Object Aliases can contain any number of:

* IP Addresses (`ipam.ipaddress`)
* IP Ranges (`ipam.iprange`)
* Prefixes (`ipam.prefix`)

If an IP Address is assigned to a device or virtual machine, that device is
also displayed.
There is no defined meaning for an empty object alias, but it can be used when:
* The aliased object is not documented in NetBox (e.g.: third party public IP addresses)
* The alias is "Any" / "Internet" destination
Binary file modified docs/media/data-model.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file removed docs/media/readme-application-dataflows.png
Binary file not shown.
Binary file modified docs/media/readme-dataflow-details.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-application-details.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-application-roles.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-applications.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-dataflow-db-new.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-dataflow-list-all.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-dataflow-new.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-dataflow-targets.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-dataflowgroups.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-device-tab.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-dfg-details.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-menu.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file removed docs/media/tuto-objectalias-addip.png
Binary file not shown.
Binary file removed docs/media/tuto-objectalias-after.png
Binary file not shown.
Binary file added docs/media/tuto-objectalias-details.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-objectalias-list-all.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-objectalias-list.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-objectalias-new.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified docs/media/tuto-vm-tab.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
22 changes: 8 additions & 14 deletions docs/quick-start.md
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ First, go the bottom of the NetBox navigation menu to find the Data Flow plugin,

![Menu of the plugin](media/tuto-menu.png)

Think of an **Object Alias** as a reusable group of IP addresses, ranges or prefixes that we can use as source and/or destination in many data flows. The technical reasons for its existence are detailed in the [data model](data-model.md).
Think of an **Object Alias** as a reusable group of IP addresses, ranges or prefixes that we can use as source and/or destination in many data flows. You can have more information in the [data model](data-model.md).

Our Object Aliases can contain any number of the following native NetBox objects:

Expand All @@ -48,19 +48,13 @@ Let's create three Object Aliases:
| acme_backend | Backend servers for Acme |
| acme_database | Database servers for Acme |

The creation is a two-step process: first, you create the alias, and then you can link aliased objects to it.
When you create or edit an object alias, you can link any IP Address, IP Range or Prefix. Zou can use the filter functions to search which addresses need to be added.

![A newly created alias](media/tuto-objectalias-new.png)
![Creation of a new alias](media/tuto-objectalias-new.png)

Create `acme_frontend` and open it, then click on "+ Add aliased objects" in the top right. There, you can select any IP Address, IP Range or Prefix to add, and you can use the filter functions to search which addresses need to be added.
Once created, your alias will look like that:

In you case, let's add the IP address of our three servers to the alias.

![Adding three IP addresses in an alias](media/tuto-objectalias-addip.png)

The objects contained in the alias are listed on its detail page.

![Detail page of the object alias](media/tuto-objectalias-after.png)
![Detail page of the object alias](media/tuto-objectalias-details.png)

Let's repeat the same process for 'acme_backend' and 'acme_database'.

Expand All @@ -78,14 +72,14 @@ For the plugin, a data flow is a network connection between some sources and som
* The sources are zero, one or more object aliases
* The destinations are also zero, one or more (potentially different) object aliases
* The protocol can be Any/ICMP/TCP/UDP/TCP+UDP/SCTP
* There can be a list of source and destination ports
* There can be a list of source and destination ports (by default, any port)
* The data flow can be marked as enabled or disabled.

> [!NOTE]
> The data flow can have an optional application and data flow group, which are explained later in the tutorial.

> [!TIP]
> By convention, if zero object aliases are specified as source (or as destination), this is interpreted (and displayed) as "Any".
> By convention, if zero object aliases are specified as source (or as destination), this is interpreted (and displayed) as "Any". Similarly, if no source or destination ports are defined, this is interpreted as "Any".

> [!TIP]
> You can change the list of available protocols in the configuration. Check [the Protocol Choices section](installation-configuration.md#protocol-choices) in the configuration documenation for details.
Expand Down Expand Up @@ -215,7 +209,7 @@ Let's create the remaining data flows.
> Because we have disabled the group "Acme Inc. external access", the data flow it contains is marked as *Disabled (Inherited)*.
> The data flow "Internal access to Acme backend API" also appears *Disabled*: its groups are all enabled, but the data flow was disabled directy.

The "Targets" tab in the data flow's detailed view can be used to resolve the object aliases and display the list of all IP address, ranges and prefixes in this data flow. The list displays when possible some context, e.g.: the device to which the IP is assigned or the VLAN a prefix is part of.
The "Targets" tab in the data flow's detailed view can be used to resolve the object aliases and display the list of all IP address, ranges and prefixes in this data flow.

![Targets tab of a data flow showing all its components](media/tuto-dataflow-targets.png)

Expand Down
7 changes: 1 addition & 6 deletions netbox_data_flows/__init__.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
from netbox.plugins import PluginConfig


__version__ = "1.0.7-dev"
__version__ = "1.1.0-dev"


class DataFlowsConfig(PluginConfig):
Expand All @@ -18,10 +18,5 @@ class DataFlowsConfig(PluginConfig):
min_version = "4.0.0"
max_version = "4.1.99"

def ready(self):
from . import signals # noqa: F401

super().ready()


config = DataFlowsConfig
69 changes: 25 additions & 44 deletions netbox_data_flows/api/serializers/objectaliases.py
Original file line number Diff line number Diff line change
@@ -1,57 +1,36 @@
from rest_framework import serializers

from core.models import ObjectType
from netbox.api.fields import SerializedPKRelatedField
from netbox.api.serializers import GenericObjectSerializer, NetBoxModelSerializer

from netbox_data_flows import models


__all__ = (
"ObjectAliasTargetSerializer",
"ObjectAliasSerializer",
)
from netbox.api.serializers import NetBoxModelSerializer

from ipam.api.serializers import IPAddressSerializer, IPRangeSerializer, PrefixSerializer
from ipam.models import IPAddress, IPRange, Prefix

class NestableGenericObjectSerializer(GenericObjectSerializer):
# Hack to get utilities.api.get_prefetches_for_serializer to
# stop prefetching our fields.

nested = True
from netbox_data_flows import models

class Meta:
model = ObjectType
fields = ["object_type", "object_id"]
brief_fields = ["object_type", "object_id"]


class ObjectAliasTargetSerializer(NetBoxModelSerializer):
url = serializers.HyperlinkedIdentityField(view_name="plugins-api:netbox_data_flows-api:objectaliastarget-detail")
target = NestableGenericObjectSerializer(
required=True,
many=False,
)

class Meta:
model = models.ObjectAliasTarget
fields = (
"display",
"id",
"target",
"url",
)
brief_fields = (
"display",
"id",
"url",
)
__all__ = ("ObjectAliasSerializer",)


class ObjectAliasSerializer(NetBoxModelSerializer):
url = serializers.HyperlinkedIdentityField(view_name="plugins-api:netbox_data_flows-api:objectalias-detail")
targets = SerializedPKRelatedField(
queryset=models.ObjectAliasTarget.objects.all(),
serializer=ObjectAliasTargetSerializer,
prefixes = SerializedPKRelatedField(
queryset=Prefix.objects.all(),
serializer=PrefixSerializer,
nested=True,
required=False,
many=True,
)
ip_ranges = SerializedPKRelatedField(
queryset=IPRange.objects.all(),
serializer=IPRangeSerializer,
nested=True,
required=False,
many=True,
)
ip_addresses = SerializedPKRelatedField(
queryset=IPAddress.objects.all(),
serializer=IPAddressSerializer,
nested=True,
required=False,
many=True,
Expand All @@ -65,7 +44,9 @@ class Meta:
"display",
"id",
"name",
"targets",
"prefixes",
"ip_ranges",
"ip_addresses",
"url",
)
brief_fields = (
Expand Down
1 change: 0 additions & 1 deletion netbox_data_flows/api/urls.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,6 @@
router.register("applications", views.ApplicationViewSet)
router.register("dataflows", views.DataFlowViewSet)
router.register("dataflow-groups", views.DataFlowGroupViewSet)
router.register("objectalias-target", views.ObjectAliasTargetViewSet)
router.register("objectalias", views.ObjectAliasViewSet)

urlpatterns = router.urls
7 changes: 0 additions & 7 deletions netbox_data_flows/api/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,13 +43,6 @@ class DataFlowGroupViewSet(NetBoxModelViewSet):
filterset_class = filtersets.DataFlowGroupFilterSet


class ObjectAliasTargetViewSet(NetBoxModelViewSet):
queryset = models.ObjectAliasTarget.objects.all()

serializer_class = serializers.ObjectAliasTargetSerializer
filterset_class = filtersets.ObjectAliasTargetFilterSet


class ObjectAliasViewSet(NetBoxModelViewSet):
queryset = models.ObjectAlias.objects.all()

Expand Down
8 changes: 4 additions & 4 deletions netbox_data_flows/filtersets/dataflows.py
Original file line number Diff line number Diff line change
Expand Up @@ -72,12 +72,12 @@ class DataFlowFilterSet(
label="Source Prefix (ID)",
method="filter_sources",
)
source_ipranges = ModelMultipleChoiceFilter(
source_ip_ranges = ModelMultipleChoiceFilter(
queryset=IPRange.objects.all(),
label="Source IP Ranges (ID)",
method="filter_sources",
)
source_ipaddresses = ModelMultipleChoiceFilter(
source_ip_addresses = ModelMultipleChoiceFilter(
queryset=IPAddress.objects.all(),
label="Source IP Addresses (ID)",
method="filter_sources",
Expand Down Expand Up @@ -107,12 +107,12 @@ class DataFlowFilterSet(
label="Destination Prefix (ID)",
method="filter_destinations",
)
destination_ipranges = ModelMultipleChoiceFilter(
destination_ip_ranges = ModelMultipleChoiceFilter(
queryset=IPRange.objects.all(),
label="Destination IP Ranges (ID)",
method="filter_destinations",
)
destination_ipaddresses = ModelMultipleChoiceFilter(
destination_ip_addresses = ModelMultipleChoiceFilter(
queryset=IPAddress.objects.all(),
label="Destination IP Addresses (ID)",
method="filter_destinations",
Expand Down
69 changes: 26 additions & 43 deletions netbox_data_flows/filtersets/objectaliases.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
from django.db.models import Q

from netbox.filtersets import BaseFilterSet, NetBoxModelFilterSet
from netbox.filtersets import NetBoxModelFilterSet

from dcim.models import Device
from ipam.models import IPAddress, IPRange, Prefix
Expand All @@ -12,55 +12,50 @@
from .filters import ModelMultipleChoiceFilter


__all__ = (
"ObjectAliasTargetFilterSet",
"ObjectAliasFilterSet",
)
__all__ = ("ObjectAliasFilterSet",)


class ObjectAliasTargetFilterSet(BaseFilterSet):
class ObjectAliasFilterSet(NetBoxModelFilterSet):
prefixes = ModelMultipleChoiceFilter(
queryset=Prefix.objects.all(),
label="Aliased Prefix (ID)",
label="Prefix (ID)",
method="filter_targets",
)
ipranges = ModelMultipleChoiceFilter(
ip_ranges = ModelMultipleChoiceFilter(
queryset=IPRange.objects.all(),
label="Aliased IP Ranges (ID)",
label="IP Ranges (ID)",
method="filter_targets",
)
ipaddresses = ModelMultipleChoiceFilter(
ip_addresses = ModelMultipleChoiceFilter(
queryset=IPAddress.objects.all(),
label="Aliased IP Addresses (ID)",
label="IP Addresses (ID)",
method="filter_targets",
)
devices = ModelMultipleChoiceFilter(
queryset=Device.objects.all(),
label="Aliased Devices (any IP address) (ID)",
label="Devices (any IP address) (ID)",
method="filter_devices",
)
virtual_machines = ModelMultipleChoiceFilter(
queryset=VirtualMachine.objects.all(),
label="Aliased Virtual Machine (any IP address) (ID)",
label="Virtual Machine (any IP address) (ID)",
method="filter_devices",
)

class Meta:
model = models.ObjectAliasTarget
fields = ("id",)
model = models.ObjectAlias
fields = (
"id",
"name",
"description",
)

def search(self, queryset, name, value):
return queryset

def filter_devices(self, queryset, name, value):
if not value:
if not value.strip():
return queryset

ip_addresses = get_device_ipaddresses(*value)
if not ip_addresses.exists():
return queryset.none()

return self.filter_targets(queryset, name, ip_addresses)
qs_filter = Q(name__icontains=value) | Q(description__icontains=value)
return queryset.filter(qs_filter)

# OR all the targets
# First, build a list
Expand All @@ -86,24 +81,12 @@ def qs(self):

return qs


class ObjectAliasFilterSet(NetBoxModelFilterSet, ObjectAliasTargetFilterSet):
targets = ModelMultipleChoiceFilter(
queryset=models.ObjectAliasTarget.objects.all(),
label="Target object (ID)",
)

class Meta:
model = models.ObjectAlias
fields = (
"id",
"name",
"description",
)

def search(self, queryset, name, value):
if not value.strip():
def filter_devices(self, queryset, name, value):
if not value:
return queryset

qs_filter = Q(name__icontains=value) | Q(description__icontains=value)
return queryset.filter(qs_filter)
ip_addresses = get_device_ipaddresses(*value)
if not ip_addresses.exists():
return queryset.none()

return self.filter_targets(queryset, name, ip_addresses)
Loading