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

Release v2.4.0 #251

Merged
merged 6 commits into from
Nov 4, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions docs/admin/release_notes/version_2.4.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
# v2.4.0

## Added

(#248) Added support for new cloud models found in Nautobot 2.3+
5 changes: 5 additions & 0 deletions docs/dev/code_reference/models/cloud.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
# Cloud

::: pynautobot.models.cloud
options:
show_submodules: true
2 changes: 2 additions & 0 deletions mkdocs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -141,6 +141,7 @@ nav:
- Uninstall: "admin/uninstall.md"
- Release Notes:
- "admin/release_notes/index.md"
- v2.4: "admin/release_notes/version_2.4.md"
- v2.3: "admin/release_notes/version_2.3.md"
- v2.2: "admin/release_notes/version_2.2.md"
- v2.1: "admin/release_notes/version_2.1.md"
Expand All @@ -163,6 +164,7 @@ nav:
- Util: "dev/code_reference/core/util.md"
- Models:
- Circuits: "dev/code_reference/models/circuits.md"
- Cloud: "dev/code_reference/models/cloud.md"
- DCIM: "dev/code_reference/models/dcim.md"
- Extras: "dev/code_reference/models/extras.md"
- IPAM: "dev/code_reference/models/ipam.md"
Expand Down
2 changes: 2 additions & 0 deletions pynautobot/core/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ class Api(object):

Attributes:
dcim: An instance of the `App` class providing access to DCIM endpoints.
cloud: An instance of the `App` class providing access to Cloud endpoints.
ipam: An instance of the `App` class providing access to IPAM endpoints.
circuits: An instance of the `App` class providing access to Circuits endpoints.
tenancy: An instance of the `App` class providing access to Tenancy endpoints.
Expand Down Expand Up @@ -103,6 +104,7 @@ def __init__(

self.dcim = App(self, "dcim")
self.ipam = App(self, "ipam")
self.cloud = App(self, "cloud")
self.circuits = App(self, "circuits")
self.tenancy = App(self, "tenancy")
self.extras = App(self, "extras")
Expand Down
3 changes: 2 additions & 1 deletion pynautobot/core/app.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@

from pynautobot.core.endpoint import Endpoint, JobsEndpoint, GraphqlEndpoint
from pynautobot.core.query import Request
from pynautobot.models import circuits, dcim, extras, ipam, users, virtualization
from pynautobot.models import circuits, cloud, dcim, extras, ipam, users, virtualization

logger = logging.getLogger(__name__)

Expand All @@ -37,6 +37,7 @@ class App(object):

models = {
"dcim": dcim,
"cloud": cloud,
"ipam": ipam,
"circuits": circuits,
"virtualization": virtualization,
Expand Down
17 changes: 17 additions & 0 deletions pynautobot/models/cloud.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
# (c) 2017 DigitalOcean
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# This file has been modified by NetworktoCode, LLC.

# Reserved for cloud models
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@

[tool.poetry]
name = "pynautobot"
version = "2.3.0"
version = "2.4.0"
description = "Nautobot API client library"
authors = ["Network to Code, LLC <[email protected]>"]
readme = "README.md"
Expand Down
129 changes: 129 additions & 0 deletions tests/integration/test_cloud.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,129 @@
from packaging import version

import pytest


class TestCloudResourceType:
@pytest.fixture
def skipif_version(self, nb_client):
"""Retrieve the current Nautobot version and skip the test if less than 2.3."""
status = nb_client.status()
nautobot_version = status.get("nautobot-version")
if version.parse(nautobot_version) < version.parse("2.3"):
pytest.skip("Cloud resources are only in Nautobot 2.3+")

return nautobot_version

def test_cloud_resource_types(self, skipif_version, nb_client):
"""Verify we can CRUD a cloud resource type."""
assert skipif_version

# Create
cloud_resource_type = nb_client.cloud.cloud_resource_types.create(
name="Test", provider="Dell", content_types=["cloud.cloudservice"]
)
assert cloud_resource_type

# Read
test_cloud_resource_type = nb_client.cloud.cloud_resource_types.get(name="Test")
assert test_cloud_resource_type.name == "Test"

# Update
test_cloud_resource_type.name = "Updated Test"
test_cloud_resource_type.save()
updated_cloud_resource_type = nb_client.cloud.cloud_resource_types.get(name="Updated Test")
assert updated_cloud_resource_type.name == "Updated Test"

# Delete
updated_cloud_resource_type.delete()
deleted_cloud_resource_type = nb_client.cloud.cloud_resource_types.get(name="Updated Test")
assert not deleted_cloud_resource_type

def test_cloud_accounts(self, skipif_version, nb_client):
"""Verify we can CRUD a cloud account."""
assert skipif_version

# Create
cloud_account = nb_client.cloud.cloud_accounts.create(name="TestAcct", provider="Dell", account_number="424242")
assert cloud_account

# Read
test_cloud_account = nb_client.cloud.cloud_accounts.get(name="TestAcct")
assert test_cloud_account.name == "TestAcct"

# Update
test_cloud_account.name = "Updated TestAcct"
test_cloud_account.save()
updated_cloud_account = nb_client.cloud.cloud_accounts.get(name="Updated TestAcct")
assert updated_cloud_account.name == "Updated TestAcct"

# Delete
updated_cloud_account.delete()
deleted_cloud_account = nb_client.cloud.cloud_accounts.get(name="Updated TestAcct")
assert not deleted_cloud_account

def test_cloud_services(self, skipif_version, nb_client):
"""Verify we can CRUD a cloud service."""
assert skipif_version

# Create
cloud_resource_type = nb_client.cloud.cloud_resource_types.create(
name="Test", provider="Dell", content_types=["cloud.cloudservice"]
)
cloud_account = nb_client.cloud.cloud_accounts.create(name="TestAcct", provider="Dell", account_number="424242")
cloud_service = nb_client.cloud.cloud_services.create(
name="TestService", cloud_resource_type="Test", cloud_account="TestAcct"
)
assert cloud_service

# Read
test_cloud_service = nb_client.cloud.cloud_services.get(name="TestService")
assert test_cloud_service.name == "TestService"

# Update
test_cloud_service.name = "Updated TestService"
test_cloud_service.save()
updated_cloud_service = nb_client.cloud.cloud_services.get(name="Updated TestService")
assert updated_cloud_service.name == "Updated TestService"

# Delete
updated_cloud_service.delete()
deleted_cloud_service = nb_client.cloud.cloud_services.get(name="Updated TestService")
assert not deleted_cloud_service

# Cleanup
cloud_resource_type.delete()
cloud_account.delete()

def test_cloud_networks(self, skipif_version, nb_client):
"""Verify we can CRUD a cloud service."""
assert skipif_version

# Create
cloud_resource_type = nb_client.cloud.cloud_resource_types.create(
name="Test", provider="Dell", content_types=["cloud.cloudnetwork"]
)
cloud_account = nb_client.cloud.cloud_accounts.create(name="TestAcct", provider="Dell", account_number="424242")
cloud_network = nb_client.cloud.cloud_networks.create(
name="TestNetwork", cloud_resource_type="Test", cloud_account="TestAcct"
)
assert cloud_network

# Read
test_cloud_network = nb_client.cloud.cloud_networks.get(name="TestNetwork")
assert test_cloud_network.name == "TestNetwork"

# Update
test_cloud_network.name = "Updated TestNetwork"
test_cloud_network.save()
updated_cloud_network = nb_client.cloud.cloud_networks.get(name="Updated TestNetwork")
assert updated_cloud_network.name == "Updated TestNetwork"

# Delete
updated_cloud_network.delete()
deleted_cloud_network = nb_client.cloud.cloud_networks.get(name="Updated TestNetwork")
assert not deleted_cloud_network

# Cleanup
cloud_resource_type.delete()
cloud_account.delete()
Loading