admin.py 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681
  1. from django.contrib import admin
  2. from django.db import models
  3. from django.db.models import Q
  4. from django.forms import ModelForm, BaseInlineFormSet
  5. from django.utils import timezone
  6. from django.core.urlresolvers import reverse
  7. from django.utils.html import format_html
  8. from django.core.mail import mail_managers
  9. from django.conf.urls import url
  10. from django.template.response import TemplateResponse
  11. from django.core.serializers import serialize
  12. from django.http import HttpResponse
  13. from django.db.models.functions import Cast
  14. from django.contrib.postgres.aggregates import StringAgg
  15. from django.db import connection
  16. from django.core.cache import cache
  17. from django.contrib.humanize.templatetags.humanize import naturaltime
  18. from djgeojson.views import GeoJSONLayerView
  19. from urllib.parse import urlencode
  20. from functools import partial, update_wrapper
  21. from datetime import timedelta
  22. from djadhere.utils import get_active_filter
  23. from adhesions.models import Adhesion
  24. from banking.models import PaymentUpdate
  25. from .models import Service, ServiceType, IPPrefix, IPResource, Route, Tunnel, \
  26. ServiceAllocation, Antenna, AntennaAllocation, Allocation, \
  27. Switch, Port
  28. from .utils import notify_allocation
  29. from .forms import AntennaForm
  30. ### Filters
  31. class ResourceInUseFilter(admin.SimpleListFilter):
  32. title = 'disponibilité'
  33. parameter_name = 'available'
  34. def lookups(self, request, model_admin):
  35. return (
  36. (1, 'Disponible'),
  37. (0, 'Non disponible'),
  38. )
  39. def queryset(self, request, queryset):
  40. available_filter = Q(reserved=False, in_use=False)
  41. if self.value() == '0': # non disponible
  42. return queryset.exclude(available_filter)
  43. if self.value() == '1': # disponible
  44. return queryset.filter(available_filter)
  45. class ResourcePingFilter(admin.SimpleListFilter):
  46. title = 'ping'
  47. parameter_name = 'ping'
  48. def lookups(self, request, model_admin):
  49. return (
  50. ('up', 'Up'),
  51. ('down', 'Down'),
  52. ('unknown', 'Inconnu'),
  53. )
  54. def queryset(self, request, queryset):
  55. known_filter = models.Q(last_time_up__isnull=False, last_check__isnull=False)
  56. if self.value() == 'unknown':
  57. return queryset.exclude(known_filter)
  58. else:
  59. queryset = queryset.filter(known_filter)
  60. if self.value() == 'up':
  61. return queryset.filter(last_check__lte=models.F('last_time_up'))
  62. if self.value() == 'down':
  63. return queryset.filter(last_check__gt=models.F('last_time_up'))
  64. class ActiveServiceFilter(admin.SimpleListFilter):
  65. title = 'actif'
  66. parameter_name = 'active'
  67. def lookups(self, request, model_admin):
  68. return (
  69. (1, 'Actif'),
  70. (0, 'Inactif'),
  71. )
  72. def queryset(self, request, queryset):
  73. if self.value() == '0': # inactif
  74. return queryset.exclude(get_active_filter('allocation')) # has_active_allocations=False ne fonctionne pas
  75. if self.value() == '1': # actif
  76. return queryset.filter(has_active_allocations=True)
  77. class RouteFilter(admin.SimpleListFilter):
  78. title = 'route'
  79. parameter_name = 'route'
  80. def lookups(self, request, model_admin):
  81. return ServiceAllocation.objects.filter(active=True).values_list('route__pk', 'route__name').distinct()
  82. def queryset(self, request, queryset):
  83. try:
  84. route = int(self.value())
  85. except (TypeError, ValueError):
  86. pass
  87. else:
  88. allocations = ServiceAllocation.objects.filter(active=True, route__pk=route).values_list('pk', flat=True)
  89. queryset = queryset.filter(service_allocation__in=allocations)
  90. return queryset
  91. class AntennaPrefixFilter(admin.SimpleListFilter):
  92. title = 'préfix'
  93. parameter_name = 'prefix'
  94. def lookups(self, request, model_admin):
  95. resources = AntennaAllocation.objects.filter(active=True).values_list('resource__pk', flat=True)
  96. prefixes = IPPrefix.objects.filter(ipresource__in=resources).values_list('pk', 'prefix').distinct()
  97. return prefixes
  98. def queryset(self, request, queryset):
  99. try:
  100. prefix = int(self.value())
  101. except (TypeError, ValueError):
  102. pass
  103. else:
  104. allocations = AntennaAllocation.objects.filter(active=True, resource__prefixes__pk=prefix).values_list('pk', flat=True)
  105. queryset = queryset.filter(allocation__in=allocations)
  106. return queryset
  107. class AntennaPositionFilter(admin.SimpleListFilter):
  108. title = 'géolocalisation'
  109. parameter_name = 'position'
  110. def lookups(self, request, model_admin):
  111. return (
  112. ('1', 'Connue'),
  113. ('0', 'Inconnue'),
  114. )
  115. def queryset(self, request, queryset):
  116. query = Q(position__isnull=True)
  117. if self.value() == '0':
  118. return queryset.filter(query)
  119. if self.value() == '1':
  120. return queryset.exclude(query)
  121. return queryset
  122. class ActiveTunnelFilter(admin.SimpleListFilter):
  123. title = 'status'
  124. parameter_name = 'active'
  125. def lookups(self, request, model_admin):
  126. return (
  127. ('1', 'Actif'),
  128. ('0', 'Désactivé'),
  129. )
  130. def queryset(self, request, queryset):
  131. query = Q(ended__isnull=True)
  132. if self.value() == '0':
  133. return queryset.exclude(query)
  134. if self.value() == '1':
  135. return queryset.filter(query)
  136. return queryset
  137. ### Inlines
  138. class AllocationInlineFormSet(BaseInlineFormSet):
  139. def save_new(self, form, commit=True):
  140. obj = super().save_new(form, commit)
  141. if type(obj) == ServiceAllocation:
  142. notify_allocation(self.request, obj)
  143. return obj
  144. def save_existing(self, form, instance, commit=True):
  145. old = type(instance).objects.get(pk=instance.pk)
  146. if type(instance) == ServiceAllocation:
  147. notify_allocation(self.request, instance, old)
  148. return super().save_existing(form, instance, commit)
  149. class AllocationInline(admin.TabularInline):
  150. formset = AllocationInlineFormSet
  151. extra = 0
  152. show_change_link = True
  153. def get_formset(self, request, obj=None, **kwargs):
  154. formset = super().get_formset(request, obj, **kwargs)
  155. formset.request = request
  156. return formset
  157. def has_delete_permission(self, request, obj=None):
  158. return False
  159. class ActiveAllocationMixin:
  160. verbose_name_plural = 'Allocations actives'
  161. def get_max_num(self, request, obj=None, **kwargs):
  162. existing = obj.allocations.count() if obj else 0
  163. # pour simplifier la validation, on ajoute qu’une allocation à la fois
  164. # il faudrait surcharger la méthode clean du formset pour supprimer cette limite
  165. return existing + 1
  166. def get_queryset(self, request):
  167. return super().get_queryset(request).filter(active=True)
  168. class InactiveAllocationMixin:
  169. verbose_name_plural = 'Anciennes allocations'
  170. max_num = 0
  171. def get_queryset(self, request):
  172. return super().get_queryset(request).filter(active=False)
  173. class ServiceAllocationMixin:
  174. model = ServiceAllocation
  175. fields = ('id', 'service', 'resource', 'route', 'start', 'end')
  176. raw_id_fields = ('service', 'resource',)
  177. class AntennaAllocationMixin:
  178. model = AntennaAllocation
  179. fields = ('id', 'antenna', 'resource', 'start', 'end')
  180. raw_id_fields = ('antenna', 'resource',)
  181. class ActiveServiceAllocationInline(ServiceAllocationMixin, ActiveAllocationMixin, AllocationInline):
  182. pass
  183. class InactiveServiceAllocationInline(ServiceAllocationMixin, InactiveAllocationMixin, AllocationInline):
  184. pass
  185. class ActiveAntennaAllocationInline(AntennaAllocationMixin, ActiveAllocationMixin, AllocationInline):
  186. pass
  187. class InactiveAntennaAllocationInline(AntennaAllocationMixin, InactiveAllocationMixin, AllocationInline):
  188. pass
  189. class PortInline(admin.TabularInline):
  190. model = Port
  191. def has_add_permission(self, request):
  192. return False
  193. def has_delete_permission(self, request, obj=None):
  194. return False
  195. class SwitchPortInline(PortInline):
  196. fields = ('port', 'reserved', 'service', 'notes',)
  197. readonly_fields = ('port',)
  198. raw_id_fields = ('service',)
  199. def get_queryset(self, request):
  200. qs = super().get_queryset(request)
  201. qs = qs.prefetch_related('switch', 'service', 'service__service_type')
  202. return qs
  203. class ServicePortInline(PortInline):
  204. fields = ('switch', 'port', 'notes',)
  205. readonly_fields = ('switch', 'port',)
  206. max_num = 0
  207. ### Actions
  208. def ends_resource(resource, request, queryset):
  209. now = timezone.now()
  210. queryset.exclude(start__lte=now, end__isnull=False).update(end=now)
  211. # TODO: send mail
  212. ends_resource.short_description = 'Terminer les allocations sélectionnées'
  213. ### ModelAdmin
  214. class ServiceAdmin(admin.ModelAdmin):
  215. list_display = ('id', 'get_adhesion_link', 'get_adherent_link', 'service_type', 'label', 'is_active',)
  216. list_select_related = ('adhesion', 'adhesion__user', 'adhesion__user__profile', 'adhesion__corporation', 'service_type')
  217. list_filter = (
  218. ActiveServiceFilter,
  219. ('service_type', admin.RelatedOnlyFieldListFilter),
  220. )
  221. search_fields = ('=id', 'service_type__name', 'label', 'notes',)
  222. fields = ('adhesion', 'service_type', 'label', 'notes', 'get_contribution_link', 'is_active',)
  223. readonly_fields = ('get_contribution_link', 'is_active',)
  224. raw_id_fields = ('adhesion',)
  225. get_adhesion_link = lambda self, service: service.adhesion.get_adhesion_link()
  226. get_adhesion_link.short_description = Adhesion.get_adhesion_link.short_description
  227. get_adherent_link = lambda self, service: service.adhesion.get_adherent_link()
  228. get_adherent_link.short_description = Adhesion.get_adherent_link.short_description
  229. def get_contribution_link(self, obj):
  230. return format_html(u'<a href="{}">{}</a>', obj.contribution.get_absolute_url(), obj.contribution)
  231. get_contribution_link.short_description = 'Contribution financière'
  232. def get_inline_instances(self, request, obj=None):
  233. inlines = []
  234. if obj and obj.ports.exists():
  235. inlines += [ServicePortInline]
  236. inlines += [ActiveServiceAllocationInline, InactiveServiceAllocationInline]
  237. return [inline(self.model, self.admin_site) for inline in inlines]
  238. def get_actions(self, request):
  239. actions = super().get_actions(request)
  240. if 'delete_selected' in actions:
  241. del actions['delete_selected']
  242. return actions
  243. def has_delete_permission(self, request, obj=None):
  244. if not obj:
  245. return False
  246. one_year_ago = timezone.now() - timedelta(days=365)
  247. contribution = obj.contribution.updates.filter(validated=True).first()
  248. # s’il y avait un paiement actif il y a moins d’un an
  249. if not contribution or contribution.payment_method != PaymentUpdate.STOP or contribution.start > one_year_ago:
  250. return False
  251. # s’il y avait une allocation active il y a moins d’un an
  252. if any(map(lambda a: a.end is None or a.end > one_year_ago, obj.allocations.all())):
  253. return False
  254. return True
  255. class IPPrefixAdmin(admin.ModelAdmin):
  256. readonly_fields = ('prefix',)
  257. def has_delete_permission(self, request, obj=None):
  258. # Interdiction de supprimer le préfix s’il est assigné à une route
  259. return obj and obj.tunnel_set.exists()
  260. # pour embêcher de by-passer le check has_delete_permission, on désactive l’action delete
  261. def get_actions(self, request):
  262. actions = super().get_actions(request)
  263. if 'delete_selected' in actions:
  264. del actions['delete_selected']
  265. return actions
  266. class IPResourceAdmin(admin.ModelAdmin):
  267. list_display = ('__str__', 'available_display', 'last_use', 'ping',)
  268. list_filter = (
  269. 'category',
  270. ResourceInUseFilter,
  271. ResourcePingFilter,
  272. 'reserved',
  273. ('prefixes', admin.RelatedOnlyFieldListFilter),
  274. RouteFilter,
  275. )
  276. search_fields = ('=ip', 'notes',)
  277. def get_fields(self, request, obj=None):
  278. return self.get_readonly_fields(request, obj) + ['notes']
  279. def get_readonly_fields(self, request, obj=None):
  280. fields = ['ip']
  281. if obj and obj.reserved:
  282. fields += ['reserved']
  283. if obj and not obj.in_use:
  284. fields += ['last_use']
  285. if obj and obj.last_time_up and obj.last_check:
  286. fields += ['last_time_up', 'last_check']
  287. if obj and obj.checkmk_label:
  288. fields += ['checkmk']
  289. return fields
  290. def get_inline_instances(self, request, obj=None):
  291. if obj:
  292. if obj.category == 0:
  293. inlines = (ActiveServiceAllocationInline, InactiveServiceAllocationInline,)
  294. elif obj.category == 1:
  295. inlines = (ActiveAntennaAllocationInline, InactiveAntennaAllocationInline,)
  296. else:
  297. inlines = ()
  298. return [inline(self.model, self.admin_site) for inline in inlines]
  299. def get_queryset(self, request):
  300. qs = super().get_queryset(request)
  301. qs = qs.annotate(last_use=models.Case(
  302. models.When(category=0, then=models.Max('service_allocation__end')),
  303. models.When(category=1, then=models.Max('antenna_allocation__end')),
  304. default=None,
  305. ))
  306. qs = qs.annotate(downtime=Cast(
  307. models.Case(
  308. models.When(last_check__isnull=False, last_time_up__isnull=False, last_check__lte=models.F('last_time_up'), then=timedelta(days=0)),
  309. models.When(last_check__isnull=False, last_time_up__isnull=False, then=models.F('last_check') - models.F('last_time_up')),
  310. default=None,
  311. ),
  312. models.DurationField(),
  313. ))
  314. return qs
  315. def available_display(self, obj):
  316. return not obj.reserved and not obj.in_use
  317. available_display.short_description = 'Disponible'
  318. available_display.boolean = True
  319. def last_use(self, obj):
  320. if obj.last_use:
  321. return naturaltime(obj.last_use)
  322. else:
  323. return '-'
  324. last_use.short_description = 'Dernière utilisation'
  325. last_use.admin_order_field = 'last_use'
  326. def ping(self, obj):
  327. if obj.downtime:
  328. label = 'down depuis >' + str(obj.downtime)
  329. elif obj.downtime == timedelta(days=0):
  330. label = 'UP'
  331. else:
  332. return None
  333. if obj.checkmk_url:
  334. return format_html('<a href="{}">{}</a>', obj.checkmk_url, label)
  335. else:
  336. return label
  337. ping.short_description = 'ping'
  338. ping.admin_order_field = 'downtime'
  339. def checkmk(self, obj):
  340. return format_html('<a href="{}">{}</a>', obj.checkmk_url, 'voir')
  341. checkmk.short_description = 'CheckMK'
  342. def get_actions(self, request):
  343. actions = super().get_actions(request)
  344. if 'delete_selected' in actions:
  345. del actions['delete_selected']
  346. return actions
  347. def has_add_permission(self, request, obj=None):
  348. return False
  349. def has_delete_permission(self, request, obj=None):
  350. return False
  351. class RouteAdmin(admin.ModelAdmin):
  352. list_display = ('name',)
  353. def get_fields(self, request, obj=None):
  354. if obj:
  355. return ('name', 'get_emails', 'get_sms', 'get_routed_ip', 'get_adh',)
  356. else:
  357. return ('name',)
  358. def get_readonly_fields(self, request, obj=None):
  359. if obj:
  360. return ('get_emails', 'get_sms', 'get_routed_ip', 'get_adh',)
  361. else:
  362. return ()
  363. def get_contacts(self, route):
  364. cache_emails_key = 'route-%d-emails' % route.pk
  365. cache_sms_key = 'route-%d-sms' % route.pk
  366. cache_adh_key = 'route-%d-adh' % route.pk
  367. cache_results = cache.get_many([cache_emails_key, cache_sms_key, cache_adh_key])
  368. if len(cache_results) == 3:
  369. return (cache_results[cache_emails_key], cache_results[cache_sms_key], cache_results[cache_adh_key])
  370. allocations = route.allocations \
  371. .order_by('service__adhesion__pk') \
  372. .distinct('service__adhesion__pk') \
  373. .select_related(
  374. 'service__adhesion__user',
  375. 'service__adhesion__corporation',
  376. )
  377. emails, sms, adh = [], [], []
  378. for allocation in allocations:
  379. adhesion = allocation.service.adhesion
  380. adherent = adhesion.adherent
  381. if adhesion.pk not in adh:
  382. adh.append(adhesion.pk)
  383. if adherent.email:
  384. email = '%s <%s>' % (str(adherent), adherent.email)
  385. if email not in emails:
  386. emails.append(email)
  387. # S’il s’agit d’une raison sociale, on contact aussi les gestionnaires
  388. if adhesion.corporation:
  389. if adhesion.corporation.phone_number:
  390. sms.append(adhesion.corporation.phone_number)
  391. for member in adhesion.corporation.members.all():
  392. if member.email:
  393. email = '%s <%s>' % (str(member), member.email)
  394. if email not in emails:
  395. emails.append(email)
  396. else: # user
  397. if adhesion.user.profile.phone_number:
  398. if adhesion.user.profile.phone_number not in sms:
  399. sms.append(adhesion.user.profile.phone_number)
  400. sms = list(filter(lambda x: x[:2] == '06' or x[:2] == '07' or x[:3] == '+336' or x[:3] == '+337', sms))
  401. cache.set_many({cache_emails_key: emails, cache_sms_key: sms, cache_adh_key: adh}, timeout=3600)
  402. return (emails, sms, adh)
  403. def get_emails(self, route):
  404. emails, _, _ = self.get_contacts(route)
  405. return '\n'.join(emails)
  406. get_emails.short_description = 'Contacts'
  407. def get_sms(self, route):
  408. _, sms, _ = self.get_contacts(route)
  409. return '\n'.join(sms)
  410. get_sms.short_description = 'SMS'
  411. def get_adh(self, route):
  412. _, _, adh = self.get_contacts(route)
  413. return '\n'.join(map(lambda a: 'ADT%d' % a, sorted(adh)))
  414. get_adh.short_description = 'Adhérents'
  415. def get_routed_ip(self, route):
  416. routed_ip = route.allocations.order_by('resource__ip').values_list('resource__ip', flat=True)
  417. return '\n'.join(routed_ip)
  418. get_routed_ip.short_description = 'IP routées'
  419. def get_actions(self, request):
  420. actions = super().get_actions(request)
  421. if 'delete_selected' in actions:
  422. del actions['delete_selected']
  423. return actions
  424. def has_delete_permission(self, request, obj=None):
  425. return False
  426. class TunnelAdmin(admin.ModelAdmin):
  427. list_display = ('name', 'description', 'created', 'active')
  428. list_filter = (
  429. ActiveTunnelFilter,
  430. )
  431. def active(self, obj):
  432. return not obj.ended
  433. active.short_description = 'Actif'
  434. active.boolean = True
  435. class ServiceTypeAdmin(admin.ModelAdmin):
  436. fields = ('name',)
  437. def get_actions(self, request):
  438. actions = super().get_actions(request)
  439. if 'delete_selected' in actions:
  440. del actions['delete_selected']
  441. return actions
  442. def has_delete_permission(self, request, obj=None):
  443. return False
  444. class AntennaAdmin(admin.ModelAdmin):
  445. inlines = (ActiveAntennaAllocationInline, InactiveAntennaAllocationInline,)
  446. list_filter = (
  447. AntennaPrefixFilter,
  448. AntennaPositionFilter,
  449. 'mode',
  450. 'ssid',
  451. )
  452. list_display_links = ('id', 'label')
  453. search_fields = ('=id', 'label', 'notes', 'ssid')
  454. raw_id_fields = ('contact',)
  455. form = AntennaForm
  456. def get_queryset(self, request):
  457. qs = super().get_queryset(request)
  458. if connection.vendor == 'postgresql':
  459. qs = qs.annotate(
  460. ip=StringAgg( # concaténation des IP avec des virgules directement par postgresql
  461. Cast( # casting en TextField car StringApp oppère sur des string mais les ip sont des inet
  462. models.Case( # seulement les IP des allocations actives
  463. models.When(
  464. get_active_filter('allocation'),
  465. then='allocation__resource__ip'
  466. ),
  467. ),
  468. models.TextField()
  469. ),
  470. delimiter=', '
  471. )
  472. )
  473. return qs
  474. def get_list_display(self, request):
  475. # ssid_display needs request to access query string and preserve filters
  476. ssid_display = partial(self.ssid_display, request=request)
  477. update_wrapper(ssid_display, self.ssid_display)
  478. return ('id', 'label', 'mode', ssid_display, 'position_display', 'ip_display')
  479. def position_display(self, obj):
  480. return obj.position is not None
  481. position_display.short_description = 'Géolocalisé'
  482. position_display.boolean = True
  483. def ip_display(self, obj):
  484. if connection.vendor == 'postgresql':
  485. return obj.ip
  486. else:
  487. # peu efficace car génère une requête par ligne
  488. allocations = obj.allocations.filter(active=True)
  489. return ', '.join(allocations.values_list('resource__ip', flat=True)) or '-'
  490. ip_display.short_description = 'IP'
  491. def ssid_display(self, obj, request):
  492. if obj.ssid:
  493. qs = request.GET.copy()
  494. qs.update({'ssid': obj.ssid})
  495. ssid_url = reverse('admin:services_antenna_changelist') + '?' + urlencode(qs)
  496. return format_html(u'<a href="{}">{}</a>', ssid_url, obj.ssid)
  497. else:
  498. return None
  499. ssid_display.short_description = 'SSID'
  500. def get_actions(self, request):
  501. actions = super().get_actions(request)
  502. if 'delete_selected' in actions:
  503. del actions['delete_selected']
  504. return actions
  505. def has_delete_permission(self, request, obj=None):
  506. return False
  507. def get_urls(self):
  508. my_urls = [
  509. url(r'^map/$', self.admin_site.admin_view(self.map_view, cacheable=True), name='antenna-map'),
  510. url(r'^map/data.json$', self.admin_site.admin_view(GeoJSONLayerView.as_view(
  511. model=Antenna,
  512. geometry_field='position',
  513. properties=('label', 'mode', 'ssid', 'orientation',),
  514. )), name='antenna-map-data'),
  515. ]
  516. return my_urls + super().get_urls()
  517. def map_view(self, request):
  518. return TemplateResponse(request, 'services/antenna_map.html', {
  519. 'json_url': reverse('admin:antenna-map-data'),
  520. })
  521. def map_data_view(self, request):
  522. geojson = serialize('geojson', Antenna.objects.all(), geometry_field='point', fields=('position',))
  523. return HttpResponse(geojson, content_type='application/json')
  524. class SwitchAdmin(admin.ModelAdmin):
  525. list_display = ('name', 'size',)
  526. fields = ('name', 'size', 'notes',)
  527. inlines = (SwitchPortInline,)
  528. def get_readonly_fields(self, request, obj=None):
  529. if obj:
  530. return ('size',)
  531. else:
  532. return ()
  533. def get_actions(self, request):
  534. actions = super().get_actions(request)
  535. if 'delete_selected' in actions:
  536. del actions['delete_selected']
  537. return actions
  538. def has_delete_permission(self, request, obj=None):
  539. return False
  540. admin.site.register(ServiceType, ServiceTypeAdmin)
  541. admin.site.register(Service, ServiceAdmin)
  542. admin.site.register(IPPrefix, IPPrefixAdmin)
  543. admin.site.register(IPResource, IPResourceAdmin)
  544. admin.site.register(Route, RouteAdmin)
  545. admin.site.register(Tunnel, TunnelAdmin)
  546. admin.site.register(Antenna, AntennaAdmin)
  547. admin.site.register(Switch, SwitchAdmin)