From 76abe4e889a804501dd7c74c5e673422e79697ac Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Sat, 8 Mar 2025 20:18:18 +0100 Subject: [PATCH 1/8] Minor changes, explain name of the folder --- tutorial/step01-initial-setup.md | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/tutorial/step01-initial-setup.md b/tutorial/step01-initial-setup.md index f75ccbd..cbcf90d 100644 --- a/tutorial/step01-initial-setup.md +++ b/tutorial/step01-initial-setup.md @@ -16,7 +16,7 @@ Be sure to enable debugging in your NetBox configuration by setting `DEBUG = Tru ### Clone the git Repository -Next, we'll clone the demo git repository from GitHub. First, `cd` into your preferred location (your home directory is probably fine), then clone the repo with `git clone`. We're checking out the `init` branch, which will provide us with an empty workspace to start. +Next, we'll clone the demo git repository from GitHub. First, `cd` into your preferred location (your home directory is probably fine), then clone the repo with `git clone`. We're checking out the `step00-empty` branch, which will provide us with an empty workspace to start. ```bash $ git clone --branch step00-empty https://github.com/netbox-community/netbox-plugin-demo @@ -34,7 +34,10 @@ Unpacking objects: 100% (58/58), done. ### Create `__init__.py` -The `PluginConfig` class holds all the information needs to know about our plugin to install it. First, we'll create a subdirectory to hold our plugin's Python code, as well as an `__init__.py` file to hold the `PluginConfig` definition. +Our plugin is for managing access lists in NetBox, so we'll give it an appropriate name, such as `netbox_access_lists`. + +First, we'll create a subdirectory to hold our plugin's Python code, as well as an `__init__.py` file to hold the `PluginConfig` definition. +The `PluginConfig` class holds all the information Netbox needs to know about our plugin to install it. ```bash $ mkdir netbox_access_lists @@ -145,7 +148,7 @@ Save the file and run the NetBox development server (if not already running): $ python netbox/manage.py runserver ``` -You should see the development server start successfully. Open NetBox in a new browser window, log in as a superuser, and navigate to the admin UI. Under **System > Installed Plugins** you should see our plugin listed. +You should see the development server start successfully. Open NetBox in a new browser window, log in as a superuser, and navigate to the admin UI. Under **Admin > System > Plugins** you should see our plugin listed. ![Django admin UI: Plugins list](/images/step01-django-admin-plugins.png) @@ -158,4 +161,3 @@ This completes our initial setup. Now, onto the fun stuff! [Step 2: Models](/tutorial/step02-models.md) :arrow_right: - From a736a9d7de2b3566753f8778f2afd843a64d672e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Wed, 12 Mar 2025 23:40:08 +0100 Subject: [PATCH 2/8] Add some links to reference materials --- tutorial/step02-models.md | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tutorial/step02-models.md b/tutorial/step02-models.md index 210a786..2968e8d 100644 --- a/tutorial/step02-models.md +++ b/tutorial/step02-models.md @@ -103,7 +103,7 @@ The protocol field is next. This will store the name of a protocol such as TCP o ) ``` -Next we need to define a source prefix. We're going to use a foreign key field to reference an instance of NetBox's `Prefix` model within its `ipam` app. Instead of importing the model class, we can instead reference it by name. And because we want this to be an _optional_ field, we'll also set `blank=True` and `null=True`. +Next we need to define a source prefix. We're going to use a foreign key field to reference an instance of NetBox's [`Prefix` model](https://netboxlabs.com/docs/netbox/en/stable/models/ipam/prefix/) within its `ipam` app. Instead of importing the model class, we can instead reference it by name. And because we want this to be an _optional_ field, we'll also set `blank=True` and `null=True`. ```python source_prefix = models.ForeignKey( @@ -119,7 +119,7 @@ Next we need to define a source prefix. We're going to use a foreign key field t Notice above that we've defined `related_name='+'`. This tells Django not to create a reverse relationship from the `Prefix` model to the `AccessListRule` model, because it wouldn't be very useful. -We also need to add a field for the source port number(s). We could use an integer field for this, however that would limit us to defining a single source port per rule. Instead, we can add an `ArrayField` to store a list of `PositiveIntegerField` values. Like `source_prefix`, this will also be an optional field, so we add `blank=True` and `null=True` as well. +We also need to add a field for the source port number(s). We could use an integer field for this, however that would limit us to defining a single source port per rule. Instead, we can add an [`ArrayField`](https://docs.djangoproject.com/en/stable/ref/contrib/postgres/fields/#arrayfield) to store a list of `PositiveIntegerField` values. Like `source_prefix`, this will also be an optional field, so we add `blank=True` and `null=True` as well. ```python source_ports = ArrayField( @@ -181,7 +181,7 @@ Looking back at our models, we see a few fields that would benefit from having p * Deny * Reject -We can define a `ChoiceSet` to store these pre-defined values for the user, to avoid the hassle of manually typing the name of the desired action each time. Back at the top of `models.py`, import NetBox's `ChoiceSet` class: +We can define a [`ChoiceSet`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/models/#choice-sets) to store these pre-defined values for the user, to avoid the hassle of manually typing the name of the desired action each time. Back at the top of `models.py`, import NetBox's `ChoiceSet` class: ```python from utilities.choices import ChoiceSet @@ -204,9 +204,9 @@ The `CHOICES` attribute must be an iterable of two- or three-value tuples, each * The raw value to be stored in the database * A human-friendly string for display -* A color for display in the UI (optional, see [available colors](https://docs.netbox.dev/en/stable/configuration/data-validation/#field_choices)) +* A color for display in the UI (optional, see [available colors](https://netboxlabs.com/docs/netbox/en/stable/configuration/data-validation/#field_choices)) -Additionally, we've added a `key` attribute: This will allow the NetBox administrator to replace or extend the plugin's default choices via NetBox's [`FIELD_CHOICES`](https://netbox.readthedocs.io/en/stable/configuration/optional-settings/#field_choices) configuration parameter. +Additionally, we've added a `key` attribute: This will allow the NetBox administrator to replace or extend the plugin's default choices via NetBox's [`FIELD_CHOICES`](https://netboxlabs.com/docs/netbox/en/stable/configuration/data-validation/#field_choices) configuration parameter. Now, we can reference this as the set of valid choices on the `default_action` and `action` model fields by passing it as the `choices` keyword argument. @@ -274,7 +274,7 @@ class AccessListRule(NetBoxModel): ## Create Schema Migrations -Now that we have our models defined, we need to generate a schema for the PostgreSQL database. While it's possible to create the tables and constraints by hand, it's _much_ easier to employ Django's [migrations feature](https://docs.djangoproject.com/en/4.0/topics/migrations/). This will inspect our model classes and generate the necessary migration files automatically. This is a two-step process: First we generate the migration file with the `makemigrations` management command, then we run `migrate` to apply it to the live database. +Now that we have our models defined, we need to generate a schema for the PostgreSQL database. While it's possible to create the tables and constraints by hand, it's _much_ easier to employ Django's [migrations feature](https://docs.djangoproject.com/en/stable/topics/migrations/). This will inspect our model classes and generate the necessary migration files automatically. This is a two-step process: First we generate the migration file with the `makemigrations` management command, then we run `migrate` to apply it to the live database. :warning: **Warning:** Before continuing, check that you've set `DEVELOPER=True` in NetBox's `configuration.py` file. This is necessary to disable a safeguard intended to prevent people from creating new migrations mistakenly. From 7e32690d40de9c0c9f22b3cef01a51f7f903a01c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Fri, 14 Mar 2025 11:50:42 +0100 Subject: [PATCH 3/8] Add link to NetBoxTable. Add image of table --- tutorial/step03-tables.md | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/tutorial/step03-tables.md b/tutorial/step03-tables.md index 1d651c3..698dbc0 100644 --- a/tutorial/step03-tables.md +++ b/tutorial/step03-tables.md @@ -15,7 +15,7 @@ $ cd netbox_access_lists/ $ edit tables.py ``` -At the top of this file, import the `django-tables2` library. This will provide the column classes for fields we wish to customize. We'll also import NetBox's `NetBoxTable` class, which will serve as the base class for our tables, and `ChoiceFieldColumn`. Finally we import our plugin's models from `models.py`. +At the top of this file, import the `django-tables2` library. This will provide the column classes for fields we wish to customize. We'll also import NetBox's [`NetBoxTable`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/tables/#netboxtable) class, which will serve as the base class for our tables, and `ChoiceFieldColumn`. Finally we import our plugin's models from `models.py`. ```python import django_tables2 as tables @@ -72,6 +72,10 @@ class AccessListTable(NetBoxTable): default_columns = ('name', 'rule_count', 'default_action') ``` +Once our plugin is finished, the table will look like this: + +![Access lists table](../images/step05-accesslist-list.png) + ### AccessListRuleTable We'll also create a table for our `AccessListRule` model using the same approach as above. Start by linkifying the `access_list` and `index` columns. The former will link to the parent access list, and the latter will link to the individual rule. We also want to declare `protocol` and `action` as `ChoiceFieldColumn` instances. @@ -106,4 +110,3 @@ This should be all we need to list these objects in the UI. Next, we'll define s :arrow_left: [Step 2: Models](/tutorial/step02-models.md) | [Step 4: Forms](/tutorial/step04-forms.md) :arrow_right: - From a48f4342b53a95c844d317d76a36e47093a2513b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Fri, 14 Mar 2025 13:27:02 +0100 Subject: [PATCH 4/8] Add links to reference material, and image of the form --- tutorial/step04-forms.md | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/tutorial/step04-forms.md b/tutorial/step04-forms.md index d3c0b51..0e5f4c5 100644 --- a/tutorial/step04-forms.md +++ b/tutorial/step04-forms.md @@ -13,7 +13,7 @@ $ cd netbox_access_lists/ $ edit forms.py ``` -At the top of the file, we'll import NetBox's `NetBoxModelForm` class, which will serve as the base class for our forms. We'll also import our plugin's models. +At the top of the file, we'll import NetBox's [`NetBoxModelForm`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/forms/#netboxmodelform) class, which will serve as the base class for our forms. We'll also import our plugin's models. ```python from netbox.forms import NetBoxModelForm @@ -32,7 +32,7 @@ class AccessListForm(NetBoxModelForm): fields = ('name', 'default_action', 'comments', 'tags') ``` -This alone is sufficient for our first model, but we can make one tweak: Instead of the default field that Django will generate for the `comments` model field, we can use NetBox's purpose-built `CommentField` class. (This handles some largely cosmetic details like setting a `help_text` and adjusting the field's layout.) To do this, simply import the `CommentField` class and override the form field: +This alone is sufficient for our first model, but we can make one tweak: Instead of the default field that Django will generate for the `comments` model field, we can use NetBox's purpose-built [`CommentField`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/forms/#utilities.forms.fields.fields.CommentField) class. (This handles some largely cosmetic details like setting a `help_text` and adjusting the field's layout.) To do this, simply import the `CommentField` class and override the form field: ```python from utilities.forms.fields import CommentField @@ -45,6 +45,10 @@ class AccessListForm(NetBoxModelForm): fields = ('name', 'default_action', 'comments', 'tags') ``` +Once our plugin is finished, the form will look like this: + +![Access lists form](../images/step05-accesslist-form.png) + ### AccessListRuleForm We'll create a form for `AccessListRule` following the same pattern. @@ -62,7 +66,7 @@ class AccessListRuleForm(NetBoxModelForm): By default, Django will create a "static" foreign key field for related objects. This renders as a dropdown list that's pre-populated with _all_ available objects. As you can imagine, in a NetBox instance with many thousands of objects this can get rather unwieldy. -To avoid this, NetBox provides the `DynamicModelChoiceField` class. This renders foreign key fields using a special dynamic widget backed by NetBox's REST API. This avoids the overhead imposed by the static field, and allows the user to conveniently search for the desired object. +To avoid this, NetBox provides the [`DynamicModelChoiceField`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/forms/#dynamic-object-fields) class. This renders foreign key fields using a special dynamic widget backed by NetBox's REST API. This avoids the overhead imposed by the static field, and allows the user to conveniently search for the desired object. :green_circle: **Tip:** The `DynamicModelMultipleChoiceField` class is also available for many-to-many fields, which support the assignment of multiple objects. @@ -97,4 +101,3 @@ With our models, tables, and forms all in place, next we'll create some views to :arrow_left: [Step 3: Tables](/tutorial/step03-tables.md) | [Step 5: Views](/tutorial/step05-views.md) :arrow_right: - From 607fd94c3c960f16c2b7db060b45b2ac48029251 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Fri, 14 Mar 2025 13:28:03 +0100 Subject: [PATCH 5/8] Explain URL parts. Fix typo. Add links --- tutorial/step05-views.md | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/tutorial/step05-views.md b/tutorial/step05-views.md index 64d8705..9fb3f31 100644 --- a/tutorial/step05-views.md +++ b/tutorial/step05-views.md @@ -2,7 +2,7 @@ Views are responsible for the business logic of your application. Generally, this means processing incoming requests, performing some action(s), and returning a response to the client. Each view typically has a URL associated with it, and can handle one or more types of HTTP requests (i.e. `GET` and/or `POST` requests). -Django provides a set of [generic view classes](https://docs.djangoproject.com/en/4.0/topics/class-based-views/generic-display/) which handle much of the boilerplate code needed to process requests. NetBox likewise provides a set of view classes to simplify the creation of views for creating, editing, deleting, and viewing objects. They also introduce support for NetBox-specific features such as custom fields and change logging. +Django provides a set of [generic view classes](https://docs.djangoproject.com/en/stable/topics/class-based-views/generic-display/) which handle much of the boilerplate code needed to process requests. NetBox likewise provides a set of view classes to simplify the creation of views for creating, editing, deleting, and viewing objects. They also introduce support for NetBox-specific features such as custom fields and change logging. In this step, we'll create a set of views for each of our plugin's models. @@ -17,7 +17,7 @@ $ cd netbox_access_lists/ $ edit views.py ``` -We'll need to import our plugin's `models`, `tables`, and `forms` modules: This is where everything we've built so far really comes together! We also need to import NetBox's generic views module, as it provides the base classes for our views. +We'll need to import our plugin's `models`, `tables`, and `forms` modules: This is where everything we've built so far really comes together! We also need to import [NetBox's generic views](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/views/#view-classes) module, as it provides the base classes for our views. ```python from netbox.views import generic @@ -56,7 +56,7 @@ class AccessListListView(generic.ObjectListView): :green_circle: **Tip:** It occurs to the author that having chosen a model name that ends with "List" might be a bit confusing here. Just remember that `AccessListView` is the _detail_ (single object) view, and `AccessListListView` is the _list_ (multiple objects) view. -Before we move on to the next view, do you remember the extra column we added to `AccessListTable` in step three? That column expects to find a count of rules assigned for each access list in the queryset, named `rule_count`. Let's add this to our queryset now. We can employ Django's `Count()` function to extend the SQL query and annotate the count of associated rules. (Don't forget to add the import statement up top.) +Before we move on to the next view, do you remember the extra column we added to `AccessListTable` in step three? That column expects to find a count of rules assigned for each access list in the queryset, named `rule_count`. Let's add this to our queryset now. We can employ Django's [`Count()`](https://docs.djangoproject.com/en/stable/ref/models/querysets/#aggregation-functions) function to extend the SQL query and annotate the count of associated rules. (Don't forget to add the import statement up top.) ```python from django.db.models import Count @@ -233,13 +233,16 @@ class AccessListRule(NetBoxModel): Now for the moment of truth: Has all our work thus far yielded functional UI views? Check that the development server is running, then open a browser and navigate to . You should see the access list list view and (if you followed in step two) a single access list named MyACL1. +The first `access-lists` in the URL is the `base_url` we defined in `__init__.py`. +The second is the path we defined in `urls.py`. + :blue_square: **Note:** This guide assumes that you're running the Django development server locally on port 8000. If your setup is different, you'll need to adjust the link above accordingly. ![Access lists list view](/images/step05-accesslist-list.png) We see that our table has successfully render the `name`, `rule_count`, and `default_action` columns that we defined in step three, and the `rule_count` column shows two rules assigned as expected. -If we click the "Add" button at top right, we'll be taken to the access list creation form. (Creating a new access list won'r work yet, but the form should render as seen below.) +If we click the "Add" button at top right, we'll be taken to the access list creation form. (Creating a new access list won't work yet, but the form should render as seen below.) ![Access list creation form](/images/step05-accesslist-form.png) From c0c6f0497a026702b0ce433a9e2959fe9092a2c2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Sat, 15 Mar 2025 20:00:17 +0100 Subject: [PATCH 6/8] Remove color buttons NetBox no longer applies color to buttons within navigation menu items --- tutorial/step07-navigation.md | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/tutorial/step07-navigation.md b/tutorial/step07-navigation.md index 9351025..7df27a4 100644 --- a/tutorial/step07-navigation.md +++ b/tutorial/step07-navigation.md @@ -13,7 +13,7 @@ $ cd netbox_access_lists/ $ edit navigation.py ``` -We'll need to import the `PluginMenuItem` class provided by NetBox to add new menu items; do this at the top of the file. +We'll need to import the [`PluginMenuItem`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/navigation/#menu-items) class provided by NetBox to add new menu items; do this at the top of the file. ```python from extras.plugins import PluginMenuItem @@ -55,11 +55,10 @@ That's much more convenient! ### Adding Menu Buttons -While we're at it, we can add direct links to the "add" views for access lists and rules as buttons. We'll need to import two additional classes at the top of `navigation.py`: `PluginMenuButton` and `ButtonColorChoices`. +While we're at it, we can add direct links to the "add" views for access lists and rules as buttons. We'll need to import one additional class at the top of `navigation.py`: `PluginMenuButton`. ```python from extras.plugins import PluginMenuButton, PluginMenuItem -from utilities.choices import ButtonColorChoices ``` `PluginMenuButton` is used similarly to `PluginMenuItem`: Instantiate it with the necessary keyword arguments to effect a menu button. These arguments are: @@ -67,7 +66,6 @@ from utilities.choices import ButtonColorChoices * `link` - The name of the URL path to which the button links * `title` - The text displayed when the user hovers over the button * `icon_class` - CSS class name(s) indicating the icon to display -* `color` - The button's color (choices are provided by `ButtonColorChoices`) Create these instances in `navigation.py` _above_ `menu_items`. Because each menu item expects to receive an iterable of button instances, we'll create each of these inside a list. @@ -77,7 +75,6 @@ accesslist_buttons = [ link='plugins:netbox_access_lists:accesslist_add', title='Add', icon_class='mdi mdi-plus-thick', - color=ButtonColorChoices.GREEN ) ] @@ -86,7 +83,6 @@ accesslistrule_buttons = [ link='plugins:netbox_access_lists:accesslistrule_add', title='Add', icon_class='mdi mdi-plus-thick', - color=ButtonColorChoices.GREEN ) ] ``` @@ -117,4 +113,3 @@ Now we should see green "add" buttons appear next to our menu links. :arrow_left: [Step 6: Templates](/tutorial/step06-templates.md) | [Step 8: Filter Sets](/tutorial/step08-filter-sets.md) :arrow_right: - From ca1c2e8283bfa6aad7fce3e846c4c674011126f1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Sat, 15 Mar 2025 20:04:29 +0100 Subject: [PATCH 7/8] Add link to NetBoxModelFilterSetForm class --- tutorial/step08-filter-sets.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tutorial/step08-filter-sets.md b/tutorial/step08-filter-sets.md index 5f9a7df..b90c61c 100644 --- a/tutorial/step08-filter-sets.md +++ b/tutorial/step08-filter-sets.md @@ -13,7 +13,7 @@ $ cd netbox_access_lists/ $ edit filtersets.py ``` -At the top of this file, we'll import NetBox's `NetBoxModelFilterSet` class, which will serve as the base class for our filter set, as well as our `AccessListRule` model. (In the interest of brevity, we're only going to create a filter set for one model, but it should be clear how to replicate this approach for the `AccessList` model as well.) +At the top of this file, we'll import NetBox's `NetBoxModelFilterSet` class, which will serve as the base class for our filter set, as well as our `AccessListRule` model. (In the interest of brevity, we're only going to create a filter set for one model, but it should be clear how to replicate this approach for the `AccessList` model as well.) ```python from netbox.filtersets import NetBoxModelFilterSet @@ -41,7 +41,7 @@ This will return all rules whose description contains the queried string. Of cou ## Create a Filter Form -The filter set handles the "behind the scenes" process of filtering queries, but we also need to create a form class to render the filter fields in the UI. We'll add this to `forms.py`. First, import Django's `forms` module (which will provide the field classes we need) and append `NetBoxModelFilterSetForm` to the existing import statement for `netbox.forms`: +The filter set handles the "behind the scenes" process of filtering queries, but we also need to create a form class to render the filter fields in the UI. We'll add this to `forms.py`. First, import Django's `forms` module (which will provide the field classes we need) and append [`NetBoxModelFilterSetForm`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/forms/#netboxmodelfiltersetform) to the existing import statement for `netbox.forms`: ```python from django import forms From 11484a69478f9dcb1384f6275279c53929599f0a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrique=20Mat=C3=ADas=20S=C3=A1nchez=20=28Quique=29?= Date: Sat, 15 Mar 2025 20:07:46 +0100 Subject: [PATCH 8/8] Nested serializer classes are no longer required Switch to using the primary serializer and pass nested=True. --- tutorial/step09-rest-api.md | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tutorial/step09-rest-api.md b/tutorial/step09-rest-api.md index 397893d..4127595 100644 --- a/tutorial/step09-rest-api.md +++ b/tutorial/step09-rest-api.md @@ -20,7 +20,7 @@ Serializers are somewhat analogous to forms: They control the translation of cli $ edit api/serializers.py ``` -At the top of this file, we need to import the `serializers` module from the `rest_framework` library, as well as NetBox's `NetBoxModelSerializer` class and our plugin's own models: +At the top of this file, we need to import the `serializers` module from the `rest_framework` library, as well as NetBox's [`NetBoxModelSerializer`](https://netboxlabs.com/docs/netbox/en/stable/plugins/development/rest-api/) class and our plugin's own models: ```python from rest_framework import serializers @@ -104,12 +104,12 @@ There's an additional consideration when referencing related objects in a serial For instance, the `source_prefix` and `destination_prefix` fields both reference NetBox's core `ipam.Prefix` model. We can extend `AccessListRuleSerializer` to use NetBox's nested serializer for this model: ```python -from ipam.api.serializers import NestedPrefixSerializer +from ipam.api.serializers import PrefixSerializer # ... class AccessListRuleSerializer(NetBoxModelSerializer): url = serializers.HyperlinkedIdentityField(view_name='plugins-api:netbox_access_lists-api:accesslistrule-detail') - source_prefix = NestedPrefixSerializer() - destination_prefix = NestedPrefixSerializer() + source_prefix = PrefixSerializer(nested=True) + destination_prefix = PrefixSerializer(nested=True) ``` Now, our serializer will include an abridged representation of the source and/or destination prefixes for the object. We should do this with the `access_list` field as well, however we'll first need to create a nested serializer for the `AccessList` model. @@ -152,8 +152,8 @@ class AccessListRuleSerializer(NetBoxModelSerializer): view_name='plugins-api:netbox_access_lists-api:accesslistrule-detail' ) access_list = NestedAccessListSerializer() - source_prefix = NestedPrefixSerializer() - destination_prefix = NestedPrefixSerializer() + source_prefix = PrefixSerializer(nested=True) + destination_prefix = PrefixSerializer(nested=True) ``` ## Create the Views