Group management form converted to crispy forms. Page for group list improved

This commit is contained in:
Eduardo Silva 2025-01-20 15:03:15 -03:00
parent eefc573c61
commit 8eb0a6d852
6 changed files with 61 additions and 85 deletions

View File

@ -1,77 +0,0 @@
{% extends "base.html" %}
{% block content %}
<div class="container mt-3">
<div class="card card-primary card-outline">
<div class="card-header">
<h3 class="card-title">{{ form.instance.pk|yesno:"Edit Peer Group,Create New Peer Group" }}</h3>
</div>
<div class="card-body">
<div class="row">
<div class="col-md-4">
<form method="post">
{% csrf_token %}
<!-- Name -->
<div class="form-group">
<label for="{{ form.name.id_for_label }}">Name</label>
<input type="text" class="form-control" id="{{ form.name.id_for_label }}" name="{{ form.name.html_name }}" placeholder="Enter Name" value="{{ form.name.value|default_if_none:'' }}">
</div>
<!-- Peers -->
<div class="form-group">
<label for="{{ form.peer.id_for_label }}">Peers</label>
<select class="form-control" id="{{ form.peer.id_for_label }}" name="{{ form.peer.html_name }}" multiple>
{% for peer in form.peer.field.queryset %}
<option value="{{ peer.pk }}" {% if peer.pk in form.peer.value %}selected{% endif %}>{{ peer }}</option>
{% endfor %}
</select>
</div>
<!-- Server Instances -->
<div class="form-group">
<label for="{{ form.server_instance.id_for_label }}">WireGuard Instances</label>
<select class="form-control" id="{{ form.server_instance.id_for_label }}" name="{{ form.server_instance.html_name }}" multiple>
{% for instance in form.server_instance.field.queryset %}
<option value="{{ instance.pk }}" {% if instance.pk in form.server_instance.value %}selected{% endif %}>{{ instance }}</option>
{% endfor %}
</select>
</div>
<div>
<button type="submit" class="btn btn-primary">Submit</button>
<a href="/user/peer-group/list/" class="btn btn-outline-secondary">Back</a>
{% if peer_group %}<a href='javascript:void(0)' class='btn btn-outline-danger' data-command='delete' onclick='openCommandDialog(this)'>Delete Peer Group</a>{% endif %}
</div>
</form>
</div>
<div class="col-md-8">
<h5>Peers</h5>
<p>Select which peers can be managed by users with this peer group.</p>
<h5>WireGuard Instances</h5>
<p>All peers in this WireGuard instance can be managed by users with this peer group, including adding or removing peers.</p>
</div>
</div>
</div>
</div>
</div>
{% endblock %}
{% block custom_page_scripts %}
<script>
function openCommandDialog(element) {
var command = element.getAttribute('data-command');
var confirmation = prompt("Please type '{{ peer_group.name }}' to remove this peer group.");
if (confirmation) {
var url = "?uuid={{ peer_group.uuid }}&action=delete&confirmation=" + encodeURIComponent(confirmation);
window.location.href = url;
}
}
</script>
{% endblock %}

View File

@ -7,6 +7,7 @@
<th>Name</th> <th>Name</th>
<th>Peers</th> <th>Peers</th>
<th>Server Instance</th> <th>Server Instance</th>
<th>Users</th>
<th></th> <th></th>
</tr> </tr>
</thead> </thead>
@ -16,12 +17,17 @@
<td>{{ peer_group.name }}</td> <td>{{ peer_group.name }}</td>
<td> <td>
{% for peer in peer_group.peer.all %} {% for peer in peer_group.peer.all %}
{{ peer }}{% if not forloop.last %}, {% endif %} <a href="/peer/manage/?peer={{ peer.uuid }}">{{ peer }}</a>{% if not forloop.last %}, {% endif %}
{% endfor %} {% endfor %}
</td> </td>
<td> <td>
{% for instance in peer_group.server_instance.all %} {% for instance in peer_group.server_instance.all %}
{{ instance }}{% if not forloop.last %}, {% endif %} <a href="/peer/list/?uuid={{ instance.uuid }}">{{ instance }}</a>{% if not forloop.last %}, {% endif %}
{% endfor %}
</td>
<td>
{% for user_acl in peer_group.useracl_set.all %}
<a href="/user/manage/?uuid={{ user_acl.uuid }}">{{ user_acl.user.username }}</a>{% if not forloop.last %}, {% endif %}
{% endfor %} {% endfor %}
</td> </td>
<td style="width: 1%; white-space: nowrap;"> <td style="width: 1%; white-space: nowrap;">

