From 7201595c83cc3eee8217331bc984385dd9288a42 Mon Sep 17 00:00:00 2001 From: Leo Kirchner Date: Wed, 12 Jun 2024 16:30:50 +0200 Subject: [PATCH] docs: overhaul development docs section --- README.md | 4 + changes/468.documentation | 1 + changes/468.fixed | 1 + docs/dev/issues.md | 21 +++ docs/dev/jobs.md | 226 ++++++++++++++----------------- docs/{user => dev}/modeling.md | 0 docs/user/app_getting_started.md | 2 + mkdocs.yml | 15 +- 8 files changed, 137 insertions(+), 133 deletions(-) create mode 100644 changes/468.documentation create mode 100644 changes/468.fixed rename docs/{user => dev}/modeling.md (100%) diff --git a/README.md b/README.md index 9d0ae25fc..4b4a73fb3 100644 --- a/README.md +++ b/README.md @@ -38,6 +38,10 @@ This Nautobot application framework includes the following integrations: Read more about integrations [here](https://docs.nautobot.com/projects/ssot/en/latest/user/integrations). To enable and configure integrations follow the instructions from [the install guide](https://docs.nautobot.com/projects/ssot/en/latest/admin/install/#integrations-configuration). +### Building Your Own Integration + +If you need an integration that is not supported, you can build your own! Check out the documentation [here](https://docs.nautobot.com/projects/ssot/en/latest/dev/jobs). + ### Screenshots --- diff --git a/changes/468.documentation b/changes/468.documentation new file mode 100644 index 000000000..adff1b0e6 --- /dev/null +++ b/changes/468.documentation @@ -0,0 +1 @@ +Overhauled developer documentation structure. \ No newline at end of file diff --git a/changes/468.fixed b/changes/468.fixed new file mode 100644 index 000000000..cc4622c03 --- /dev/null +++ b/changes/468.fixed @@ -0,0 +1 @@ +Fixed SSoT jobs only allowing dry run unless you overwrote the `run` method. \ No newline at end of file diff --git a/docs/dev/issues.md b/docs/dev/issues.md index e69de29bb..ed6af5bb4 100644 --- a/docs/dev/issues.md +++ b/docs/dev/issues.md @@ -0,0 +1,21 @@ +# Reference: Common Issues and Solutions + +This pages describes common issues when implementing SSoT integrations and their respective solutions. + +## Converting Types Between Database and Pydantic + +Developers are able to override the default loading of basic parameters to control how that parameter is loaded from Nautobot. + +This only works with basic parameters belonging to the model and does not override more complex parameters (foreign keys, custom fields, custom relationships, etc.). + +To override a parameter, add a method with the name `load_param_{param_key}` to your adapter class inheriting from `NautobotAdapter`: + +```python +from nautobot_ssot.contrib import NautobotAdapter + +class YourSSoTNautobotAdapter(NautobotAdapter): + ... + def load_param_time_zone(self, parameter_name, database_object): + """Custom loader for `time_zone` parameter.""" + return str(getattr(database_object, parameter_name)) +``` diff --git a/docs/dev/jobs.md b/docs/dev/jobs.md index 17deadab1..9a8f60e89 100644 --- a/docs/dev/jobs.md +++ b/docs/dev/jobs.md @@ -1,181 +1,151 @@ -# Developing Data Source and Data Target Jobs +# Tutorial: Developing a Data Source Integration -A goal of this Nautobot app is to make it relatively quick and straightforward to develop and integrate your own system-specific Data Sources and Data Targets into Nautobot with a common UI and user experience. +This tutorial will walk you through building a custom data source (i.e. synchronizing data _to_ Nautobot) SSoT integration with a remote system. We will be using static data as the remote system, but it should be easy enough to substitute this later for any external systemy ou want to integrate with. -Familiarity with [DiffSync](https://diffsync.readthedocs.io/en/latest/) and with developing [Nautobot Jobs](https://nautobot.readthedocs.io/en/latest/additional-features/jobs/) is recommended. +*Familiarity with [DiffSync](https://diffsync.readthedocs.io/en/latest/) and with developing [Nautobot Jobs](https://nautobot.readthedocs.io/en/latest/additional-features/jobs/) is a plus, but important concepts will be explained or linked to along the way.** -## Quickstart Example +## Creating a New Nautobot App -The following code presents a minimum viable example for syncing VLANs from a remote system into Nautobot. You will have to adapt the following things to make this work for your use case: +To start building your own SSoT integration, you first need a project as well as the accompanying development environment. Network to Code provides the [Cookiecutter Nautobot App](https://github.com/nautobot/cookiecutter-nautobot-app) to make this as easy as possible for you. Check out the README of that project and bake a `nautobot-app` cookie. -- Swap out any mention of a "remote system" for your actual remote system, such as your IPAM tool -- Implement any models that you need, unless you are actually only interested in VLAN data -- Your `load` function in the remote adapter will probably be a little harder to implement, check the example integrations in this repository (under `nautobot_ssot/integrations`) +!!! note + The `nautobot-app-ssot` cookie is not yet updated to support the newest paradigms of building SSoT integrations, which is why it is recommended to instead use the `nautobot-app` cookie. + +Resume this tutorial whenever you have a folder on your development device with an up and running development environment. + +## Building the model -It contains 3 steps: +To synchronize data from a remote system into Nautobot, you need to first write a common model to bridge between the two systems. This model will be a [Pydantic](https://docs.pydantic.dev/latest/) model with built-in SSoT intelligence on top, specifically a subclass of `nautobot_ssot.contrib.NautobotModel`. -- Data modeling -- Adapters - - Nautobot - - Remote -- Nautobot Job +The following code snippet shows a basic VLAN model that includes the `name`, `vid` and `description` fields for the synchronization: ```python -# example_ssot_app/jobs.py +# tutorial_ssot_app/ssot/models.py from typing import Optional -from diffsync import Adapter from nautobot.ipam.models import VLAN -from nautobot.extras.jobs import Job -from nautobot_ssot.contrib import NautobotModel, NautobotAdapter -from nautobot_ssot.jobs import DataSource -from remote_system import APIClient # This is only an example +from nautobot_ssot.contrib import NautobotModel -# Step 1 - data modeling class VLANModel(NautobotModel): - """DiffSync model for VLANs.""" - _model = VLAN - _modelname = "vlan" - _identifiers = ("vid", "group__name") - _attributes = ("description",) - + _model = VLAN # SSoT models need to have a 1-to-1 mapping with a concrete Nautobot model + _modelname = "vlan" # This is the model name diffsync uses for its logging output + _identifiers = ("name",) # These are the fields that uniquely identify a model instance + _attributes = ("vid", "status__name", "description",) # The rest of the data fields go here + + # The field names here need to match those in the Nautobot model exactly, otherwise it doesn't work. + name: str vid: int - group__name: Optional[str] = None description: Optional[str] = None - -# Step 2.1 - the Nautobot adapter -class MySSoTNautobotAdapter(NautobotAdapter): - """DiffSync adapter for Nautobot.""" - vlan = VLANModel - top_level = ("vlan",) - -# Step 2.2 - the remote adapter -class MySSoTRemoteAdapter(Adapter): - """DiffSync adapter for remote system.""" - vlan = VLANModel - top_level = ("vlan",) - - def __init__(self, *args, api_client, job=None, **kwargs): - super().__init__(*args, **kwargs) - self.api_client = api_client - self.job = job - - def load(self): - for vlan in self.api_client.get_vlans(): - loaded_vlan = self.vlan(vid=vlan["vlan_id"], group__name=vlan["grouping"], description=vlan["description"]) - self.add(loaded_vlan) - -# Step 3 - the job -class ExampleDataSource(DataSource, Job): - """SSoT Job class.""" - class Meta: - name = "Example Data Source" - - def load_source_adapter(self): - self.source_adapter = MySSoTRemoteAdapter(api_client=APIClient(), job=self) - self.source_adapter.load() - - def load_target_adapter(self): - self.target_adapter = MySSoTNautobotAdapter(job=self) - self.target_adapter.load() - -jobs = [ExampleDataSource] + + # This is a foreign key to the `extras.status` model - its instances can be uniquely identified through its `name` + # field, which is why we specify this here + status__name: str ``` !!! note - This example is able to be so brief because usage of the `NautobotModel` class provides `create`, `update` and `delete` out of the box for the `VLANModel`. If you want to sync data _from_ Nautobot to another remote system, you will need to implement those yourself using whatever client or SDK provides the ability to write to that system. For examples, check out the existing integrations under `nautobot_ssot/integrations` + This example is able to be so brief because usage of the `NautobotModel` class provides `create`, `update` and `delete` out of the box for the `VLANModel`. If you want to sync data _from_ Nautobot _to_ another remote system, you will need to implement those yourself using whatever client or SDK provides the ability to write to that system. For examples, check out the existing integrations under `nautobot_ssot/integrations`. -The following sections describe the individual steps in more detail. -### Step 1 - Defining the Models +## Building the Adapters -The models use DiffSync, which in turn uses [Pydantic](https://docs.pydantic.dev/latest/) for its data modeling. Nautobot SSoT comes with a set of classes in `nautobot_ssot.contrib` that implement a lot of functionality for you automatically, provided you adhere to a set of rules. +On its own, the model is not particularly interesting. We have to combine it with an adapter to make it do interesting things. The adapter is responsible for pulling data out of Nautobot or your remote system (which is also Nautobot if you're following this tutorial). -The first rule is to define your models tightly after the Nautobot models themselves. Example: +### Building the Local adapter + +First, we define the adapter to load data from the local Nautobot instance - this is very straight-forward. ```python -from nautobot.tenancy.models import Tenant -from nautobot_ssot.contrib import NautobotModel +# tutorial_ssot_app/ssot/adapters.py +from nautobot_ssot.contrib import NautobotAdapter +from tutorial_ssot_app.ssot.models import VLANModel -class DiffSyncTenant(NautobotModel): - """An example model of a tenant.""" - _model = Tenant - _modelname = "tenant" - _identifiers = ("name",) - _attributes = ("description",) - name: str - description: str +class MySSoTNautobotAdapter(NautobotAdapter): + vlan = VLANModel # Map your model into the adapter + top_level = ("vlan",) # Tell the adapter to consider your model ``` -As you can see when looking at the [source code](https://github.com/nautobot/nautobot/blob/develop/nautobot/tenancy/models.py#L81) of the `nautobot.tenancy.models.Tenant` model, the fields on this model (`name` and `description`) match the names of the fields on the Nautobot model exactly. This enables the base class `NautobotModel` to dynamically load, create, update and delete your tenants without you needing to implement this functionality yourself. - -The above example shows the simplest field type (an attribute on the model), however, to build a production implementation you will need to understand how to identify different variants of fields by following the [modeling docs](../user/modeling.md). - -!!! warning - Currently, only normal fields, forwards foreign key fields and custom fields may be used in identifiers. Anything else is unsupported and will likely fail in unintuitive ways. - -### Step 2.1 - Creating the Nautobot Adapter - -Having created all your models, creating the Nautobot side adapter is very straight-forward: +Now create a couple of VLANs in the web GUI of Nautobot so that the adapter has some data to load. Once you're done, you can try out the adapter as follows: ```python -from nautobot_ssot.contrib import NautobotAdapter - -from your_ssot_app.models import DiffSyncDevice, DiffSyncPrefix, DiffSyncIPAddress +from tutorial_ssot_app.ssot.adapters import MySSoTNautobotAdapter +adapter = MySSoTNautobotAdapter(job=None) +adapter.load() +adapter.count("vlan") # This will return the amount of VLANs that were loaded from Nautobot -class YourSSoTNautobotAdapter(NautobotAdapter): - top_level = ("device", "prefix") - - device = DiffSyncDevice - prefix = DiffSyncPrefix - ip_address = DiffSyncIPAddress # Not in the `top_level` tuple, since it's a child of the prefix model +test_vlan = adapter.get_all("vlan")[0] # Now we retrieve an example VLAN to test the model with +test_vlan.update(attrs={"description": "My updated description"}) # Verify the update has worked using the web GUI +test_vlan.delete() # Verify the deletion has worked using the web GUI ``` -The `load` function is already implemented on this adapter and will automatically and recursively traverse any children relationships for you, provided the models are [defined correctly](../user/modeling.md). - -Developers are able to override the default loading of basic parameters to control how that parameter is loaded from Nautobot. +### Building the Remote Adapter -This only works with basic parameters belonging to the model and does not override more complex parameters (foreign keys, custom fields, custom relationships, etc.). +To synchronize a diff, we need to pull data from both sides. Therefore, we now need to build the remote adapter. In this example we are reading static data, but feel free to substitute this adapter with your remote system of choice. -To override a parameter, simply add a method with the name `load_param_{param_key}` to your adapter class inheriting from `NautobotAdapter`: +!!! note + Note that we are not subclassing from `nautobot_ssot.contrib.NautobotAdapter` here as we don't want the SSoT framework to auto-derive the loading implementation of the adapter for us, but rather want to handle this ourselves. ```python -from nautobot_ssot.contrib import NautobotAdapter +# tutorial_ssot_app/ssot/adapters.py +from diffsync import DiffSync -class YourSSoTNautobotAdapter(NautobotAdapter): - ... - def load_param_time_zone(self, parameter_name, database_object): - """Custom loader for `time_zone` parameter.""" - return str(getattr(database_object, parameter_name)) -``` +from tutorial_ssot_app.ssot.models import VLANModel -### Step 2.2 - Creating the Remote Adapter +class MySSoTRemoteAdapter(DiffSync): + """DiffSync adapter for remote system.""" + vlan = VLANModel + top_level = ("vlan",) + + def __init__(self, *args, api_client, job=None, **kwargs): + super().__init__(*args, **kwargs) + self.api_client = api_client + self.job = job -Regardless of which direction you are synchronizing data in, you need to write the `load` method for the remote adapter yourself. You can find many examples of how this can be done in the `nautobot_ssot.integrations` module, which contains pre-existing integrations with remote systems. + def load(self): + vlans = [ + {"name": "Servers", "vid": 100, "description": "Server VLAN Datacenter", "status__name": "Active"}, + {"name": "Printers", "vid": 200, "description": "Printer VLAN Office", "status__name": "Deprecated"}, + {"name": "Clients", "vid": 300, "description": "Client VLAN Office", "status__name": "Active"}, + ] + for vlan in vlans: + loaded_vlan = self.vlan( + name=vlan["name"], + vid=vlan["vid"], + description=vlan["description"], + status__name=vlan["status__name"] + ) + self.add(loaded_vlan) +``` -!!! note - As the features in `nautobot_ssot.contrib` are still very new, most existing integrations do not use them. The example job in `nautobot_ssot/jobs/examples.py` can serve as a fully working example. +Now you can verify that this adapter is working in a similar manner to how we did with the Nautobot adapter, feel free to try this! -### Step 3 - The Job +## Putting it Together in a Job -Develop a Job class, derived from either the `nautobot_ssot.jobs.base.DataSource` or `nautobot_ssot.jobs.base.DataTarget` classes provided by this Nautobot app, and implement the methods to populate the `self.source_adapter` and `self.target_adapter` attributes that are used by the built-in implementation of `sync_data`. This `sync_data` method is an opinionated way of running the process including some performance data (more about this in the next section), but you could overwrite it completely or any of the key hooks that it calls. +Having built the model and both adapters, we can now put everything together into a job to finish up the integration. -The methods [`calculate_diff`][nautobot_ssot.jobs.base.DataSyncBaseJob.calculate_diff] and [`execute_sync`][nautobot_ssot.jobs.base.DataSyncBaseJob.execute_sync] are both implemented by default, using the data that is loaded into the adapters through the respective methods. Note that `execute_sync` will _only_ execute when dry-run is set to false. +```python +# tutorial_ssot_app/jobs.py +from nautobot_ssot.jobs.base import DataSource +from nautobot.apps.jobs import Job, register_jobs +from tutorial_ssot_app.ssot.adapters import MySSoTRemoteAdapter, MySSoTNautobotAdapter -Optionally, on your Job class, also implement the [`lookup_object`][nautobot_ssot.jobs.base.DataSyncBaseJob.lookup_object], [`data_mapping`][nautobot_ssot.jobs.base.DataSyncBaseJob.data_mappings], and/or [`config_information`][nautobot_ssot.jobs.base.DataSyncBaseJob.config_information] APIs (to provide more information to the end user about the details of this Job), as well as the various metadata properties on your Job's Meta inner class. Refer to the example Jobs provided in this Nautobot app for examples and further details. -Install your Job via any of the supported Nautobot methods (installation into the `JOBS_ROOT` directory, inclusion in a Git repository, or packaging as part of an app) and it should automatically become available! +class TutorialDataSource(DataSource, Job): + class Meta: + name = "Example Data Source" -### Extra Step: Implementing `create`, `update` and `delete` + def load_source_adapter(self): + self.source_adapter = MySSoTRemoteAdapter(api_client=APIClient(), job=self) + self.source_adapter.load() -If you are synchronizing data _to_ Nautobot and not _from_ Nautobot, you can entirely skip this step. The `nautobot_ssot.contrib.NautobotModel` class provides this functionality automatically. + def load_target_adapter(self): + self.target_adapter = MySSoTNautobotAdapter(job=self) + self.target_adapter.load() -If you need to perform the `create`, `update` and `delete` operations on the remote system however, it will be up to you to implement those on your model. +register_jobs(TutorialDataSource) +``` -!!! note - You still want your models to adhere to the [modeling guide](../user/modeling.md), since it provides you the auto-generated `load` function for the diffsync adapter on the Nautobot side. +At this point, the job will show up in the web GUI, and you can enable it and even run it! You should see the objects you specified in the remote adapter being synchronized into Nautobot now. Also take a look at the "SSoT Sync Details" button in the top right of the job result page to get some more information on what diffsync is doing. -!!! warning - Special care should be taken when synchronizing new Devices with children Interfaces into a Nautobot instance that also defines Device Types with Interface components of the same name. When the new Device is created in Nautobot, its Interfaces will also be created as defined in the respective Device Type. As a result, when SSoT will attempt to create the children Interfaces loaded by the remote adapter, these will already exist in the target Nautobot system. In this scenario, if not properly handled, the sync will fail! Possible remediation steps may vary depending on the specific use-case, therefore this is left as an exercise to the reader/developer to solve for their specific context. +The above example shows the simplest field type (an attribute on the model), however, to build a production implementation you will need to understand how to identify different variants of fields by following the [modeling docs](../user/modeling.md). diff --git a/docs/user/modeling.md b/docs/dev/modeling.md similarity index 100% rename from docs/user/modeling.md rename to docs/dev/modeling.md diff --git a/docs/user/app_getting_started.md b/docs/user/app_getting_started.md index 741069976..3f5fac67f 100644 --- a/docs/user/app_getting_started.md +++ b/docs/user/app_getting_started.md @@ -44,3 +44,5 @@ PLUGINS_CONFIG = { ## What are the next steps? You can check out the [Use Cases](app_use_cases.md) section for more examples. + +Alternatively, if you intend to build your own integration, check out [Developing Data Source and Data Target Jobs](../dev/jobs.md). diff --git a/mkdocs.yml b/mkdocs.yml index 2bac83284..277ecca02 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -118,7 +118,6 @@ nav: - Itential: "user/integrations/itential.md" - Cisco Meraki: "user/integrations/meraki.md" - ServiceNow: "user/integrations/servicenow.md" - - Modeling: "user/modeling.md" - Performance: "user/performance.md" - Frequently Asked Questions: "user/faq.md" - External Interactions: "user/external_interactions.md" @@ -161,12 +160,18 @@ nav: - v1.1: "admin/release_notes/version_1.1.md" - v1.0: "admin/release_notes/version_1.0.md" - Developer Guide: - - Extending the App: "dev/extending.md" - - Developing Jobs: "dev/jobs.md" - - Debugging Jobs: "dev/debugging.md" - - Contributing to the App: "dev/contributing.md" + - "Tutorial: Developing Jobs": "dev/jobs.md" + - "How To: Debugging Jobs": "dev/debugging.md" + - "Reference: Issues": "dev/issues.md" + - "Reference: Modeling": "dev/modeling.md" - Development Environment: "dev/dev_environment.md" +<<<<<<< HEAD - Upgrading SSoT Apps: "dev/upgrade.md" +||||||| parent of c4fd5acc (docs: overhaul development docs section) +======= + - Extending the App: "dev/extending.md" + - Contributing to the App: "dev/contributing.md" +>>>>>>> c4fd5acc (docs: overhaul development docs section) - Code Reference: - "dev/code_reference/index.md" - Package: "dev/code_reference/package.md"