from django.contrib.contenttypes.models import ContentType from django.test import TestCase from dcim.models import Device, DeviceRole, DeviceType, Location, Manufacturer, Platform, Region, Site, SiteGroup from extras.models import ConfigContext, Tag from tenancy.models import Tenant, TenantGroup from utilities.exceptions import AbortRequest from virtualization.models import Cluster, ClusterGroup, ClusterType, VirtualMachine class TagTest(TestCase): def test_create_tag_unicode(self): tag = Tag(name='Testing Unicode: 台灣') tag.save() self.assertEqual(tag.slug, 'testing-unicode-台灣') def test_object_type_validation(self): region = Region.objects.create(name='Region 1', slug='region-1') sitegroup = SiteGroup.objects.create(name='Site Group 1', slug='site-group-1') # Create a Tag that can only be applied to Regions tag = Tag.objects.create(name='Tag 1', slug='tag-1') tag.object_types.add(ContentType.objects.get_by_natural_key('dcim', 'region')) # Apply the Tag to a Region region.tags.add(tag) self.assertIn(tag, region.tags.all()) # Apply the Tag to a SiteGroup with self.assertRaises(AbortRequest): sitegroup.tags.add(tag) class ConfigContextTest(TestCase): """ These test cases deal with the weighting, ordering, and deep merge logic of config context data. It also ensures the various config context querysets are consistent. """ @classmethod def setUpTestData(cls): manufacturer = Manufacturer.objects.create(name='Manufacturer 1', slug='manufacturer-1') devicetype = DeviceType.objects.create(manufacturer=manufacturer, model='Device Type 1', slug='device-type-1') role = DeviceRole.objects.create(name='Device Role 1', slug='device-role-1') region = Region.objects.create(name='Region') sitegroup = SiteGroup.objects.create(name='Site Group') site = Site.objects.create(name='Site 1', slug='site-1', region=region, group=sitegroup) location = Location.objects.create(name='Location 1', slug='location-1', site=site) platform = Platform.objects.create(name='Platform') tenantgroup = TenantGroup.objects.create(name='Tenant Group') tenant = Tenant.objects.create(name='Tenant', group=tenantgroup) tag1 = Tag.objects.create(name='Tag', slug='tag') tag2 = Tag.objects.create(name='Tag2', slug='tag2') Device.objects.create( name='Device 1', device_type=devicetype, role=role, site=site, location=location ) def test_higher_weight_wins(self): device = Device.objects.first() context1 = ConfigContext( name="context 1", weight=101, data={ "a": 123, "b": 456, "c": 777 } ) context2 = ConfigContext( name="context 2", weight=100, data={ "a": 123, "b": 456, "c": 789 } ) ConfigContext.objects.bulk_create([context1, context2]) expected_data = { "a": 123, "b": 456, "c": 777 } self.assertEqual(device.get_config_context(), expected_data) def test_name_ordering_after_weight(self): device = Device.objects.first() context1 = ConfigContext( name="context 1", weight=100, data={ "a": 123, "b": 456, "c": 777 } ) context2 = ConfigContext( name="context 2", weight=100, data={ "a": 123, "b": 456, "c": 789 } ) ConfigContext.objects.bulk_create([context1, context2]) expected_data = { "a": 123, "b": 456, "c": 789 } self.assertEqual(device.get_config_context(), expected_data) def test_annotation_same_as_get_for_object(self): """ This test incorporates features from all of the above tests cases to ensure the annotate_config_context_data() and get_for_object() queryset methods are the same. """ device = Device.objects.first() context1 = ConfigContext( name="context 1", weight=101, data={ "a": 123, "b": 456, "c": 777 } ) context2 = ConfigContext( name="context 2", weight=100, data={ "a": 123, "b": 456, "c": 789 } ) context3 = ConfigContext( name="context 3", weight=99, data={ "d": 1 } ) context4 = ConfigContext( name="context 4", weight=99, data={ "d": 2 } ) ConfigContext.objects.bulk_create([context1, context2, context3, context4]) annotated_queryset = Device.objects.filter(name=device.name).annotate_config_context_data() self.assertEqual(device.get_config_context(), annotated_queryset[0].get_config_context()) def test_annotation_same_as_get_for_object_device_relations(self): region = Region.objects.first() sitegroup = SiteGroup.objects.first() site = Site.objects.first() location = Location.objects.first() platform = Platform.objects.first() tenantgroup = TenantGroup.objects.first() tenant = Tenant.objects.first() tag = Tag.objects.first() region_context = ConfigContext.objects.create( name="region", weight=100, data={ "region": 1 } ) region_context.regions.add(region) sitegroup_context = ConfigContext.objects.create( name="sitegroup", weight=100, data={ "sitegroup": 1 } ) sitegroup_context.site_groups.add(sitegroup) site_context = ConfigContext.objects.create( name="site", weight=100, data={ "site": 1 } ) site_context.sites.add(site) location_context = ConfigContext.objects.create( name="location", weight=100, data={ "location": 1 } ) location_context.locations.add(location) platform_context = ConfigContext.objects.create( name="platform", weight=100, data={ "platform": 1 } ) platform_context.platforms.add(platform) tenant_group_context = ConfigContext.objects.create( name="tenant group", weight=100, data={ "tenant_group": 1 } ) tenant_group_context.tenant_groups.add(tenantgroup) tenant_context = ConfigContext.objects.create( name="tenant", weight=100, data={ "tenant": 1 } ) tenant_context.tenants.add(tenant) tag_context = ConfigContext.objects.create( name="tag", weight=100, data={ "tag": 1 } ) tag_context.tags.add(tag) device = Device.objects.create( name="Device 2", site=site, location=location, tenant=tenant, platform=platform, role=DeviceRole.objects.first(), device_type=DeviceType.objects.first() ) device.tags.add(tag) annotated_queryset = Device.objects.filter(name=device.name).annotate_config_context_data() self.assertEqual(device.get_config_context(), annotated_queryset[0].get_config_context()) def test_annotation_same_as_get_for_object_virtualmachine_relations(self): region = Region.objects.first() sitegroup = SiteGroup.objects.first() site = Site.objects.first() platform = Platform.objects.first() tenantgroup = TenantGroup.objects.first() tenant = Tenant.objects.first() tag = Tag.objects.first() cluster_type = ClusterType.objects.create(name="Cluster Type") cluster_group = ClusterGroup.objects.create(name="Cluster Group") cluster = Cluster.objects.create( name="Cluster", group=cluster_group, type=cluster_type, site=site, ) region_context = ConfigContext.objects.create( name="region", weight=100, data={"region": 1} ) region_context.regions.add(region) sitegroup_context = ConfigContext.objects.create( name="sitegroup", weight=100, data={"sitegroup": 1} ) sitegroup_context.site_groups.add(sitegroup) site_context = ConfigContext.objects.create( name="site", weight=100, data={"site": 1} ) site_context.sites.add(site) platform_context = ConfigContext.objects.create( name="platform", weight=100, data={"platform": 1} ) platform_context.platforms.add(platform) tenant_group_context = ConfigContext.objects.create( name="tenant group", weight=100, data={"tenant_group": 1} ) tenant_group_context.tenant_groups.add(tenantgroup) tenant_context = ConfigContext.objects.create( name="tenant", weight=100, data={"tenant": 1} ) tenant_context.tenants.add(tenant) tag_context = ConfigContext.objects.create( name="tag", weight=100, data={"tag": 1} ) tag_context.tags.add(tag) cluster_type_context = ConfigContext.objects.create( name="cluster type", weight=100, data={"cluster_type": 1} ) cluster_type_context.cluster_types.add(cluster_type) cluster_group_context = ConfigContext.objects.create( name="cluster group", weight=100, data={"cluster_group": 1} ) cluster_group_context.cluster_groups.add(cluster_group) cluster_context = ConfigContext.objects.create( name="cluster", weight=100, data={"cluster": 1} ) cluster_context.clusters.add(cluster) virtual_machine = VirtualMachine.objects.create( name="VM 1", cluster=cluster, tenant=tenant, platform=platform, role=DeviceRole.objects.first() ) virtual_machine.tags.add(tag) annotated_queryset = VirtualMachine.objects.filter(name=virtual_machine.name).annotate_config_context_data() self.assertEqual(virtual_machine.get_config_context(), annotated_queryset[0].get_config_context()) def test_virtualmachine_site_context(self): """ Check that config context associated with a site applies to a VM whether the VM is assigned directly to that site or via its cluster. """ site = Site.objects.first() cluster_type = ClusterType.objects.create(name="Cluster Type") cluster = Cluster.objects.create(name="Cluster", type=cluster_type, site=site) vm_role = DeviceRole.objects.first() # Create a ConfigContext associated with the site context = ConfigContext.objects.create( name="context1", weight=100, data={"foo": True} ) context.sites.add(site) # Create one VM assigned directly to the site, and one assigned via the cluster vm1 = VirtualMachine.objects.create(name="VM 1", site=site, role=vm_role) vm2 = VirtualMachine.objects.create(name="VM 2", cluster=cluster, role=vm_role) # Check that their individually-rendered config contexts are identical self.assertEqual( vm1.get_config_context(), vm2.get_config_context() ) # Check that their annotated config contexts are identical vms = VirtualMachine.objects.filter(pk__in=(vm1.pk, vm2.pk)).annotate_config_context_data() self.assertEqual( vms[0].get_config_context(), vms[1].get_config_context() ) def test_multiple_tags_return_distinct_objects(self): """ Tagged items use a generic relationship, which results in duplicate rows being returned when queried. This is combated by appending distinct() to the config context querysets. This test creates a config context assigned to two tags and ensures objects related by those same two tags result in only a single config context record being returned. See https://github.com/netbox-community/netbox/issues/5314 """ site = Site.objects.first() platform = Platform.objects.first() tenant = Tenant.objects.first() tags = Tag.objects.all() tag_context = ConfigContext.objects.create( name="tag", weight=100, data={ "tag": 1 } ) tag_context.tags.set(tags) device = Device.objects.create( name="Device 3", site=site, tenant=tenant, platform=platform, role=DeviceRole.objects.first(), device_type=DeviceType.objects.first() ) device.tags.set(tags) annotated_queryset = Device.objects.filter(name=device.name).annotate_config_context_data() self.assertEqual(ConfigContext.objects.get_for_object(device).count(), 1) self.assertEqual(device.get_config_context(), annotated_queryset[0].get_config_context()) def test_multiple_tags_return_distinct_objects_with_seperate_config_contexts(self): """ Tagged items use a generic relationship, which results in duplicate rows being returned when queried. This is combatted by by appending distinct() to the config context querysets. This test creates a config context assigned to two tags and ensures objects related by those same two tags result in only a single config context record being returned. This test case is seperate from the above in that it deals with multiple config context objects in play. See https://github.com/netbox-community/netbox/issues/5387 """ site = Site.objects.first() platform = Platform.objects.first() tenant = Tenant.objects.first() tag1, tag2 = list(Tag.objects.all()) tag_context_1 = ConfigContext.objects.create( name="tag-1", weight=100, data={ "tag": 1 } ) tag_context_1.tags.add(tag1) tag_context_2 = ConfigContext.objects.create( name="tag-2", weight=100, data={ "tag": 1 } ) tag_context_2.tags.add(tag2) device = Device.objects.create( name="Device 3", site=site, tenant=tenant, platform=platform, role=DeviceRole.objects.first(), device_type=DeviceType.objects.first() ) device.tags.set([tag1, tag2]) annotated_queryset = Device.objects.filter(name=device.name).annotate_config_context_data() self.assertEqual(ConfigContext.objects.get_for_object(device).count(), 2) self.assertEqual(device.get_config_context(), annotated_queryset[0].get_config_context())