test_api.py 36 KB

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