mirror of
https://github.com/eduardogsilva/wireguard_webadmin.git
synced 2025-04-19 00:45:16 +00:00
add DNS Filter List management functionality
This commit is contained in:
parent
a63c5643b0
commit
3fc70cbb58
25
dns/admin.py
25
dns/admin.py
@ -1,3 +1,26 @@
|
|||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
|
|
||||||
# Register your models here.
|
from .models import DNSFilterList, DNSSettings, StaticHost
|
||||||
|
|
||||||
|
|
||||||
|
class DNSFilterListAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('name', 'description', 'enabled', 'list_url', 'host_count', 'created', 'updated')
|
||||||
|
list_filter = ('enabled', 'created', 'updated')
|
||||||
|
search_fields = ('name', 'description', 'list_url')
|
||||||
|
ordering = ('name', 'created')
|
||||||
|
admin.site.register(DNSFilterList, DNSFilterListAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
class DNSSettingsAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('dns_primary', 'dns_secondary', 'pending_changes', 'created', 'updated')
|
||||||
|
list_filter = ('pending_changes', 'created', 'updated')
|
||||||
|
search_fields = ('dns_primary', 'dns_secondary')
|
||||||
|
ordering = ('created', 'updated')
|
||||||
|
admin.site.register(DNSSettings, DNSSettingsAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
class StaticHostAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('hostname', 'ip_address', 'created', 'updated')
|
||||||
|
search_fields = ('hostname', 'ip_address')
|
||||||
|
ordering = ('hostname', 'created')
|
||||||
|
admin.site.register(StaticHost, StaticHostAdmin)
|
55
dns/forms.py
55
dns/forms.py
@ -1,15 +1,14 @@
|
|||||||
import requests
|
|
||||||
from .models import DNSSettings, StaticHost
|
|
||||||
|
|
||||||
from django import forms
|
|
||||||
from crispy_forms.helper import FormHelper
|
|
||||||
from crispy_forms.layout import Layout, Fieldset, ButtonHolder, Submit, Div, Field, HTML
|
|
||||||
from crispy_forms.bootstrap import FormActions, StrictButton
|
|
||||||
from django.core.exceptions import ValidationError
|
|
||||||
from datetime import datetime
|
|
||||||
from django.core.exceptions import ValidationError
|
|
||||||
import re
|
import re
|
||||||
|
|
||||||
|
from crispy_forms.bootstrap import FormActions
|
||||||
|
from crispy_forms.helper import FormHelper
|
||||||
|
from crispy_forms.layout import Layout, Fieldset, Div, Field, Submit, HTML
|
||||||
|
from django import forms
|
||||||
|
from django.core.exceptions import ValidationError
|
||||||
|
|
||||||
|
from .models import DNSFilterList
|
||||||
|
from .models import DNSSettings, StaticHost
|
||||||
|
|
||||||
|
|
||||||
class DNSSettingsForm(forms.ModelForm):
|
class DNSSettingsForm(forms.ModelForm):
|
||||||
class Meta:
|
class Meta:
|
||||||
@ -89,3 +88,39 @@ class StaticHostForm(forms.ModelForm):
|
|||||||
if not re.match(regex, hostname):
|
if not re.match(regex, hostname):
|
||||||
raise ValidationError('Invalid hostname')
|
raise ValidationError('Invalid hostname')
|
||||||
return
|
return
|
||||||
|
|
||||||
|
|
||||||
|
class DNSFilterListForm(forms.ModelForm):
|
||||||
|
class Meta:
|
||||||
|
model = DNSFilterList
|
||||||
|
# Only allow editable fields
|
||||||
|
fields = ['name', 'description', 'list_url']
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
super(DNSFilterListForm, self).__init__(*args, **kwargs)
|
||||||
|
self.helper = FormHelper()
|
||||||
|
self.helper.form_method = 'post'
|
||||||
|
# Add a delete button if editing an existing instance
|
||||||
|
if self.instance.pk:
|
||||||
|
delete_html = (
|
||||||
|
"<a href='javascript:void(0)' class='btn btn-outline-danger' "
|
||||||
|
"data-command='delete' onclick='openCommandDialog(this)'>Delete</a>"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
delete_html = ''
|
||||||
|
self.helper.layout = Layout(
|
||||||
|
Fieldset(
|
||||||
|
'DNS Filter List Details',
|
||||||
|
Div(
|
||||||
|
Div(Field('name', css_class='form-control'), css_class='col-md-12'),
|
||||||
|
Div(Field('description', css_class='form-control'), css_class='col-md-12'),
|
||||||
|
Div(Field('list_url', css_class='form-control'), css_class='col-md-12'),
|
||||||
|
css_class='row'
|
||||||
|
),
|
||||||
|
),
|
||||||
|
FormActions(
|
||||||
|
Submit('save', 'Save', css_class='btn btn-primary'),
|
||||||
|
HTML('<a class="btn btn-outline-primary" href="/dns/">Back</a>'),
|
||||||
|
HTML(delete_html),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
29
dns/migrations/0002_dnsfilterlist.py
Normal file
29
dns/migrations/0002_dnsfilterlist.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-03-01 21:05
|
||||||
|
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('dns', '0001_initial'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='DNSFilterList',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('name', models.SlugField(max_length=100, unique=True)),
|
||||||
|
('description', models.CharField(max_length=100)),
|
||||||
|
('enabled', models.BooleanField(default=False)),
|
||||||
|
('list_url', models.URLField()),
|
||||||
|
('last_updated', models.DateTimeField(blank=True, null=True)),
|
||||||
|
('created', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated', models.DateTimeField(auto_now=True)),
|
||||||
|
('uuid', models.UUIDField(default=uuid.uuid4, editable=False, unique=True)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
]
|
18
dns/migrations/0003_dnsfilterlist_host_count.py
Normal file
18
dns/migrations/0003_dnsfilterlist_host_count.py
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-03-01 21:37
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('dns', '0002_dnsfilterlist'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='dnsfilterlist',
|
||||||
|
name='host_count',
|
||||||
|
field=models.IntegerField(default=0),
|
||||||
|
),
|
||||||
|
]
|
@ -1,6 +1,7 @@
|
|||||||
from django.db import models
|
|
||||||
import uuid
|
import uuid
|
||||||
|
|
||||||
|
from django.db import models
|
||||||
|
|
||||||
|
|
||||||
class DNSSettings(models.Model):
|
class DNSSettings(models.Model):
|
||||||
name = models.CharField(default='dns_settings', max_length=100)
|
name = models.CharField(default='dns_settings', max_length=100)
|
||||||
@ -23,3 +24,19 @@ class StaticHost(models.Model):
|
|||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.hostname
|
return self.hostname
|
||||||
|
|
||||||
|
|
||||||
|
class DNSFilterList(models.Model):
|
||||||
|
name = models.SlugField(max_length=100, unique=True)
|
||||||
|
description = models.CharField(max_length=100)
|
||||||
|
enabled = models.BooleanField(default=False)
|
||||||
|
list_url = models.URLField()
|
||||||
|
last_updated = models.DateTimeField(blank=True, null=True)
|
||||||
|
host_count = models.IntegerField(default=0)
|
||||||
|
|
||||||
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated = models.DateTimeField(auto_now=True)
|
||||||
|
uuid = models.UUIDField(unique=True, default=uuid.uuid4, editable=False)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
58
dns/views.py
58
dns/views.py
@ -1,11 +1,14 @@
|
|||||||
|
from django.conf import settings
|
||||||
|
from django.contrib import messages
|
||||||
from django.contrib.auth.decorators import login_required
|
from django.contrib.auth.decorators import login_required
|
||||||
from django.shortcuts import render, get_object_or_404, redirect
|
from django.shortcuts import render, get_object_or_404, redirect
|
||||||
from django.contrib import messages
|
|
||||||
from user_manager.models import UserAcl
|
from user_manager.models import UserAcl
|
||||||
from .models import DNSSettings, StaticHost
|
from .forms import DNSFilterListForm
|
||||||
from .forms import StaticHostForm, DNSSettingsForm
|
from .forms import StaticHostForm, DNSSettingsForm
|
||||||
from .functions import generate_dnsmasq_config
|
from .functions import generate_dnsmasq_config
|
||||||
from django.conf import settings
|
from .models import DNSSettings, DNSFilterList
|
||||||
|
from .models import StaticHost
|
||||||
|
|
||||||
|
|
||||||
def export_dns_configuration():
|
def export_dns_configuration():
|
||||||
@ -29,11 +32,21 @@ def view_apply_dns_config(request):
|
|||||||
def view_static_host_list(request):
|
def view_static_host_list(request):
|
||||||
dns_settings, _ = DNSSettings.objects.get_or_create(name='dns_settings')
|
dns_settings, _ = DNSSettings.objects.get_or_create(name='dns_settings')
|
||||||
static_host_list = StaticHost.objects.all().order_by('hostname')
|
static_host_list = StaticHost.objects.all().order_by('hostname')
|
||||||
|
filter_lists = DNSFilterList.objects.all().order_by('name')
|
||||||
|
if not filter_lists:
|
||||||
|
DNSFilterList.objects.create(
|
||||||
|
name='stevenblack-hosts', list_url='https://raw.githubusercontent.com/StevenBlack/hosts/master/hosts',
|
||||||
|
description='adware and malware domains', enabled=False
|
||||||
|
)
|
||||||
|
filter_lists = DNSFilterList.objects.all().order_by('name')
|
||||||
|
messages.success(request, 'Default DNS Filter List created successfully')
|
||||||
|
|
||||||
if dns_settings.pending_changes:
|
if dns_settings.pending_changes:
|
||||||
messages.warning(request, 'Pending Changes|There are pending DNS changes that have not been applied')
|
messages.warning(request, 'Pending Changes|There are pending DNS changes that have not been applied')
|
||||||
context = {
|
context = {
|
||||||
'dns_settings': dns_settings,
|
'dns_settings': dns_settings,
|
||||||
'static_host_list': static_host_list,
|
'static_host_list': static_host_list,
|
||||||
|
'filter_lists': filter_lists
|
||||||
}
|
}
|
||||||
return render(request, 'dns/static_host_list.html', context=context)
|
return render(request, 'dns/static_host_list.html', context=context)
|
||||||
|
|
||||||
@ -101,3 +114,42 @@ def view_manage_static_host(request):
|
|||||||
'instance': static_dns,
|
'instance': static_dns,
|
||||||
}
|
}
|
||||||
return render(request, 'generic_form.html', context=context)
|
return render(request, 'generic_form.html', context=context)
|
||||||
|
|
||||||
|
|
||||||
|
@login_required
|
||||||
|
def view_manage_filter_list(request):
|
||||||
|
if not UserAcl.objects.filter(user=request.user, user_level__gte=50).exists():
|
||||||
|
return render(request, 'access_denied.html', {'page_title': 'Access Denied'})
|
||||||
|
|
||||||
|
dns_settings, _ = DNSSettings.objects.get_or_create(name='dns_settings')
|
||||||
|
|
||||||
|
if request.GET.get('uuid'):
|
||||||
|
filter_list = get_object_or_404(DNSFilterList, uuid=request.GET.get('uuid'))
|
||||||
|
if request.GET.get('action') == 'delete':
|
||||||
|
if request.GET.get('confirmation') == 'delete':
|
||||||
|
if filter_list.enabled:
|
||||||
|
messages.warning(request, 'DNS Filter List not deleted | Filter List is enabled')
|
||||||
|
return redirect('/dns/')
|
||||||
|
filter_list.delete()
|
||||||
|
messages.success(request, 'DNS Filter List deleted successfully')
|
||||||
|
return redirect('/dns/')
|
||||||
|
else:
|
||||||
|
messages.warning(request, 'DNS Filter List not deleted | Invalid confirmation')
|
||||||
|
return redirect('/dns/')
|
||||||
|
else:
|
||||||
|
filter_list = None
|
||||||
|
|
||||||
|
form = DNSFilterListForm(request.POST or None, instance=filter_list)
|
||||||
|
if form.is_valid():
|
||||||
|
form.save()
|
||||||
|
dns_settings.pending_changes = True
|
||||||
|
dns_settings.save()
|
||||||
|
messages.success(request, 'DNS Filter List saved successfully')
|
||||||
|
return redirect('/dns/')
|
||||||
|
|
||||||
|
context = {
|
||||||
|
'dns_settings': dns_settings,
|
||||||
|
'form': form,
|
||||||
|
'instance': filter_list,
|
||||||
|
}
|
||||||
|
return render(request, 'generic_form.html', context=context)
|
@ -7,46 +7,105 @@
|
|||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
<h3 class="card-title">Static Host List</h3>
|
<h3 class="card-title">Static Host List</h3>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<table class="table table-hover">
|
<table class="table table-hover">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
|
||||||
<th>Hostname</th>
|
|
||||||
<th>IP Address</th>
|
|
||||||
<th></th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{% if static_host_list %}
|
|
||||||
{% for static_host in static_host_list %}
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ static_host.hostname }}</td>
|
<th>Hostname</th>
|
||||||
<td>{{ static_host.ip_address }}</td>
|
<th>IP Address</th>
|
||||||
<td class="min-width">
|
<th></th>
|
||||||
<a href="/dns/manage_static_host/?uuid={{ static_host.uuid }}"><i class="fas fa-edit"></i></a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
</thead>
|
||||||
{% else %}
|
<tbody>
|
||||||
<tr>
|
{% if static_host_list %}
|
||||||
<td colspan="3"></td>
|
{% for static_host in static_host_list %}
|
||||||
</tr>
|
<tr>
|
||||||
{% endif %}
|
<td>{{ static_host.hostname }}</td>
|
||||||
</tbody>
|
<td>{{ static_host.ip_address }}</td>
|
||||||
</table>
|
<td class="min-width">
|
||||||
|
<a href="/dns/manage_static_host/?uuid={{ static_host.uuid }}"><i class="fas fa-edit"></i></a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
{% else %}
|
||||||
|
<tr>
|
||||||
|
<td colspan="3"></td>
|
||||||
|
</tr>
|
||||||
|
{% endif %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-12">
|
||||||
|
<a href="/dns/manage_static_host/" class="btn btn-primary">Add Static Host</a>
|
||||||
|
<a href="/dns/manage_settings/" class="btn btn-primary">DNS Settings</a>
|
||||||
|
<a href="/dns/apply_config/" class="btn btn-primary">Apply Config</a>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
<div class="row">
|
</div>
|
||||||
<div class="col-lg-12">
|
</div>
|
||||||
<a href="/dns/manage_static_host/" class="btn btn-primary">Add Static Host</a>
|
|
||||||
<a href="/dns/manage_settings/" class="btn btn-primary">DNS Settings</a>
|
<div class="row">
|
||||||
<a href="/dns/apply_config/" class="btn btn-primary">Apply Config</a>
|
<div class='col-lg-12'>
|
||||||
</div>
|
<div class="card card-primary card-outline">
|
||||||
|
<div class="card-header">
|
||||||
|
<h3 class="card-title">DNS Filter Lists</h3>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="card-body">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-12">
|
||||||
|
<table class="table table-hover">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Name</th>
|
||||||
|
<th>Description</th>
|
||||||
|
<th>Hosts</th>
|
||||||
|
<th>Last Update</th>
|
||||||
|
<th class="min-width"></th>
|
||||||
|
<th class="min-width"></th>
|
||||||
|
<th class="min-width"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{% if filter_lists %}
|
||||||
|
{% for filter_list in filter_lists %}
|
||||||
|
<tr>
|
||||||
|
<td>{{ filter_list.name }}</td>
|
||||||
|
<td>{{ filter_list.description }}</td>
|
||||||
|
<td>{{ filter_list.host_count }}</td>
|
||||||
|
<td>{{ filter_list.last_updated|default_if_none:"" }}</td>
|
||||||
|
<td>
|
||||||
|
<!-- refresh -->
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<!-- toggle enable/disable -->
|
||||||
|
</td>
|
||||||
|
<td class="min-width">
|
||||||
|
<a href="/dns/manage_filter_list/?uuid={{ filter_list.uuid }}"><i class="fas fa-edit"></i></a>
|
||||||
|
{% comment %}<a class="text-danger" href="/dns/manage_filter_list/?uuid={{ filter_list.uuid }}&action=delete" onclick="return confirm('Proceed removing the dns filter list: {{ filter_list.name }}?')"><i class="fas fa-trash-alt"></i></a>{% endcomment %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
{% else %}
|
||||||
|
<tr>
|
||||||
|
<td colspan="7"></td>
|
||||||
|
</tr>
|
||||||
|
{% endif %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-12">
|
||||||
|
<a href="/dns/manage_filter_list/" class="btn btn-primary">Add Filter List</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -21,7 +21,8 @@ from accounts.views import view_create_first_user, view_login, view_logout
|
|||||||
from api.views import wireguard_status, cron_check_updates, cron_update_peer_latest_handshake, \
|
from api.views import wireguard_status, cron_check_updates, cron_update_peer_latest_handshake, \
|
||||||
routerfleet_get_user_token, routerfleet_authenticate_session, peer_info, api_peer_invite
|
routerfleet_get_user_token, routerfleet_authenticate_session, peer_info, api_peer_invite
|
||||||
from console.views import view_console
|
from console.views import view_console
|
||||||
from dns.views import view_static_host_list, view_manage_static_host, view_manage_dns_settings, view_apply_dns_config
|
from dns.views import view_static_host_list, view_manage_static_host, view_manage_dns_settings, view_apply_dns_config, \
|
||||||
|
view_manage_filter_list
|
||||||
from firewall.views import view_redirect_rule_list, manage_redirect_rule, view_firewall_rule_list, manage_firewall_rule, \
|
from firewall.views import view_redirect_rule_list, manage_redirect_rule, view_firewall_rule_list, manage_firewall_rule, \
|
||||||
view_manage_firewall_settings, view_generate_iptables_script, view_reset_firewall, view_firewall_migration_required
|
view_manage_firewall_settings, view_generate_iptables_script, view_reset_firewall, view_firewall_migration_required
|
||||||
from user_manager.views import view_user_list, view_manage_user, view_peer_group_list, view_peer_group_manage
|
from user_manager.views import view_user_list, view_manage_user, view_peer_group_list, view_peer_group_manage
|
||||||
@ -41,6 +42,7 @@ urlpatterns = [
|
|||||||
path('dns/apply_config/', view_apply_dns_config, name='apply_dns_config'),
|
path('dns/apply_config/', view_apply_dns_config, name='apply_dns_config'),
|
||||||
path('dns/manage_static_host/', view_manage_static_host, name='manage_static_host'),
|
path('dns/manage_static_host/', view_manage_static_host, name='manage_static_host'),
|
||||||
path('dns/manage_settings/', view_manage_dns_settings, name='manage_dns_settings'),
|
path('dns/manage_settings/', view_manage_dns_settings, name='manage_dns_settings'),
|
||||||
|
path('dns/manage_filter_list/', view_manage_filter_list, name='manage_filter_list'),
|
||||||
path('peer/list/', view_wireguard_peer_list, name='wireguard_peer_list'),
|
path('peer/list/', view_wireguard_peer_list, name='wireguard_peer_list'),
|
||||||
path('peer/sort/', view_wireguard_peer_sort, name='wireguard_peer_sort'),
|
path('peer/sort/', view_wireguard_peer_sort, name='wireguard_peer_sort'),
|
||||||
path('peer/manage/', view_wireguard_peer_manage, name='wireguard_peer_manage'),
|
path('peer/manage/', view_wireguard_peer_manage, name='wireguard_peer_manage'),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user