-
-
Notifications
You must be signed in to change notification settings - Fork 98
Expand file tree
/
Copy pathadmin.py
More file actions
231 lines (178 loc) · 7.55 KB
/
admin.py
File metadata and controls
231 lines (178 loc) · 7.55 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
from django.contrib.admin import ModelAdmin, StackedInline
from django.core.exceptions import FieldError
from django.urls import reverse
from django.utils.translation import gettext_lazy as _
from openwisp_utils.widgets import Select2Widget
class TimeReadonlyAdminMixin(object):
"""A mixin that automatically flags `created` and `modified` as readonly."""
def __init__(self, *args, **kwargs):
self.readonly_fields += ("created", "modified")
super().__init__(*args, **kwargs)
class ReadOnlyAdmin(ModelAdmin):
"""Disables all editing capabilities."""
exclude = tuple()
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
exclude = self.exclude
self.readonly_fields = [
f.name for f in self.model._meta.fields if f.name not in exclude
]
def get_actions(self, request):
actions = super().get_actions(request)
if "delete_selected" in actions: # pragma: no cover
del actions["delete_selected"]
return actions
def has_add_permission(self, request):
return False
def has_delete_permission(self, request, obj=None):
return False
def save_model(self, request, obj, form, change): # pragma: nocover
pass
def delete_model(self, request, obj): # pragma: nocover
pass
def save_related(self, request, form, formsets, change): # pragma: nocover
pass
def change_view(self, request, object_id, extra_context=None):
extra_context = extra_context or {}
extra_context["show_save_and_continue"] = False
extra_context["show_save"] = False
return super().change_view(request, object_id, extra_context=extra_context)
class AlwaysHasChangedMixin(object):
def has_changed(self):
"""Returns true for new objects.
This django-admin trick ensures the inline item is saved even if
default values are unchanged (without this trick new objects won't
be created unless users change the default values).
"""
if self.instance._state.adding:
return True
return super().has_changed()
class CopyableFieldError(FieldError):
pass
class CopyableFieldsAdmin(ModelAdmin):
"""Allows to set fields as read-only and easy to copy.
Useful for auto-generated fields such as UUIDs, secret keys, tokens,
etc.
"""
copyable_fields = ()
change_form_template = "admin/change_form.html"
def _check_copyable_subset_fields(self, copyable_fields, fields):
if not set(copyable_fields).issubset(fields):
class_name = self.__class__.__name__
raise CopyableFieldError(
(
f"{copyable_fields} not in {class_name}.fields {fields}, "
f"Check copyable_fields attribute of class {class_name}."
)
)
def get_fields(self, request, obj=None):
fields = super(ModelAdmin, self).get_fields(request, obj)
self._check_copyable_subset_fields(self.copyable_fields, fields)
# We should exclude `copyable_fields` fields
# when the object doesn't exist for example,
# in the case of `add_view` because `copyable_fields`
# can't be edited and are auto generated by the system
if not obj:
# Make sure field order remains the same when excluding `copyable_fields`
return [field for field in fields if field not in self.copyable_fields]
return fields
def get_readonly_fields(self, request, obj=None):
readonly_fields = super(ModelAdmin, self).get_readonly_fields(request, obj)
if not obj:
return readonly_fields
# If any `copyable_fields` are not included
# in the `read-only` fields, then include them
if not any(field in readonly_fields for field in self.copyable_fields):
return tuple([*readonly_fields, *self.copyable_fields])
return readonly_fields
def add_view(self, request, form_url="", extra_context=None):
extra_context = extra_context or {}
extra_context["copyable_fields"] = []
return super().add_view(
request,
form_url,
extra_context=extra_context,
)
def change_view(self, request, object_id, form_url="", extra_context=None):
extra_context = extra_context or {}
extra_context["copyable_fields"] = list(self.copyable_fields)
return super().change_view(
request,
object_id,
form_url,
extra_context=extra_context,
)
class Media:
js = ("admin/js/jquery.init.js", "openwisp-utils/js/copyable.js")
class UUIDAdmin(CopyableFieldsAdmin):
"""Sets `uuid` as copyable field.
Subclass of `CopyableFieldsAdmin`. This class is kept for backward
compatibility and convenience, since different models of various
OpenWISP modules show `uuid` as the only copyable field.
"""
copyable_fields = ("uuid",)
def uuid(self, obj):
return obj.pk
uuid.short_description = _("UUID")
class ReceiveUrlAdmin(ModelAdmin):
"""Adds a receive_url field.
The receive_url method will build the URL using the parameters:
- receive_url_name
- receive_url_object_arg
- receive_url_object_arg
"""
receive_url_querystring_arg = "key"
receive_url_object_arg = "pk"
receive_url_name = None
receive_url_urlconf = None
receive_url_baseurl = None
def add_view(self, request, *args, **kwargs):
self.request = request
return super().add_view(request, *args, **kwargs)
def change_view(self, request, *args, **kwargs):
self.request = request
return super().change_view(request, *args, **kwargs)
def receive_url(self, obj):
""":param obj: Object for which the url is generated"""
if self.receive_url_name is None:
raise ValueError("receive_url_name is not set up")
reverse_kwargs = {}
if self.receive_url_object_arg:
reverse_kwargs = {
self.receive_url_object_arg: getattr(obj, self.receive_url_object_arg)
}
receive_path = reverse(
self.receive_url_name,
urlconf=self.receive_url_urlconf,
kwargs=reverse_kwargs,
)
baseurl = self.receive_url_baseurl
if not baseurl:
baseurl = "{0}://{1}".format(self.request.scheme, self.request.get_host())
if self.receive_url_querystring_arg:
url = "{0}{1}?{2}={3}".format(
baseurl,
receive_path,
self.receive_url_querystring_arg,
getattr(obj, self.receive_url_querystring_arg),
)
return url
class Media:
js = ("admin/js/jquery.init.js", "openwisp-utils/js/receive_url.js")
receive_url.short_description = _("URL")
class HelpTextStackedInline(StackedInline):
help_text = None
template = "admin/edit_inline/help_text_stacked.html"
class Media:
css = {"all": ["admin/css/help-text-stacked.css"]}
def get_formset(self, request, obj=None, **kwargs):
formset = super().get_formset(request, obj, **kwargs)
formset.help_text = self.help_text
return formset
class Select2AdminMixin:
"""Mixin that applies Select2Widget to specified choice fields."""
select2_fields = ()
def formfield_for_choice_field(self, db_field, request, **kwargs):
if db_field.name in self.select2_fields:
kwargs["widget"] = Select2Widget()
return super().formfield_for_choice_field(db_field, request, **kwargs)