filters.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  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
  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. method='filter_mask_length',
  128. label='Mask length',
  129. )
  130. vrf_id = django_filters.ModelMultipleChoiceFilter(
  131. queryset=VRF.objects.all(),
  132. label='VRF',
  133. )
  134. vrf = django_filters.ModelMultipleChoiceFilter(
  135. field_name='vrf__rd',
  136. queryset=VRF.objects.all(),
  137. to_field_name='rd',
  138. label='VRF (RD)',
  139. )
  140. region_id = TreeNodeMultipleChoiceFilter(
  141. queryset=Region.objects.all(),
  142. field_name='site__region',
  143. lookup_expr='in',
  144. label='Region (ID)',
  145. )
  146. region = TreeNodeMultipleChoiceFilter(
  147. queryset=Region.objects.all(),
  148. field_name='site__region',
  149. lookup_expr='in',
  150. to_field_name='slug',
  151. label='Region (slug)',
  152. )
  153. site_id = django_filters.ModelMultipleChoiceFilter(
  154. queryset=Site.objects.all(),
  155. label='Site (ID)',
  156. )
  157. site = django_filters.ModelMultipleChoiceFilter(
  158. field_name='site__slug',
  159. queryset=Site.objects.all(),
  160. to_field_name='slug',
  161. label='Site (slug)',
  162. )
  163. vlan_id = django_filters.ModelMultipleChoiceFilter(
  164. queryset=VLAN.objects.all(),
  165. label='VLAN (ID)',
  166. )
  167. vlan_vid = django_filters.NumberFilter(
  168. field_name='vlan__vid',
  169. label='VLAN number (1-4095)',
  170. )
  171. role_id = django_filters.ModelMultipleChoiceFilter(
  172. queryset=Role.objects.all(),
  173. label='Role (ID)',
  174. )
  175. role = django_filters.ModelMultipleChoiceFilter(
  176. field_name='role__slug',
  177. queryset=Role.objects.all(),
  178. to_field_name='slug',
  179. label='Role (slug)',
  180. )
  181. status = django_filters.MultipleChoiceFilter(
  182. choices=PrefixStatusChoices,
  183. null_value=None
  184. )
  185. tag = TagFilter()
  186. class Meta:
  187. model = Prefix
  188. fields = ['id', 'is_pool']
  189. def search(self, queryset, name, value):
  190. if not value.strip():
  191. return queryset
  192. qs_filter = Q(description__icontains=value)
  193. try:
  194. prefix = str(netaddr.IPNetwork(value.strip()).cidr)
  195. qs_filter |= Q(prefix__net_contains_or_equals=prefix)
  196. except (AddrFormatError, ValueError):
  197. pass
  198. return queryset.filter(qs_filter)
  199. def filter_prefix(self, queryset, name, value):
  200. if not value.strip():
  201. return queryset
  202. try:
  203. query = str(netaddr.IPNetwork(value).cidr)
  204. return queryset.filter(prefix=query)
  205. except (AddrFormatError, ValueError):
  206. return queryset.none()
  207. def search_within(self, queryset, name, value):
  208. value = value.strip()
  209. if not value:
  210. return queryset
  211. try:
  212. query = str(netaddr.IPNetwork(value).cidr)
  213. return queryset.filter(prefix__net_contained=query)
  214. except (AddrFormatError, ValueError):
  215. return queryset.none()
  216. def search_within_include(self, queryset, name, value):
  217. value = value.strip()
  218. if not value:
  219. return queryset
  220. try:
  221. query = str(netaddr.IPNetwork(value).cidr)
  222. return queryset.filter(prefix__net_contained_or_equal=query)
  223. except (AddrFormatError, ValueError):
  224. return queryset.none()
  225. def search_contains(self, queryset, name, value):
  226. value = value.strip()
  227. if not value:
  228. return queryset
  229. try:
  230. # Searching by prefix
  231. if '/' in value:
  232. return queryset.filter(prefix__net_contains_or_equals=str(netaddr.IPNetwork(value).cidr))
  233. # Searching by IP address
  234. else:
  235. return queryset.filter(prefix__net_contains=str(netaddr.IPAddress(value)))
  236. except (AddrFormatError, ValueError):
  237. return queryset.none()
  238. def filter_mask_length(self, queryset, name, value):
  239. if not value:
  240. return queryset
  241. return queryset.filter(prefix__net_mask_length=value)
  242. class IPAddressFilterSet(BaseFilterSet, TenancyFilterSet, CustomFieldFilterSet, CreatedUpdatedFilterSet):
  243. q = django_filters.CharFilter(
  244. method='search',
  245. label='Search',
  246. )
  247. family = django_filters.NumberFilter(
  248. field_name='address',
  249. lookup_expr='family'
  250. )
  251. parent = django_filters.CharFilter(
  252. method='search_by_parent',
  253. label='Parent prefix',
  254. )
  255. address = MultiValueCharFilter(
  256. method='filter_address',
  257. label='Address',
  258. )
  259. mask_length = django_filters.NumberFilter(
  260. method='filter_mask_length',
  261. label='Mask length',
  262. )
  263. vrf_id = django_filters.ModelMultipleChoiceFilter(
  264. queryset=VRF.objects.all(),
  265. label='VRF',
  266. )
  267. vrf = django_filters.ModelMultipleChoiceFilter(
  268. field_name='vrf__rd',
  269. queryset=VRF.objects.all(),
  270. to_field_name='rd',
  271. label='VRF (RD)',
  272. )
  273. device = MultiValueCharFilter(
  274. method='filter_device',
  275. field_name='name',
  276. label='Device (name)',
  277. )
  278. device_id = MultiValueNumberFilter(
  279. method='filter_device',
  280. field_name='pk',
  281. label='Device (ID)',
  282. )
  283. virtual_machine_id = django_filters.ModelMultipleChoiceFilter(
  284. field_name='interface__virtual_machine',
  285. queryset=VirtualMachine.objects.all(),
  286. label='Virtual machine (ID)',
  287. )
  288. virtual_machine = django_filters.ModelMultipleChoiceFilter(
  289. field_name='interface__virtual_machine__name',
  290. queryset=VirtualMachine.objects.all(),
  291. to_field_name='name',
  292. label='Virtual machine (name)',
  293. )
  294. interface = django_filters.ModelMultipleChoiceFilter(
  295. field_name='interface__name',
  296. queryset=Interface.objects.all(),
  297. to_field_name='name',
  298. label='Interface (ID)',
  299. )
  300. interface_id = django_filters.ModelMultipleChoiceFilter(
  301. queryset=Interface.objects.all(),
  302. label='Interface (ID)',
  303. )
  304. assigned_to_interface = django_filters.BooleanFilter(
  305. method='_assigned_to_interface',
  306. label='Is assigned to an interface',
  307. )
  308. status = django_filters.MultipleChoiceFilter(
  309. choices=IPAddressStatusChoices,
  310. null_value=None
  311. )
  312. role = django_filters.MultipleChoiceFilter(
  313. choices=IPAddressRoleChoices
  314. )
  315. tag = TagFilter()
  316. class Meta:
  317. model = IPAddress
  318. fields = ['id', 'dns_name']
  319. def search(self, queryset, name, value):
  320. if not value.strip():
  321. return queryset
  322. qs_filter = (
  323. Q(dns_name__icontains=value) |
  324. Q(description__icontains=value) |
  325. Q(address__istartswith=value)
  326. )
  327. return queryset.filter(qs_filter)
  328. def search_by_parent(self, queryset, name, value):
  329. value = value.strip()
  330. if not value:
  331. return queryset
  332. try:
  333. query = str(netaddr.IPNetwork(value.strip()).cidr)
  334. return queryset.filter(address__net_host_contained=query)
  335. except (AddrFormatError, ValueError):
  336. return queryset.none()
  337. def filter_address(self, queryset, name, value):
  338. try:
  339. return queryset.filter(address__net_in=value)
  340. except ValidationError:
  341. return queryset.none()
  342. def filter_mask_length(self, queryset, name, value):
  343. if not value:
  344. return queryset
  345. return queryset.filter(address__net_mask_length=value)
  346. def filter_device(self, queryset, name, value):
  347. try:
  348. devices = Device.objects.prefetch_related('device_type').filter(**{'{}__in'.format(name): value})
  349. vc_interface_ids = []
  350. for device in devices:
  351. vc_interface_ids.extend([i['id'] for i in device.vc_interfaces.values('id')])
  352. return queryset.filter(interface_id__in=vc_interface_ids)
  353. except Device.DoesNotExist:
  354. return queryset.none()
  355. def _assigned_to_interface(self, queryset, name, value):
  356. return queryset.exclude(interface__isnull=value)
  357. class VLANGroupFilterSet(BaseFilterSet, NameSlugSearchFilterSet):
  358. region_id = TreeNodeMultipleChoiceFilter(
  359. queryset=Region.objects.all(),
  360. field_name='site__region',
  361. lookup_expr='in',
  362. label='Region (ID)',
  363. )
  364. region = TreeNodeMultipleChoiceFilter(
  365. queryset=Region.objects.all(),
  366. field_name='site__region',
  367. lookup_expr='in',
  368. to_field_name='slug',
  369. label='Region (slug)',
  370. )
  371. site_id = django_filters.ModelMultipleChoiceFilter(
  372. queryset=Site.objects.all(),
  373. label='Site (ID)',
  374. )
  375. site = django_filters.ModelMultipleChoiceFilter(
  376. field_name='site__slug',
  377. queryset=Site.objects.all(),
  378. to_field_name='slug',
  379. label='Site (slug)',
  380. )
  381. class Meta:
  382. model = VLANGroup
  383. fields = ['id', 'name', 'slug', 'description']
  384. class VLANFilterSet(BaseFilterSet, TenancyFilterSet, CustomFieldFilterSet, CreatedUpdatedFilterSet):
  385. q = django_filters.CharFilter(
  386. method='search',
  387. label='Search',
  388. )
  389. region_id = TreeNodeMultipleChoiceFilter(
  390. queryset=Region.objects.all(),
  391. field_name='site__region',
  392. lookup_expr='in',
  393. label='Region (ID)',
  394. )
  395. region = TreeNodeMultipleChoiceFilter(
  396. queryset=Region.objects.all(),
  397. field_name='site__region',
  398. lookup_expr='in',
  399. to_field_name='slug',
  400. label='Region (slug)',
  401. )
  402. site_id = django_filters.ModelMultipleChoiceFilter(
  403. queryset=Site.objects.all(),
  404. label='Site (ID)',
  405. )
  406. site = django_filters.ModelMultipleChoiceFilter(
  407. field_name='site__slug',
  408. queryset=Site.objects.all(),
  409. to_field_name='slug',
  410. label='Site (slug)',
  411. )
  412. group_id = django_filters.ModelMultipleChoiceFilter(
  413. queryset=VLANGroup.objects.all(),
  414. label='Group (ID)',
  415. )
  416. group = django_filters.ModelMultipleChoiceFilter(
  417. field_name='group__slug',
  418. queryset=VLANGroup.objects.all(),
  419. to_field_name='slug',
  420. label='Group',
  421. )
  422. role_id = django_filters.ModelMultipleChoiceFilter(
  423. queryset=Role.objects.all(),
  424. label='Role (ID)',
  425. )
  426. role = django_filters.ModelMultipleChoiceFilter(
  427. field_name='role__slug',
  428. queryset=Role.objects.all(),
  429. to_field_name='slug',
  430. label='Role (slug)',
  431. )
  432. status = django_filters.MultipleChoiceFilter(
  433. choices=VLANStatusChoices,
  434. null_value=None
  435. )
  436. tag = TagFilter()
  437. class Meta:
  438. model = VLAN
  439. fields = ['id', 'vid', 'name']
  440. def search(self, queryset, name, value):
  441. if not value.strip():
  442. return queryset
  443. qs_filter = Q(name__icontains=value) | Q(description__icontains=value)
  444. try:
  445. qs_filter |= Q(vid=int(value.strip()))
  446. except ValueError:
  447. pass
  448. return queryset.filter(qs_filter)
  449. class ServiceFilterSet(BaseFilterSet, CreatedUpdatedFilterSet):
  450. q = django_filters.CharFilter(
  451. method='search',
  452. label='Search',
  453. )
  454. device_id = django_filters.ModelMultipleChoiceFilter(
  455. queryset=Device.objects.all(),
  456. label='Device (ID)',
  457. )
  458. device = django_filters.ModelMultipleChoiceFilter(
  459. field_name='device__name',
  460. queryset=Device.objects.all(),
  461. to_field_name='name',
  462. label='Device (name)',
  463. )
  464. virtual_machine_id = django_filters.ModelMultipleChoiceFilter(
  465. queryset=VirtualMachine.objects.all(),
  466. label='Virtual machine (ID)',
  467. )
  468. virtual_machine = django_filters.ModelMultipleChoiceFilter(
  469. field_name='virtual_machine__name',
  470. queryset=VirtualMachine.objects.all(),
  471. to_field_name='name',
  472. label='Virtual machine (name)',
  473. )
  474. tag = TagFilter()
  475. class Meta:
  476. model = Service
  477. fields = ['id', 'name', 'protocol', 'port']
  478. def search(self, queryset, name, value):
  479. if not value.strip():
  480. return queryset
  481. qs_filter = Q(name__icontains=value) | Q(description__icontains=value)
  482. return queryset.filter(qs_filter)