test_api.py 40 KB

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