| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217 |
- import datetime
- from django.contrib.contenttypes.models import ContentType
- from django.test import override_settings
- from django.urls import reverse
- from netaddr import IPNetwork
- from core.models import ObjectType
- from dcim.constants import InterfaceTypeChoices
- from dcim.models import Device, DeviceRole, DeviceType, Manufacturer, Site, Interface
- from ipam.choices import *
- from ipam.models import *
- from netbox.choices import CSVDelimiterChoices, ImportFormatChoices
- from tenancy.models import Tenant
- from users.models import ObjectPermission
- from utilities.testing import ViewTestCases, create_tags
- class ASNRangeTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = ASNRange
- @classmethod
- def setUpTestData(cls):
- rirs = [
- RIR(name='RIR 1', slug='rir-1', is_private=True),
- RIR(name='RIR 2', slug='rir-2', is_private=True),
- ]
- RIR.objects.bulk_create(rirs)
- tenants = [
- Tenant(name='Tenant 1', slug='tenant-1'),
- Tenant(name='Tenant 2', slug='tenant-2'),
- ]
- Tenant.objects.bulk_create(tenants)
- asn_ranges = (
- ASNRange(name='ASN Range 1', slug='asn-range-1', rir=rirs[0], tenant=tenants[0], start=100, end=199),
- ASNRange(name='ASN Range 2', slug='asn-range-2', rir=rirs[0], tenant=tenants[0], start=200, end=299),
- ASNRange(name='ASN Range 3', slug='asn-range-3', rir=rirs[0], tenant=tenants[0], start=300, end=399),
- )
- ASNRange.objects.bulk_create(asn_ranges)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'name': 'ASN Range X',
- 'slug': 'asn-range-x',
- 'rir': rirs[1].pk,
- 'tenant': tenants[1].pk,
- 'start': 1000,
- 'end': 1099,
- 'description': 'A new ASN range',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name,slug,rir,tenant,start,end,description",
- f"ASN Range 4,asn-range-4,{rirs[1].name},{tenants[1].name},400,499,Fourth range",
- f"ASN Range 5,asn-range-5,{rirs[1].name},{tenants[1].name},500,599,Fifth range",
- f"ASN Range 6,asn-range-6,{rirs[1].name},{tenants[1].name},600,699,Sixth range",
- )
- cls.csv_update_data = (
- "id,description",
- f"{asn_ranges[0].pk},New description 1",
- f"{asn_ranges[1].pk},New description 2",
- f"{asn_ranges[2].pk},New description 3",
- )
- cls.bulk_edit_data = {
- 'rir': rirs[1].pk,
- 'description': 'Next description',
- }
- class ASNTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = ASN
- @classmethod
- def setUpTestData(cls):
- rirs = [
- RIR(name='RIR 1', slug='rir-1', is_private=True),
- RIR(name='RIR 2', slug='rir-2', is_private=True),
- ]
- RIR.objects.bulk_create(rirs)
- sites = (
- Site(name='Site 1', slug='site-1'),
- Site(name='Site 2', slug='site-2')
- )
- Site.objects.bulk_create(sites)
- tenants = (
- Tenant(name='Tenant 1', slug='tenant-1'),
- Tenant(name='Tenant 2', slug='tenant-2'),
- )
- Tenant.objects.bulk_create(tenants)
- asns = (
- ASN(asn=65001, rir=rirs[0], tenant=tenants[0]),
- ASN(asn=65002, rir=rirs[1], tenant=tenants[1]),
- ASN(asn=4200000001, rir=rirs[0], tenant=tenants[0]),
- ASN(asn=4200000002, rir=rirs[1], tenant=tenants[1]),
- )
- ASN.objects.bulk_create(asns)
- asns[0].sites.set([sites[0]])
- asns[1].sites.set([sites[1]])
- asns[2].sites.set([sites[0]])
- asns[3].sites.set([sites[1]])
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'asn': 65000,
- 'rir': rirs[0].pk,
- 'tenant': tenants[0].pk,
- 'site': sites[0].pk,
- 'description': 'A new ASN',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "asn,rir",
- "65003,RIR 1",
- "65004,RIR 2",
- "4200000003,RIR 1",
- "4200000004,RIR 2",
- )
- cls.csv_update_data = (
- "id,description",
- f"{asns[0].pk},New description1",
- f"{asns[1].pk},New description2",
- f"{asns[2].pk},New description3",
- )
- cls.bulk_edit_data = {
- 'rir': rirs[1].pk,
- 'description': 'Next description',
- }
- class VRFTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = VRF
- @classmethod
- def setUpTestData(cls):
- tenants = (
- Tenant(name='Tenant A', slug='tenant-a'),
- Tenant(name='Tenant B', slug='tenant-b'),
- )
- Tenant.objects.bulk_create(tenants)
- vrfs = (
- VRF(name='VRF 1', rd='65000:1'),
- VRF(name='VRF 2', rd='65000:2'),
- VRF(name='VRF 3', rd='65000:3'),
- )
- VRF.objects.bulk_create(vrfs)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'name': 'VRF X',
- 'rd': '65000:999',
- 'tenant': tenants[0].pk,
- 'enforce_unique': True,
- 'description': 'A new VRF',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name",
- "VRF 4",
- "VRF 5",
- "VRF 6",
- )
- cls.csv_update_data = (
- "id,name",
- f"{vrfs[0].pk},VRF 7",
- f"{vrfs[1].pk},VRF 8",
- f"{vrfs[2].pk},VRF 9",
- )
- cls.bulk_edit_data = {
- 'tenant': tenants[1].pk,
- 'enforce_unique': False,
- 'description': 'New description',
- }
- class RouteTargetTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = RouteTarget
- @classmethod
- def setUpTestData(cls):
- tenants = (
- Tenant(name='Tenant A', slug='tenant-a'),
- Tenant(name='Tenant B', slug='tenant-b'),
- )
- Tenant.objects.bulk_create(tenants)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- route_targets = (
- RouteTarget(name='65000:1001', tenant=tenants[0]),
- RouteTarget(name='65000:1002', tenant=tenants[1]),
- RouteTarget(name='65000:1003'),
- )
- RouteTarget.objects.bulk_create(route_targets)
- cls.form_data = {
- 'name': '65000:100',
- 'description': 'A new route target',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name,tenant,description",
- "65000:1004,Tenant A,Foo",
- "65000:1005,Tenant B,Bar",
- "65000:1006,,No tenant",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{route_targets[0].pk},65000:1007,New description1",
- f"{route_targets[1].pk},65000:1008,New description2",
- f"{route_targets[2].pk},65000:1009,New description3",
- )
- cls.bulk_edit_data = {
- 'tenant': tenants[1].pk,
- 'description': 'New description',
- }
- class RIRTestCase(ViewTestCases.OrganizationalObjectViewTestCase):
- model = RIR
- @classmethod
- def setUpTestData(cls):
- rirs = (
- RIR(name='RIR 1', slug='rir-1'),
- RIR(name='RIR 2', slug='rir-2'),
- RIR(name='RIR 3', slug='rir-3'),
- )
- RIR.objects.bulk_create(rirs)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'name': 'RIR X',
- 'slug': 'rir-x',
- 'is_private': True,
- 'description': 'A new RIR',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name,slug,description",
- "RIR 4,rir-4,Fourth RIR",
- "RIR 5,rir-5,Fifth RIR",
- "RIR 6,rir-6,Sixth RIR",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{rirs[0].pk},RIR 7,Fourth RIR7",
- f"{rirs[1].pk},RIR 8,Fifth RIR8",
- f"{rirs[2].pk},RIR 9,Sixth RIR9",
- )
- cls.bulk_edit_data = {
- 'description': 'New description',
- }
- class AggregateTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = Aggregate
- @classmethod
- def setUpTestData(cls):
- rirs = (
- RIR(name='RIR 1', slug='rir-1'),
- RIR(name='RIR 2', slug='rir-2'),
- )
- RIR.objects.bulk_create(rirs)
- aggregates = (
- Aggregate(prefix=IPNetwork('10.1.0.0/16'), rir=rirs[0]),
- Aggregate(prefix=IPNetwork('10.2.0.0/16'), rir=rirs[0]),
- Aggregate(prefix=IPNetwork('10.3.0.0/16'), rir=rirs[0]),
- )
- Aggregate.objects.bulk_create(aggregates)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'prefix': IPNetwork('10.99.0.0/16'),
- 'rir': rirs[1].pk,
- 'date_added': datetime.date(2020, 1, 1),
- 'description': 'A new aggregate',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "prefix,rir",
- "10.4.0.0/16,RIR 1",
- "10.5.0.0/16,RIR 1",
- "10.6.0.0/16,RIR 1",
- )
- cls.csv_update_data = (
- "id,description",
- f"{aggregates[0].pk},New description1",
- f"{aggregates[1].pk},New description2",
- f"{aggregates[2].pk},New description3",
- )
- cls.bulk_edit_data = {
- 'rir': rirs[1].pk,
- 'date_added': datetime.date(2020, 1, 1),
- 'description': 'New description',
- }
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_aggregate_prefixes(self):
- rir = RIR.objects.first()
- aggregate = Aggregate.objects.create(prefix=IPNetwork('192.168.0.0/16'), rir=rir)
- prefixes = (
- Prefix(prefix=IPNetwork('192.168.1.0/24')),
- Prefix(prefix=IPNetwork('192.168.2.0/24')),
- Prefix(prefix=IPNetwork('192.168.3.0/24')),
- )
- Prefix.objects.bulk_create(prefixes)
- self.assertEqual(aggregate.get_child_prefixes().count(), 3)
- url = reverse('ipam:aggregate_prefixes', kwargs={'pk': aggregate.pk})
- self.assertHttpStatus(self.client.get(url), 200)
- class RoleTestCase(ViewTestCases.OrganizationalObjectViewTestCase):
- model = Role
- @classmethod
- def setUpTestData(cls):
- roles = (
- Role(name='Role 1', slug='role-1'),
- Role(name='Role 2', slug='role-2'),
- Role(name='Role 3', slug='role-3'),
- )
- Role.objects.bulk_create(roles)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'name': 'Role X',
- 'slug': 'role-x',
- 'weight': 200,
- 'description': 'A new role',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name,slug,weight",
- "Role 4,role-4,1000",
- "Role 5,role-5,1000",
- "Role 6,role-6,1000",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{roles[0].pk},Role 7,New description7",
- f"{roles[1].pk},Role 8,New description8",
- f"{roles[2].pk},Role 9,New description9",
- )
- cls.bulk_edit_data = {
- 'description': 'New description',
- }
- class PrefixTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = Prefix
- @classmethod
- def setUpTestData(cls):
- sites = (
- Site(name='Site 1', slug='site-1'),
- Site(name='Site 2', slug='site-2'),
- )
- Site.objects.bulk_create(sites)
- vrfs = (
- VRF(name='VRF 1', rd='65000:1'),
- VRF(name='VRF 2', rd='65000:2'),
- )
- VRF.objects.bulk_create(vrfs)
- roles = (
- Role(name='Role 1', slug='role-1'),
- Role(name='Role 2', slug='role-2'),
- )
- Role.objects.bulk_create(roles)
- prefixes = (
- Prefix(prefix=IPNetwork('10.1.0.0/16'), vrf=vrfs[0], scope=sites[0], role=roles[0]),
- Prefix(prefix=IPNetwork('10.2.0.0/16'), vrf=vrfs[0], scope=sites[0], role=roles[0]),
- Prefix(prefix=IPNetwork('10.3.0.0/16'), vrf=vrfs[0], scope=sites[0], role=roles[0]),
- )
- Prefix.objects.bulk_create(prefixes)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- # TODO: Alter for prefix
- cls.form_data = {
- 'prefix': IPNetwork('192.0.2.0/24'),
- 'scope_type': ContentType.objects.get_for_model(Site).pk,
- 'scope': sites[1].pk,
- 'vrf': vrfs[1].pk,
- 'tenant': None,
- 'vlan': None,
- 'status': PrefixStatusChoices.STATUS_RESERVED,
- 'role': roles[1].pk,
- 'is_pool': True,
- 'description': 'A new prefix',
- 'tags': [t.pk for t in tags],
- }
- site = sites[0].pk
- # TODO: Alter for prefix
- cls.csv_data = (
- "vrf,prefix,status,scope_type,scope_id",
- f"VRF 1,10.4.0.0/16,active,dcim.site,{site}",
- f"VRF 1,10.5.0.0/16,active,dcim.site,{site}",
- f"VRF 1,10.6.0.0/16,active,dcim.site,{site}",
- )
- # TODO: Alter for prefix
- cls.csv_update_data = (
- "id,description,status",
- f"{prefixes[0].pk},New description 7,{PrefixStatusChoices.STATUS_RESERVED}",
- f"{prefixes[1].pk},New description 8,{PrefixStatusChoices.STATUS_RESERVED}",
- f"{prefixes[2].pk},New description 9,{PrefixStatusChoices.STATUS_RESERVED}",
- )
- # TODO: Alter for prefix
- cls.bulk_edit_data = {
- 'vrf': vrfs[1].pk,
- 'tenant': None,
- 'status': PrefixStatusChoices.STATUS_RESERVED,
- 'role': roles[1].pk,
- 'is_pool': False,
- 'description': 'New description',
- }
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_prefix_prefixes(self):
- prefixes = (
- Prefix(prefix=IPNetwork('192.168.0.0/16')),
- Prefix(prefix=IPNetwork('192.168.1.0/24')),
- Prefix(prefix=IPNetwork('192.168.2.0/24')),
- Prefix(prefix=IPNetwork('192.168.3.0/24')),
- )
- Prefix.objects.bulk_create(prefixes)
- self.assertEqual(prefixes[0].get_child_prefixes().count(), 3)
- url = reverse('ipam:prefix_prefixes', kwargs={'pk': prefixes[0].pk})
- self.assertHttpStatus(self.client.get(url), 200)
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_prefix_ipranges(self):
- prefix = Prefix.objects.create(prefix=IPNetwork('192.168.0.0/16'))
- ip_ranges = (
- IPRange(start_address='192.168.0.1/24', end_address='192.168.0.100/24', size=99),
- IPRange(start_address='192.168.1.1/24', end_address='192.168.1.100/24', size=99),
- IPRange(start_address='192.168.2.1/24', end_address='192.168.2.100/24', size=99),
- )
- IPRange.objects.bulk_create(ip_ranges)
- self.assertEqual(prefix.get_child_ranges().count(), 3)
- url = reverse('ipam:prefix_ipranges', kwargs={'pk': prefix.pk})
- self.assertHttpStatus(self.client.get(url), 200)
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_prefix_ipaddresses(self):
- prefix = Prefix.objects.create(prefix=IPNetwork('192.168.0.0/16'))
- ip_addresses = (
- IPAddress(address=IPNetwork('192.168.0.1/16')),
- IPAddress(address=IPNetwork('192.168.0.2/16')),
- IPAddress(address=IPNetwork('192.168.0.3/16')),
- )
- IPAddress.objects.bulk_create(ip_addresses)
- self.assertEqual(prefix.ip_addresses.all().count(), 3)
- url = reverse('ipam:prefix_ipaddresses', kwargs={'pk': prefix.pk})
- self.assertHttpStatus(self.client.get(url), 200)
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_prefix_import(self):
- """
- Custom import test for YAML-based imports (versus CSV)
- """
- site = Site.objects.get(name='Site 1')
- IMPORT_DATA = f"""
- prefix: 10.1.1.0/24
- status: active
- vlan: 101
- scope_type: dcim.site
- scope_id: {site.pk}
- """
- # Note, a site is not tied to the VLAN to verify the fix for #12622
- VLAN.objects.create(vid=101, name='VLAN101')
- # Add all required permissions to the test user
- self.add_permissions('ipam.view_prefix', 'ipam.add_prefix')
- form_data = {
- 'data': IMPORT_DATA,
- 'format': 'yaml'
- }
- response = self.client.post(reverse('ipam:prefix_bulk_import'), data=form_data, follow=True)
- self.assertHttpStatus(response, 200)
- prefix = Prefix.objects.get(prefix='10.1.1.0/24')
- self.assertEqual(prefix.status, PrefixStatusChoices.STATUS_ACTIVE)
- self.assertEqual(prefix.vlan.vid, 101)
- self.assertEqual(prefix.scope, site)
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_prefix_import_with_vlan_group(self):
- """
- This test covers a unique import edge case where VLAN group is specified during the import.
- """
- site = Site.objects.get(name='Site 1')
- IMPORT_DATA = f"""
- prefix: 10.1.2.0/24
- status: active
- scope_type: dcim.site
- scope_id: {site.pk}
- vlan_group: Group 1
- vlan: 102
- """
- vlan_group = VLANGroup.objects.create(name='Group 1', slug='group-1', scope=Site.objects.get(name="Site 1"))
- VLAN.objects.create(vid=102, name='VLAN102', group=vlan_group)
- # Add all required permissions to the test user
- self.add_permissions('ipam.view_prefix', 'ipam.add_prefix')
- form_data = {
- 'data': IMPORT_DATA,
- 'format': 'yaml'
- }
- response = self.client.post(reverse('ipam:prefix_bulk_import'), data=form_data, follow=True)
- self.assertHttpStatus(response, 200)
- prefix = Prefix.objects.get(prefix='10.1.2.0/24')
- self.assertEqual(prefix.status, PrefixStatusChoices.STATUS_ACTIVE)
- self.assertEqual(prefix.vlan.vid, 102)
- self.assertEqual(prefix.scope, site)
- class IPRangeTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = IPRange
- @classmethod
- def setUpTestData(cls):
- vrfs = (
- VRF(name='VRF 1', rd='65000:1'),
- VRF(name='VRF 2', rd='65000:2'),
- )
- VRF.objects.bulk_create(vrfs)
- roles = (
- Role(name='Role 1', slug='role-1'),
- Role(name='Role 2', slug='role-2'),
- )
- Role.objects.bulk_create(roles)
- ip_ranges = (
- IPRange(start_address='192.168.0.10/24', end_address='192.168.0.100/24', size=91),
- IPRange(start_address='192.168.1.10/24', end_address='192.168.1.100/24', size=91),
- IPRange(start_address='192.168.2.10/24', end_address='192.168.2.100/24', size=91),
- IPRange(start_address='192.168.3.10/24', end_address='192.168.3.100/24', size=91),
- IPRange(start_address='192.168.4.10/24', end_address='192.168.4.100/24', size=91),
- )
- IPRange.objects.bulk_create(ip_ranges)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- # TODO: Alter for prefix
- cls.form_data = {
- 'start_address': IPNetwork('192.0.5.10/24'),
- 'end_address': IPNetwork('192.0.5.100/24'),
- 'vrf': vrfs[1].pk,
- 'tenant': None,
- 'vlan': None,
- 'status': IPRangeStatusChoices.STATUS_RESERVED,
- 'role': roles[1].pk,
- 'is_pool': True,
- 'description': 'A new IP range',
- 'tags': [t.pk for t in tags],
- }
- # TODO: Alter for prefix
- cls.csv_data = (
- "vrf,start_address,end_address,status",
- "VRF 1,10.1.0.1/16,10.1.9.254/16,active",
- "VRF 1,10.2.0.1/16,10.2.9.254/16,active",
- "VRF 1,10.3.0.1/16,10.3.9.254/16,active",
- )
- # TODO: Alter for prefix
- cls.csv_update_data = (
- "id,description,status",
- f"{ip_ranges[0].pk},New description 7,{IPRangeStatusChoices.STATUS_RESERVED}",
- f"{ip_ranges[1].pk},New description 8,{IPRangeStatusChoices.STATUS_RESERVED}",
- f"{ip_ranges[2].pk},New description 9,{IPRangeStatusChoices.STATUS_RESERVED}",
- )
- # TODO: Alter for prefix
- cls.bulk_edit_data = {
- 'vrf': vrfs[1].pk,
- 'tenant': None,
- 'status': IPRangeStatusChoices.STATUS_RESERVED,
- 'role': roles[1].pk,
- 'description': 'New description',
- }
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_iprange_ipaddresses(self):
- iprange = IPRange.objects.create(
- start_address=IPNetwork('192.168.0.1/24'),
- end_address=IPNetwork('192.168.0.100/24'),
- size=99
- )
- ip_addresses = (
- IPAddress(address=IPNetwork('192.168.0.1/24')),
- IPAddress(address=IPNetwork('192.168.0.2/24')),
- IPAddress(address=IPNetwork('192.168.0.3/24')),
- )
- IPAddress.objects.bulk_create(ip_addresses)
- self.assertEqual(iprange.get_child_ips().count(), 3)
- url = reverse('ipam:iprange_ipaddresses', kwargs={'pk': iprange.pk})
- self.assertHttpStatus(self.client.get(url), 200)
- class IPAddressTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = IPAddress
- @classmethod
- def setUpTestData(cls):
- vrfs = (
- VRF(name='VRF 1', rd='65000:1'),
- VRF(name='VRF 2', rd='65000:2'),
- )
- VRF.objects.bulk_create(vrfs)
- ipaddresses = (
- IPAddress(address=IPNetwork('192.0.2.1/24'), vrf=vrfs[0]),
- IPAddress(address=IPNetwork('192.0.2.2/24'), vrf=vrfs[0]),
- IPAddress(address=IPNetwork('192.0.2.3/24'), vrf=vrfs[0]),
- )
- IPAddress.objects.bulk_create(ipaddresses)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- fhrp_groups = (
- FHRPGroup(
- name='FHRP Group 1',
- protocol=FHRPGroupProtocolChoices.PROTOCOL_HSRP,
- group_id=10
- ),
- FHRPGroup(
- name='FHRP Group 2',
- protocol=FHRPGroupProtocolChoices.PROTOCOL_HSRP,
- group_id=20
- ),
- FHRPGroup(
- name='FHRP Group 3',
- protocol=FHRPGroupProtocolChoices.PROTOCOL_HSRP,
- group_id=30
- ),
- )
- FHRPGroup.objects.bulk_create(fhrp_groups)
- # TODO: Alter for prefix
- cls.form_data = {
- 'vrf': vrfs[1].pk,
- 'address': IPNetwork('192.0.2.99/24'),
- 'tenant': None,
- 'status': IPAddressStatusChoices.STATUS_RESERVED,
- 'role': IPAddressRoleChoices.ROLE_ANYCAST,
- 'nat_inside': None,
- 'dns_name': 'example',
- 'description': 'A new IP address',
- 'tags': [t.pk for t in tags],
- }
- # TODO: Alter for prefix
- cls.csv_data = (
- "vrf,address,status,fhrp_group",
- "VRF 1,192.0.2.4/24,active,FHRP Group 1",
- "VRF 1,192.0.2.5/24,active,FHRP Group 2",
- "VRF 1,192.0.2.6/24,active,FHRP Group 3",
- )
- # TODO: Alter for prefix
- cls.csv_update_data = (
- "id,description,status",
- f"{ipaddresses[0].pk},New description 7,{IPAddressStatusChoices.STATUS_RESERVED}",
- f"{ipaddresses[1].pk},New description 8,{IPAddressStatusChoices.STATUS_RESERVED}",
- f"{ipaddresses[2].pk},New description 9,{IPAddressStatusChoices.STATUS_RESERVED}",
- )
- # TODO: Alter for prefix
- cls.bulk_edit_data = {
- 'vrf': vrfs[1].pk,
- 'tenant': None,
- 'status': IPAddressStatusChoices.STATUS_RESERVED,
- 'role': IPAddressRoleChoices.ROLE_ANYCAST,
- 'dns_name': 'example',
- 'description': 'New description',
- }
- class FHRPGroupTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = FHRPGroup
- @classmethod
- def setUpTestData(cls):
- fhrp_groups = (
- FHRPGroup(
- protocol=FHRPGroupProtocolChoices.PROTOCOL_VRRP2,
- group_id=10,
- auth_type=FHRPGroupAuthTypeChoices.AUTHENTICATION_PLAINTEXT,
- auth_key='foobar123',
- ),
- FHRPGroup(
- protocol=FHRPGroupProtocolChoices.PROTOCOL_VRRP3,
- group_id=20,
- auth_type=FHRPGroupAuthTypeChoices.AUTHENTICATION_MD5,
- auth_key='foobar123',
- ),
- FHRPGroup(
- protocol=FHRPGroupProtocolChoices.PROTOCOL_HSRP,
- group_id=30
- ),
- )
- FHRPGroup.objects.bulk_create(fhrp_groups)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'protocol': FHRPGroupProtocolChoices.PROTOCOL_VRRP2,
- 'group_id': 99,
- 'auth_type': FHRPGroupAuthTypeChoices.AUTHENTICATION_MD5,
- 'auth_key': 'abc123def456',
- 'description': 'Blah blah blah',
- 'name': 'test123 name',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "protocol,group_id,auth_type,auth_key,description",
- "vrrp2,40,plaintext,foobar123,Foo",
- "vrrp3,50,md5,foobar123,Bar",
- "hsrp,60,,,",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{fhrp_groups[0].pk},FHRP Group 1,New description 1",
- f"{fhrp_groups[1].pk},FHRP Group 2,New description 2",
- f"{fhrp_groups[2].pk},FHRP Group 3,New description 3",
- )
- cls.bulk_edit_data = {
- 'protocol': FHRPGroupProtocolChoices.PROTOCOL_CARP,
- }
- class VLANGroupTestCase(ViewTestCases.OrganizationalObjectViewTestCase):
- model = VLANGroup
- @classmethod
- def setUpTestData(cls):
- sites = (
- Site(name='Site 1', slug='site-1'),
- Site(name='Site 2', slug='site-2'),
- )
- Site.objects.bulk_create(sites)
- vlan_groups = (
- VLANGroup(name='VLAN Group 1', slug='vlan-group-1', scope=sites[0]),
- VLANGroup(name='VLAN Group 2', slug='vlan-group-2', scope=sites[0]),
- VLANGroup(name='VLAN Group 3', slug='vlan-group-3', scope=sites[0]),
- )
- VLANGroup.objects.bulk_create(vlan_groups)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'name': 'VLAN Group X',
- 'slug': 'vlan-group-x',
- 'description': 'A new VLAN group',
- 'vid_ranges': '100-199,300-399',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name,slug,scope_type,scope_id,description",
- "VLAN Group 4,vlan-group-4,,,Fourth VLAN group",
- f"VLAN Group 5,vlan-group-5,dcim.site,{sites[0].pk},Fifth VLAN group",
- f"VLAN Group 6,vlan-group-6,dcim.site,{sites[1].pk},Sixth VLAN group",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{vlan_groups[0].pk},VLAN Group 7,Fourth VLAN group7",
- f"{vlan_groups[1].pk},VLAN Group 8,Fifth VLAN group8",
- f"{vlan_groups[2].pk},VLAN Group 9,Sixth VLAN group9",
- )
- cls.bulk_edit_data = {
- 'description': 'New description',
- }
- class VLANTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = VLAN
- @classmethod
- def setUpTestData(cls):
- sites = (
- Site(name='Site 1', slug='site-1'),
- Site(name='Site 2', slug='site-2'),
- )
- Site.objects.bulk_create(sites)
- vlangroups = (
- VLANGroup(name='VLAN Group 1', slug='vlan-group-1', scope=sites[0]),
- VLANGroup(name='VLAN Group 2', slug='vlan-group-2', scope=sites[1]),
- )
- VLANGroup.objects.bulk_create(vlangroups)
- roles = (
- Role(name='Role 1', slug='role-1'),
- Role(name='Role 2', slug='role-2'),
- )
- Role.objects.bulk_create(roles)
- vlans = (
- VLAN(group=vlangroups[0], vid=101, name='VLAN101', site=sites[0], role=roles[0]),
- VLAN(group=vlangroups[0], vid=102, name='VLAN102', site=sites[0], role=roles[0]),
- VLAN(group=vlangroups[0], vid=103, name='VLAN103', site=sites[0], role=roles[0]),
- )
- VLAN.objects.bulk_create(vlans)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'site': sites[1].pk,
- 'group': vlangroups[1].pk,
- 'vid': 999,
- 'name': 'VLAN999',
- 'tenant': None,
- 'status': VLANStatusChoices.STATUS_RESERVED,
- 'role': roles[1].pk,
- 'description': 'A new VLAN',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "vid,name,status",
- "104,VLAN104,active",
- "105,VLAN105,active",
- "106,VLAN106,active",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{vlans[0].pk},VLAN107,New description 7",
- f"{vlans[1].pk},VLAN108,New description 8",
- f"{vlans[2].pk},VLAN109,New description 9",
- )
- cls.bulk_edit_data = {
- 'site': sites[1].pk,
- 'group': vlangroups[1].pk,
- 'tenant': None,
- 'status': VLANStatusChoices.STATUS_RESERVED,
- 'role': roles[1].pk,
- 'description': 'New description',
- }
- class VLANTranslationPolicyTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = VLANTranslationPolicy
- @classmethod
- def setUpTestData(cls):
- vlan_translation_policies = (
- VLANTranslationPolicy(
- name='Policy 1',
- description='foobar1',
- ),
- VLANTranslationPolicy(
- name='Policy 2',
- description='foobar2',
- ),
- VLANTranslationPolicy(
- name='Policy 3',
- description='foobar3',
- ),
- )
- VLANTranslationPolicy.objects.bulk_create(vlan_translation_policies)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'name': 'Policy999',
- 'description': 'A new VLAN Translation Policy',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name,description",
- "Policy101,foobar1",
- "Policy102,foobar2",
- "Policy103,foobar3",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{vlan_translation_policies[0].pk},Policy101,New description 1",
- f"{vlan_translation_policies[1].pk},Policy102,New description 2",
- f"{vlan_translation_policies[2].pk},Policy103,New description 3",
- )
- cls.bulk_edit_data = {
- 'description': 'New description',
- }
- class VLANTranslationRuleTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = VLANTranslationRule
- @classmethod
- def setUpTestData(cls):
- vlan_translation_policies = (
- VLANTranslationPolicy(
- name='Policy 1',
- description='foobar1',
- ),
- VLANTranslationPolicy(
- name='Policy 2',
- description='foobar2',
- ),
- VLANTranslationPolicy(
- name='Policy 3',
- description='foobar3',
- ),
- )
- VLANTranslationPolicy.objects.bulk_create(vlan_translation_policies)
- vlan_translation_rules = (
- VLANTranslationRule(
- policy=vlan_translation_policies[0],
- local_vid=100,
- remote_vid=200,
- ),
- VLANTranslationRule(
- policy=vlan_translation_policies[0],
- local_vid=101,
- remote_vid=201,
- ),
- VLANTranslationRule(
- policy=vlan_translation_policies[1],
- local_vid=102,
- remote_vid=202,
- ),
- )
- VLANTranslationRule.objects.bulk_create(vlan_translation_rules)
- cls.form_data = {
- 'policy': vlan_translation_policies[0].pk,
- 'local_vid': 300,
- 'remote_vid': 400,
- }
- cls.csv_data = (
- "policy,local_vid,remote_vid",
- f"{vlan_translation_policies[0].name},103,203",
- f"{vlan_translation_policies[0].name},104,204",
- f"{vlan_translation_policies[1].name},105,205",
- )
- cls.csv_update_data = (
- "id,local_vid,remote_vid",
- f"{vlan_translation_rules[0].pk},105,205",
- f"{vlan_translation_rules[1].pk},106,206",
- f"{vlan_translation_rules[2].pk},107,207",
- )
- cls.bulk_edit_data = {
- 'policy': vlan_translation_policies[2].pk,
- }
- class ServiceTemplateTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = ServiceTemplate
- @classmethod
- def setUpTestData(cls):
- service_templates = (
- ServiceTemplate(name='Service Template 1', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[101]),
- ServiceTemplate(name='Service Template 2', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[102]),
- ServiceTemplate(name='Service Template 3', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[103]),
- )
- ServiceTemplate.objects.bulk_create(service_templates)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'name': 'Service Template X',
- 'protocol': ServiceProtocolChoices.PROTOCOL_UDP,
- 'ports': '104,105',
- 'description': 'A new service template',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "name,protocol,ports,description",
- "Service Template 4,tcp,1,First service template",
- "Service Template 5,tcp,2,Second service template",
- "Service Template 6,tcp,3,Third service template",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{service_templates[0].pk},Service Template 7,First service template7",
- f"{service_templates[1].pk},Service Template 8,Second service template8",
- f"{service_templates[2].pk},Service Template 9,Third service template9",
- )
- cls.bulk_edit_data = {
- 'protocol': ServiceProtocolChoices.PROTOCOL_UDP,
- 'ports': '106,107',
- 'description': 'New description',
- }
- class ServiceTestCase(ViewTestCases.PrimaryObjectViewTestCase):
- model = Service
- # TODO, related to #9816, cannot validate GFK
- validation_excluded_fields = ('device',)
- @classmethod
- def setUpTestData(cls):
- site = Site.objects.create(name='Site 1', slug='site-1')
- manufacturer = Manufacturer.objects.create(name='Manufacturer 1', slug='manufacturer-1')
- devicetype = DeviceType.objects.create(manufacturer=manufacturer, model='Device Type 1')
- role = DeviceRole.objects.create(name='Device Role 1', slug='device-role-1')
- device = Device.objects.create(name='Device 1', site=site, device_type=devicetype, role=role)
- interface = Interface.objects.create(device=device, name='Interface 1', type=InterfaceTypeChoices.TYPE_VIRTUAL)
- services = (
- Service(parent=device, name='Service 1', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[101]),
- Service(parent=device, name='Service 2', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[102]),
- Service(parent=device, name='Service 3', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[103]),
- )
- Service.objects.bulk_create(services)
- ip_addresses = (
- IPAddress(assigned_object=interface, address='192.0.2.1/24'),
- IPAddress(assigned_object=interface, address='192.0.2.2/24'),
- )
- IPAddress.objects.bulk_create(ip_addresses)
- tags = create_tags('Alpha', 'Bravo', 'Charlie')
- cls.form_data = {
- 'parent_object_type': ContentType.objects.get_for_model(Device).pk,
- 'parent': device.pk,
- 'name': 'Service X',
- 'protocol': ServiceProtocolChoices.PROTOCOL_TCP,
- 'ports': '104,105',
- 'ipaddresses': [],
- 'description': 'A new service',
- 'tags': [t.pk for t in tags],
- }
- cls.csv_data = (
- "parent_object_type,parent,name,protocol,ports,ipaddresses,description",
- "dcim.device,Device 1,Service 1,tcp,1,192.0.2.1/24,First service",
- "dcim.device,Device 1,Service 2,tcp,2,192.0.2.2/24,Second service",
- "dcim.device,Device 1,Service 3,udp,3,,Third service",
- )
- cls.csv_update_data = (
- "id,name,description",
- f"{services[0].pk},Service 7,First service7",
- f"{services[1].pk},Service 8,Second service8",
- f"{services[2].pk},Service 9,Third service9",
- )
- cls.bulk_edit_data = {
- 'protocol': ServiceProtocolChoices.PROTOCOL_UDP,
- 'ports': '106,107',
- 'description': 'New description',
- }
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'], EXEMPT_EXCLUDE_MODELS=[])
- def test_unassigned_ip_addresses(self):
- device = Device.objects.first()
- addr = IPAddress.objects.create(address='192.0.2.4/24')
- csv_data = (
- "parent_object_type,parent_object_id,name,protocol,ports,ipaddresses,description",
- f"dcim.device,{device.pk},Service 11,tcp,10,{addr.address},Eleventh service",
- )
- initial_count = self._get_queryset().count()
- data = {
- 'data': '\n'.join(csv_data),
- 'format': ImportFormatChoices.CSV,
- 'csv_delimiter': CSVDelimiterChoices.AUTO,
- }
- # Assign model-level permission
- obj_perm = ObjectPermission.objects.create(name='Test permission', actions=['add'])
- obj_perm.users.add(self.user)
- obj_perm.object_types.add(ObjectType.objects.get_for_model(self.model))
- # Test POST with permission
- response = self.client.post(self._get_url('bulk_import'), data)
- self.assertHttpStatus(response, 200)
- form_errors = response.context['form'].errors
- self.assertEqual(len(form_errors), 1)
- self.assertIn(addr.address, form_errors['__all__'][0])
- self.assertEqual(self._get_queryset().count(), initial_count)
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'], EXEMPT_EXCLUDE_MODELS=[])
- def test_alternate_csv_import(self):
- device = Device.objects.first()
- interface = device.interfaces.first()
- addr = IPAddress.objects.create(assigned_object=interface, address='192.0.2.3/24')
- csv_data = (
- "parent_object_type,parent_object_id,name,protocol,ports,ipaddresses,description",
- f"dcim.device,{device.pk},Service 11,tcp,10,{addr.address},Eleventh service",
- )
- initial_count = self._get_queryset().count()
- data = {
- 'data': '\n'.join(csv_data),
- 'format': ImportFormatChoices.CSV,
- 'csv_delimiter': CSVDelimiterChoices.AUTO,
- }
- # Assign model-level permission
- obj_perm = ObjectPermission.objects.create(name='Test permission', actions=['add'])
- obj_perm.users.add(self.user)
- obj_perm.object_types.add(ObjectType.objects.get_for_model(self.model))
- # Test POST with permission
- response = self.client.post(self._get_url('bulk_import'), data)
- if response.status_code != 302:
- self.assertEqual(response.context['form'].errors, {}) # debugging aid
- self.assertHttpStatus(response, 302)
- self.assertEqual(self._get_queryset().count(), initial_count + len(csv_data) - 1)
- @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
- def test_create_from_template(self):
- self.add_permissions('ipam.add_service')
- device = Device.objects.first()
- service_template = ServiceTemplate.objects.create(
- name='HTTP',
- protocol=ServiceProtocolChoices.PROTOCOL_TCP,
- ports=[80],
- description='Hypertext transfer protocol'
- )
- request = {
- 'path': self._get_url('add'),
- 'data': {
- 'parent_object_type': ContentType.objects.get_for_model(Device).pk,
- 'parent': device.pk,
- 'service_template': service_template.pk,
- },
- }
- self.assertHttpStatus(self.client.post(**request), 302)
- instance = self._get_queryset().order_by('pk').last()
- self.assertEqual(instance.parent, device)
- self.assertEqual(instance.name, service_template.name)
- self.assertEqual(instance.protocol, service_template.protocol)
- self.assertEqual(instance.ports, service_template.ports)
- self.assertEqual(instance.description, service_template.description)
|