A proof of concept for sending the intended state of objects to a Job in Nautobot
This POC exists solely to present an alternative solution to creating or updating many items in Nautobot. Instead of doing a GET
call for each object, determining the current state of that object, sending a PUT/PATCH
call to update an object or POST
to create an object, you can instead send in a large amount of data in a single API call to a Job and let it deal with determining state and what needs to be updated.
You can add this Job to any Nautobot instance by add it under Extensibility > Git Repositories.
Once you have synced the repo, you can run the Job via the GUI or an API call like any other Job. The only job data that is required is a JSON serialized object. Each top level key of the dictionary must be the app name and model in dot notation that you would like to create or update. The value of each app_label.model
must be a list of dictionaries with field attributes. Here is an example:
{
"dcim.location": [
{
"name": "New Location",
"status": ...,
"parent": ...,
}
],
"dcim.device": [
{
"name": "New Device",
"role": ...,
"device_type": ...,
}
]
}
Note: You can find the app_label and model for all current content types by going to https://your.nautobot.instance/api/extras/content-types/.
This payload must be sent as a string that will be serialized. Here is an example of an API payload:
{
"data": {
"json_payload": "{\"extras.status\": [{\"name\": \"Test Status\"}]}"
}
}
In Nautobot v1.X, all Jobs ran inside an atomic transaction which means if the job fails (for any reason) it automatically rolls back all of the changes since the Job started. In Nautobot v2.X, this is no longer the case, but the feature is helpful in some situations so I have added an option to the Job to enable atomic transactions if it is desired. Simply add atomic=True
to the data
that is sent in the request:
{
"data": {
"atomic": True,
"json_payload": {
...
}
}
}
Some item fields are foreign key relationships to other objects. In those cases, you will need to look up the object before trying to use it. For this, you will use a dictionary with the key being #ref
and the value being the object type and details. You will need to format the reference dictionary as such:
{
"#ref": {
"dcim.location": {
"name": "Location 1",
# "parent__name": "Parent Location 1",
# etc.
# Add as many fields as necessary to uniquely identify the referenced object
}
}
}
Some item fields (M2M) cannot be directly assigned. An example of this would be Content Types or Tags. In this case, you can use the #set
or #add
keys to manage these relationships in the same manner as #ref
.
If you want to replace all of the items you would use the #set
key in the item dictionary. They will be assigned to the object after it has been created or updated.
Example:
{
"extras.status": [
{
"name": "Active",
"#set": {
"content_types": [
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "location",
}
}
},
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "device",
}
}
},
]
},
}
]
}
If you, however, don't want to provide the entire list of items every time but rather want to ensure something is in the list you can use the #add
key in the same way:
{
"extras.status": [
{
"name": "Active",
"#add": {
"content_types": [
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "location",
}
}
},
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "device",
}
}
},
]
},
}
]
}
The underlying method that gets called in this job is the Django update_or_create
method. When using the update_or_create
method, it will first attempt to use the fields passed in as a .get()
call looking to see if the item exists with all of the provided fields and if not it will create it. If you are running the Job a second time to update a field on an item (i.e. description
) then it will fail to find an object with that description and try and create a second instance. This will either fail with an IntegrityError due to duplicate unique fields or it will succeed and it will create a duplicate item which is undesired. To combat this, you should nest all fields that are not used to uniquely identify an object in a defaults
key.
In this example, only the name
field is used to uniquely identify the device:
{
"dcim.device": [
{
"name": "Device 1",
"defaults": {
"role": {"#ref": {"extras.role": {"name": "Role 1"}}},
"device_type": {"#ref": {"dcim.devicetype": {"model": "Model 1"}}},
"location": {"#ref": {"dcim.location": {"name": "Location 1"}}},
"status": {"#ref": {"extras.status": {"name": "Active"}}},
},
}
]
}
In addition, if you don't want to send in the entire payload on subsequent calls (after the item has been created), you can simply include the fields you want to update. For example, this payload simply changes the status of Device 1 to Decommissioned and leaves the role, device type and location the same:
{
"dcim.device": [
{
"name": "Device 1",
"defaults": {
"status": {"#ref": {"extras.status": {"name": "Decommissioned"}}},
},
}
]
}
Here is an example of a working (albeit rudimentary) example in python:
import json
import requests
url = "https://nautobot.example.com"
token = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
headers = {"Authorization": f"Token {token}"}
json_payload = {
"extras.status": [
{
"name": "Active",
"#add": {
"content_types": [
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "location",
}
}
},
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "device",
}
}
},
]
},
}
],
"dcim.manufacturer": [
{"name": "Manufacturer 1"},
{"name": "Manufacturer 2"},
],
"dcim.devicetype": [
{
"model": "Model 1",
"manufacturer": {"#ref": {"dcim.manufacturer": {"name": "Manufacturer 1"}}},
},
{
"model": "Model 2",
"manufacturer": {"#ref": {"dcim.manufacturer": {"name": "Manufacturer 2"}}},
},
],
"extras.role": [
{
"name": "Role 1",
"#set": {
"content_types": [
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "device",
}
}
},
]
},
},
{
"name": "Role 2",
"#set": {
"content_types": [
{
"#ref": {
"contenttypes.contenttype": {
"app_label": "dcim",
"model": "device",
}
}
},
]
},
},
],
"dcim.locationtype": [
{"name": "Location Type 1"},
{"name": "Location Type 2"},
],
"dcim.location": [
{
"name": "Location 1",
"location_type": {
"#ref": {"dcim.locationtype": {"name": "Location Type 1"}}
},
"defaults": {
"status": {"#ref": {"extras.status": {"name": "Active"}}},
},
},
{
"name": "Location 2",
"location_type": {
"#ref": {"dcim.locationtype": {"name": "Location Type 2"}}
},
"defaults": {
"status": {"#ref": {"extras.status": {"name": "Active"}}},
},
},
],
"dcim.device": [
{
"name": "Device 1",
"defaults": {
"role": {"#ref": {"extras.role": {"name": "Role 1"}}},
"device_type": {"#ref": {"dcim.devicetype": {"model": "Model 1"}}},
"location": {"#ref": {"dcim.location": {"name": "Location 1"}}},
"status": {"#ref": {"extras.status": {"name": "Active"}}},
},
},
{
"name": "Device 2",
"defaults": {
"role": {"#ref": {"extras.role": {"name": "Role 2"}}},
"device_type": {"#ref": {"dcim.devicetype": {"model": "Model 2"}}},
"location": {"#ref": {"dcim.location": {"name": "Location 2"}}},
"status": {"#ref": {"extras.status": {"name": "Active"}}},
},
},
],
}
payload = {"data": {"json_payload": json.dumps(json_payload)}}
requests.post(f"{url}/api/extras/jobs/Intended State Job/run/", headers=headers, json=payload)
Note, this repo is not meant to be a perfect representation of how to implement this for every environment, but rather just a simple POC on how it could be done. Feel free to fork/copy this code and modify it as you see fit to your required specifications.