test_api.py 45 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253
  1. import json
  2. from django.urls import reverse
  3. from netaddr import IPNetwork
  4. from rest_framework import status
  5. from dcim.models import Device, DeviceRole, DeviceType, Interface, Manufacturer, Site
  6. from ipam.choices import *
  7. from ipam.models import *
  8. from tenancy.models import Tenant
  9. from utilities.data import string_to_ranges
  10. from utilities.testing import APITestCase, APIViewTestCases, create_test_device, disable_warnings
  11. class AppTest(APITestCase):
  12. def test_root(self):
  13. url = reverse('ipam-api:api-root')
  14. response = self.client.get('{}?format=api'.format(url), **self.header)
  15. self.assertEqual(response.status_code, 200)
  16. class ASNRangeTest(APIViewTestCases.APIViewTestCase):
  17. model = ASNRange
  18. brief_fields = ['description', 'display', 'id', 'name', 'url']
  19. bulk_update_data = {
  20. 'description': 'New description',
  21. }
  22. @classmethod
  23. def setUpTestData(cls):
  24. rirs = (
  25. RIR(name='RIR 1', slug='rir-1', is_private=True),
  26. RIR(name='RIR 2', slug='rir-2', is_private=True),
  27. )
  28. RIR.objects.bulk_create(rirs)
  29. tenants = (
  30. Tenant(name='Tenant 1', slug='tenant-1'),
  31. Tenant(name='Tenant 2', slug='tenant-2'),
  32. )
  33. Tenant.objects.bulk_create(tenants)
  34. asn_ranges = (
  35. ASNRange(name='ASN Range 1', slug='asn-range-1', rir=rirs[0], tenant=tenants[0], start=100, end=199),
  36. ASNRange(name='ASN Range 2', slug='asn-range-2', rir=rirs[0], tenant=tenants[0], start=200, end=299),
  37. ASNRange(name='ASN Range 3', slug='asn-range-3', rir=rirs[0], tenant=tenants[0], start=300, end=399),
  38. )
  39. ASNRange.objects.bulk_create(asn_ranges)
  40. cls.create_data = [
  41. {
  42. 'name': 'ASN Range 4',
  43. 'slug': 'asn-range-4',
  44. 'rir': rirs[1].pk,
  45. 'start': 400,
  46. 'end': 499,
  47. 'tenant': tenants[1].pk,
  48. },
  49. {
  50. 'name': 'ASN Range 5',
  51. 'slug': 'asn-range-5',
  52. 'rir': rirs[1].pk,
  53. 'start': 500,
  54. 'end': 599,
  55. 'tenant': tenants[1].pk,
  56. },
  57. {
  58. 'name': 'ASN Range 6',
  59. 'slug': 'asn-range-6',
  60. 'rir': rirs[1].pk,
  61. 'start': 600,
  62. 'end': 699,
  63. 'tenant': tenants[1].pk,
  64. },
  65. ]
  66. def test_list_available_asns(self):
  67. """
  68. Test retrieval of all available ASNs within a parent range.
  69. """
  70. rir = RIR.objects.first()
  71. asnrange = ASNRange.objects.create(name='Range 1', slug='range-1', rir=rir, start=101, end=110)
  72. url = reverse('ipam-api:asnrange-available-asns', kwargs={'pk': asnrange.pk})
  73. self.add_permissions('ipam.view_asnrange', 'ipam.view_asn')
  74. response = self.client.get(url, **self.header)
  75. self.assertHttpStatus(response, status.HTTP_200_OK)
  76. self.assertEqual(len(response.data), 10)
  77. def test_create_single_available_asn(self):
  78. """
  79. Test creation of the first available ASN within a range.
  80. """
  81. rir = RIR.objects.first()
  82. asnrange = ASNRange.objects.create(name='Range 1', slug='range-1', rir=rir, start=101, end=110)
  83. url = reverse('ipam-api:asnrange-available-asns', kwargs={'pk': asnrange.pk})
  84. self.add_permissions('ipam.view_asnrange', 'ipam.add_asn')
  85. data = {
  86. 'description': 'New ASN'
  87. }
  88. response = self.client.post(url, data, format='json', **self.header)
  89. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  90. self.assertEqual(response.data['rir']['id'], asnrange.rir.pk)
  91. self.assertEqual(response.data['description'], data['description'])
  92. def test_create_multiple_available_asns(self):
  93. """
  94. Test the creation of several available ASNs within a parent range.
  95. """
  96. rir = RIR.objects.first()
  97. asnrange = ASNRange.objects.create(name='Range 1', slug='range-1', rir=rir, start=101, end=110)
  98. url = reverse('ipam-api:asnrange-available-asns', kwargs={'pk': asnrange.pk})
  99. self.add_permissions('ipam.view_asnrange', 'ipam.add_asn')
  100. # Try to create eleven ASNs (only ten are available)
  101. data = [
  102. {'description': f'New ASN {i}'}
  103. for i in range(1, 12)
  104. ]
  105. assert len(data) == 11
  106. response = self.client.post(url, data, format='json', **self.header)
  107. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  108. self.assertIn('detail', response.data)
  109. # Create all ten available ASNs in a single request
  110. data.pop()
  111. assert len(data) == 10
  112. response = self.client.post(url, data, format='json', **self.header)
  113. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  114. self.assertEqual(len(response.data), 10)
  115. class ASNTest(APIViewTestCases.APIViewTestCase):
  116. model = ASN
  117. brief_fields = ['asn', 'description', 'display', 'id', 'url']
  118. bulk_update_data = {
  119. 'description': 'New description',
  120. }
  121. @classmethod
  122. def setUpTestData(cls):
  123. rirs = (
  124. RIR(name='RIR 1', slug='rir-1', is_private=True),
  125. RIR(name='RIR 2', slug='rir-2', is_private=True),
  126. )
  127. RIR.objects.bulk_create(rirs)
  128. sites = (
  129. Site(name='Site 1', slug='site-1'),
  130. Site(name='Site 2', slug='site-2')
  131. )
  132. Site.objects.bulk_create(sites)
  133. tenants = (
  134. Tenant(name='Tenant 1', slug='tenant-1'),
  135. Tenant(name='Tenant 2', slug='tenant-2'),
  136. )
  137. Tenant.objects.bulk_create(tenants)
  138. asns = (
  139. ASN(asn=65000, rir=rirs[0], tenant=tenants[0]),
  140. ASN(asn=65001, rir=rirs[0], tenant=tenants[1]),
  141. ASN(asn=4200000000, rir=rirs[1], tenant=tenants[0]),
  142. ASN(asn=4200000001, rir=rirs[1], tenant=tenants[1]),
  143. )
  144. ASN.objects.bulk_create(asns)
  145. asns[0].sites.set([sites[0]])
  146. asns[1].sites.set([sites[1]])
  147. asns[2].sites.set([sites[0]])
  148. asns[3].sites.set([sites[1]])
  149. cls.create_data = [
  150. {
  151. 'asn': 64512,
  152. 'rir': rirs[0].pk,
  153. },
  154. {
  155. 'asn': 65002,
  156. 'rir': rirs[0].pk,
  157. },
  158. {
  159. 'asn': 4200000002,
  160. 'rir': rirs[1].pk,
  161. },
  162. ]
  163. class VRFTest(APIViewTestCases.APIViewTestCase):
  164. model = VRF
  165. brief_fields = ['description', 'display', 'id', 'name', 'prefix_count', 'rd', 'url']
  166. create_data = [
  167. {
  168. 'name': 'VRF 4',
  169. 'rd': '65000:4',
  170. },
  171. {
  172. 'name': 'VRF 5',
  173. 'rd': '65000:5',
  174. },
  175. {
  176. 'name': 'VRF 6',
  177. 'rd': '65000:6',
  178. },
  179. ]
  180. bulk_update_data = {
  181. 'description': 'New description',
  182. }
  183. @classmethod
  184. def setUpTestData(cls):
  185. vrfs = (
  186. VRF(name='VRF 1', rd='65000:1'),
  187. VRF(name='VRF 2', rd='65000:2'),
  188. VRF(name='VRF 3'), # No RD
  189. )
  190. VRF.objects.bulk_create(vrfs)
  191. class RouteTargetTest(APIViewTestCases.APIViewTestCase):
  192. model = RouteTarget
  193. brief_fields = ['description', 'display', 'id', 'name', 'url']
  194. create_data = [
  195. {
  196. 'name': '65000:1004',
  197. },
  198. {
  199. 'name': '65000:1005',
  200. },
  201. {
  202. 'name': '65000:1006',
  203. },
  204. ]
  205. bulk_update_data = {
  206. 'description': 'New description',
  207. }
  208. @classmethod
  209. def setUpTestData(cls):
  210. route_targets = (
  211. RouteTarget(name='65000:1001'),
  212. RouteTarget(name='65000:1002'),
  213. RouteTarget(name='65000:1003'),
  214. )
  215. RouteTarget.objects.bulk_create(route_targets)
  216. class RIRTest(APIViewTestCases.APIViewTestCase):
  217. model = RIR
  218. brief_fields = ['aggregate_count', 'description', 'display', 'id', 'name', 'slug', 'url']
  219. create_data = [
  220. {
  221. 'name': 'RIR 4',
  222. 'slug': 'rir-4',
  223. },
  224. {
  225. 'name': 'RIR 5',
  226. 'slug': 'rir-5',
  227. },
  228. {
  229. 'name': 'RIR 6',
  230. 'slug': 'rir-6',
  231. },
  232. ]
  233. bulk_update_data = {
  234. 'description': 'New description',
  235. }
  236. @classmethod
  237. def setUpTestData(cls):
  238. rirs = (
  239. RIR(name='RIR 1', slug='rir-1'),
  240. RIR(name='RIR 2', slug='rir-2'),
  241. RIR(name='RIR 3', slug='rir-3'),
  242. )
  243. RIR.objects.bulk_create(rirs)
  244. class AggregateTest(APIViewTestCases.APIViewTestCase):
  245. model = Aggregate
  246. brief_fields = ['description', 'display', 'family', 'id', 'prefix', 'url']
  247. bulk_update_data = {
  248. 'description': 'New description',
  249. }
  250. @classmethod
  251. def setUpTestData(cls):
  252. rirs = (
  253. RIR(name='RIR 1', slug='rir-1'),
  254. RIR(name='RIR 2', slug='rir-2'),
  255. )
  256. RIR.objects.bulk_create(rirs)
  257. aggregates = (
  258. Aggregate(prefix=IPNetwork('10.0.0.0/8'), rir=rirs[0]),
  259. Aggregate(prefix=IPNetwork('172.16.0.0/12'), rir=rirs[0]),
  260. Aggregate(prefix=IPNetwork('192.168.0.0/16'), rir=rirs[0]),
  261. )
  262. Aggregate.objects.bulk_create(aggregates)
  263. cls.create_data = [
  264. {
  265. 'prefix': '100.0.0.0/8',
  266. 'rir': rirs[1].pk,
  267. },
  268. {
  269. 'prefix': '101.0.0.0/8',
  270. 'rir': rirs[1].pk,
  271. },
  272. {
  273. 'prefix': '102.0.0.0/8',
  274. 'rir': rirs[1].pk,
  275. },
  276. ]
  277. class RoleTest(APIViewTestCases.APIViewTestCase):
  278. model = Role
  279. brief_fields = ['description', 'display', 'id', 'name', 'prefix_count', 'slug', 'url', 'vlan_count']
  280. create_data = [
  281. {
  282. 'name': 'Role 4',
  283. 'slug': 'role-4',
  284. },
  285. {
  286. 'name': 'Role 5',
  287. 'slug': 'role-5',
  288. },
  289. {
  290. 'name': 'Role 6',
  291. 'slug': 'role-6',
  292. },
  293. ]
  294. bulk_update_data = {
  295. 'description': 'New description',
  296. }
  297. @classmethod
  298. def setUpTestData(cls):
  299. roles = (
  300. Role(name='Role 1', slug='role-1'),
  301. Role(name='Role 2', slug='role-2'),
  302. Role(name='Role 3', slug='role-3'),
  303. )
  304. Role.objects.bulk_create(roles)
  305. class PrefixTest(APIViewTestCases.APIViewTestCase):
  306. model = Prefix
  307. brief_fields = ['_depth', 'description', 'display', 'family', 'id', 'prefix', 'url']
  308. create_data = [
  309. {
  310. 'prefix': '192.168.4.0/24',
  311. },
  312. {
  313. 'prefix': '192.168.5.0/24',
  314. },
  315. {
  316. 'prefix': '192.168.6.0/24',
  317. },
  318. ]
  319. bulk_update_data = {
  320. 'description': 'New description',
  321. }
  322. @classmethod
  323. def setUpTestData(cls):
  324. prefixes = (
  325. Prefix(prefix=IPNetwork('192.168.1.0/24')),
  326. Prefix(prefix=IPNetwork('192.168.2.0/24')),
  327. Prefix(prefix=IPNetwork('192.168.3.0/24')),
  328. )
  329. Prefix.objects.bulk_create(prefixes)
  330. def test_list_available_prefixes(self):
  331. """
  332. Test retrieval of all available prefixes within a parent prefix.
  333. """
  334. vrf = VRF.objects.create(name='VRF 1')
  335. prefix = Prefix.objects.create(prefix=IPNetwork('192.0.2.0/24'), vrf=vrf)
  336. Prefix.objects.create(prefix=IPNetwork('192.0.2.64/26'), vrf=vrf)
  337. Prefix.objects.create(prefix=IPNetwork('192.0.2.192/27'), vrf=vrf)
  338. url = reverse('ipam-api:prefix-available-prefixes', kwargs={'pk': prefix.pk})
  339. self.add_permissions('ipam.view_prefix')
  340. # Retrieve all available IPs
  341. response = self.client.get(url, **self.header)
  342. available_prefixes = ['192.0.2.0/26', '192.0.2.128/26', '192.0.2.224/27']
  343. for i, p in enumerate(response.data):
  344. self.assertEqual(p['prefix'], available_prefixes[i])
  345. def test_create_single_available_prefix(self):
  346. """
  347. Test retrieval of the first available prefix within a parent prefix.
  348. """
  349. vrf = VRF.objects.create(name='VRF 1')
  350. prefix = Prefix.objects.create(prefix=IPNetwork('192.0.2.0/28'), vrf=vrf, is_pool=True)
  351. url = reverse('ipam-api:prefix-available-prefixes', kwargs={'pk': prefix.pk})
  352. self.add_permissions('ipam.view_prefix', 'ipam.add_prefix')
  353. # Create four available prefixes with individual requests
  354. prefixes_to_be_created = [
  355. '192.0.2.0/30',
  356. '192.0.2.4/30',
  357. '192.0.2.8/30',
  358. '192.0.2.12/30',
  359. ]
  360. for i in range(4):
  361. data = {
  362. 'prefix_length': 30,
  363. 'description': 'Test Prefix {}'.format(i + 1)
  364. }
  365. response = self.client.post(url, data, format='json', **self.header)
  366. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  367. self.assertEqual(response.data['prefix'], prefixes_to_be_created[i])
  368. self.assertEqual(response.data['vrf']['id'], vrf.pk)
  369. self.assertEqual(response.data['description'], data['description'])
  370. # Try to create one more prefix
  371. response = self.client.post(url, {'prefix_length': 30}, format='json', **self.header)
  372. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  373. self.assertIn('detail', response.data)
  374. # Try to create invalid prefix type
  375. response = self.client.post(url, {'prefix_length': '30'}, format='json', **self.header)
  376. self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
  377. self.assertIn('prefix_length', response.data[0])
  378. def test_create_multiple_available_prefixes(self):
  379. """
  380. Test the creation of available prefixes within a parent prefix.
  381. """
  382. vrf = VRF.objects.create(name='VRF 1')
  383. prefix = Prefix.objects.create(prefix=IPNetwork('192.0.2.0/28'), vrf=vrf, is_pool=True)
  384. url = reverse('ipam-api:prefix-available-prefixes', kwargs={'pk': prefix.pk})
  385. self.add_permissions('ipam.view_prefix', 'ipam.add_prefix')
  386. # Try to create five /30s (only four are available)
  387. data = [
  388. {'prefix_length': 30, 'description': 'Prefix 1'},
  389. {'prefix_length': 30, 'description': 'Prefix 2'},
  390. {'prefix_length': 30, 'description': 'Prefix 3'},
  391. {'prefix_length': 30, 'description': 'Prefix 4'},
  392. {'prefix_length': 30, 'description': 'Prefix 5'},
  393. ]
  394. response = self.client.post(url, data, format='json', **self.header)
  395. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  396. self.assertIn('detail', response.data)
  397. # Verify that no prefixes were created (the entire /28 is still available)
  398. response = self.client.get(url, **self.header)
  399. self.assertHttpStatus(response, status.HTTP_200_OK)
  400. self.assertEqual(response.data[0]['prefix'], '192.0.2.0/28')
  401. # Create four /30s in a single request
  402. response = self.client.post(url, data[:4], format='json', **self.header)
  403. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  404. self.assertEqual(len(response.data), 4)
  405. def test_list_available_ips(self):
  406. """
  407. Test retrieval of all available IP addresses within a parent prefix.
  408. """
  409. vrf = VRF.objects.create(name='VRF 1')
  410. prefix = Prefix.objects.create(prefix=IPNetwork('192.0.2.0/29'), vrf=vrf, is_pool=True)
  411. url = reverse('ipam-api:prefix-available-ips', kwargs={'pk': prefix.pk})
  412. self.add_permissions('ipam.view_prefix', 'ipam.view_ipaddress')
  413. # Retrieve all available IPs
  414. response = self.client.get(url, **self.header)
  415. self.assertHttpStatus(response, status.HTTP_200_OK)
  416. self.assertEqual(len(response.data), 8) # 8 because prefix.is_pool = True
  417. # Change the prefix to not be a pool and try again
  418. prefix.is_pool = False
  419. prefix.save()
  420. response = self.client.get(url, **self.header)
  421. self.assertEqual(len(response.data), 6) # 8 - 2 because prefix.is_pool = False
  422. def test_create_single_available_ip(self):
  423. """
  424. Test retrieval of the first available IP address within a parent prefix.
  425. """
  426. vrf = VRF.objects.create(name='VRF 1')
  427. prefix = Prefix.objects.create(prefix=IPNetwork('192.0.2.0/30'), vrf=vrf, is_pool=True)
  428. url = reverse('ipam-api:prefix-available-ips', kwargs={'pk': prefix.pk})
  429. self.add_permissions('ipam.view_prefix', 'ipam.add_ipaddress')
  430. # Create all four available IPs with individual requests
  431. for i in range(1, 5):
  432. data = {
  433. 'description': 'Test IP {}'.format(i)
  434. }
  435. response = self.client.post(url, data, format='json', **self.header)
  436. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  437. self.assertEqual(response.data['vrf']['id'], vrf.pk)
  438. self.assertEqual(response.data['description'], data['description'])
  439. # Try to create one more IP
  440. response = self.client.post(url, {}, format='json', **self.header)
  441. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  442. self.assertIn('detail', response.data)
  443. def test_create_multiple_available_ips(self):
  444. """
  445. Test the creation of available IP addresses within a parent prefix.
  446. """
  447. vrf = VRF.objects.create(name='VRF 1')
  448. prefix = Prefix.objects.create(prefix=IPNetwork('192.0.2.0/29'), vrf=vrf, is_pool=True)
  449. url = reverse('ipam-api:prefix-available-ips', kwargs={'pk': prefix.pk})
  450. self.add_permissions('ipam.view_prefix', 'ipam.add_ipaddress')
  451. # Try to create nine IPs (only eight are available)
  452. data = [{'description': f'Test IP {i}'} for i in range(1, 10)] # 9 IPs
  453. response = self.client.post(url, data, format='json', **self.header)
  454. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  455. self.assertIn('detail', response.data)
  456. # Create all eight available IPs in a single request
  457. data = [{'description': 'Test IP {}'.format(i)} for i in range(1, 9)] # 8 IPs
  458. response = self.client.post(url, data, format='json', **self.header)
  459. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  460. self.assertEqual(len(response.data), 8)
  461. def test_get_prefix_with_aggregate_and_rir(self):
  462. self.add_permissions('ipam.view_prefix')
  463. rir = RIR.objects.create(name='RFC 1918', slug='rfc-1918')
  464. aggregate = Aggregate.objects.create(prefix=IPNetwork('192.168.0.0/16'), rir=rir)
  465. prefixes = [
  466. Prefix.objects.filter(prefix=IPNetwork('192.168.2.0/24')).first(),
  467. Prefix.objects.create(prefix=IPNetwork('10.0.0.0/24'))
  468. ]
  469. self.assertIsNotNone(prefixes[0])
  470. url = self._get_detail_url(prefixes[0])
  471. response = self.client.get(url, **self.header)
  472. self.assertIsNotNone(prefixes[0].aggregate)
  473. self.assertHttpStatus(response, status.HTTP_200_OK)
  474. self.assertIsNotNone(response.data.get('aggregate'))
  475. self.assertIsNotNone(response.data.get('rir'))
  476. self.assertEqual(response.data.get('aggregate').get('id', None), aggregate.pk)
  477. self.assertEqual(response.data.get('rir').get('id', None), rir.pk)
  478. url = self._get_detail_url(prefixes[1])
  479. response = self.client.get(url, **self.header)
  480. self.assertIsNone(prefixes[1].aggregate)
  481. self.assertHttpStatus(response, status.HTTP_200_OK)
  482. self.assertIsNone(response.data.get('aggregate'))
  483. self.assertIsNone(response.data.get('rir'))
  484. class IPRangeTest(APIViewTestCases.APIViewTestCase):
  485. model = IPRange
  486. brief_fields = ['description', 'display', 'end_address', 'family', 'id', 'start_address', 'url']
  487. create_data = [
  488. {
  489. 'start_address': '192.168.4.10/24',
  490. 'end_address': '192.168.4.50/24',
  491. },
  492. {
  493. 'start_address': '192.168.5.10/24',
  494. 'end_address': '192.168.5.50/24',
  495. },
  496. {
  497. 'start_address': '192.168.6.10/24',
  498. 'end_address': '192.168.6.50/24',
  499. },
  500. ]
  501. bulk_update_data = {
  502. 'description': 'New description',
  503. }
  504. @classmethod
  505. def setUpTestData(cls):
  506. ip_ranges = (
  507. IPRange(start_address=IPNetwork('192.168.1.10/24'), end_address=IPNetwork('192.168.1.50/24'), size=51),
  508. IPRange(start_address=IPNetwork('192.168.2.10/24'), end_address=IPNetwork('192.168.2.50/24'), size=51),
  509. IPRange(start_address=IPNetwork('192.168.3.10/24'), end_address=IPNetwork('192.168.3.50/24'), size=51),
  510. )
  511. IPRange.objects.bulk_create(ip_ranges)
  512. def test_list_available_ips(self):
  513. """
  514. Test retrieval of all available IP addresses within a parent IP range.
  515. """
  516. iprange = IPRange.objects.create(
  517. start_address=IPNetwork('192.0.2.10/24'),
  518. end_address=IPNetwork('192.0.2.19/24')
  519. )
  520. url = reverse('ipam-api:iprange-available-ips', kwargs={'pk': iprange.pk})
  521. self.add_permissions('ipam.view_iprange', 'ipam.view_ipaddress')
  522. # Retrieve all available IPs
  523. response = self.client.get(url, **self.header)
  524. self.assertHttpStatus(response, status.HTTP_200_OK)
  525. self.assertEqual(len(response.data), 10)
  526. def test_create_single_available_ip(self):
  527. """
  528. Test retrieval of the first available IP address within a parent IP range.
  529. """
  530. vrf = VRF.objects.create(name='Test VRF 1', rd='1234')
  531. iprange = IPRange.objects.create(
  532. start_address=IPNetwork('192.0.2.1/24'),
  533. end_address=IPNetwork('192.0.2.3/24'),
  534. vrf=vrf
  535. )
  536. url = reverse('ipam-api:iprange-available-ips', kwargs={'pk': iprange.pk})
  537. self.add_permissions('ipam.view_iprange', 'ipam.add_ipaddress')
  538. # Create all three available IPs with individual requests
  539. for i in range(1, 4):
  540. data = {
  541. 'description': f'Test IP #{i}'
  542. }
  543. response = self.client.post(url, data, format='json', **self.header)
  544. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  545. self.assertEqual(response.data['vrf']['id'], vrf.pk)
  546. self.assertEqual(response.data['description'], data['description'])
  547. # Try to create one more IP
  548. response = self.client.post(url, {}, format='json', **self.header)
  549. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  550. self.assertIn('detail', response.data)
  551. def test_create_multiple_available_ips(self):
  552. """
  553. Test the creation of available IP addresses within a parent IP range.
  554. """
  555. iprange = IPRange.objects.create(
  556. start_address=IPNetwork('192.0.2.1/24'),
  557. end_address=IPNetwork('192.0.2.8/24')
  558. )
  559. url = reverse('ipam-api:iprange-available-ips', kwargs={'pk': iprange.pk})
  560. self.add_permissions('ipam.view_iprange', 'ipam.add_ipaddress')
  561. # Try to create nine IPs (only eight are available)
  562. data = [{'description': f'Test IP #{i}'} for i in range(1, 10)] # 9 IPs
  563. response = self.client.post(url, data, format='json', **self.header)
  564. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  565. self.assertIn('detail', response.data)
  566. # Create all eight available IPs in a single request
  567. data = [{'description': f'Test IP #{i}'} for i in range(1, 9)] # 8 IPs
  568. response = self.client.post(url, data, format='json', **self.header)
  569. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  570. self.assertEqual(len(response.data), 8)
  571. class IPAddressTest(APIViewTestCases.APIViewTestCase):
  572. model = IPAddress
  573. brief_fields = ['address', 'description', 'display', 'family', 'id', 'url']
  574. create_data = [
  575. {
  576. 'address': '192.168.0.4/24',
  577. },
  578. {
  579. 'address': '192.168.0.5/24',
  580. },
  581. {
  582. 'address': '192.168.0.6/24',
  583. },
  584. ]
  585. bulk_update_data = {
  586. 'description': 'New description',
  587. }
  588. graphql_filter = {
  589. 'address': {'lookup': 'i_exact', 'value': '192.168.0.1/24'},
  590. }
  591. @classmethod
  592. def setUpTestData(cls):
  593. ip_addresses = (
  594. IPAddress(address=IPNetwork('192.168.0.1/24')),
  595. IPAddress(address=IPNetwork('192.168.0.2/24')),
  596. IPAddress(address=IPNetwork('192.168.0.3/24')),
  597. )
  598. IPAddress.objects.bulk_create(ip_addresses)
  599. def test_assign_object(self):
  600. """
  601. Test the creation of available IP addresses within a parent IP range.
  602. """
  603. site = Site.objects.create(name='Site 1')
  604. manufacturer = Manufacturer.objects.create(name='Manufacturer 1')
  605. device_type = DeviceType.objects.create(model='Device Type 1', manufacturer=manufacturer)
  606. role = DeviceRole.objects.create(name='Switch')
  607. device1 = Device.objects.create(
  608. name='Device 1',
  609. site=site,
  610. device_type=device_type,
  611. role=role,
  612. status='active'
  613. )
  614. interface1 = Interface.objects.create(name='Interface 1', device=device1, type='1000baset')
  615. interface2 = Interface.objects.create(name='Interface 2', device=device1, type='1000baset')
  616. device2 = Device.objects.create(
  617. name='Device 2',
  618. site=site,
  619. device_type=device_type,
  620. role=role,
  621. status='active'
  622. )
  623. interface3 = Interface.objects.create(name='Interface 3', device=device2, type='1000baset')
  624. ip_addresses = (
  625. IPAddress(address=IPNetwork('192.168.0.4/24'), assigned_object=interface1),
  626. IPAddress(address=IPNetwork('192.168.1.4/24')),
  627. )
  628. IPAddress.objects.bulk_create(ip_addresses)
  629. ip1 = ip_addresses[0]
  630. ip1.assigned_object = interface1
  631. device1.primary_ip4 = ip_addresses[0]
  632. device1.save()
  633. url = reverse('ipam-api:ipaddress-detail', kwargs={'pk': ip1.pk})
  634. self.add_permissions('ipam.change_ipaddress')
  635. # assign to same parent
  636. data = {
  637. 'assigned_object_id': interface2.pk
  638. }
  639. response = self.client.patch(url, data, format='json', **self.header)
  640. self.assertHttpStatus(response, status.HTTP_200_OK)
  641. # assign to same different parent - should error
  642. data = {
  643. 'assigned_object_id': interface3.pk
  644. }
  645. response = self.client.patch(url, data, format='json', **self.header)
  646. self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
  647. class FHRPGroupTest(APIViewTestCases.APIViewTestCase):
  648. model = FHRPGroup
  649. brief_fields = ['description', 'display', 'group_id', 'id', 'protocol', 'url']
  650. bulk_update_data = {
  651. 'protocol': FHRPGroupProtocolChoices.PROTOCOL_GLBP,
  652. 'group_id': 200,
  653. 'auth_type': FHRPGroupAuthTypeChoices.AUTHENTICATION_MD5,
  654. 'auth_key': 'foobarbaz999',
  655. 'name': 'foobar-999',
  656. 'description': 'New description',
  657. }
  658. @classmethod
  659. def setUpTestData(cls):
  660. fhrp_groups = (
  661. FHRPGroup(
  662. protocol=FHRPGroupProtocolChoices.PROTOCOL_VRRP2,
  663. group_id=10,
  664. auth_type=FHRPGroupAuthTypeChoices.AUTHENTICATION_PLAINTEXT,
  665. auth_key='foobar123',
  666. ),
  667. FHRPGroup(
  668. protocol=FHRPGroupProtocolChoices.PROTOCOL_VRRP3,
  669. group_id=20,
  670. auth_type=FHRPGroupAuthTypeChoices.AUTHENTICATION_MD5,
  671. auth_key='foobar123',
  672. ),
  673. FHRPGroup(protocol=FHRPGroupProtocolChoices.PROTOCOL_HSRP, group_id=30),
  674. )
  675. FHRPGroup.objects.bulk_create(fhrp_groups)
  676. cls.create_data = [
  677. {
  678. 'protocol': FHRPGroupProtocolChoices.PROTOCOL_VRRP2,
  679. 'group_id': 110,
  680. 'auth_type': FHRPGroupAuthTypeChoices.AUTHENTICATION_PLAINTEXT,
  681. 'auth_key': 'foobar123',
  682. },
  683. {
  684. 'protocol': FHRPGroupProtocolChoices.PROTOCOL_VRRP3,
  685. 'group_id': 120,
  686. 'auth_type': FHRPGroupAuthTypeChoices.AUTHENTICATION_MD5,
  687. 'auth_key': 'barfoo456',
  688. },
  689. {
  690. 'protocol': FHRPGroupProtocolChoices.PROTOCOL_GLBP,
  691. 'group_id': 130,
  692. },
  693. ]
  694. class FHRPGroupAssignmentTest(APIViewTestCases.APIViewTestCase):
  695. model = FHRPGroupAssignment
  696. brief_fields = ['display', 'group', 'id', 'interface_id', 'interface_type', 'priority', 'url']
  697. bulk_update_data = {
  698. 'priority': 100,
  699. }
  700. user_permissions = ('ipam.view_fhrpgroup', )
  701. @classmethod
  702. def setUpTestData(cls):
  703. device1 = create_test_device('device1')
  704. device2 = create_test_device('device2')
  705. device3 = create_test_device('device3')
  706. interfaces = (
  707. Interface(device=device1, name='eth0', type='other'),
  708. Interface(device=device1, name='eth1', type='other'),
  709. Interface(device=device1, name='eth2', type='other'),
  710. Interface(device=device2, name='eth0', type='other'),
  711. Interface(device=device2, name='eth1', type='other'),
  712. Interface(device=device2, name='eth2', type='other'),
  713. Interface(device=device3, name='eth0', type='other'),
  714. Interface(device=device3, name='eth1', type='other'),
  715. Interface(device=device3, name='eth2', type='other'),
  716. )
  717. Interface.objects.bulk_create(interfaces)
  718. ip_addresses = (
  719. IPAddress(address=IPNetwork('192.168.0.2/24'), assigned_object=interfaces[0]),
  720. IPAddress(address=IPNetwork('192.168.1.2/24'), assigned_object=interfaces[1]),
  721. IPAddress(address=IPNetwork('192.168.2.2/24'), assigned_object=interfaces[2]),
  722. IPAddress(address=IPNetwork('192.168.0.3/24'), assigned_object=interfaces[3]),
  723. IPAddress(address=IPNetwork('192.168.1.3/24'), assigned_object=interfaces[4]),
  724. IPAddress(address=IPNetwork('192.168.2.3/24'), assigned_object=interfaces[5]),
  725. IPAddress(address=IPNetwork('192.168.0.4/24'), assigned_object=interfaces[6]),
  726. IPAddress(address=IPNetwork('192.168.1.4/24'), assigned_object=interfaces[7]),
  727. IPAddress(address=IPNetwork('192.168.2.4/24'), assigned_object=interfaces[8]),
  728. )
  729. IPAddress.objects.bulk_create(ip_addresses)
  730. fhrp_groups = (
  731. FHRPGroup(protocol=FHRPGroupProtocolChoices.PROTOCOL_VRRP2, group_id=10),
  732. FHRPGroup(protocol=FHRPGroupProtocolChoices.PROTOCOL_VRRP2, group_id=20),
  733. FHRPGroup(protocol=FHRPGroupProtocolChoices.PROTOCOL_VRRP2, group_id=30),
  734. )
  735. FHRPGroup.objects.bulk_create(fhrp_groups)
  736. fhrp_group_assignments = (
  737. FHRPGroupAssignment(group=fhrp_groups[0], interface=interfaces[0], priority=10),
  738. FHRPGroupAssignment(group=fhrp_groups[1], interface=interfaces[1], priority=10),
  739. FHRPGroupAssignment(group=fhrp_groups[2], interface=interfaces[2], priority=10),
  740. FHRPGroupAssignment(group=fhrp_groups[0], interface=interfaces[3], priority=20),
  741. FHRPGroupAssignment(group=fhrp_groups[1], interface=interfaces[4], priority=20),
  742. FHRPGroupAssignment(group=fhrp_groups[2], interface=interfaces[5], priority=20),
  743. )
  744. FHRPGroupAssignment.objects.bulk_create(fhrp_group_assignments)
  745. cls.create_data = [
  746. {
  747. 'group': fhrp_groups[0].pk,
  748. 'interface_type': 'dcim.interface',
  749. 'interface_id': interfaces[6].pk,
  750. 'priority': 30,
  751. },
  752. {
  753. 'group': fhrp_groups[1].pk,
  754. 'interface_type': 'dcim.interface',
  755. 'interface_id': interfaces[7].pk,
  756. 'priority': 30,
  757. },
  758. {
  759. 'group': fhrp_groups[2].pk,
  760. 'interface_type': 'dcim.interface',
  761. 'interface_id': interfaces[8].pk,
  762. 'priority': 30,
  763. },
  764. ]
  765. class VLANGroupTest(APIViewTestCases.APIViewTestCase):
  766. model = VLANGroup
  767. brief_fields = ['description', 'display', 'id', 'name', 'slug', 'url', 'vlan_count']
  768. create_data = [
  769. {
  770. 'name': 'VLAN Group 4',
  771. 'slug': 'vlan-group-4',
  772. },
  773. {
  774. 'name': 'VLAN Group 5',
  775. 'slug': 'vlan-group-5',
  776. },
  777. {
  778. 'name': 'VLAN Group 6',
  779. 'slug': 'vlan-group-6',
  780. },
  781. ]
  782. bulk_update_data = {
  783. 'description': 'New description',
  784. }
  785. @classmethod
  786. def setUpTestData(cls):
  787. vlan_groups = (
  788. VLANGroup(name='VLAN Group 1', slug='vlan-group-1'),
  789. VLANGroup(name='VLAN Group 2', slug='vlan-group-2'),
  790. VLANGroup(name='VLAN Group 3', slug='vlan-group-3'),
  791. )
  792. VLANGroup.objects.bulk_create(vlan_groups)
  793. def test_list_available_vlans(self):
  794. """
  795. Test retrieval of all available VLANs within a group.
  796. """
  797. MIN_VID = 100
  798. MAX_VID = 199
  799. self.add_permissions('ipam.view_vlangroup', 'ipam.view_vlan')
  800. vlangroup = VLANGroup.objects.create(
  801. name='VLAN Group X',
  802. slug='vlan-group-x',
  803. vid_ranges=string_to_ranges(f"{MIN_VID}-{MAX_VID}")
  804. )
  805. # Create a set of VLANs within the group
  806. vlans = (
  807. VLAN(vid=10, name='VLAN 10', group=vlangroup),
  808. VLAN(vid=20, name='VLAN 20', group=vlangroup),
  809. VLAN(vid=30, name='VLAN 30', group=vlangroup),
  810. )
  811. VLAN.objects.bulk_create(vlans)
  812. # Retrieve all available VLANs
  813. url = reverse('ipam-api:vlangroup-available-vlans', kwargs={'pk': vlangroup.pk})
  814. response = self.client.get(f'{url}?limit=0', **self.header)
  815. self.assertEqual(len(response.data), MAX_VID - MIN_VID + 1)
  816. available_vlans = {vlan['vid'] for vlan in response.data}
  817. for vlan in vlans:
  818. self.assertNotIn(vlan.vid, available_vlans)
  819. # Retrieve a maximum number of available VLANs
  820. url = reverse('ipam-api:vlangroup-available-vlans', kwargs={'pk': vlangroup.pk})
  821. response = self.client.get(f'{url}?limit=10', **self.header)
  822. self.assertEqual(len(response.data), 10)
  823. def test_create_single_available_vlan(self):
  824. """
  825. Test the creation of a single available VLAN.
  826. """
  827. self.add_permissions('ipam.view_vlangroup', 'ipam.view_vlan', 'ipam.add_vlan')
  828. vlangroup = VLANGroup.objects.first()
  829. VLAN.objects.create(vid=1, name='VLAN 1', group=vlangroup)
  830. data = {
  831. "name": "First VLAN",
  832. }
  833. url = reverse('ipam-api:vlangroup-available-vlans', kwargs={'pk': vlangroup.pk})
  834. response = self.client.post(url, data, format='json', **self.header)
  835. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  836. self.assertEqual(response.data['name'], data['name'])
  837. self.assertEqual(response.data['group']['id'], vlangroup.pk)
  838. self.assertEqual(response.data['vid'], 2)
  839. def test_create_multiple_available_vlans(self):
  840. """
  841. Test the creation of multiple available VLANs.
  842. """
  843. self.add_permissions('ipam.view_vlangroup', 'ipam.view_vlan', 'ipam.add_vlan')
  844. vlangroup = VLANGroup.objects.first()
  845. vlans = (
  846. VLAN(vid=1, name='VLAN 1', group=vlangroup),
  847. VLAN(vid=3, name='VLAN 3', group=vlangroup),
  848. VLAN(vid=5, name='VLAN 5', group=vlangroup),
  849. )
  850. VLAN.objects.bulk_create(vlans)
  851. data = (
  852. {"name": "First VLAN"},
  853. {"name": "Second VLAN"},
  854. {"name": "Third VLAN"},
  855. )
  856. url = reverse('ipam-api:vlangroup-available-vlans', kwargs={'pk': vlangroup.pk})
  857. response = self.client.post(url, data, format='json', **self.header)
  858. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  859. self.assertEqual(len(response.data), 3)
  860. self.assertEqual(response.data[0]['name'], data[0]['name'])
  861. self.assertEqual(response.data[0]['group']['id'], vlangroup.pk)
  862. self.assertEqual(response.data[0]['vid'], 2)
  863. self.assertEqual(response.data[1]['name'], data[1]['name'])
  864. self.assertEqual(response.data[1]['group']['id'], vlangroup.pk)
  865. self.assertEqual(response.data[1]['vid'], 4)
  866. self.assertEqual(response.data[2]['name'], data[2]['name'])
  867. self.assertEqual(response.data[2]['group']['id'], vlangroup.pk)
  868. self.assertEqual(response.data[2]['vid'], 6)
  869. class VLANTest(APIViewTestCases.APIViewTestCase):
  870. model = VLAN
  871. brief_fields = ['description', 'display', 'id', 'name', 'url', 'vid']
  872. bulk_update_data = {
  873. 'description': 'New description',
  874. }
  875. @classmethod
  876. def setUpTestData(cls):
  877. vlan_groups = (
  878. VLANGroup(name='VLAN Group 1', slug='vlan-group-1'),
  879. VLANGroup(name='VLAN Group 2', slug='vlan-group-2'),
  880. )
  881. VLANGroup.objects.bulk_create(vlan_groups)
  882. vlans = (
  883. VLAN(name='VLAN 1', vid=1, group=vlan_groups[0]),
  884. VLAN(name='VLAN 2', vid=2, group=vlan_groups[0]),
  885. VLAN(name='VLAN 3', vid=3, group=vlan_groups[0]),
  886. VLAN(name='SVLAN 1', vid=1001, qinq_role=VLANQinQRoleChoices.ROLE_SERVICE),
  887. )
  888. VLAN.objects.bulk_create(vlans)
  889. cls.create_data = [
  890. {
  891. 'vid': 4,
  892. 'name': 'VLAN 4',
  893. 'group': vlan_groups[1].pk,
  894. },
  895. {
  896. 'vid': 5,
  897. 'name': 'VLAN 5',
  898. 'group': vlan_groups[1].pk,
  899. },
  900. {
  901. 'vid': 6,
  902. 'name': 'VLAN 6',
  903. 'group': vlan_groups[1].pk,
  904. },
  905. {
  906. 'vid': 2001,
  907. 'name': 'CVLAN 1',
  908. 'qinq_role': VLANQinQRoleChoices.ROLE_CUSTOMER,
  909. 'qinq_svlan': vlans[3].pk,
  910. },
  911. ]
  912. def test_delete_vlan_with_prefix(self):
  913. """
  914. Attempt and fail to delete a VLAN with a Prefix assigned to it.
  915. """
  916. vlan = VLAN.objects.first()
  917. Prefix.objects.create(prefix=IPNetwork('192.0.2.0/24'), vlan=vlan)
  918. self.add_permissions('ipam.delete_vlan')
  919. url = reverse('ipam-api:vlan-detail', kwargs={'pk': vlan.pk})
  920. with disable_warnings('netbox.api.views.ModelViewSet'):
  921. response = self.client.delete(url, **self.header)
  922. self.assertHttpStatus(response, status.HTTP_409_CONFLICT)
  923. content = json.loads(response.content.decode('utf-8'))
  924. self.assertIn('detail', content)
  925. self.assertTrue(content['detail'].startswith('Unable to delete object.'))
  926. class VLANTranslationPolicyTest(APIViewTestCases.APIViewTestCase):
  927. model = VLANTranslationPolicy
  928. brief_fields = ['description', 'display', 'id', 'name', 'url',]
  929. bulk_update_data = {
  930. 'description': 'New description',
  931. }
  932. @classmethod
  933. def setUpTestData(cls):
  934. vlan_translation_policies = (
  935. VLANTranslationPolicy(
  936. name='Policy 1',
  937. description='foobar1',
  938. ),
  939. VLANTranslationPolicy(
  940. name='Policy 2',
  941. description='foobar2',
  942. ),
  943. VLANTranslationPolicy(
  944. name='Policy 3',
  945. description='foobar3',
  946. ),
  947. )
  948. VLANTranslationPolicy.objects.bulk_create(vlan_translation_policies)
  949. cls.create_data = [
  950. {
  951. 'name': 'Policy 4',
  952. 'description': 'foobar4',
  953. },
  954. {
  955. 'name': 'Policy 5',
  956. 'description': 'foobar5',
  957. },
  958. {
  959. 'name': 'Policy 6',
  960. 'description': 'foobar6',
  961. },
  962. ]
  963. class VLANTranslationRuleTest(APIViewTestCases.APIViewTestCase):
  964. model = VLANTranslationRule
  965. brief_fields = ['description', 'display', 'id', 'local_vid', 'policy', 'remote_vid', 'url']
  966. @classmethod
  967. def setUpTestData(cls):
  968. vlan_translation_policies = (
  969. VLANTranslationPolicy(
  970. name='Policy 1',
  971. description='foobar1',
  972. ),
  973. VLANTranslationPolicy(
  974. name='Policy 2',
  975. description='foobar2',
  976. ),
  977. )
  978. VLANTranslationPolicy.objects.bulk_create(vlan_translation_policies)
  979. vlan_translation_rules = (
  980. VLANTranslationRule(
  981. policy=vlan_translation_policies[0],
  982. local_vid=100,
  983. remote_vid=200,
  984. description='foo',
  985. ),
  986. VLANTranslationRule(
  987. policy=vlan_translation_policies[0],
  988. local_vid=101,
  989. remote_vid=201,
  990. description='bar',
  991. ),
  992. VLANTranslationRule(
  993. policy=vlan_translation_policies[1],
  994. local_vid=102,
  995. remote_vid=202,
  996. description='baz',
  997. ),
  998. )
  999. VLANTranslationRule.objects.bulk_create(vlan_translation_rules)
  1000. cls.create_data = [
  1001. {
  1002. 'policy': vlan_translation_policies[0].pk,
  1003. 'local_vid': 300,
  1004. 'remote_vid': 400,
  1005. },
  1006. {
  1007. 'policy': vlan_translation_policies[0].pk,
  1008. 'local_vid': 301,
  1009. 'remote_vid': 401,
  1010. },
  1011. {
  1012. 'policy': vlan_translation_policies[1].pk,
  1013. 'local_vid': 302,
  1014. 'remote_vid': 402,
  1015. },
  1016. ]
  1017. cls.bulk_update_data = {
  1018. 'policy': vlan_translation_policies[1].pk,
  1019. }
  1020. class ServiceTemplateTest(APIViewTestCases.APIViewTestCase):
  1021. model = ServiceTemplate
  1022. brief_fields = ['description', 'display', 'id', 'name', 'ports', 'protocol', 'url']
  1023. bulk_update_data = {
  1024. 'description': 'New description',
  1025. }
  1026. @classmethod
  1027. def setUpTestData(cls):
  1028. service_templates = (
  1029. ServiceTemplate(name='Service Template 1', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[1, 2]),
  1030. ServiceTemplate(name='Service Template 2', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[3, 4]),
  1031. ServiceTemplate(name='Service Template 3', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[5, 6]),
  1032. )
  1033. ServiceTemplate.objects.bulk_create(service_templates)
  1034. cls.create_data = [
  1035. {
  1036. 'name': 'Service Template 4',
  1037. 'protocol': ServiceProtocolChoices.PROTOCOL_TCP,
  1038. 'ports': [7, 8],
  1039. },
  1040. {
  1041. 'name': 'Service Template 5',
  1042. 'protocol': ServiceProtocolChoices.PROTOCOL_TCP,
  1043. 'ports': [9, 10],
  1044. },
  1045. {
  1046. 'name': 'Service Template 6',
  1047. 'protocol': ServiceProtocolChoices.PROTOCOL_TCP,
  1048. 'ports': [11, 12],
  1049. },
  1050. ]
  1051. class ServiceTest(APIViewTestCases.APIViewTestCase):
  1052. model = Service
  1053. brief_fields = ['description', 'display', 'id', 'name', 'ports', 'protocol', 'url']
  1054. bulk_update_data = {
  1055. 'description': 'New description',
  1056. }
  1057. @classmethod
  1058. def setUpTestData(cls):
  1059. site = Site.objects.create(name='Site 1', slug='site-1')
  1060. manufacturer = Manufacturer.objects.create(name='Manufacturer 1', slug='manufacturer-1')
  1061. devicetype = DeviceType.objects.create(manufacturer=manufacturer, model='Device Type 1')
  1062. role = DeviceRole.objects.create(name='Device Role 1', slug='device-role-1')
  1063. devices = (
  1064. Device(name='Device 1', site=site, device_type=devicetype, role=role),
  1065. Device(name='Device 2', site=site, device_type=devicetype, role=role),
  1066. )
  1067. Device.objects.bulk_create(devices)
  1068. services = (
  1069. Service(device=devices[0], name='Service 1', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[1]),
  1070. Service(device=devices[0], name='Service 2', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[2]),
  1071. Service(device=devices[0], name='Service 3', protocol=ServiceProtocolChoices.PROTOCOL_TCP, ports=[3]),
  1072. )
  1073. Service.objects.bulk_create(services)
  1074. cls.create_data = [
  1075. {
  1076. 'device': devices[1].pk,
  1077. 'name': 'Service 4',
  1078. 'protocol': ServiceProtocolChoices.PROTOCOL_TCP,
  1079. 'ports': [4],
  1080. },
  1081. {
  1082. 'device': devices[1].pk,
  1083. 'name': 'Service 5',
  1084. 'protocol': ServiceProtocolChoices.PROTOCOL_TCP,
  1085. 'ports': [5],
  1086. },
  1087. {
  1088. 'device': devices[1].pk,
  1089. 'name': 'Service 6',
  1090. 'protocol': ServiceProtocolChoices.PROTOCOL_TCP,
  1091. 'ports': [6],
  1092. },
  1093. ]