View File

@ -4,6 +4,8 @@ from django.contrib.auth.models import User
from .models import UserAcl from .models import UserAcl
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from wireguard.models import PeerGroup from wireguard.models import PeerGroup
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Layout, Row, Column, Submit, HTML
class UserAclForm(UserCreationForm): class UserAclForm(UserCreationForm):
@ -56,8 +58,37 @@ class PeerGroupForm(forms.ModelForm):
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
self.user_id = kwargs.pop('user_id', None) self.user_id = kwargs.pop('user_id', None)
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.helper = FormHelper()
self.helper.form_method = 'post'
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(
Row(
Column('name', css_class='form-group col-md-12 mb-0'),
css_class='form-row'
),
Row(
Column('peer', css_class='form-group col-md-12 mb-0'),
css_class='form-row'
),
Row(
Column('server_instance', css_class='form-group col-md-12 mb-0'),
css_class='form-row'
),
Row(
Column(
Submit('submit', 'Save', css_class='btn btn-success'),
HTML(' <a class="btn btn-secondary" href="/user/peer-group/list/">Back</a> '),
HTML(delete_html),
css_class='col-md-12'),
css_class='form-row'
)
)
def clean(self): def clean(self):
cleaned_data = super().clean() cleaned_data = super().clean()
name = cleaned_data.get('name') name = cleaned_data.get('name')
@ -68,6 +99,7 @@ class PeerGroupForm(forms.ModelForm):
raise ValidationError("A peer group with that name already exists.") raise ValidationError("A peer group with that name already exists.")
return cleaned_data return cleaned_data
def save(self, commit=True): def save(self, commit=True):
peer_group = super().save(commit=False) peer_group = super().save(commit=False)

View File

@ -30,11 +30,12 @@ def view_peer_group_manage(request):
page_title = 'Edit Peer Group ' + peer_group.name page_title = 'Edit Peer Group ' + peer_group.name
if request.GET.get('action') == 'delete': if request.GET.get('action') == 'delete':
group_name = peer_group.name group_name = peer_group.name
if request.GET.get('confirmation') == group_name: if request.GET.get('confirmation') == 'delete':
peer_group.delete() peer_group.delete()
messages.success(request, 'Peer Group deleted|The peer group ' + group_name + ' has been deleted.') messages.success(request, 'Peer Group deleted|The peer group ' + group_name + ' has been deleted.')
return redirect('/user/peer-group/list/') return redirect('/user/peer-group/list/')
else:
messages.warning(request, 'Peer Group not deleted|Invalid confirmation.')
return redirect('/user/peer-group/list/') return redirect('/user/peer-group/list/')
else: else:
form = PeerGroupForm(user_id=request.user.id) form = PeerGroupForm(user_id=request.user.id)
@ -50,8 +51,19 @@ def view_peer_group_manage(request):
peer_group = form.save() peer_group = form.save()
form.save_m2m() form.save_m2m()
return redirect('/user/peer-group/list/') return redirect('/user/peer-group/list/')
context = {'page_title': page_title, 'form': form, 'peer_group': peer_group}
return render(request, 'user_manager/manage_peer_group.html', context) form_description = {
'size': '',
'content': '''
<h5>Peers</h5>
<p>Select which peers can be managed by users with this peer group.</p>
<h5>WireGuard Instances</h5>
<p>All peers in this WireGuard instance can be managed by users with this peer group, including adding or removing peers.</p>
'''
}
context = {'page_title': page_title, 'form': form, 'peer_group': peer_group, 'instance': peer_group, 'form_description': form_description}
return render(request, 'generic_form.html', context)
@login_required @login_required

View File

@ -131,3 +131,6 @@ class PeerGroup(models.Model):
updated = models.DateTimeField(auto_now=True) updated = models.DateTimeField(auto_now=True)
uuid = models.UUIDField(primary_key=True, editable=False, default=uuid.uuid4) uuid = models.UUIDField(primary_key=True, editable=False, default=uuid.uuid4)
def __str__(self):
return self.name

View File

@ -28,7 +28,7 @@ from dns.views import view_static_host_list, view_manage_static_host, view_manag
urlpatterns = [ urlpatterns = [
# path('admin/', admin.site.urls), path('admin/', admin.site.urls),
path('', view_welcome, name='welcome'), path('', view_welcome, name='welcome'),
path('status/', view_wireguard_status, name='wireguard_status'), path('status/', view_wireguard_status, name='wireguard_status'),
path('dns/', view_static_host_list, name='static_host_list'), path('dns/', view_static_host_list, name='static_host_list'),