filters.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725
  1. from __future__ import unicode_literals
  2. import django_filters
  3. from django.contrib.auth.models import User
  4. from django.db.models import Q
  5. from netaddr import EUI
  6. from netaddr.core import AddrFormatError
  7. from extras.filters import CustomFieldFilterSet
  8. from tenancy.models import Tenant
  9. from utilities.filters import NullableCharFieldFilter, NumericInFilter
  10. from virtualization.models import Cluster
  11. from .constants import (
  12. IFACE_FF_LAG, NONCONNECTABLE_IFACE_TYPES, STATUS_CHOICES, VIRTUAL_IFACE_TYPES, WIRELESS_IFACE_TYPES,
  13. )
  14. from .models import (
  15. ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate, Device, DeviceBay,
  16. DeviceBayTemplate, DeviceRole, DeviceType, Interface, InterfaceConnection, InterfaceTemplate, Manufacturer,
  17. InventoryItem, Platform, PowerOutlet, PowerOutletTemplate, PowerPort, PowerPortTemplate, Rack, RackGroup,
  18. RackReservation, RackRole, Region, Site, VCMembership,
  19. )
  20. class RegionFilter(django_filters.FilterSet):
  21. parent_id = django_filters.ModelMultipleChoiceFilter(
  22. queryset=Region.objects.all(),
  23. label='Parent region (ID)',
  24. )
  25. parent = django_filters.ModelMultipleChoiceFilter(
  26. name='parent__slug',
  27. queryset=Region.objects.all(),
  28. to_field_name='slug',
  29. label='Parent region (slug)',
  30. )
  31. class Meta:
  32. model = Region
  33. fields = ['name', 'slug']
  34. class SiteFilter(CustomFieldFilterSet, django_filters.FilterSet):
  35. id__in = NumericInFilter(name='id', lookup_expr='in')
  36. q = django_filters.CharFilter(
  37. method='search',
  38. label='Search',
  39. )
  40. region_id = django_filters.ModelMultipleChoiceFilter(
  41. queryset=Region.objects.all(),
  42. label='Region (ID)',
  43. )
  44. region = django_filters.ModelMultipleChoiceFilter(
  45. name='region__slug',
  46. queryset=Region.objects.all(),
  47. to_field_name='slug',
  48. label='Region (slug)',
  49. )
  50. tenant_id = django_filters.ModelMultipleChoiceFilter(
  51. queryset=Tenant.objects.all(),
  52. label='Tenant (ID)',
  53. )
  54. tenant = django_filters.ModelMultipleChoiceFilter(
  55. name='tenant__slug',
  56. queryset=Tenant.objects.all(),
  57. to_field_name='slug',
  58. label='Tenant (slug)',
  59. )
  60. class Meta:
  61. model = Site
  62. fields = ['q', 'name', 'slug', 'facility', 'asn', 'contact_name', 'contact_phone', 'contact_email']
  63. def search(self, queryset, name, value):
  64. if not value.strip():
  65. return queryset
  66. qs_filter = (
  67. Q(name__icontains=value) |
  68. Q(facility__icontains=value) |
  69. Q(physical_address__icontains=value) |
  70. Q(shipping_address__icontains=value) |
  71. Q(contact_name__icontains=value) |
  72. Q(contact_phone__icontains=value) |
  73. Q(contact_email__icontains=value) |
  74. Q(comments__icontains=value)
  75. )
  76. try:
  77. qs_filter |= Q(asn=int(value.strip()))
  78. except ValueError:
  79. pass
  80. return queryset.filter(qs_filter)
  81. class RackGroupFilter(django_filters.FilterSet):
  82. site_id = django_filters.ModelMultipleChoiceFilter(
  83. queryset=Site.objects.all(),
  84. label='Site (ID)',
  85. )
  86. site = django_filters.ModelMultipleChoiceFilter(
  87. name='site__slug',
  88. queryset=Site.objects.all(),
  89. to_field_name='slug',
  90. label='Site (slug)',
  91. )
  92. class Meta:
  93. model = RackGroup
  94. fields = ['site_id', 'name', 'slug']
  95. class RackRoleFilter(django_filters.FilterSet):
  96. class Meta:
  97. model = RackRole
  98. fields = ['name', 'slug', 'color']
  99. class RackFilter(CustomFieldFilterSet, django_filters.FilterSet):
  100. id__in = NumericInFilter(name='id', lookup_expr='in')
  101. q = django_filters.CharFilter(
  102. method='search',
  103. label='Search',
  104. )
  105. facility_id = NullableCharFieldFilter()
  106. site_id = django_filters.ModelMultipleChoiceFilter(
  107. queryset=Site.objects.all(),
  108. label='Site (ID)',
  109. )
  110. site = django_filters.ModelMultipleChoiceFilter(
  111. name='site__slug',
  112. queryset=Site.objects.all(),
  113. to_field_name='slug',
  114. label='Site (slug)',
  115. )
  116. group_id = django_filters.ModelMultipleChoiceFilter(
  117. queryset=RackGroup.objects.all(),
  118. label='Group (ID)',
  119. )
  120. group = django_filters.ModelMultipleChoiceFilter(
  121. name='group__slug',
  122. queryset=RackGroup.objects.all(),
  123. to_field_name='slug',
  124. label='Group',
  125. )
  126. tenant_id = django_filters.ModelMultipleChoiceFilter(
  127. queryset=Tenant.objects.all(),
  128. label='Tenant (ID)',
  129. )
  130. tenant = django_filters.ModelMultipleChoiceFilter(
  131. name='tenant__slug',
  132. queryset=Tenant.objects.all(),
  133. to_field_name='slug',
  134. label='Tenant (slug)',
  135. )
  136. role_id = django_filters.ModelMultipleChoiceFilter(
  137. queryset=RackRole.objects.all(),
  138. label='Role (ID)',
  139. )
  140. role = django_filters.ModelMultipleChoiceFilter(
  141. name='role__slug',
  142. queryset=RackRole.objects.all(),
  143. to_field_name='slug',
  144. label='Role (slug)',
  145. )
  146. class Meta:
  147. model = Rack
  148. fields = ['serial', 'type', 'width', 'u_height', 'desc_units']
  149. def search(self, queryset, name, value):
  150. if not value.strip():
  151. return queryset
  152. return queryset.filter(
  153. Q(name__icontains=value) |
  154. Q(facility_id__icontains=value) |
  155. Q(serial__icontains=value.strip()) |
  156. Q(comments__icontains=value)
  157. )
  158. class RackReservationFilter(django_filters.FilterSet):
  159. id__in = NumericInFilter(name='id', lookup_expr='in')
  160. q = django_filters.CharFilter(
  161. method='search',
  162. label='Search',
  163. )
  164. rack_id = django_filters.ModelMultipleChoiceFilter(
  165. queryset=Rack.objects.all(),
  166. label='Rack (ID)',
  167. )
  168. site_id = django_filters.ModelMultipleChoiceFilter(
  169. name='rack__site',
  170. queryset=Site.objects.all(),
  171. label='Site (ID)',
  172. )
  173. site = django_filters.ModelMultipleChoiceFilter(
  174. name='rack__site__slug',
  175. queryset=Site.objects.all(),
  176. to_field_name='slug',
  177. label='Site (slug)',
  178. )
  179. group_id = django_filters.ModelMultipleChoiceFilter(
  180. name='rack__group',
  181. queryset=RackGroup.objects.all(),
  182. label='Group (ID)',
  183. )
  184. group = django_filters.ModelMultipleChoiceFilter(
  185. name='rack__group__slug',
  186. queryset=RackGroup.objects.all(),
  187. to_field_name='slug',
  188. label='Group',
  189. )
  190. tenant_id = django_filters.ModelMultipleChoiceFilter(
  191. queryset=Tenant.objects.all(),
  192. label='Tenant (ID)',
  193. )
  194. tenant = django_filters.ModelMultipleChoiceFilter(
  195. name='tenant__slug',
  196. queryset=Tenant.objects.all(),
  197. to_field_name='slug',
  198. label='Tenant (slug)',
  199. )
  200. user_id = django_filters.ModelMultipleChoiceFilter(
  201. queryset=User.objects.all(),
  202. label='User (ID)',
  203. )
  204. user = django_filters.ModelMultipleChoiceFilter(
  205. name='user',
  206. queryset=User.objects.all(),
  207. to_field_name='username',
  208. label='User (name)',
  209. )
  210. class Meta:
  211. model = RackReservation
  212. fields = ['created']
  213. def search(self, queryset, name, value):
  214. if not value.strip():
  215. return queryset
  216. return queryset.filter(
  217. Q(rack__name__icontains=value) |
  218. Q(rack__facility_id__icontains=value) |
  219. Q(user__username__icontains=value) |
  220. Q(description__icontains=value)
  221. )
  222. class ManufacturerFilter(django_filters.FilterSet):
  223. class Meta:
  224. model = Manufacturer
  225. fields = ['name', 'slug']
  226. class DeviceTypeFilter(CustomFieldFilterSet, django_filters.FilterSet):
  227. id__in = NumericInFilter(name='id', lookup_expr='in')
  228. q = django_filters.CharFilter(
  229. method='search',
  230. label='Search',
  231. )
  232. manufacturer_id = django_filters.ModelMultipleChoiceFilter(
  233. queryset=Manufacturer.objects.all(),
  234. label='Manufacturer (ID)',
  235. )
  236. manufacturer = django_filters.ModelMultipleChoiceFilter(
  237. name='manufacturer__slug',
  238. queryset=Manufacturer.objects.all(),
  239. to_field_name='slug',
  240. label='Manufacturer (slug)',
  241. )
  242. class Meta:
  243. model = DeviceType
  244. fields = [
  245. 'model', 'slug', 'part_number', 'u_height', 'is_full_depth', 'is_console_server', 'is_pdu',
  246. 'is_network_device', 'subdevice_role',
  247. ]
  248. def search(self, queryset, name, value):
  249. if not value.strip():
  250. return queryset
  251. return queryset.filter(
  252. Q(manufacturer__name__icontains=value) |
  253. Q(model__icontains=value) |
  254. Q(part_number__icontains=value) |
  255. Q(comments__icontains=value)
  256. )
  257. class DeviceTypeComponentFilterSet(django_filters.FilterSet):
  258. devicetype_id = django_filters.ModelMultipleChoiceFilter(
  259. queryset=DeviceType.objects.all(),
  260. name='device_type_id',
  261. label='Device type (ID)',
  262. )
  263. class ConsolePortTemplateFilter(DeviceTypeComponentFilterSet):
  264. class Meta:
  265. model = ConsolePortTemplate
  266. fields = ['name']
  267. class ConsoleServerPortTemplateFilter(DeviceTypeComponentFilterSet):
  268. class Meta:
  269. model = ConsoleServerPortTemplate
  270. fields = ['name']
  271. class PowerPortTemplateFilter(DeviceTypeComponentFilterSet):
  272. class Meta:
  273. model = PowerPortTemplate
  274. fields = ['name']
  275. class PowerOutletTemplateFilter(DeviceTypeComponentFilterSet):
  276. class Meta:
  277. model = PowerOutletTemplate
  278. fields = ['name']
  279. class InterfaceTemplateFilter(DeviceTypeComponentFilterSet):
  280. class Meta:
  281. model = InterfaceTemplate
  282. fields = ['name', 'form_factor', 'mgmt_only']
  283. class DeviceBayTemplateFilter(DeviceTypeComponentFilterSet):
  284. class Meta:
  285. model = DeviceBayTemplate
  286. fields = ['name']
  287. class DeviceRoleFilter(django_filters.FilterSet):
  288. class Meta:
  289. model = DeviceRole
  290. fields = ['name', 'slug', 'color']
  291. class PlatformFilter(django_filters.FilterSet):
  292. class Meta:
  293. model = Platform
  294. fields = ['name', 'slug']
  295. class DeviceFilter(CustomFieldFilterSet, django_filters.FilterSet):
  296. id__in = NumericInFilter(name='id', lookup_expr='in')
  297. q = django_filters.CharFilter(
  298. method='search',
  299. label='Search',
  300. )
  301. manufacturer_id = django_filters.ModelMultipleChoiceFilter(
  302. name='device_type__manufacturer',
  303. queryset=Manufacturer.objects.all(),
  304. label='Manufacturer (ID)',
  305. )
  306. manufacturer = django_filters.ModelMultipleChoiceFilter(
  307. name='device_type__manufacturer__slug',
  308. queryset=Manufacturer.objects.all(),
  309. to_field_name='slug',
  310. label='Manufacturer (slug)',
  311. )
  312. device_type_id = django_filters.ModelMultipleChoiceFilter(
  313. queryset=DeviceType.objects.all(),
  314. label='Device type (ID)',
  315. )
  316. role_id = django_filters.ModelMultipleChoiceFilter(
  317. name='device_role_id',
  318. queryset=DeviceRole.objects.all(),
  319. label='Role (ID)',
  320. )
  321. role = django_filters.ModelMultipleChoiceFilter(
  322. name='device_role__slug',
  323. queryset=DeviceRole.objects.all(),
  324. to_field_name='slug',
  325. label='Role (slug)',
  326. )
  327. tenant_id = django_filters.ModelMultipleChoiceFilter(
  328. queryset=Tenant.objects.all(),
  329. label='Tenant (ID)',
  330. )
  331. tenant = django_filters.ModelMultipleChoiceFilter(
  332. name='tenant__slug',
  333. queryset=Tenant.objects.all(),
  334. to_field_name='slug',
  335. label='Tenant (slug)',
  336. )
  337. platform_id = django_filters.ModelMultipleChoiceFilter(
  338. queryset=Platform.objects.all(),
  339. label='Platform (ID)',
  340. )
  341. platform = django_filters.ModelMultipleChoiceFilter(
  342. name='platform__slug',
  343. queryset=Platform.objects.all(),
  344. to_field_name='slug',
  345. label='Platform (slug)',
  346. )
  347. name = NullableCharFieldFilter()
  348. asset_tag = NullableCharFieldFilter()
  349. site_id = django_filters.ModelMultipleChoiceFilter(
  350. queryset=Site.objects.all(),
  351. label='Site (ID)',
  352. )
  353. site = django_filters.ModelMultipleChoiceFilter(
  354. name='site__slug',
  355. queryset=Site.objects.all(),
  356. to_field_name='slug',
  357. label='Site name (slug)',
  358. )
  359. rack_group_id = django_filters.ModelMultipleChoiceFilter(
  360. name='rack__group',
  361. queryset=RackGroup.objects.all(),
  362. label='Rack group (ID)',
  363. )
  364. rack_id = django_filters.ModelMultipleChoiceFilter(
  365. name='rack',
  366. queryset=Rack.objects.all(),
  367. label='Rack (ID)',
  368. )
  369. cluster_id = django_filters.ModelMultipleChoiceFilter(
  370. queryset=Cluster.objects.all(),
  371. label='VM cluster (ID)',
  372. )
  373. model = django_filters.ModelMultipleChoiceFilter(
  374. name='device_type__slug',
  375. queryset=DeviceType.objects.all(),
  376. to_field_name='slug',
  377. label='Device model (slug)',
  378. )
  379. status = django_filters.MultipleChoiceFilter(
  380. choices=STATUS_CHOICES,
  381. null_value=None
  382. )
  383. is_full_depth = django_filters.BooleanFilter(
  384. name='device_type__is_full_depth',
  385. label='Is full depth',
  386. )
  387. is_console_server = django_filters.BooleanFilter(
  388. name='device_type__is_console_server',
  389. label='Is a console server',
  390. )
  391. is_pdu = django_filters.BooleanFilter(
  392. name='device_type__is_pdu',
  393. label='Is a PDU',
  394. )
  395. is_network_device = django_filters.BooleanFilter(
  396. name='device_type__is_network_device',
  397. label='Is a network device',
  398. )
  399. mac_address = django_filters.CharFilter(
  400. method='_mac_address',
  401. label='MAC address',
  402. )
  403. has_primary_ip = django_filters.BooleanFilter(
  404. method='_has_primary_ip',
  405. label='Has a primary IP',
  406. )
  407. class Meta:
  408. model = Device
  409. fields = ['serial']
  410. def search(self, queryset, name, value):
  411. if not value.strip():
  412. return queryset
  413. return queryset.filter(
  414. Q(name__icontains=value) |
  415. Q(serial__icontains=value.strip()) |
  416. Q(inventory_items__serial__icontains=value.strip()) |
  417. Q(asset_tag=value.strip()) |
  418. Q(comments__icontains=value)
  419. ).distinct()
  420. def _mac_address(self, queryset, name, value):
  421. value = value.strip()
  422. if not value:
  423. return queryset
  424. try:
  425. mac = EUI(value.strip())
  426. return queryset.filter(interfaces__mac_address=mac).distinct()
  427. except AddrFormatError:
  428. return queryset.none()
  429. def _has_primary_ip(self, queryset, name, value):
  430. if value:
  431. return queryset.filter(
  432. Q(primary_ip4__isnull=False) |
  433. Q(primary_ip6__isnull=False)
  434. )
  435. else:
  436. return queryset.exclude(
  437. Q(primary_ip4__isnull=False) |
  438. Q(primary_ip6__isnull=False)
  439. )
  440. class DeviceComponentFilterSet(django_filters.FilterSet):
  441. device_id = django_filters.ModelChoiceFilter(
  442. queryset=Device.objects.all(),
  443. label='Device (ID)',
  444. )
  445. device = django_filters.ModelChoiceFilter(
  446. queryset=Device.objects.all(),
  447. to_field_name='name',
  448. label='Device (name)',
  449. )
  450. class ConsolePortFilter(DeviceComponentFilterSet):
  451. class Meta:
  452. model = ConsolePort
  453. fields = ['name']
  454. class ConsoleServerPortFilter(DeviceComponentFilterSet):
  455. class Meta:
  456. model = ConsoleServerPort
  457. fields = ['name']
  458. class PowerPortFilter(DeviceComponentFilterSet):
  459. class Meta:
  460. model = PowerPort
  461. fields = ['name']
  462. class PowerOutletFilter(DeviceComponentFilterSet):
  463. class Meta:
  464. model = PowerOutlet
  465. fields = ['name']
  466. class InterfaceFilter(django_filters.FilterSet):
  467. """
  468. Not using DeviceComponentFilterSet for Interfaces because we need to glean the ordering logic from the parent
  469. Device's DeviceType.
  470. """
  471. device = django_filters.CharFilter(
  472. method='filter_device',
  473. name='name',
  474. label='Device',
  475. )
  476. device_id = django_filters.NumberFilter(
  477. method='filter_device',
  478. name='pk',
  479. label='Device (ID)',
  480. )
  481. type = django_filters.CharFilter(
  482. method='filter_type',
  483. label='Interface type',
  484. )
  485. lag_id = django_filters.ModelMultipleChoiceFilter(
  486. name='lag',
  487. queryset=Interface.objects.all(),
  488. label='LAG interface (ID)',
  489. )
  490. mac_address = django_filters.CharFilter(
  491. method='_mac_address',
  492. label='MAC address',
  493. )
  494. class Meta:
  495. model = Interface
  496. fields = ['name', 'form_factor', 'enabled', 'mtu', 'mgmt_only']
  497. def filter_device(self, queryset, name, value):
  498. try:
  499. device = Device.objects.select_related('device_type').get(**{name: value})
  500. ordering = device.device_type.interface_ordering
  501. return queryset.filter(device=device).order_naturally(ordering)
  502. except Device.DoesNotExist:
  503. return queryset.none()
  504. def filter_type(self, queryset, name, value):
  505. value = value.strip().lower()
  506. return {
  507. 'physical': queryset.exclude(form_factor__in=NONCONNECTABLE_IFACE_TYPES),
  508. 'virtual': queryset.filter(form_factor__in=VIRTUAL_IFACE_TYPES),
  509. 'wireless': queryset.filter(form_factor__in=WIRELESS_IFACE_TYPES),
  510. 'lag': queryset.filter(form_factor=IFACE_FF_LAG),
  511. }.get(value, queryset.none())
  512. def _mac_address(self, queryset, name, value):
  513. value = value.strip()
  514. if not value:
  515. return queryset
  516. try:
  517. mac = EUI(value.strip())
  518. return queryset.filter(mac_address=mac)
  519. except AddrFormatError:
  520. return queryset.none()
  521. class DeviceBayFilter(DeviceComponentFilterSet):
  522. class Meta:
  523. model = DeviceBay
  524. fields = ['name']
  525. class InventoryItemFilter(DeviceComponentFilterSet):
  526. parent_id = django_filters.ModelMultipleChoiceFilter(
  527. queryset=InventoryItem.objects.all(),
  528. label='Parent inventory item (ID)',
  529. )
  530. manufacturer_id = django_filters.ModelMultipleChoiceFilter(
  531. queryset=Manufacturer.objects.all(),
  532. label='Manufacturer (ID)',
  533. )
  534. manufacturer = django_filters.ModelMultipleChoiceFilter(
  535. name='manufacturer__slug',
  536. queryset=Manufacturer.objects.all(),
  537. to_field_name='slug',
  538. label='Manufacturer (slug)',
  539. )
  540. asset_tag = NullableCharFieldFilter()
  541. class Meta:
  542. model = InventoryItem
  543. fields = ['name', 'part_id', 'serial', 'discovered']
  544. class VCMembershipFilter(django_filters.FilterSet):
  545. class Meta:
  546. model = VCMembership
  547. fields = ['virtual_chassis']
  548. class ConsoleConnectionFilter(django_filters.FilterSet):
  549. site = django_filters.CharFilter(
  550. method='filter_site',
  551. label='Site (slug)',
  552. )
  553. device = django_filters.CharFilter(
  554. method='filter_device',
  555. label='Device',
  556. )
  557. class Meta:
  558. model = ConsolePort
  559. fields = ['name', 'connection_status']
  560. def filter_site(self, queryset, name, value):
  561. if not value.strip():
  562. return queryset
  563. return queryset.filter(cs_port__device__site__slug=value)
  564. def filter_device(self, queryset, name, value):
  565. if not value.strip():
  566. return queryset
  567. return queryset.filter(
  568. Q(device__name__icontains=value) |
  569. Q(cs_port__device__name__icontains=value)
  570. )
  571. class PowerConnectionFilter(django_filters.FilterSet):
  572. site = django_filters.CharFilter(
  573. method='filter_site',
  574. label='Site (slug)',
  575. )
  576. device = django_filters.CharFilter(
  577. method='filter_device',
  578. label='Device',
  579. )
  580. class Meta:
  581. model = PowerPort
  582. fields = ['name', 'connection_status']
  583. def filter_site(self, queryset, name, value):
  584. if not value.strip():
  585. return queryset
  586. return queryset.filter(power_outlet__device__site__slug=value)
  587. def filter_device(self, queryset, name, value):
  588. if not value.strip():
  589. return queryset
  590. return queryset.filter(
  591. Q(device__name__icontains=value) |
  592. Q(power_outlet__device__name__icontains=value)
  593. )
  594. class InterfaceConnectionFilter(django_filters.FilterSet):
  595. site = django_filters.CharFilter(
  596. method='filter_site',
  597. label='Site (slug)',
  598. )
  599. device = django_filters.CharFilter(
  600. method='filter_device',
  601. label='Device',
  602. )
  603. class Meta:
  604. model = InterfaceConnection
  605. fields = ['connection_status']
  606. def filter_site(self, queryset, name, value):
  607. if not value.strip():
  608. return queryset
  609. return queryset.filter(
  610. Q(interface_a__device__site__slug=value) |
  611. Q(interface_b__device__site__slug=value)
  612. )
  613. def filter_device(self, queryset, name, value):
  614. if not value.strip():
  615. return queryset
  616. return queryset.filter(
  617. Q(interface_a__device__name__icontains=value) |
  618. Q(interface_b__device__name__icontains=value)
  619. )