|
@@ -26,9 +26,9 @@ from datetime import timedelta
|
|
|
from djadhere.utils import get_active_filter
|
|
|
from adhesions.models import Adhesion
|
|
|
from banking.models import PaymentUpdate
|
|
|
-from .models import Service, ServiceType, IPPrefix, IPResource, Route, Tunnel, \
|
|
|
+from .models import Service, ServiceType, IPPrefix, IPResource, IPResourceState, \
|
|
|
ServiceAllocation, Antenna, AntennaAllocation, Allocation, \
|
|
|
- Switch, Port
|
|
|
+ Route, Tunnel, Switch, Port
|
|
|
from .utils.notifications import notify_allocation
|
|
|
from .forms import AntennaForm
|
|
|
|
|
@@ -67,13 +67,15 @@ class ResourcePingFilter(admin.SimpleListFilter):
|
|
|
|
|
|
def queryset(self, request, queryset):
|
|
|
if self.value() == 'up':
|
|
|
- return queryset.filter(up=True)
|
|
|
+ return queryset.filter(last_state__state=IPResourceState.STATE_UP)
|
|
|
if self.value() == 'down':
|
|
|
- return queryset.filter(models.Q(up__isnull=True) | models.Q(up=False))
|
|
|
+ return queryset.exclude(last_state__state=IPResourceState.STATE_UP) # DOWN + UNKNOWN
|
|
|
if self.value() == 'down-since':
|
|
|
- return queryset.filter(up=False)
|
|
|
+ queryset = queryset.exclude(last_state__state=IPResourceState.STATE_UP)
|
|
|
+ return queryset.filter(last_time_up__isnull=False)
|
|
|
if self.value() == 'never-up':
|
|
|
- return queryset.filter(up__isnull=True)
|
|
|
+ queryset = queryset.exclude(last_state__state=IPResourceState.STATE_UP) # DOWN + UNKWON
|
|
|
+ return queryset.filter(last_time_up__isnull=True)
|
|
|
|
|
|
|
|
|
class ActiveServiceFilter(admin.SimpleListFilter):
|
|
@@ -265,6 +267,22 @@ class InactiveAntennaAllocationInline(AntennaAllocationMixin, InactiveAllocation
|
|
|
pass
|
|
|
|
|
|
|
|
|
+class IPResourceStateInline(admin.TabularInline):
|
|
|
+ model = IPResourceState
|
|
|
+ verbose_name_plural = 'Historique des changements d’état'
|
|
|
+ fields = ['date']
|
|
|
+ ordering = ['-date']
|
|
|
+
|
|
|
+ def has_add_permission(self, request):
|
|
|
+ return False
|
|
|
+
|
|
|
+ def has_change_permission(self, request, obj):
|
|
|
+ return False
|
|
|
+
|
|
|
+ def has_delete_permission(self, request, obj=None):
|
|
|
+ return False
|
|
|
+
|
|
|
+
|
|
|
class PortInline(admin.TabularInline):
|
|
|
model = Port
|
|
|
max_num = 0
|
|
@@ -385,27 +403,32 @@ class IPResourceAdmin(admin.ModelAdmin):
|
|
|
)
|
|
|
search_fields = ('=ip', 'notes',)
|
|
|
actions = ['contact_ip_owners']
|
|
|
+ ordering = ['ip']
|
|
|
+ inlines = [ IPResourceStateInline ]
|
|
|
|
|
|
def get_fields(self, request, obj=None):
|
|
|
return self.get_readonly_fields(request, obj)
|
|
|
|
|
|
def get_readonly_fields(self, request, obj=None):
|
|
|
fields = ['ip']
|
|
|
- if obj and obj.reserved:
|
|
|
- fields += ['reserved']
|
|
|
- if obj and not obj.in_use:
|
|
|
- fields += ['last_use']
|
|
|
- if obj and obj.last_time_up and obj.last_check:
|
|
|
- fields += ['last_time_up', 'last_check']
|
|
|
- if obj.category == IPResource.CATEGORY_PUBLIC:
|
|
|
- fields += ['password']
|
|
|
- if obj and obj.checkmk_label:
|
|
|
- fields += ['checkmk']
|
|
|
- if obj and obj.notes:
|
|
|
- fields += ['notes']
|
|
|
+ if obj:
|
|
|
+ if obj.reserved:
|
|
|
+ fields += ['reserved']
|
|
|
+ if not obj.in_use:
|
|
|
+ fields += ['last_use']
|
|
|
+ fields += ['last_state']
|
|
|
+ if obj.last_state.state != IPResourceState.STATE_UP:
|
|
|
+ fields += ['last_time_up']
|
|
|
+ if obj.category == IPResource.CATEGORY_PUBLIC:
|
|
|
+ fields += ['password']
|
|
|
+ if obj.checkmk_label:
|
|
|
+ fields += ['checkmk']
|
|
|
+ if obj.notes:
|
|
|
+ fields += ['notes']
|
|
|
return fields
|
|
|
|
|
|
def get_inline_instances(self, request, obj=None):
|
|
|
+ super_inlines = super().get_inline_instances(request, obj)
|
|
|
if obj:
|
|
|
if obj.category == 0:
|
|
|
inlines = (ActiveServiceAllocationInline, InactiveServiceAllocationInline,)
|
|
@@ -413,7 +436,7 @@ class IPResourceAdmin(admin.ModelAdmin):
|
|
|
inlines = (ActiveAntennaAllocationInline, InactiveAntennaAllocationInline,)
|
|
|
else:
|
|
|
inlines = ()
|
|
|
- return [inline(self.model, self.admin_site) for inline in inlines]
|
|
|
+ return [inline(self.model, self.admin_site) for inline in inlines] + super_inlines
|
|
|
|
|
|
def get_queryset(self, request):
|
|
|
qs = super().get_queryset(request)
|
|
@@ -426,13 +449,12 @@ class IPResourceAdmin(admin.ModelAdmin):
|
|
|
default=None,
|
|
|
))
|
|
|
qs = qs.annotate(
|
|
|
- up=models.Case(
|
|
|
- models.When(last_check__isnull=False, last_time_up__isnull=False, last_time_up=models.F('last_check'), then=True),
|
|
|
- models.When(last_check__isnull=False, last_time_up__isnull=False, then=False),
|
|
|
+ downtime=models.Case(
|
|
|
+ models.When(last_state__state=IPResourceState.STATE_UP, then=models.F('last_state__date')-models.Value(now)),
|
|
|
+ models.When(last_state__state=IPResourceState.STATE_DOWN, then=models.Value(now)-models.F('last_time_up')),
|
|
|
default=None,
|
|
|
- output_field=models.NullBooleanField(),
|
|
|
+ output_field=models.DurationField(),
|
|
|
))
|
|
|
- qs = qs.annotate(downtime=models.F('last_check') - models.F('last_time_up'))
|
|
|
qs = qs.annotate(
|
|
|
route=models.Case(
|
|
|
models.When(
|
|
@@ -461,12 +483,13 @@ class IPResourceAdmin(admin.ModelAdmin):
|
|
|
last_use.admin_order_field = 'last_use'
|
|
|
|
|
|
def ping(self, obj):
|
|
|
- if obj.up:
|
|
|
+ if obj.last_state.state == IPResourceState.STATE_UP:
|
|
|
label = 'UP'
|
|
|
- elif obj.last_time_up:
|
|
|
- label = 'dernier ping : ' + naturaltime(obj.last_time_up)
|
|
|
else:
|
|
|
- label = 'DOWN'
|
|
|
+ if obj.last_time_up:
|
|
|
+ label = 'dernier ping : ' + naturaltime(obj.last_time_up)
|
|
|
+ else:
|
|
|
+ label = 'DOWN'
|
|
|
if obj.checkmk_url:
|
|
|
return format_html('<a href="{}">{}</a>', obj.checkmk_url, label)
|
|
|
else:
|