filters.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. import django_filters
  2. import netaddr
  3. from django.core.exceptions import ValidationError
  4. from django.db.models import Q
  5. from netaddr.core import AddrFormatError
  6. from dcim.models import Device, Interface, Region, Site
  7. from extras.filters import CustomFieldFilterSet, CreatedUpdatedFilterSet
  8. from tenancy.filters import TenancyFilterSet
  9. from utilities.filters import (
  10. BaseFilterSet, MultiValueCharFilter, MultiValueNumberFilter, NameSlugSearchFilterSet, TagFilter,
  11. TreeNodeMultipleChoiceFilter,
  12. )
  13. from virtualization.models import VirtualMachine, VMInterface
  14. from .choices import *
  15. from .models import Aggregate, IPAddress, Prefix, RIR, Role, Service, VLAN, VLANGroup, VRF
  16. __all__ = (
  17. 'AggregateFilterSet',
  18. 'IPAddressFilterSet',
  19. 'PrefixFilterSet',
  20. 'RIRFilterSet',
  21. 'RoleFilterSet',
  22. 'ServiceFilterSet',
  23. 'VLANFilterSet',
  24. 'VLANGroupFilterSet',
  25. 'VRFFilterSet',
  26. )
  27. class VRFFilterSet(BaseFilterSet, TenancyFilterSet, CustomFieldFilterSet, CreatedUpdatedFilterSet):
  28. q = django_filters.CharFilter(
  29. method='search',
  30. label='Search',
  31. )
  32. tag = TagFilter()
  33. def search(self, queryset, name, value):
  34. if not value.strip():
  35. return queryset
  36. return queryset.filter(
  37. Q(name__icontains=value) |
  38. Q(rd__icontains=value) |
  39. Q(description__icontains=value)
  40. )
  41. class Meta:
  42. model = VRF
  43. fields = ['id', 'name', 'rd', 'enforce_unique']
  44. class RIRFilterSet(BaseFilterSet, NameSlugSearchFilterSet):
  45. class Meta:
  46. model = RIR
  47. fields = ['id', 'name', 'slug', 'is_private', 'description']
  48. class AggregateFilterSet(BaseFilterSet, CustomFieldFilterSet, CreatedUpdatedFilterSet):
  49. q = django_filters.CharFilter(
  50. method='search',
  51. label='Search',
  52. )
  53. family = django_filters.NumberFilter(
  54. field_name='prefix',
  55. lookup_expr='family'
  56. )
  57. prefix = django_filters.CharFilter(
  58. method='filter_prefix',
  59. label='Prefix',
  60. )
  61. rir_id = django_filters.ModelMultipleChoiceFilter(
  62. queryset=RIR.objects.all(),
  63. label='RIR (ID)',
  64. )
  65. rir = django_filters.ModelMultipleChoiceFilter(
  66. field_name='rir__slug',
  67. queryset=RIR.objects.all(),
  68. to_field_name='slug',
  69. label='RIR (slug)',
  70. )
  71. tag = TagFilter()
  72. class Meta:
  73. model = Aggregate
  74. fields = ['id', 'date_added']
  75. def search(self, queryset, name, value):
  76. if not value.strip():
  77. return queryset
  78. qs_filter = Q(description__icontains=value)
  79. try:
  80. prefix = str(netaddr.IPNetwork(value.strip()).cidr)
  81. qs_filter |= Q(prefix__net_contains_or_equals=prefix)
  82. except (AddrFormatError, ValueError):
  83. pass
  84. return queryset.filter(qs_filter)
  85. def filter_prefix(self, queryset, name, value):
  86. if not value.strip():
  87. return queryset
  88. try:
  89. query = str(netaddr.IPNetwork(value).cidr)
  90. return queryset.filter(prefix=query)
  91. except (AddrFormatError, ValueError):
  92. return queryset.none()
  93. class RoleFilterSet(BaseFilterSet, NameSlugSearchFilterSet):
  94. q = django_filters.CharFilter(
  95. method='search',
  96. label='Search',
  97. )
  98. class Meta:
  99. model = Role
  100. fields = ['id', 'name', 'slug']
  101. class PrefixFilterSet(BaseFilterSet, TenancyFilterSet, CustomFieldFilterSet, CreatedUpdatedFilterSet):
  102. q = django_filters.CharFilter(
  103. method='search',
  104. label='Search',
  105. )
  106. family = django_filters.NumberFilter(
  107. field_name='prefix',
  108. lookup_expr='family'
  109. )
  110. prefix = django_filters.CharFilter(
  111. method='filter_prefix',
  112. label='Prefix',
  113. )
  114. within = django_filters.CharFilter(
  115. method='search_within',
  116. label='Within prefix',
  117. )
  118. within_include = django_filters.CharFilter(
  119. method='search_within_include',
  120. label='Within and including prefix',
  121. )
  122. contains = django_filters.CharFilter(
  123. method='search_contains',
  124. label='Prefixes which contain this prefix or IP',
  125. )
  126. mask_length = django_filters.NumberFilter(
  127. field_name='prefix',
  128. lookup_expr='net_mask_length'
  129. )
  130. mask_length__gte = django_filters.NumberFilter(
  131. field_name='prefix',
  132. lookup_expr='net_mask_length__gte'
  133. )
  134. mask_length__lte = django_filters.NumberFilter(
  135. field_name='prefix',
  136. lookup_expr='net_mask_length__lte'
  137. )
  138. vrf_id = django_filters.ModelMultipleChoiceFilter(
  139. queryset=VRF.objects.all(),
  140. label='VRF',
  141. )
  142. vrf = django_filters.ModelMultipleChoiceFilter(
  143. field_name='vrf__rd',
  144. queryset=VRF.objects.all(),
  145. to_field_name='rd',
  146. label='VRF (RD)',
  147. )
  148. region_id = TreeNodeMultipleChoiceFilter(
  149. queryset=Region.objects.all(),
  150. field_name='site__region',
  151. lookup_expr='in',
  152. label='Region (ID)',
  153. )
  154. region = TreeNodeMultipleChoiceFilter(
  155. queryset=Region.objects.all(),
  156. field_name='site__region',
  157. lookup_expr='in',
  158. to_field_name='slug',
  159. label='Region (slug)',
  160. )
  161. site_id = django_filters.ModelMultipleChoiceFilter(
  162. queryset=Site.objects.all(),
  163. label='Site (ID)',
  164. )
  165. site = django_filters.ModelMultipleChoiceFilter(
  166. field_name='site__slug',
  167. queryset=Site.objects.all(),
  168. to_field_name='slug',
  169. label='Site (slug)',
  170. )
  171. vlan_id = django_filters.ModelMultipleChoiceFilter(
  172. queryset=VLAN.objects.all(),
  173. label='VLAN (ID)',
  174. )
  175. vlan_vid = django_filters.NumberFilter(
  176. field_name='vlan__vid',
  177. label='VLAN number (1-4095)',
  178. )
  179. role_id = django_filters.ModelMultipleChoiceFilter(
  180. queryset=Role.objects.all(),
  181. label='Role (ID)',
  182. )
  183. role = django_filters.ModelMultipleChoiceFilter(
  184. field_name='role__slug',
  185. queryset=Role.objects.all(),
  186. to_field_name='slug',
  187. label='Role (slug)',
  188. )
  189. status = django_filters.MultipleChoiceFilter(
  190. choices=PrefixStatusChoices,
  191. null_value=None
  192. )
  193. tag = TagFilter()
  194. class Meta:
  195. model = Prefix
  196. fields = ['id', 'is_pool']
  197. def search(self, queryset, name, value):
  198. if not value.strip():
  199. return queryset
  200. qs_filter = Q(description__icontains=value)
  201. try:
  202. prefix = str(netaddr.IPNetwork(value.strip()).cidr)
  203. qs_filter |= Q(prefix__net_contains_or_equals=prefix)
  204. except (AddrFormatError, ValueError):
  205. pass
  206. return queryset.filter(qs_filter)
  207. def filter_prefix(self, queryset, name, value):
  208. if not value.strip():
  209. return queryset
  210. try:
  211. query = str(netaddr.IPNetwork(value).cidr)
  212. return queryset.filter(prefix=query)
  213. except (AddrFormatError, ValueError):
  214. return queryset.none()
  215. def search_within(self, queryset, name, value):
  216. value = value.strip()
  217. if not value:
  218. return queryset
  219. try:
  220. query = str(netaddr.IPNetwork(value).cidr)
  221. return queryset.filter(prefix__net_contained=query)
  222. except (AddrFormatError, ValueError):
  223. return queryset.none()
  224. def search_within_include(self, queryset, name, value):
  225. value = value.strip()
  226. if not value:
  227. return queryset
  228. try:
  229. query = str(netaddr.IPNetwork(value).cidr)
  230. return queryset.filter(prefix__net_contained_or_equal=query)
  231. except (AddrFormatError, ValueError):
  232. return queryset.none()
  233. def search_contains(self, queryset, name, value):
  234. value = value.strip()
  235. if not value:
  236. return queryset
  237. try:
  238. # Searching by prefix
  239. if '/' in value:
  240. return queryset.filter(prefix__net_contains_or_equals=str(netaddr.IPNetwork(value).cidr))
  241. # Searching by IP address
  242. else:
  243. return queryset.filter(prefix__net_contains=str(netaddr.IPAddress(value)))
  244. except (AddrFormatError, ValueError):
  245. return queryset.none()
  246. class IPAddressFilterSet(BaseFilterSet, TenancyFilterSet, CustomFieldFilterSet, CreatedUpdatedFilterSet):
  247. q = django_filters.CharFilter(
  248. method='search',
  249. label='Search',
  250. )
  251. family = django_filters.NumberFilter(
  252. field_name='address',
  253. lookup_expr='family'
  254. )
  255. parent = django_filters.CharFilter(
  256. method='search_by_parent',
  257. label='Parent prefix',
  258. )
  259. address = MultiValueCharFilter(
  260. method='filter_address',
  261. label='Address',
  262. )
  263. mask_length = django_filters.NumberFilter(
  264. method='filter_mask_length',
  265. label='Mask length',
  266. )
  267. vrf_id = django_filters.ModelMultipleChoiceFilter(
  268. queryset=VRF.objects.all(),
  269. label='VRF',
  270. )
  271. vrf = django_filters.ModelMultipleChoiceFilter(
  272. field_name='vrf__rd',
  273. queryset=VRF.objects.all(),
  274. to_field_name='rd',
  275. label='VRF (RD)',
  276. )
  277. device = MultiValueCharFilter(
  278. method='filter_device',
  279. field_name='name',
  280. label='Device (name)',
  281. )
  282. device_id = MultiValueNumberFilter(
  283. method='filter_device',
  284. field_name='pk',
  285. label='Device (ID)',
  286. )
  287. virtual_machine = MultiValueCharFilter(
  288. method='filter_virtual_machine',
  289. field_name='name',
  290. label='Virtual machine (name)',
  291. )
  292. virtual_machine_id = MultiValueNumberFilter(
  293. method='filter_virtual_machine',
  294. field_name='pk',
  295. label='Virtual machine (ID)',
  296. )
  297. interface = django_filters.ModelMultipleChoiceFilter(
  298. field_name='interface__name',
  299. queryset=Interface.objects.all(),
  300. to_field_name='name',
  301. label='Interface (name)',
  302. )
  303. interface_id = django_filters.ModelMultipleChoiceFilter(
  304. field_name='interface',
  305. queryset=Interface.objects.all(),
  306. label='Interface (ID)',
  307. )
  308. vminterface = django_filters.ModelMultipleChoiceFilter(
  309. field_name='vminterface__name',
  310. queryset=VMInterface.objects.all(),
  311. to_field_name='name',
  312. label='VM interface (name)',
  313. )
  314. vminterface_id = django_filters.ModelMultipleChoiceFilter(
  315. field_name='vminterface',
  316. queryset=VMInterface.objects.all(),
  317. label='VM interface (ID)',
  318. )
  319. assigned_to_interface = django_filters.BooleanFilter(
  320. method='_assigned_to_interface',
  321. label='Is assigned to an interface',
  322. )
  323. status = django_filters.MultipleChoiceFilter(
  324. choices=IPAddressStatusChoices,
  325. null_value=None
  326. )
  327. role = django_filters.MultipleChoiceFilter(
  328. choices=IPAddressRoleChoices
  329. )
  330. tag = TagFilter()
  331. class Meta:
  332. model = IPAddress
  333. fields = ['id', 'dns_name']
  334. def search(self, queryset, name, value):
  335. if not value.strip():
  336. return queryset
  337. qs_filter = (
  338. Q(dns_name__icontains=value) |
  339. Q(description__icontains=value) |
  340. Q(address__istartswith=value)
  341. )
  342. return queryset.filter(qs_filter)
  343. def search_by_parent(self, queryset, name, value):
  344. value = value.strip()
  345. if not value:
  346. return queryset
  347. try:
  348. query = str(netaddr.IPNetwork(value.strip()).cidr)
  349. return queryset.filter(address__net_host_contained=query)
  350. except (AddrFormatError, ValueError):
  351. return queryset.none()
  352. def filter_address(self, queryset, name, value):
  353. try:
  354. return queryset.filter(address__net_in=value)
  355. except ValidationError:
  356. return queryset.none()
  357. def filter_mask_length(self, queryset, name, value):
  358. if not value:
  359. return queryset
  360. return queryset.filter(address__net_mask_length=value)
  361. def filter_device(self, queryset, name, value):
  362. devices = Device.objects.filter(**{'{}__in'.format(name): value})
  363. if not devices.exists():
  364. return queryset.none()
  365. interface_ids = []
  366. for device in devices:
  367. interface_ids.extend(device.vc_interfaces.values_list('id', flat=True))
  368. return queryset.filter(
  369. interface__in=interface_ids
  370. )
  371. def filter_virtual_machine(self, queryset, name, value):
  372. virtual_machines = VirtualMachine.objects.filter(**{'{}__in'.format(name): value})
  373. if not virtual_machines.exists():
  374. return queryset.none()
  375. interface_ids = []
  376. for vm in virtual_machines:
  377. interface_ids.extend(vm.interfaces.values_list('id', flat=True))
  378. return queryset.filter(
  379. vminterface__in=interface_ids
  380. )
  381. def _assigned_to_interface(self, queryset, name, value):
  382. return queryset.exclude(assigned_object_id__isnull=value)
  383. class VLANGroupFilterSet(BaseFilterSet, NameSlugSearchFilterSet):
  384. region_id = TreeNodeMultipleChoiceFilter(
  385. queryset=Region.objects.all(),
  386. field_name='site__region',
  387. lookup_expr='in',
  388. label='Region (ID)',
  389. )
  390. region = TreeNodeMultipleChoiceFilter(
  391. queryset=Region.objects.all(),
  392. field_name='site__region',
  393. lookup_expr='in',
  394. to_field_name='slug',
  395. label='Region (slug)',
  396. )
  397. site_id = django_filters.ModelMultipleChoiceFilter(
  398. queryset=Site.objects.all(),
  399. label='Site (ID)',
  400. )
  401. site = django_filters.ModelMultipleChoiceFilter(
  402. field_name='site__slug',
  403. queryset=Site.objects.all(),
  404. to_field_name='slug',
  405. label='Site (slug)',
  406. )
  407. class Meta:
  408. model = VLANGroup
  409. fields = ['id', 'name', 'slug', 'description']
  410. class VLANFilterSet(BaseFilterSet, TenancyFilterSet, CustomFieldFilterSet, CreatedUpdatedFilterSet):
  411. q = django_filters.CharFilter(
  412. method='search',
  413. label='Search',
  414. )
  415. region_id = TreeNodeMultipleChoiceFilter(
  416. queryset=Region.objects.all(),
  417. field_name='site__region',
  418. lookup_expr='in',
  419. label='Region (ID)',
  420. )
  421. region = TreeNodeMultipleChoiceFilter(
  422. queryset=Region.objects.all(),
  423. field_name='site__region',
  424. lookup_expr='in',
  425. to_field_name='slug',
  426. label='Region (slug)',
  427. )
  428. site_id = django_filters.ModelMultipleChoiceFilter(
  429. queryset=Site.objects.all(),
  430. label='Site (ID)',
  431. )
  432. site = django_filters.ModelMultipleChoiceFilter(
  433. field_name='site__slug',
  434. queryset=Site.objects.all(),
  435. to_field_name='slug',
  436. label='Site (slug)',
  437. )
  438. group_id = django_filters.ModelMultipleChoiceFilter(
  439. queryset=VLANGroup.objects.all(),
  440. label='Group (ID)',
  441. )
  442. group = django_filters.ModelMultipleChoiceFilter(
  443. field_name='group__slug',
  444. queryset=VLANGroup.objects.all(),
  445. to_field_name='slug',
  446. label='Group',
  447. )
  448. role_id = django_filters.ModelMultipleChoiceFilter(
  449. queryset=Role.objects.all(),
  450. label='Role (ID)',
  451. )
  452. role = django_filters.ModelMultipleChoiceFilter(
  453. field_name='role__slug',
  454. queryset=Role.objects.all(),
  455. to_field_name='slug',
  456. label='Role (slug)',
  457. )
  458. status = django_filters.MultipleChoiceFilter(
  459. choices=VLANStatusChoices,
  460. null_value=None
  461. )
  462. tag = TagFilter()
  463. class Meta:
  464. model = VLAN
  465. fields = ['id', 'vid', 'name']
  466. def search(self, queryset, name, value):
  467. if not value.strip():
  468. return queryset
  469. qs_filter = Q(name__icontains=value) | Q(description__icontains=value)
  470. try:
  471. qs_filter |= Q(vid=int(value.strip()))
  472. except ValueError:
  473. pass
  474. return queryset.filter(qs_filter)
  475. class ServiceFilterSet(BaseFilterSet, CreatedUpdatedFilterSet):
  476. q = django_filters.CharFilter(
  477. method='search',
  478. label='Search',
  479. )
  480. device_id = django_filters.ModelMultipleChoiceFilter(
  481. queryset=Device.objects.all(),
  482. label='Device (ID)',
  483. )
  484. device = django_filters.ModelMultipleChoiceFilter(
  485. field_name='device__name',
  486. queryset=Device.objects.all(),
  487. to_field_name='name',
  488. label='Device (name)',
  489. )
  490. virtual_machine_id = django_filters.ModelMultipleChoiceFilter(
  491. queryset=VirtualMachine.objects.all(),
  492. label='Virtual machine (ID)',
  493. )
  494. virtual_machine = django_filters.ModelMultipleChoiceFilter(
  495. field_name='virtual_machine__name',
  496. queryset=VirtualMachine.objects.all(),
  497. to_field_name='name',
  498. label='Virtual machine (name)',
  499. )
  500. tag = TagFilter()
  501. class Meta:
  502. model = Service
  503. fields = ['id', 'name', 'protocol', 'port']
  504. def search(self, queryset, name, value):
  505. if not value.strip():
  506. return queryset
  507. qs_filter = Q(name__icontains=value) | Q(description__icontains=value)
  508. return queryset.filter(qs_filter)