@ -53,6 +53,12 @@ class View(OrganizationPermission, generic.View):
|
||||
|
||||
self.model = Cluster
|
||||
|
||||
case 'clustertype':
|
||||
|
||||
from itim.models.clusters import ClusterType
|
||||
|
||||
self.model = ClusterType
|
||||
|
||||
case 'configgroups':
|
||||
|
||||
self.model = ConfigGroups
|
||||
|
84
app/itim/forms/cluster_type.py
Normal file
84
app/itim/forms/cluster_type.py
Normal file
@ -0,0 +1,84 @@
|
||||
from django import forms
|
||||
from django.forms import ValidationError
|
||||
from django.urls import reverse
|
||||
|
||||
from itim.models.clusters import ClusterType
|
||||
|
||||
from app import settings
|
||||
|
||||
from core.forms.common import CommonModelForm
|
||||
|
||||
|
||||
|
||||
class ClusterTypeForm(CommonModelForm):
|
||||
|
||||
|
||||
class Meta:
|
||||
|
||||
fields = '__all__'
|
||||
|
||||
model = ClusterType
|
||||
|
||||
prefix = 'cluster_type'
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
|
||||
super().__init__(*args, **kwargs)
|
||||
|
||||
|
||||
|
||||
class DetailForm(ClusterTypeForm):
|
||||
|
||||
|
||||
tabs: dict = {
|
||||
"details": {
|
||||
"name": "Details",
|
||||
"slug": "details",
|
||||
"sections": [
|
||||
{
|
||||
"layout": "double",
|
||||
"left": [
|
||||
'name',
|
||||
'organization',
|
||||
'c_created',
|
||||
'c_modified'
|
||||
],
|
||||
"right": [
|
||||
'model_notes',
|
||||
]
|
||||
},
|
||||
]
|
||||
},
|
||||
"notes": {
|
||||
"name": "Notes",
|
||||
"slug": "notes",
|
||||
"sections": []
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
|
||||
super().__init__(*args, **kwargs)
|
||||
|
||||
|
||||
self.fields['c_created'] = forms.DateTimeField(
|
||||
label = 'Created',
|
||||
input_formats=settings.DATETIME_FORMAT,
|
||||
disabled = True,
|
||||
initial = self.instance.created,
|
||||
)
|
||||
|
||||
self.fields['c_modified'] = forms.DateTimeField(
|
||||
label = 'Modified',
|
||||
input_formats=settings.DATETIME_FORMAT,
|
||||
disabled = True,
|
||||
initial = self.instance.modified,
|
||||
)
|
||||
|
||||
self.tabs['details'].update({
|
||||
"edit_url": reverse('Settings:_cluster_type_change', args=(self.instance.pk,))
|
||||
})
|
||||
|
||||
self.url_index_view = reverse('Settings:_cluster_types')
|
||||
|
@ -29,8 +29,8 @@ class Migration(migrations.Migration):
|
||||
('organization', models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='access.organization', validators=[access.models.TenancyObject.validatate_organization_exists])),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'ClusterType',
|
||||
'verbose_name_plural': 'ClusterTypes',
|
||||
'verbose_name': 'Cluster Type',
|
||||
'verbose_name_plural': 'Cluster Types',
|
||||
'ordering': ['name'],
|
||||
},
|
||||
),
|
||||
|
@ -0,0 +1,25 @@
|
||||
# Generated by Django 5.0.7 on 2024-08-18 03:57
|
||||
|
||||
import access.fields
|
||||
import django.utils.timezone
|
||||
from django.db import migrations
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('itim', '0001_initial'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='clustertype',
|
||||
name='created',
|
||||
field=access.fields.AutoCreatedField(default=django.utils.timezone.now, editable=False),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='clustertype',
|
||||
name='modified',
|
||||
field=access.fields.AutoLastModifiedField(default=django.utils.timezone.now, editable=False),
|
||||
),
|
||||
]
|
@ -18,9 +18,9 @@ class ClusterType(TenancyObject):
|
||||
'name',
|
||||
]
|
||||
|
||||
verbose_name = "ClusterType"
|
||||
verbose_name = "Cluster Type"
|
||||
|
||||
verbose_name_plural = "ClusterTypes"
|
||||
verbose_name_plural = "Cluster Types"
|
||||
|
||||
|
||||
id = models.AutoField(
|
||||
@ -39,6 +39,15 @@ class ClusterType(TenancyObject):
|
||||
|
||||
slug = AutoSlugField()
|
||||
|
||||
created = AutoCreatedField()
|
||||
|
||||
modified = AutoLastModifiedField()
|
||||
|
||||
|
||||
def __str__(self):
|
||||
|
||||
return self.name
|
||||
|
||||
|
||||
|
||||
class Cluster(TenancyObject):
|
||||
|
22
app/itim/templates/itim/cluster_type.html.j2
Normal file
22
app/itim/templates/itim/cluster_type.html.j2
Normal file
@ -0,0 +1,22 @@
|
||||
{% extends 'detail.html.j2' %}
|
||||
|
||||
{% load json %}
|
||||
{% load markdown %}
|
||||
|
||||
|
||||
{% block tabs %}
|
||||
|
||||
<div id="details" class="content-tab">
|
||||
|
||||
{% include 'content/section.html.j2' with tab=form.tabs.details %}
|
||||
|
||||
</div>
|
||||
|
||||
|
||||
<div id="notes" class="content-tab">
|
||||
|
||||
{% include 'content/section.html.j2' with tab=form.tabs.notes %}
|
||||
|
||||
</div>
|
||||
|
||||
{% endblock %}
|
47
app/itim/templates/itim/cluster_type_index.html.j2
Normal file
47
app/itim/templates/itim/cluster_type_index.html.j2
Normal file
@ -0,0 +1,47 @@
|
||||
{% extends 'base.html.j2' %}
|
||||
|
||||
{% block content %}
|
||||
|
||||
<input type="button" value="New Cluster Type" onclick="window.location='{% url 'Settings:_cluster_type_add' %}';">
|
||||
<table class="data">
|
||||
<tr>
|
||||
<th>Title</th>
|
||||
<th> </th>
|
||||
<th>Organization</th>
|
||||
<th> </th>
|
||||
</tr>
|
||||
{% if items %}
|
||||
{% for item in items %}
|
||||
<tr>
|
||||
<td><a href="{% url 'Settings:_cluster_type_view' pk=item.id %}">{{ item.name }}</a></td>
|
||||
<td> </td>
|
||||
<td>{{ item.organization }}</td>
|
||||
<td> </td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
{% else %}
|
||||
<tr>
|
||||
<td colspan="4">Nothing Found</td>
|
||||
</tr>
|
||||
{% endif %}
|
||||
</table>
|
||||
<br>
|
||||
<div class="pagination">
|
||||
<span class="step-links">
|
||||
{% if page_obj.has_previous %}
|
||||
<a href="?page=1">« first</a>
|
||||
<a href="?page={{ page_obj.previous_page_number }}">previous</a>
|
||||
{% endif %}
|
||||
|
||||
<span class="current">
|
||||
Page {{ page_obj.number }} of {{ page_obj.paginator.num_pages }}.
|
||||
</span>
|
||||
|
||||
{% if page_obj.has_next %}
|
||||
<a href="?page={{ page_obj.next_page_number }}">next</a>
|
||||
<a href="?page={{ page_obj.paginator.num_pages }}">last »</a>
|
||||
{% endif %}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
{% endblock %}
|
185
app/itim/views/cluster_types.py
Normal file
185
app/itim/views/cluster_types.py
Normal file
@ -0,0 +1,185 @@
|
||||
from django.contrib.auth import decorators as auth_decorator
|
||||
from django.urls import reverse
|
||||
from django.utils.decorators import method_decorator
|
||||
|
||||
from core.forms.comment import AddNoteForm
|
||||
from core.models.notes import Notes
|
||||
from core.views.common import AddView, ChangeView, DeleteView, IndexView
|
||||
|
||||
from itim.forms.cluster_type import ClusterTypeForm, DetailForm
|
||||
from itim.models.clusters import ClusterType
|
||||
|
||||
from settings.models.user_settings import UserSettings
|
||||
|
||||
|
||||
|
||||
class Add(AddView):
|
||||
|
||||
form_class = ClusterTypeForm
|
||||
|
||||
model = ClusterType
|
||||
|
||||
permission_required = [
|
||||
'itim.add_clustertype',
|
||||
]
|
||||
|
||||
|
||||
def get_initial(self):
|
||||
|
||||
initial: dict = {
|
||||
'organization': UserSettings.objects.get(user = self.request.user).default_organization
|
||||
}
|
||||
|
||||
if 'pk' in self.kwargs:
|
||||
|
||||
if self.kwargs['pk']:
|
||||
|
||||
initial.update({'parent': self.kwargs['pk']})
|
||||
|
||||
self.model.parent.field.hidden = True
|
||||
|
||||
return initial
|
||||
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
|
||||
return reverse('Settings:_cluster_type_types')
|
||||
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
context = super().get_context_data(**kwargs)
|
||||
|
||||
context['content_title'] = 'New Cluster Type'
|
||||
|
||||
return context
|
||||
|
||||
|
||||
|
||||
class Change(ChangeView):
|
||||
|
||||
form_class = ClusterTypeForm
|
||||
|
||||
model = ClusterType
|
||||
|
||||
permission_required = [
|
||||
'itim.change_clustertype',
|
||||
]
|
||||
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
|
||||
context = super().get_context_data(**kwargs)
|
||||
|
||||
context['content_title'] = str(self.object)
|
||||
|
||||
return context
|
||||
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
|
||||
return reverse('Settings:_cluster_type_view', args=(self.kwargs['pk'],))
|
||||
|
||||
|
||||
|
||||
class Delete(DeleteView):
|
||||
|
||||
model = ClusterType
|
||||
|
||||
permission_required = [
|
||||
'itim.delete_clustertype',
|
||||
]
|
||||
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
|
||||
context = super().get_context_data(**kwargs)
|
||||
|
||||
context['content_title'] = 'Delete ' + str(self.object)
|
||||
|
||||
return context
|
||||
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
|
||||
return reverse('Settings:_cluster_types')
|
||||
|
||||
|
||||
|
||||
class Index(IndexView):
|
||||
|
||||
context_object_name = "items"
|
||||
|
||||
model = ClusterType
|
||||
|
||||
paginate_by = 10
|
||||
|
||||
permission_required = [
|
||||
'itim.view_clustertype'
|
||||
]
|
||||
|
||||
template_name = 'itim/cluster_type_index.html.j2'
|
||||
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
|
||||
context = super().get_context_data(**kwargs)
|
||||
|
||||
context['model_docs_path'] = self.model._meta.app_label + '/' + self.model._meta.model_name
|
||||
|
||||
context['content_title'] = 'Cluster Types'
|
||||
|
||||
return context
|
||||
|
||||
|
||||
|
||||
class View(ChangeView):
|
||||
|
||||
context_object_name = "item"
|
||||
|
||||
form_class = DetailForm
|
||||
|
||||
model = ClusterType
|
||||
|
||||
permission_required = [
|
||||
'itim.view_clustertype',
|
||||
]
|
||||
|
||||
template_name = 'itim/cluster_type.html.j2'
|
||||
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
|
||||
context = super().get_context_data(**kwargs)
|
||||
|
||||
context['notes_form'] = AddNoteForm(prefix='note')
|
||||
context['notes'] = Notes.objects.filter(service=self.kwargs['pk'])
|
||||
|
||||
context['model_pk'] = self.kwargs['pk']
|
||||
context['model_name'] = self.model._meta.model_name
|
||||
|
||||
context['model_delete_url'] = reverse('Settings:_cluster_type_delete', args=(self.kwargs['pk'],))
|
||||
|
||||
|
||||
context['content_title'] = self.object.name
|
||||
|
||||
return context
|
||||
|
||||
|
||||
def post(self, request, *args, **kwargs):
|
||||
|
||||
item = ClusterType.objects.get(pk=self.kwargs['pk'])
|
||||
|
||||
notes = AddNoteForm(request.POST, prefix='note')
|
||||
|
||||
if notes.is_bound and notes.is_valid() and notes.instance.note != '':
|
||||
|
||||
notes.instance.service = item
|
||||
|
||||
notes.instance.organization = item.organization
|
||||
|
||||
notes.save()
|
||||
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
|
||||
return reverse('Settings:_cluster_type_view', args=(self.kwargs['pk'],))
|
@ -62,6 +62,7 @@ div#content article h3 {
|
||||
<article style="">
|
||||
<h3>ITIM</h3>
|
||||
<ul>
|
||||
<li><a href="{% url 'Settings:_cluster_types' %}">Cluster Types</a></li>
|
||||
<li><a href="{% url 'Settings:_ports' %}">Service Ports</a></li>
|
||||
</ul>
|
||||
</article>
|
||||
|
@ -8,24 +8,20 @@ from settings.views import app_settings, home, device_models, device_types, exte
|
||||
|
||||
from itam.views import device_type, device_model, software_category
|
||||
|
||||
from itim.views import ports
|
||||
from itim.views import cluster_types, ports
|
||||
|
||||
app_name = "Settings"
|
||||
urlpatterns = [
|
||||
|
||||
path("", home.View.as_view(), name="Settings"),
|
||||
|
||||
path("external_links", external_link.Index.as_view(), name="External Links"),
|
||||
path("external_links/add", external_link.Add.as_view(), name="_external_link_add"),
|
||||
path("external_links/<int:pk>", external_link.View.as_view(), name="_external_link_view"),
|
||||
path("external_links/<int:pk>/edit", external_link.Change.as_view(), name="_external_link_change"),
|
||||
path("external_links/<int:pk>/delete", external_link.Delete.as_view(), name="_external_link_delete"),
|
||||
|
||||
|
||||
path('application', app_settings.View.as_view(), name="_settings_application"),
|
||||
|
||||
path("task_results", celery_log.Index.as_view(), name="_task_results"),
|
||||
path("task_result/<int:pk>", celery_log.View.as_view(), name="_task_result_view"),
|
||||
path("cluster_types", cluster_types.Index.as_view(), name="_cluster_types"),
|
||||
path("cluster_types/add", cluster_types.Add.as_view(), name="_cluster_type_add"),
|
||||
path("cluster_types/<int:pk>/edit", cluster_types.Change.as_view(), name="_cluster_type_change"),
|
||||
path("cluster_types/<int:pk>/delete", cluster_types.Delete.as_view(), name="_cluster_type_delete"),
|
||||
path("cluster_types/<int:pk>", cluster_types.View.as_view(), name="_cluster_type_view"),
|
||||
|
||||
path("device_models", device_models.Index.as_view(), name="_device_models"),
|
||||
path("device_model/<int:pk>", device_model.View.as_view(), name="_device_model_view"),
|
||||
@ -39,18 +35,18 @@ urlpatterns = [
|
||||
path("device_type/<int:pk>/delete", device_type.Delete.as_view(), name="_device_type_delete"),
|
||||
path("device_type/<int:pk>/edit", device_type.Change.as_view(), name="_device_type_change"),
|
||||
|
||||
path("external_links", external_link.Index.as_view(), name="External Links"),
|
||||
path("external_links/add", external_link.Add.as_view(), name="_external_link_add"),
|
||||
path("external_links/<int:pk>", external_link.View.as_view(), name="_external_link_view"),
|
||||
path("external_links/<int:pk>/edit", external_link.Change.as_view(), name="_external_link_change"),
|
||||
path("external_links/<int:pk>/delete", external_link.Delete.as_view(), name="_external_link_delete"),
|
||||
|
||||
path("kb/category", knowledge_base_category.Index.as_view(), name="KB Categories"),
|
||||
path("kb/category/add", knowledge_base_category.Add.as_view(), name="_knowledge_base_category_add"),
|
||||
path("kb/category/<int:pk>/edit", knowledge_base_category.Change.as_view(), name="_knowledge_base_category_change"),
|
||||
path("kb/category/<int:pk>/delete", knowledge_base_category.Delete.as_view(), name="_knowledge_base_category_delete"),
|
||||
path("kb/category/<int:pk>", knowledge_base_category.View.as_view(), name="_knowledge_base_category_view"),
|
||||
|
||||
path("software_category", software_categories.Index.as_view(), name="_software_categories"),
|
||||
path("software_category/<int:pk>", software_category.View.as_view(), name="_software_category_view"),
|
||||
path("software_category/add/", software_category.Add.as_view(), name="_software_category_add"),
|
||||
path("software_category/<int:pk>/edit", software_category.Change.as_view(), name="_software_category_change"),
|
||||
path("software_category/<int:pk>/delete", software_category.Delete.as_view(), name="_software_category_delete"),
|
||||
|
||||
path("manufacturers", manufacturer.Index.as_view(), name="_manufacturers"),
|
||||
path("manufacturer/<int:pk>", manufacturer.View.as_view(), name="_manufacturer_view"),
|
||||
path("manufacturer/add/", manufacturer.Add.as_view(), name="_manufacturer_add"),
|
||||
@ -63,4 +59,13 @@ urlpatterns = [
|
||||
path("port/<int:pk>/delete", ports.Delete.as_view(), name="_port_delete"),
|
||||
path("port/<int:pk>", ports.View.as_view(), name="_port_view"),
|
||||
|
||||
path("software_category", software_categories.Index.as_view(), name="_software_categories"),
|
||||
path("software_category/<int:pk>", software_category.View.as_view(), name="_software_category_view"),
|
||||
path("software_category/add/", software_category.Add.as_view(), name="_software_category_add"),
|
||||
path("software_category/<int:pk>/edit", software_category.Change.as_view(), name="_software_category_change"),
|
||||
path("software_category/<int:pk>/delete", software_category.Delete.as_view(), name="_software_category_delete"),
|
||||
|
||||
path("task_results", celery_log.Index.as_view(), name="_task_results"),
|
||||
path("task_result/<int:pk>", celery_log.View.as_view(), name="_task_result_view"),
|
||||
|
||||
]
|
||||
|
Reference in New Issue
Block a user