blob: 73296e092f73b7b5a2f11004e3c76eaaa02e8bf1 [file] [log] [blame]
Siobhan Tully30fd4292013-05-10 08:59:56 -04001from core.models import Site
2from core.models import *
3from openstack.manager import OpenStackManager
Tony Macke59a7c82013-04-27 11:08:10 -04004
Tony Mack7130ac32013-03-22 21:58:00 -04005from django.contrib import admin
Siobhan Tully53437282013-04-26 19:30:27 -04006from django.contrib.auth.models import Group
Siobhan Tully4bc09f22013-04-10 21:15:21 -04007from django import forms
Tony Mackd90cdbf2013-04-16 22:48:40 -04008from django.utils.safestring import mark_safe
Tony Mack7130ac32013-03-22 21:58:00 -04009from django.contrib.auth.admin import UserAdmin
Scott Baker9f6b8ed2014-11-17 23:44:03 -080010from django.contrib.admin.widgets import FilteredSelectMultiple, AdminTextareaWidget
Scott Bakercbfb6002014-10-03 00:32:37 -070011from django.contrib.auth.forms import ReadOnlyPasswordHashField, AdminPasswordChangeForm
Scott Bakeracd45142013-05-19 16:19:16 -070012from django.contrib.auth.signals import user_logged_in
13from django.utils import timezone
Siobhan Tullyde5450d2013-06-21 11:35:33 -040014from django.contrib.contenttypes import generic
Siobhan Tullybfd11dc2013-09-03 12:59:24 -040015from suit.widgets import LinkedSelect
Siobhan Tullycf04fb62014-01-11 11:25:57 -050016from django.core.exceptions import PermissionDenied
Scott Bakere2bbf7e2014-01-13 12:09:31 -080017from django.core.urlresolvers import reverse, NoReverseMatch
Scott Baker9f6b8ed2014-11-17 23:44:03 -080018from django.utils.encoding import force_text, python_2_unicode_compatible
19from django.utils.html import conditional_escape, format_html
20from django.forms.utils import flatatt, to_current_timezone
Scott Baker3cde7372014-10-21 21:03:08 -070021from cgi import escape as html_escape
Tony Mack7130ac32013-03-22 21:58:00 -040022
Scott Baker36f50872014-08-21 13:01:25 -070023import django_evolution
Scott Baker0a5633b2014-10-06 17:51:20 -070024import threading
25
26# thread locals necessary to work around a django-suit issue
27_thread_locals = threading.local()
Scott Baker36f50872014-08-21 13:01:25 -070028
Scott Baker40c00762014-08-21 16:55:59 -070029def backend_icon(obj): # backend_status, enacted, updated):
30 #return "%s %s %s" % (str(obj.updated), str(obj.enacted), str(obj.backend_status))
Sapan Bhatia42a291f2015-01-27 03:49:42 +000031 if (obj.enacted is not None) and obj.enacted >= obj.updated or obj.backend_status.startswith("1 -"):
Scott Bakerfbf06642014-09-09 10:38:15 -070032 return '<span style="min-width:16px;"><img src="/static/admin/img/icon_success.gif"></span>'
Scott Baker40c00762014-08-21 16:55:59 -070033 else:
Scott Baker5ff796b2015-01-22 15:14:50 -080034 if ((obj.backend_status is not None) and obj.backend_status.startswith("0 -")) or obj.backend_status == "Provisioning in progress" or obj.backend_status=="":
Scott Bakerfbf06642014-09-09 10:38:15 -070035 return '<span style="min-width:16px;" title="%s"><img src="/static/admin/img/icon_clock.gif"></span>' % obj.backend_status
Scott Baker63d1a552014-08-21 15:19:07 -070036 else:
Scott Baker3cde7372014-10-21 21:03:08 -070037 return '<span style="min-width:16px;" title="%s"><img src="/static/admin/img/icon_error.gif"></span>' % html_escape(obj.backend_status, quote=True)
Scott Baker40c00762014-08-21 16:55:59 -070038
39def backend_text(obj):
40 icon = backend_icon(obj)
41 if (obj.enacted is not None) and obj.enacted >= obj.updated:
Scott Baker3cde7372014-10-21 21:03:08 -070042 return "%s %s" % (icon, "successfully enacted")
Scott Baker40c00762014-08-21 16:55:59 -070043 else:
Scott Baker3cde7372014-10-21 21:03:08 -070044 return "%s %s" % (icon, html_escape(obj.backend_status, quote=True))
Scott Baker63d1a552014-08-21 15:19:07 -070045
Scott Baker9f6b8ed2014-11-17 23:44:03 -080046class UploadTextareaWidget(AdminTextareaWidget):
47 def render(self, name, value, attrs=None):
48 if value is None:
49 value = ''
50 final_attrs = self.build_attrs(attrs, name=name)
51 return format_html('<input type="file" style="width: 0; height: 0" id="btn_upload_%s" onChange="uploadTextarea(event,\'%s\');">' \
52 '<button onClick="$(\'#btn_upload_%s\').click(); return false;">Upload</button>' \
53 '<br><textarea{0}>\r\n{1}</textarea>' % (attrs["id"], attrs["id"], attrs["id"]),
54 flatatt(final_attrs),
55 force_text(value))
56
Scott Baker36f50872014-08-21 13:01:25 -070057class PlainTextWidget(forms.HiddenInput):
58 input_type = 'hidden'
59
60 def render(self, name, value, attrs=None):
61 if value is None:
62 value = ''
63 return mark_safe(str(value) + super(PlainTextWidget, self).render(name, value, attrs))
64
Scott Baker86c83ab2014-10-03 13:10:47 -070065class PermissionCheckingAdminMixin(object):
Scott Bakercbfb6002014-10-03 00:32:37 -070066 # call save_by_user and delete_by_user instead of save and delete
Siobhan Tullycf04fb62014-01-11 11:25:57 -050067
68 def has_add_permission(self, request, obj=None):
69 return (not self.__user_is_readonly(request))
Scott Baker36f50872014-08-21 13:01:25 -070070
Siobhan Tullycf04fb62014-01-11 11:25:57 -050071 def has_delete_permission(self, request, obj=None):
72 return (not self.__user_is_readonly(request))
73
74 def save_model(self, request, obj, form, change):
75 if self.__user_is_readonly(request):
Scott Bakercbfb6002014-10-03 00:32:37 -070076 # this 'if' might be redundant if save_by_user is implemented right
Siobhan Tullycf04fb62014-01-11 11:25:57 -050077 raise PermissionDenied
Scott Bakercbfb6002014-10-03 00:32:37 -070078
79 obj.caller = request.user
80 # update openstack connection to use this site/tenant
81 obj.save_by_user(request.user)
82
83 def delete_model(self, request, obj):
84 obj.delete_by_user(request.user)
85
86 def save_formset(self, request, form, formset, change):
87 instances = formset.save(commit=False)
88 for instance in instances:
89 instance.save_by_user(request.user)
90
91 # BUG in django 1.7? Objects are not deleted by formset.save if
92 # commit is False. So let's delete them ourselves.
93 #
94 # code from forms/models.py save_existing_objects()
95 try:
96 forms_to_delete = formset.deleted_forms
97 except AttributeError:
98 forms_to_delete = []
99 if formset.initial_forms:
100 for form in formset.initial_forms:
101 obj = form.instance
102 if form in forms_to_delete:
103 if obj.pk is None:
104 continue
105 formset.deleted_objects.append(obj)
106 obj.delete()
107
108 formset.save_m2m()
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500109
110 def get_actions(self,request):
Scott Baker86c83ab2014-10-03 13:10:47 -0700111 actions = super(PermissionCheckingAdminMixin,self).get_actions(request)
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500112
113 if self.__user_is_readonly(request):
114 if 'delete_selected' in actions:
115 del actions['delete_selected']
116
117 return actions
118
119 def change_view(self,request,object_id, extra_context=None):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500120 if self.__user_is_readonly(request):
Scott Bakeraf73e102014-04-22 22:40:07 -0700121 if not hasattr(self, "readonly_save"):
122 # save the original readonly fields
123 self.readonly_save = self.readonly_fields
124 self.inlines_save = self.inlines
Scott Bakere8859f92014-05-23 12:42:40 -0700125 if hasattr(self, "user_readonly_fields"):
126 self.readonly_fields=self.user_readonly_fields
127 if hasattr(self, "user_readonly_inlines"):
128 self.inlines = self.user_readonly_inlines
Scott Bakeraf73e102014-04-22 22:40:07 -0700129 else:
130 if hasattr(self, "readonly_save"):
131 # restore the original readonly fields
132 self.readonly_fields = self.readonly_save
Scott Bakere8859f92014-05-23 12:42:40 -0700133 if hasattr(self, "inlines_save"):
Scott Bakeraf73e102014-04-22 22:40:07 -0700134 self.inlines = self.inlines_save
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500135
136 try:
Scott Baker86c83ab2014-10-03 13:10:47 -0700137 return super(PermissionCheckingAdminMixin, self).change_view(request, object_id, extra_context=extra_context)
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500138 except PermissionDenied:
139 pass
140 if request.method == 'POST':
141 raise PermissionDenied
142 request.readonly = True
Scott Baker86c83ab2014-10-03 13:10:47 -0700143 return super(PermissionCheckingAdminMixin, self).change_view(request, object_id, extra_context=extra_context)
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500144
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500145 def __user_is_readonly(self, request):
146 return request.user.isReadOnlyUser()
147
Scott Baker40c00762014-08-21 16:55:59 -0700148 def backend_status_text(self, obj):
149 return mark_safe(backend_text(obj))
Scott Baker36f50872014-08-21 13:01:25 -0700150
Scott Baker63d1a552014-08-21 15:19:07 -0700151 def backend_status_icon(self, obj):
Scott Baker40c00762014-08-21 16:55:59 -0700152 return mark_safe(backend_icon(obj))
Scott Baker63d1a552014-08-21 15:19:07 -0700153 backend_status_icon.short_description = ""
154
Scott Baker24ded6a2014-11-05 09:05:38 -0800155 def get_form(self, request, obj=None, **kwargs):
Scott Baker5c432692014-10-16 00:57:55 -0700156 # Save obj and request in thread-local storage, so suit_form_tabs can
157 # use it to determine whether we're in edit or add mode, and can
158 # determine whether the user is an admin.
159 _thread_locals.request = request
160 _thread_locals.obj = obj
Scott Baker24ded6a2014-11-05 09:05:38 -0800161 return super(PermissionCheckingAdminMixin, self).get_form(request, obj, **kwargs)
Scott Baker5c432692014-10-16 00:57:55 -0700162
163 def get_inline_instances(self, request, obj=None):
164 inlines = super(PermissionCheckingAdminMixin, self).get_inline_instances(request, obj)
165
166 # inlines that should only be shown to an admin user
167 if request.user.is_admin:
168 for inline_class in getattr(self, "admin_inlines", []):
169 inlines.append(inline_class(self.model, self.admin_site))
170
171 return inlines
172
Scott Baker86c83ab2014-10-03 13:10:47 -0700173class ReadOnlyAwareAdmin(PermissionCheckingAdminMixin, admin.ModelAdmin):
174 # Note: Make sure PermissionCheckingAdminMixin is listed before
175 # admin.ModelAdmin in the class declaration.
176
Scott Bakercbfb6002014-10-03 00:32:37 -0700177 pass
178
179class PlanetStackBaseAdmin(ReadOnlyAwareAdmin):
180 save_on_top = False
Scott Baker36f50872014-08-21 13:01:25 -0700181
Scott Bakere8859f92014-05-23 12:42:40 -0700182class SingletonAdmin (ReadOnlyAwareAdmin):
Siobhan Tullyce652d02013-10-08 21:52:35 -0400183 def has_add_permission(self, request):
Scott Bakere8859f92014-05-23 12:42:40 -0700184 if not super(SingletonAdmin, self).has_add_permission(request):
185 return False
186
Siobhan Tullyce652d02013-10-08 21:52:35 -0400187 num_objects = self.model.objects.count()
188 if num_objects >= 1:
189 return False
190 else:
191 return True
192
Siobhan Tullyd3515752013-06-21 16:34:53 -0400193class PlStackTabularInline(admin.TabularInline):
Scott Baker86568322014-01-12 16:53:31 -0800194 def __init__(self, *args, **kwargs):
195 super(PlStackTabularInline, self).__init__(*args, **kwargs)
196
197 # InlineModelAdmin as no get_fields() method, so in order to add
198 # the selflink field, we override __init__ to modify self.fields and
199 # self.readonly_fields.
200
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800201 self.setup_selflink()
202
Scott Baker874936e2014-01-13 18:15:34 -0800203 def get_change_url(self, model, id):
204 """ Get the URL to a change form in the admin for this model """
205 reverse_path = "admin:%s_change" % (model._meta.db_table)
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800206 try:
Scott Baker874936e2014-01-13 18:15:34 -0800207 url = reverse(reverse_path, args=(id,))
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800208 except NoReverseMatch:
Scott Baker874936e2014-01-13 18:15:34 -0800209 return None
210
211 return url
212
213 def setup_selflink(self):
214 if hasattr(self, "selflink_fieldname"):
215 """ self.selflink_model can be defined to punch through a relation
216 to its target object. For example, in SliceNetworkInline, set
217 selflink_model = "network", and the URL will lead to the Network
218 object instead of trying to bring up a change view of the
219 SliceNetwork object.
220 """
221 self.selflink_model = getattr(self.model,self.selflink_fieldname).field.rel.to
222 else:
223 self.selflink_model = self.model
224
225 url = self.get_change_url(self.selflink_model, 0)
226
227 # We don't have an admin for this object, so don't create the
228 # selflink.
229 if (url == None):
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800230 return
231
Scott Baker874936e2014-01-13 18:15:34 -0800232 # Since we need to add "selflink" to the field list, we need to create
233 # self.fields if it is None.
Scott Baker0165fac2014-01-13 11:49:26 -0800234 if (self.fields is None):
235 self.fields = []
236 for f in self.model._meta.fields:
237 if f.editable and f.name != "id":
238 self.fields.append(f.name)
Scott Baker86568322014-01-12 16:53:31 -0800239
Scott Baker874936e2014-01-13 18:15:34 -0800240 self.fields = tuple(self.fields) + ("selflink", )
Scott Baker86568322014-01-12 16:53:31 -0800241
Scott Baker874936e2014-01-13 18:15:34 -0800242 if self.readonly_fields is None:
243 self.readonly_fields = ()
Scott Baker86568322014-01-12 16:53:31 -0800244
Scott Baker874936e2014-01-13 18:15:34 -0800245 self.readonly_fields = tuple(self.readonly_fields) + ("selflink", )
Scott Baker86568322014-01-12 16:53:31 -0800246
247 def selflink(self, obj):
Scott Baker874936e2014-01-13 18:15:34 -0800248 if hasattr(self, "selflink_fieldname"):
249 obj = getattr(obj, self.selflink_fieldname)
250
Scott Baker86568322014-01-12 16:53:31 -0800251 if obj.id:
Scott Baker874936e2014-01-13 18:15:34 -0800252 url = self.get_change_url(self.selflink_model, obj.id)
253 return "<a href='%s'>Details</a>" % str(url)
Scott Baker86568322014-01-12 16:53:31 -0800254 else:
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800255 return "Not present"
Scott Baker86568322014-01-12 16:53:31 -0800256
257 selflink.allow_tags = True
258 selflink.short_description = "Details"
Siobhan Tullyd3515752013-06-21 16:34:53 -0400259
Scott Bakerb27b62c2014-08-15 16:29:16 -0700260 def has_add_permission(self, request):
261 return not request.user.isReadOnlyUser()
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500262
263 def get_readonly_fields(self, request, obj=None):
Scott Bakerb27b62c2014-08-15 16:29:16 -0700264 readonly_fields = list(self.readonly_fields)[:]
265 if request.user.isReadOnlyUser():
266 for field in self.fields:
267 if not field in readonly_fields:
268 readonly_fields.append(field)
269 return readonly_fields
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500270
Scott Baker40c00762014-08-21 16:55:59 -0700271 def backend_status_icon(self, obj):
272 return mark_safe(backend_icon(obj))
273 backend_status_icon.short_description = ""
Scott Baker36f50872014-08-21 13:01:25 -0700274
Scott Bakerb27b62c2014-08-15 16:29:16 -0700275class PlStackGenericTabularInline(generic.GenericTabularInline):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500276 def has_add_permission(self, request):
Scott Bakerb27b62c2014-08-15 16:29:16 -0700277 return not request.user.isReadOnlyUser()
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500278
Scott Bakerb27b62c2014-08-15 16:29:16 -0700279 def get_readonly_fields(self, request, obj=None):
280 readonly_fields = list(self.readonly_fields)[:]
281 if request.user.isReadOnlyUser():
282 for field in self.fields:
283 if not field in readonly_fields:
284 readonly_fields.append(field)
285 return readonly_fields
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500286
Scott Baker40c00762014-08-21 16:55:59 -0700287 def backend_status_icon(self, obj):
288 return mark_safe(backend_icon(obj))
289 backend_status_icon.short_description = ""
290
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400291class ReservationInline(PlStackTabularInline):
292 model = Reservation
293 extra = 0
294 suit_classes = 'suit-tab suit-tab-reservations'
Scott Baker36f50872014-08-21 13:01:25 -0700295
Tony Mack5b061472014-02-04 07:57:10 -0500296 def queryset(self, request):
297 return Reservation.select_by_user(request.user)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400298
Scott Bakerb27b62c2014-08-15 16:29:16 -0700299class TagInline(PlStackGenericTabularInline):
Siobhan Tullyde5450d2013-06-21 11:35:33 -0400300 model = Tag
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400301 extra = 0
302 suit_classes = 'suit-tab suit-tab-tags'
Tony Mack5b061472014-02-04 07:57:10 -0500303 fields = ['service', 'name', 'value']
304
305 def queryset(self, request):
306 return Tag.select_by_user(request.user)
Siobhan Tullyde5450d2013-06-21 11:35:33 -0400307
Scott Baker74d8e622013-07-29 16:04:22 -0700308class NetworkLookerUpper:
Siobhan Tully2c780ad2013-09-06 11:22:40 -0400309 """ This is a callable that looks up a network name in a sliver and returns
310 the ip address for that network.
311 """
312
Scott Baker434ca7e2014-08-15 12:29:20 -0700313 byNetworkName = {} # class variable
314
Siobhan Tully2c780ad2013-09-06 11:22:40 -0400315 def __init__(self, name):
316 self.short_description = name
317 self.__name__ = name
318 self.network_name = name
319
320 def __call__(self, obj):
321 if obj is not None:
322 for nbs in obj.networksliver_set.all():
323 if (nbs.network.name == self.network_name):
324 return nbs.ip
Scott Baker74d8e622013-07-29 16:04:22 -0700325 return ""
326
327 def __str__(self):
328 return self.network_name
329
Scott Baker434ca7e2014-08-15 12:29:20 -0700330 @staticmethod
331 def get(network_name):
332 """ We want to make sure we alwars return the same NetworkLookerUpper
333 because sometimes django will cause them to be instantiated multiple
334 times (and we don't want different ones in form.fields vs
335 SliverInline.readonly_fields).
336 """
337 if network_name not in NetworkLookerUpper.byNetworkName:
338 NetworkLookerUpper.byNetworkName[network_name] = NetworkLookerUpper(network_name)
339 return NetworkLookerUpper.byNetworkName[network_name]
340
Siobhan Tullyd3515752013-06-21 16:34:53 -0400341class SliverInline(PlStackTabularInline):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400342 model = Sliver
Scott Baker887d4a82015-01-19 11:32:20 -0800343 fields = ['backend_status_icon', 'all_ips_string', 'instance_id', 'instance_name', 'slice', 'deployment', 'flavor', 'image', 'node']
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400344 extra = 0
Scott Baker887d4a82015-01-19 11:32:20 -0800345 readonly_fields = ['backend_status_icon', 'all_ips_string', 'instance_id', 'instance_name']
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400346 suit_classes = 'suit-tab suit-tab-slivers'
Scott Baker74d8e622013-07-29 16:04:22 -0700347
Tony Mack5b061472014-02-04 07:57:10 -0500348 def queryset(self, request):
349 return Sliver.select_by_user(request.user)
350
Scott Bakerb24cc932014-06-09 10:51:16 -0700351 def formfield_for_foreignkey(self, db_field, request=None, **kwargs):
Tony Mackbf6aa302014-12-26 13:38:02 -0500352 if db_field.name == 'deployment':
Scott Baker3b678742014-06-09 13:11:54 -0700353 kwargs['queryset'] = Deployment.select_by_acl(request.user)
Scott Baker7a61dc42014-09-02 17:08:20 -0700354 kwargs['widget'] = forms.Select(attrs={'onChange': "sliver_deployment_changed(this);"})
Tony Mackbf6aa302014-12-26 13:38:02 -0500355 if db_field.name == 'flavor':
Scott Baker32481312014-09-08 12:14:14 -0700356 kwargs['widget'] = forms.Select(attrs={'onChange': "sliver_flavor_changed(this);"})
Scott Baker3b678742014-06-09 13:11:54 -0700357
358 field = super(SliverInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
Scott Bakerb24cc932014-06-09 10:51:16 -0700359
360 return field
361
Siobhan Tullyd3515752013-06-21 16:34:53 -0400362class SiteInline(PlStackTabularInline):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400363 model = Site
364 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400365 suit_classes = 'suit-tab suit-tab-sites'
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400366
Tony Mack5b061472014-02-04 07:57:10 -0500367 def queryset(self, request):
368 return Site.select_by_user(request.user)
369
Siobhan Tullyd3515752013-06-21 16:34:53 -0400370class UserInline(PlStackTabularInline):
Siobhan Tully30fd4292013-05-10 08:59:56 -0400371 model = User
Scott Baker40c00762014-08-21 16:55:59 -0700372 fields = ['backend_status_icon', 'email', 'firstname', 'lastname']
373 readonly_fields = ('backend_status_icon', )
Siobhan Tully30fd4292013-05-10 08:59:56 -0400374 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400375 suit_classes = 'suit-tab suit-tab-users'
Siobhan Tully30fd4292013-05-10 08:59:56 -0400376
Tony Mack5b061472014-02-04 07:57:10 -0500377 def queryset(self, request):
378 return User.select_by_user(request.user)
379
Siobhan Tullyd3515752013-06-21 16:34:53 -0400380class SliceInline(PlStackTabularInline):
Tony Mack00d361f2013-04-28 10:28:42 -0400381 model = Slice
Scott Baker40c00762014-08-21 16:55:59 -0700382 fields = ['backend_status_icon', 'name', 'site', 'serviceClass', 'service']
383 readonly_fields = ('backend_status_icon', )
Tony Mack00d361f2013-04-28 10:28:42 -0400384 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400385 suit_classes = 'suit-tab suit-tab-slices'
386
Tony Mack5b061472014-02-04 07:57:10 -0500387 def queryset(self, request):
388 return Slice.select_by_user(request.user)
389
Siobhan Tullyd3515752013-06-21 16:34:53 -0400390class NodeInline(PlStackTabularInline):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400391 model = Node
392 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400393 suit_classes = 'suit-tab suit-tab-nodes'
Tony Mack93d1b032014-12-08 16:43:02 -0500394 fields = ['backend_status_icon', 'name', 'site_deployment']
Scott Baker40c00762014-08-21 16:55:59 -0700395 readonly_fields = ('backend_status_icon', )
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400396
Tony Mack93d1b032014-12-08 16:43:02 -0500397class DeploymentPrivilegeInline(PlStackTabularInline):
398 model = DeploymentPrivilege
399 extra = 0
400 suit_classes = 'suit-tab suit-tab-admin-only'
401 fields = ['backend_status_icon', 'user','role','deployment']
402 readonly_fields = ('backend_status_icon', )
403
404 def queryset(self, request):
405 return DeploymentPrivilege.select_by_user(request.user)
406
Tony Macka7dbd422015-01-05 22:48:11 -0500407class ControllerSiteInline(PlStackTabularInline):
408 model = ControllerSite
409 extra = 0
410 suit_classes = 'suit-tab suit-tab-admin-only'
Tony Mack8f30ebe2015-01-06 15:08:20 -0500411 fields = ['controller', 'site', 'tenant_id']
Tony Macka7dbd422015-01-05 22:48:11 -0500412
413
Siobhan Tullyd3515752013-06-21 16:34:53 -0400414class SitePrivilegeInline(PlStackTabularInline):
Siobhan Tullyaa1bcd52013-05-10 12:43:09 -0400415 model = SitePrivilege
416 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400417 suit_classes = 'suit-tab suit-tab-siteprivileges'
Scott Baker40c00762014-08-21 16:55:59 -0700418 fields = ['backend_status_icon', 'user','site', 'role']
419 readonly_fields = ('backend_status_icon', )
Siobhan Tullyaa1bcd52013-05-10 12:43:09 -0400420
Tony Mackc2835a92013-05-28 09:18:49 -0400421 def formfield_for_foreignkey(self, db_field, request, **kwargs):
422 if db_field.name == 'site':
Tony Mack5b061472014-02-04 07:57:10 -0500423 kwargs['queryset'] = Site.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400424
425 if db_field.name == 'user':
Tony Mack5b061472014-02-04 07:57:10 -0500426 kwargs['queryset'] = User.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400427 return super(SitePrivilegeInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
428
Tony Mack5b061472014-02-04 07:57:10 -0500429 def queryset(self, request):
430 return SitePrivilege.select_by_user(request.user)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400431
Tony Macka7dbd422015-01-05 22:48:11 -0500432class SiteDeploymentInline(PlStackTabularInline):
433 model = SiteDeployment
Tony Macke4be32f2014-03-11 20:45:25 -0400434 extra = 0
435 suit_classes = 'suit-tab suit-tab-deployments'
Tony Mack528d4222014-12-05 17:13:08 -0500436 fields = ['backend_status_icon', 'deployment','site', 'controller']
Scott Baker40c00762014-08-21 16:55:59 -0700437 readonly_fields = ('backend_status_icon', )
Tony Macke4be32f2014-03-11 20:45:25 -0400438
439 def formfield_for_foreignkey(self, db_field, request, **kwargs):
440 if db_field.name == 'site':
441 kwargs['queryset'] = Site.select_by_user(request.user)
442
443 if db_field.name == 'deployment':
444 kwargs['queryset'] = Deployment.select_by_user(request.user)
Tony Mack528d4222014-12-05 17:13:08 -0500445
446 if db_field.name == 'controller':
447 kwargs['queryset'] = Controller.select_by_user(request.user)
448
Tony Macka7dbd422015-01-05 22:48:11 -0500449 return super(SiteDeploymentInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
Tony Macke4be32f2014-03-11 20:45:25 -0400450
451 def queryset(self, request):
Tony Macka7dbd422015-01-05 22:48:11 -0500452 return SiteDeployment.select_by_user(request.user)
Tony Macke4be32f2014-03-11 20:45:25 -0400453
454
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400455class SlicePrivilegeInline(PlStackTabularInline):
456 model = SlicePrivilege
457 suit_classes = 'suit-tab suit-tab-sliceprivileges'
458 extra = 0
Scott Baker40c00762014-08-21 16:55:59 -0700459 fields = ('backend_status_icon', 'user', 'slice', 'role')
460 readonly_fields = ('backend_status_icon', )
Siobhan Tullyaa1bcd52013-05-10 12:43:09 -0400461
Tony Mackc2835a92013-05-28 09:18:49 -0400462 def formfield_for_foreignkey(self, db_field, request, **kwargs):
463 if db_field.name == 'slice':
Scott Baker36f50872014-08-21 13:01:25 -0700464 kwargs['queryset'] = Slice.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400465 if db_field.name == 'user':
Scott Baker36f50872014-08-21 13:01:25 -0700466 kwargs['queryset'] = User.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400467
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400468 return super(SlicePrivilegeInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
Tony Mackc2835a92013-05-28 09:18:49 -0400469
Tony Mack5b061472014-02-04 07:57:10 -0500470 def queryset(self, request):
471 return SlicePrivilege.select_by_user(request.user)
472
Scott Bakera0015eb2013-08-14 17:28:14 -0700473class SliceNetworkInline(PlStackTabularInline):
Scott Baker74d8e622013-07-29 16:04:22 -0700474 model = Network.slices.through
Scott Baker874936e2014-01-13 18:15:34 -0800475 selflink_fieldname = "network"
Scott Baker74d8e622013-07-29 16:04:22 -0700476 extra = 0
477 verbose_name = "Network Connection"
478 verbose_name_plural = "Network Connections"
Siobhan Tully2d95e482013-09-06 10:56:06 -0400479 suit_classes = 'suit-tab suit-tab-slicenetworks'
Scott Baker40c00762014-08-21 16:55:59 -0700480 fields = ['backend_status_icon', 'network']
481 readonly_fields = ('backend_status_icon', )
Scott Baker2170b972014-06-03 12:14:07 -0700482
Sapan Bhatiae9f96f62014-11-19 15:10:16 -0500483class ImageDeploymentsInline(PlStackTabularInline):
484 model = ImageDeployments
Scott Baker2170b972014-06-03 12:14:07 -0700485 extra = 0
486 verbose_name = "Image Deployments"
487 verbose_name_plural = "Image Deployments"
488 suit_classes = 'suit-tab suit-tab-imagedeployments'
Tony Mack336e0f92014-11-30 15:53:08 -0500489 fields = ['backend_status_icon', 'image', 'deployment']
490 readonly_fields = ['backend_status_icon']
491
492class ControllerImagesInline(PlStackTabularInline):
493 model = ControllerImages
494 extra = 0
495 verbose_name = "Controller Images"
496 verbose_name_plural = "Controller Images"
497 suit_classes = 'suit-tab suit-tab-admin-only'
498 fields = ['backend_status_icon', 'image', 'controller', 'glance_image_id']
Scott Baker40c00762014-08-21 16:55:59 -0700499 readonly_fields = ['backend_status_icon', 'glance_image_id']
Scott Baker74d8e622013-07-29 16:04:22 -0700500
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400501class SliceRoleAdmin(PlanetStackBaseAdmin):
502 model = SliceRole
503 pass
504
505class SiteRoleAdmin(PlanetStackBaseAdmin):
506 model = SiteRole
507 pass
508
Siobhan Tullybf1153a2013-05-27 20:53:48 -0400509class DeploymentAdminForm(forms.ModelForm):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400510 sites = forms.ModelMultipleChoiceField(
511 queryset=Site.objects.all(),
512 required=False,
Scott Baker70983182014-06-09 22:10:00 -0700513 help_text="Select which sites are allowed to host nodes in this deployment",
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400514 widget=FilteredSelectMultiple(
515 verbose_name=('Sites'), is_stacked=False
516 )
517 )
Scott Bakerde0f4412014-06-11 15:40:26 -0700518 images = forms.ModelMultipleChoiceField(
519 queryset=Image.objects.all(),
520 required=False,
521 help_text="Select which images should be deployed on this deployment",
522 widget=FilteredSelectMultiple(
523 verbose_name=('Images'), is_stacked=False
524 )
525 )
Scott Baker37b47902014-09-02 14:37:41 -0700526 flavors = forms.ModelMultipleChoiceField(
527 queryset=Flavor.objects.all(),
528 required=False,
529 help_text="Select which flavors should be usable on this deployment",
530 widget=FilteredSelectMultiple(
531 verbose_name=('Flavors'), is_stacked=False
532 )
533 )
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400534 class Meta:
Siobhan Tullybf1153a2013-05-27 20:53:48 -0400535 model = Deployment
Scott Baker37b47902014-09-02 14:37:41 -0700536 many_to_many = ["flavors",]
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400537
Siobhan Tully320b4622014-01-17 15:11:14 -0500538 def __init__(self, *args, **kwargs):
Scott Baker5380c522014-06-06 14:49:43 -0700539 request = kwargs.pop('request', None)
Siobhan Tully320b4622014-01-17 15:11:14 -0500540 super(DeploymentAdminForm, self).__init__(*args, **kwargs)
541
Scott Baker5380c522014-06-06 14:49:43 -0700542 self.fields['accessControl'].initial = "allow site " + request.user.site.name
543
Siobhan Tully320b4622014-01-17 15:11:14 -0500544 if self.instance and self.instance.pk:
Scott Bakerba5c2f22015-01-20 17:59:55 -0800545 self.fields['sites'].initial = [x.site for x in self.instance.sitedeployments.all()]
Scott Baker9f6b8ed2014-11-17 23:44:03 -0800546 self.fields['images'].initial = [x.image for x in self.instance.imagedeployments.all()]
Scott Baker37b47902014-09-02 14:37:41 -0700547 self.fields['flavors'].initial = self.instance.flavors.all()
Scott Bakerde0f4412014-06-11 15:40:26 -0700548
549 def manipulate_m2m_objs(self, this_obj, selected_objs, all_relations, relation_class, local_attrname, foreign_attrname):
550 """ helper function for handling m2m relations from the MultipleChoiceField
551
552 this_obj: the source object we want to link from
553
554 selected_objs: a list of destination objects we want to link to
555
556 all_relations: the full set of relations involving this_obj, including ones we don't want
557
558 relation_class: the class that implements the relation from source to dest
559
560 local_attrname: field name representing this_obj in relation_class
561
562 foreign_attrname: field name representing selected_objs in relation_class
563
564 This function will remove all newobjclass relations from this_obj
565 that are not contained in selected_objs, and add any relations that
566 are in selected_objs but don't exist in the data model yet.
567 """
568
569 existing_dest_objs = []
570 for relation in list(all_relations):
571 if getattr(relation, foreign_attrname) not in selected_objs:
572 #print "deleting site", sdp.site
573 relation.delete()
574 else:
575 existing_dest_objs.append(getattr(relation, foreign_attrname))
576
577 for dest_obj in selected_objs:
578 if dest_obj not in existing_dest_objs:
579 #print "adding site", site
580 kwargs = {foreign_attrname: dest_obj, local_attrname: this_obj}
581 relation = relation_class(**kwargs)
582 relation.save()
Siobhan Tully320b4622014-01-17 15:11:14 -0500583
584 def save(self, commit=True):
585 deployment = super(DeploymentAdminForm, self).save(commit=False)
586
587 if commit:
588 deployment.save()
Scott Baker61b6aec2014-10-06 17:17:40 -0700589 # this has to be done after save() if/when a deployment is first created
590 deployment.flavors = self.cleaned_data['flavors']
Siobhan Tully320b4622014-01-17 15:11:14 -0500591
592 if deployment.pk:
Scott Bakerc9b14f72014-05-22 13:44:20 -0700593 # save_m2m() doesn't seem to work with 'through' relations. So we
594 # create/destroy the through models ourselves. There has to be
595 # a better way...
596
Scott Baker8864d472015-01-20 17:12:43 -0800597 self.manipulate_m2m_objs(deployment, self.cleaned_data['sites'], deployment.sitedeployments.all(), SiteDeployment, "deployment", "site")
Tony Mack592aa952014-12-15 11:45:02 -0500598 self.manipulate_m2m_objs(deployment, self.cleaned_data['images'], deployment.imagedeployments.all(), ImageDeployments, "deployment", "image")
599 # manipulate_m2m_objs doesn't work for Flavor/Deployment relationship
600 # so well handle that manually here
601 for flavor in deployment.flavors.all():
602 if getattr(flavor, 'name') not in self.cleaned_data['flavors']:
Tony Mack11f4d202014-12-15 12:37:59 -0500603 deployment.flavors.remove(flavor)
Tony Mack592aa952014-12-15 11:45:02 -0500604 for flavor in self.cleaned_data['flavors']:
605 if flavor not in deployment.flavors.all():
606 flavor.deployments.add(deployment)
Scott Bakerc9b14f72014-05-22 13:44:20 -0700607
Scott Baker37b47902014-09-02 14:37:41 -0700608 self.save_m2m()
Siobhan Tully320b4622014-01-17 15:11:14 -0500609
610 return deployment
611
Scott Bakerff5e0f32014-05-22 14:40:27 -0700612class DeploymentAdminROForm(DeploymentAdminForm):
613 def save(self, commit=True):
614 raise PermissionDenied
615
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500616class SiteAssocInline(PlStackTabularInline):
617 model = Site.deployments.through
618 extra = 0
619 suit_classes = 'suit-tab suit-tab-sites'
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400620
Siobhan Tullybf1153a2013-05-27 20:53:48 -0400621class DeploymentAdmin(PlanetStackBaseAdmin):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500622 model = Deployment
Tony Mack93d1b032014-12-08 16:43:02 -0500623 fieldList = ['backend_status_text', 'name', 'sites', 'images', 'flavors', 'accessControl']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500624 fieldsets = [(None, {'fields': fieldList, 'classes':['suit-tab suit-tab-sites']})]
Tony Mack93d1b032014-12-08 16:43:02 -0500625 # node no longer directly connected to deployment
626 #inlines = [DeploymentPrivilegeInline,NodeInline,TagInline,ImageDeploymentsInline]
627 inlines = [DeploymentPrivilegeInline,TagInline,ImageDeploymentsInline]
Scott Baker63d1a552014-08-21 15:19:07 -0700628 list_display = ['backend_status_icon', 'name']
629 list_display_links = ('backend_status_icon', 'name', )
Scott Baker40c00762014-08-21 16:55:59 -0700630 readonly_fields = ('backend_status_text', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500631
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500632 user_readonly_fields = ['name']
633
Tony Mack93d1b032014-12-08 16:43:02 -0500634 # nodes no longer direclty connected to deployments
635 #suit_form_tabs =(('sites','Deployment Details'),('nodes','Nodes'),('deploymentprivileges','Privileges'),('tags','Tags'),('imagedeployments','Images'))
Tony Macka9be0102015-01-15 14:49:14 -0500636 suit_form_tabs =(('sites','Deployment Details'),('deploymentprivileges','Privileges'))
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500637
Scott Bakerff5e0f32014-05-22 14:40:27 -0700638 def get_form(self, request, obj=None, **kwargs):
639 if request.user.isReadOnlyUser():
640 kwargs["form"] = DeploymentAdminROForm
641 else:
642 kwargs["form"] = DeploymentAdminForm
Scott Baker5380c522014-06-06 14:49:43 -0700643 adminForm = super(DeploymentAdmin,self).get_form(request, obj, **kwargs)
644
645 # from stackexchange: pass the request object into the form
646
647 class AdminFormMetaClass(adminForm):
648 def __new__(cls, *args, **kwargs):
649 kwargs['request'] = request
650 return adminForm(*args, **kwargs)
651
652 return AdminFormMetaClass
653
Tony Mack528d4222014-12-05 17:13:08 -0500654class ControllerAdminForm(forms.ModelForm):
Tony Mack26564362015-01-06 17:49:25 -0500655 sites = forms.ModelMultipleChoiceField(
656 queryset=Site.objects.all(),
Tony Mack528d4222014-12-05 17:13:08 -0500657 required=False,
Tony Mack26564362015-01-06 17:49:25 -0500658 help_text="Select which sites are managed by this controller",
Tony Mack528d4222014-12-05 17:13:08 -0500659 widget=FilteredSelectMultiple(
Tony Mack26564362015-01-06 17:49:25 -0500660 verbose_name=('Sites'), is_stacked=False
Tony Mack528d4222014-12-05 17:13:08 -0500661 )
662 )
663
664 class Meta:
665 model = Controller
666
Tony Mack93d1b032014-12-08 16:43:02 -0500667 def __init__(self, *args, **kwargs):
Tony Mack528d4222014-12-05 17:13:08 -0500668 request = kwargs.pop('request', None)
669 super(ControllerAdminForm, self).__init__(*args, **kwargs)
670
671 if self.instance and self.instance.pk:
Tony Mack10328a12015-01-14 12:11:05 -0500672 self.fields['sites'].initial = [x.site for x in self.instance.controllersite.all()]
Tony Mack528d4222014-12-05 17:13:08 -0500673
Tony Mack93d1b032014-12-08 16:43:02 -0500674 def manipulate_m2m_objs(self, this_obj, selected_objs, all_relations, relation_class, local_attrname, foreign_attrname):
Tony Mack528d4222014-12-05 17:13:08 -0500675 """ helper function for handling m2m relations from the MultipleChoiceField
676 this_obj: the source object we want to link from
677 selected_objs: a list of destination objects we want to link to
678 all_relations: the full set of relations involving this_obj, including ones we don't want
679 relation_class: the class that implements the relation from source to dest
680 local_attrname: field name representing this_obj in relation_class
681 foreign_attrname: field name representing selected_objs in relation_class
682 This function will remove all newobjclass relations from this_obj
683 that are not contained in selected_objs, and add any relations that
684 are in selected_objs but don't exist in the data model yet.
685 """
686 existing_dest_objs = []
687 for relation in list(all_relations):
688 if getattr(relation, foreign_attrname) not in selected_objs:
689 #print "deleting site", sdp.site
690 relation.delete()
691 else:
692 existing_dest_objs.append(getattr(relation, foreign_attrname))
693
694 for dest_obj in selected_objs:
695 if dest_obj not in existing_dest_objs:
696 #print "adding site", site
697 kwargs = {foreign_attrname: dest_obj, local_attrname: this_obj}
698 relation = relation_class(**kwargs)
699 relation.save()
700
Tony Mack93d1b032014-12-08 16:43:02 -0500701 def save(self, commit=True):
702 controller = super(ControllerAdminForm, self).save(commit=False)
703 if commit:
704 controller.save()
Tony Mack528d4222014-12-05 17:13:08 -0500705
Tony Mack93d1b032014-12-08 16:43:02 -0500706 if controller.pk:
707 # save_m2m() doesn't seem to work with 'through' relations. So we
708 # create/destroy the through models ourselves. There has to be
709 # a better way...
Tony Mack26564362015-01-06 17:49:25 -0500710 self.manipulate_m2m_objs(controller, self.cleaned_data['sites'], controller.controllersite.all(), ControllerSite, "controller", "site")
Tony Mack1e828282015-01-03 17:40:42 -0500711 pass
712
Tony Mack93d1b032014-12-08 16:43:02 -0500713 self.save_m2m()
Tony Mack528d4222014-12-05 17:13:08 -0500714
Tony Mack93d1b032014-12-08 16:43:02 -0500715 return controller
Tony Mack528d4222014-12-05 17:13:08 -0500716
717class ControllerAdmin(PlanetStackBaseAdmin):
718 model = Controller
Tony Mack78d4c3d2015-01-19 15:05:42 -0500719 fieldList = ['name', 'backend_type', 'version', 'auth_url', 'admin_user', 'admin_tenant','admin_password']
Tony Mack93d1b032014-12-08 16:43:02 -0500720 #fieldsets = [(None, {'fields': fieldList, 'classes':['suit-tab suit-tab-general']})]
Tony Macka7dbd422015-01-05 22:48:11 -0500721 inlines = [ControllerSiteInline] # ,ControllerImagesInline]
Tony Mack528d4222014-12-05 17:13:08 -0500722 list_display = ['backend_status_icon', 'name', 'version', 'backend_type']
723 list_display_links = ('backend_status_icon', 'name', )
724 readonly_fields = ('backend_status_text',)
725
726 user_readonly_fields = []
727
728 def get_form(self, request, obj=None, **kwargs):
Tony Mack93d1b032014-12-08 16:43:02 -0500729 print self.fieldsets
Tony Mack528d4222014-12-05 17:13:08 -0500730 if request.user.isReadOnlyUser():
731 kwargs["form"] = ControllerAdminROForm
732 else:
733 kwargs["form"] = ControllerAdminForm
Tony Mack93d1b032014-12-08 16:43:02 -0500734 adminForm = super(ControllerAdmin,self).get_form(request, obj, **kwargs)
Tony Mack528d4222014-12-05 17:13:08 -0500735
736 # from stackexchange: pass the request object into the form
737
738 class AdminFormMetaClass(adminForm):
739 def __new__(cls, *args, **kwargs):
740 kwargs['request'] = request
741 return adminForm(*args, **kwargs)
742
743 return AdminFormMetaClass
744
Tony Mackc36cafb2015-01-13 17:33:08 -0500745 def save_model(self, request, obj, form, change):
746 # update openstack connection to use this site/tenant
747 obj.save_by_user(request.user)
748
749 def delete_model(self, request, obj):
750 obj.delete_by_user(request.user)
751
Siobhan Tullyce652d02013-10-08 21:52:35 -0400752class ServiceAttrAsTabInline(PlStackTabularInline):
753 model = ServiceAttribute
754 fields = ['name','value']
755 extra = 0
756 suit_classes = 'suit-tab suit-tab-serviceattrs'
757
Siobhan Tullyce652d02013-10-08 21:52:35 -0400758class ServiceAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -0700759 list_display = ("backend_status_icon","name","description","versionNumber","enabled","published")
760 list_display_links = ('backend_status_icon', 'name', )
Scott Baker40c00762014-08-21 16:55:59 -0700761 fieldList = ["backend_status_text","name","description","versionNumber","enabled","published"]
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500762 fieldsets = [(None, {'fields': fieldList, 'classes':['suit-tab suit-tab-general']})]
763 inlines = [ServiceAttrAsTabInline,SliceInline]
Scott Baker40c00762014-08-21 16:55:59 -0700764 readonly_fields = ('backend_status_text', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500765
766 user_readonly_fields = fieldList
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500767
768 suit_form_tabs =(('general', 'Service Details'),
769 ('slices','Slices'),
770 ('serviceattrs','Additional Attributes'),
771 )
Siobhan Tullyce652d02013-10-08 21:52:35 -0400772
Tony Mack82df1d02015-01-14 20:58:38 -0500773class SiteNodeInline(PlStackTabularInline):
774 model = Node
775 fields = ['name', 'site_deployment']
776 extra = 0
777 suit_classes = 'suit-tab suit-tab-nodes'
778
Tony Mack0553f282013-06-10 22:54:50 -0400779class SiteAdmin(PlanetStackBaseAdmin):
Tony Mack598eaf22015-01-25 12:35:29 -0500780 #fieldList = ['backend_status_text', 'name', 'site_url', 'enabled', 'is_public', 'login_base', 'accountLink','location']
781 fieldList = ['backend_status_text', 'name', 'site_url', 'enabled', 'is_public', 'login_base', 'location']
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400782 fieldsets = [
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500783 (None, {'fields': fieldList, 'classes':['suit-tab suit-tab-general']}),
Tony Macke4be32f2014-03-11 20:45:25 -0400784 #('Deployment Networks', {'fields': ['deployments'], 'classes':['suit-tab suit-tab-deployments']}),
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400785 ]
Tony Mack598eaf22015-01-25 12:35:29 -0500786 #readonly_fields = ['backend_status_text', 'accountLink']
787 readonly_fields = ['backend_status_text']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500788
Tony Mack598eaf22015-01-25 12:35:29 -0500789 #user_readonly_fields = ['name', 'deployments','site_url', 'enabled', 'is_public', 'login_base', 'accountLink']
790 user_readonly_fields = ['name', 'deployments','site_url', 'enabled', 'is_public', 'login_base']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500791
Scott Baker63d1a552014-08-21 15:19:07 -0700792 list_display = ('backend_status_icon', 'name', 'login_base','site_url', 'enabled')
793 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400794 filter_horizontal = ('deployments',)
Tony Mack82df1d02015-01-14 20:58:38 -0500795 inlines = [SliceInline,UserInline,TagInline, SitePrivilegeInline, SiteDeploymentInline, SiteNodeInline]
Tony Mack10328a12015-01-14 12:11:05 -0500796 admin_inlines = [ControllerSiteInline]
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400797 search_fields = ['name']
798
Tony Mack3c01ff92015-01-10 23:08:10 -0500799 @property
800 def suit_form_tabs(self):
801 tabs = [('general', 'Site Details'),
802 ('users','Users'),
803 ('siteprivileges','Privileges'),
804 ('deployments','Deployments'),
805 ('slices','Slices'),
Tony Mack82df1d02015-01-14 20:58:38 -0500806 ('nodes','Nodes'),
Tony Mack3c01ff92015-01-10 23:08:10 -0500807 ]
808
809 request=getattr(_thread_locals, "request", None)
810 if request and request.user.is_admin:
811 tabs.append( ('admin-only', 'Admin-Only') )
812
813 return tabs
814
Tony Mack04062832013-05-10 08:22:44 -0400815 def queryset(self, request):
Tony Mack5b061472014-02-04 07:57:10 -0500816 return Site.select_by_user(request.user)
Tony Mack04062832013-05-10 08:22:44 -0400817
Tony Mack5cd13202013-05-01 21:48:38 -0400818 def get_formsets(self, request, obj=None):
819 for inline in self.get_inline_instances(request, obj):
820 # hide MyInline in the add view
821 if obj is None:
822 continue
Tony Mack2bd5b412013-06-11 21:05:06 -0400823 if isinstance(inline, SliverInline):
824 inline.model.caller = request.user
Tony Mack5cd13202013-05-01 21:48:38 -0400825 yield inline.get_formset(request, obj)
826
Scott Baker545db2a2013-12-09 18:44:43 -0800827 def accountLink(self, obj):
828 link_obj = obj.accounts.all()
829 if link_obj:
830 reverse_path = "admin:core_account_change"
831 url = reverse(reverse_path, args =(link_obj[0].id,))
832 return "<a href='%s'>%s</a>" % (url, "view billing details")
833 else:
834 return "no billing data for this site"
835 accountLink.allow_tags = True
836 accountLink.short_description = "Billing"
837
Tony Mack332ee1d2014-02-04 15:33:45 -0500838 def save_model(self, request, obj, form, change):
839 # update openstack connection to use this site/tenant
840 obj.save_by_user(request.user)
841
842 def delete_model(self, request, obj):
843 obj.delete_by_user(request.user)
844
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500845
Tony Mack9bcbe4f2013-04-29 08:13:27 -0400846class SitePrivilegeAdmin(PlanetStackBaseAdmin):
Scott Baker40c00762014-08-21 16:55:59 -0700847 fieldList = ['backend_status_text', 'user', 'site', 'role']
Tony Mack00d361f2013-04-28 10:28:42 -0400848 fieldsets = [
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500849 (None, {'fields': fieldList, 'classes':['collapse']})
Tony Mack00d361f2013-04-28 10:28:42 -0400850 ]
Scott Baker40c00762014-08-21 16:55:59 -0700851 readonly_fields = ('backend_status_text', )
Scott Baker63d1a552014-08-21 15:19:07 -0700852 list_display = ('backend_status_icon', 'user', 'site', 'role')
853 list_display_links = list_display
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500854 user_readonly_fields = fieldList
855 user_readonly_inlines = []
Tony Mack00d361f2013-04-28 10:28:42 -0400856
Tony Mackc2835a92013-05-28 09:18:49 -0400857 def formfield_for_foreignkey(self, db_field, request, **kwargs):
858 if db_field.name == 'site':
859 if not request.user.is_admin:
860 # only show sites where user is an admin or pi
861 sites = set()
862 for site_privilege in SitePrivilege.objects.filer(user=request.user):
863 if site_privilege.role.role_type in ['admin', 'pi']:
864 sites.add(site_privilege.site)
865 kwargs['queryset'] = Site.objects.filter(site__in=list(sites))
866
867 if db_field.name == 'user':
868 if not request.user.is_admin:
869 # only show users from sites where caller has admin or pi role
870 roles = Role.objects.filter(role_type__in=['admin', 'pi'])
871 site_privileges = SitePrivilege.objects.filter(user=request.user).filter(role__in=roles)
872 sites = [site_privilege.site for site_privilege in site_privileges]
873 site_privileges = SitePrivilege.objects.filter(site__in=sites)
874 emails = [site_privilege.user.email for site_privilege in site_privileges]
875 users = User.objects.filter(email__in=emails)
876 kwargs['queryset'] = users
877
878 return super(SitePrivilegeAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
879
Tony Mack04062832013-05-10 08:22:44 -0400880 def queryset(self, request):
881 # admins can see all privileges. Users can only see privileges at sites
Tony Mackc2835a92013-05-28 09:18:49 -0400882 # where they have the admin role or pi role.
Tony Mack04062832013-05-10 08:22:44 -0400883 qs = super(SitePrivilegeAdmin, self).queryset(request)
Tony Mack5b061472014-02-04 07:57:10 -0500884 #if not request.user.is_admin:
885 # roles = Role.objects.filter(role_type__in=['admin', 'pi'])
886 # site_privileges = SitePrivilege.objects.filter(user=request.user).filter(role__in=roles)
887 # login_bases = [site_privilege.site.login_base for site_privilege in site_privileges]
888 # sites = Site.objects.filter(login_base__in=login_bases)
889 # qs = qs.filter(site__in=sites)
Tony Mack04062832013-05-10 08:22:44 -0400890 return qs
891
Siobhan Tullyce652d02013-10-08 21:52:35 -0400892class SliceForm(forms.ModelForm):
893 class Meta:
894 model = Slice
895 widgets = {
Scott Baker36f50872014-08-21 13:01:25 -0700896 'service': LinkedSelect
Siobhan Tullyce652d02013-10-08 21:52:35 -0400897 }
898
Tony Mack2cbd3802014-09-29 16:10:52 -0400899 def clean(self):
900 cleaned_data = super(SliceForm, self).clean()
901 name = cleaned_data.get('name')
Scott Baker6efad462014-10-06 23:09:59 -0700902 site = cleaned_data.get('site')
Tony Mack585cb192014-10-22 12:54:19 -0400903 slice_id = self.instance.id
904 if not site and slice_id:
905 site = Slice.objects.get(id=slice_id).site
Scott Baker6efad462014-10-06 23:09:59 -0700906 if (not isinstance(site,Site)):
907 # previous code indicates 'site' could be a site_id and not a site?
908 site = Slice.objects.get(id=site.id)
Tony Mack2cbd3802014-09-29 16:10:52 -0400909 if not name.startswith(site.login_base):
910 raise forms.ValidationError('slice name must begin with %s' % site.login_base)
911 return cleaned_data
912
Tony Macka7dbd422015-01-05 22:48:11 -0500913class ControllerSliceInline(PlStackTabularInline):
914 model = ControllerSlice
Scott Bakerc4efdc72014-10-15 16:54:04 -0700915 extra = 0
Tony Mack336e0f92014-11-30 15:53:08 -0500916 verbose_name = "Controller Slices"
917 verbose_name_plural = "Controller Slices"
Scott Bakerc4efdc72014-10-15 16:54:04 -0700918 suit_classes = 'suit-tab suit-tab-admin-only'
Tony Mack336e0f92014-11-30 15:53:08 -0500919 fields = ['backend_status_icon', 'controller', 'tenant_id']
Tony Mack3c01ff92015-01-10 23:08:10 -0500920 readonly_fields = ('backend_status_icon', 'controller' )
Scott Bakerc4efdc72014-10-15 16:54:04 -0700921
Tony Mack2bd5b412013-06-11 21:05:06 -0400922class SliceAdmin(PlanetStackBaseAdmin):
Siobhan Tullyce652d02013-10-08 21:52:35 -0400923 form = SliceForm
Tony Mackfbb26fc2014-09-02 07:03:27 -0400924 fieldList = ['backend_status_text', 'site', 'name', 'serviceClass', 'enabled','description', 'service', 'slice_url', 'max_slivers']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500925 fieldsets = [('Slice Details', {'fields': fieldList, 'classes':['suit-tab suit-tab-general']}),]
Scott Baker40c00762014-08-21 16:55:59 -0700926 readonly_fields = ('backend_status_text', )
Tony Mack7d459902014-09-03 13:18:57 -0400927 list_display = ('backend_status_icon', 'name', 'site','serviceClass', 'slice_url', 'max_slivers')
928 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tully2d95e482013-09-06 10:56:06 -0400929 inlines = [SlicePrivilegeInline,SliverInline, TagInline, ReservationInline,SliceNetworkInline]
Tony Macka7dbd422015-01-05 22:48:11 -0500930 admin_inlines = [ControllerSliceInline]
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400931
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500932 user_readonly_fields = fieldList
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400933
Scott Bakerc4efdc72014-10-15 16:54:04 -0700934 @property
935 def suit_form_tabs(self):
936 tabs =[('general', 'Slice Details'),
937 ('slicenetworks','Networks'),
938 ('sliceprivileges','Privileges'),
939 ('slivers','Slivers'),
Tony Mack598eaf22015-01-25 12:35:29 -0500940 #('reservations','Reservations'),
Tony Mackb34553e2015-01-15 14:44:06 -0500941 ('tags','Tags'),
Scott Bakerc4efdc72014-10-15 16:54:04 -0700942 ]
943
944 request=getattr(_thread_locals, "request", None)
945 if request and request.user.is_admin:
946 tabs.append( ('admin-only', 'Admin-Only') )
947
948 return tabs
Tony Mack7b8505a2014-10-22 11:54:29 -0400949
950 def add_view(self, request, form_url='', extra_context=None):
951 # revert to default read-only fields
952 self.readonly_fields = ('backend_status_text',)
953 return super(SliceAdmin, self).add_view(request, form_url, extra_context=extra_context)
954
955 def change_view(self, request, object_id, form_url='', extra_context=None):
Tony Mack7b8505a2014-10-22 11:54:29 -0400956 # cannot change the site of an existing slice so make the site field read only
957 if object_id:
958 self.readonly_fields = ('backend_status_text','site')
959 return super(SliceAdmin, self).change_view(request, object_id, form_url)
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400960
Scott Baker510fdbb2014-08-05 17:19:24 -0700961 def render_change_form(self, request, context, add=False, change=False, form_url='', obj=None):
Scott Baker510fdbb2014-08-05 17:19:24 -0700962 deployment_nodes = []
963 for node in Node.objects.all():
Scott Baker39293d72015-01-21 16:24:07 -0800964 deployment_nodes.append( (node.site_deployment.deployment.id, node.id, node.name) )
Scott Baker510fdbb2014-08-05 17:19:24 -0700965
Scott Baker7a61dc42014-09-02 17:08:20 -0700966 deployment_flavors = []
967 for flavor in Flavor.objects.all():
968 for deployment in flavor.deployments.all():
969 deployment_flavors.append( (deployment.id, flavor.id, flavor.name) )
970
Tony Mack93d1b032014-12-08 16:43:02 -0500971 deployment_images = []
Scott Baker93e80cd2014-09-09 09:58:49 -0700972 for image in Image.objects.all():
Tony Mack93d1b032014-12-08 16:43:02 -0500973 for deployment_image in image.imagedeployments.all():
Scott Bakera6a0c772014-12-22 17:35:34 -0800974 deployment_images.append( (deployment_image.deployment.id, image.id, image.name) )
Scott Baker93e80cd2014-09-09 09:58:49 -0700975
Tony Mackec23b992014-09-02 21:18:45 -0400976 site_login_bases = []
977 for site in Site.objects.all():
Scott Baker93e80cd2014-09-09 09:58:49 -0700978 site_login_bases.append((site.id, site.login_base))
979
Scott Baker510fdbb2014-08-05 17:19:24 -0700980 context["deployment_nodes"] = deployment_nodes
Scott Baker7a61dc42014-09-02 17:08:20 -0700981 context["deployment_flavors"] = deployment_flavors
Scott Baker93e80cd2014-09-09 09:58:49 -0700982 context["deployment_images"] = deployment_images
Tony Mackec23b992014-09-02 21:18:45 -0400983 context["site_login_bases"] = site_login_bases
Scott Baker510fdbb2014-08-05 17:19:24 -0700984 return super(SliceAdmin, self).render_change_form(request, context, add, change, form_url, obj)
985
Tony Mackc2835a92013-05-28 09:18:49 -0400986 def formfield_for_foreignkey(self, db_field, request, **kwargs):
987 if db_field.name == 'site':
Tony Mack5b061472014-02-04 07:57:10 -0500988 kwargs['queryset'] = Site.select_by_user(request.user)
Tony Mackec23b992014-09-02 21:18:45 -0400989 kwargs['widget'] = forms.Select(attrs={'onChange': "update_slice_prefix(this, $($(this).closest('fieldset')[0]).find('.field-name input')[0].id)"})
Scott Baker40c00762014-08-21 16:55:59 -0700990
Tony Mackc2835a92013-05-28 09:18:49 -0400991 return super(SliceAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
992
Tony Mack04062832013-05-10 08:22:44 -0400993 def queryset(self, request):
994 # admins can see all keys. Users can only see slices they belong to.
Tony Mack5b061472014-02-04 07:57:10 -0500995 return Slice.select_by_user(request.user)
Tony Mack04062832013-05-10 08:22:44 -0400996
Tony Mack79748612013-05-01 14:52:03 -0400997 def get_formsets(self, request, obj=None):
998 for inline in self.get_inline_instances(request, obj):
999 # hide MyInline in the add view
1000 if obj is None:
1001 continue
Tony Mack2bd5b412013-06-11 21:05:06 -04001002 if isinstance(inline, SliverInline):
1003 inline.model.caller = request.user
Tony Mack79748612013-05-01 14:52:03 -04001004 yield inline.get_formset(request, obj)
1005
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001006class SlicePrivilegeAdmin(PlanetStackBaseAdmin):
Tony Mack00d361f2013-04-28 10:28:42 -04001007 fieldsets = [
Scott Baker40c00762014-08-21 16:55:59 -07001008 (None, {'fields': ['backend_status_text', 'user', 'slice', 'role']})
Tony Mack00d361f2013-04-28 10:28:42 -04001009 ]
Scott Baker40c00762014-08-21 16:55:59 -07001010 readonly_fields = ('backend_status_text', )
Scott Baker63d1a552014-08-21 15:19:07 -07001011 list_display = ('backend_status_icon', 'user', 'slice', 'role')
1012 list_display_links = list_display
Tony Mack00d361f2013-04-28 10:28:42 -04001013
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001014 user_readonly_fields = ['user', 'slice', 'role']
1015 user_readonly_inlines = []
1016
Tony Mackc2835a92013-05-28 09:18:49 -04001017 def formfield_for_foreignkey(self, db_field, request, **kwargs):
1018 if db_field.name == 'slice':
Tony Mack5b061472014-02-04 07:57:10 -05001019 kwargs['queryset'] = Slice.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -04001020
1021 if db_field.name == 'user':
Tony Mack5b061472014-02-04 07:57:10 -05001022 kwargs['queryset'] = User.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -04001023
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001024 return super(SlicePrivilegeAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
Tony Mackc2835a92013-05-28 09:18:49 -04001025
Tony Mack04062832013-05-10 08:22:44 -04001026 def queryset(self, request):
1027 # admins can see all memberships. Users can only see memberships of
1028 # slices where they have the admin role.
Tony Mack5b061472014-02-04 07:57:10 -05001029 return SlicePrivilege.select_by_user(request.user)
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001030
Tony Mack9bcbe4f2013-04-29 08:13:27 -04001031 def save_model(self, request, obj, form, change):
Tony Mack951dab42013-05-02 19:51:45 -04001032 # update openstack connection to use this site/tenant
1033 auth = request.session.get('auth', {})
Tony Mackf7f79a12014-08-11 11:21:42 -04001034 auth['tenant'] = obj.slice.slicename
Tony Mack951dab42013-05-02 19:51:45 -04001035 obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
Tony Mack9bcbe4f2013-04-29 08:13:27 -04001036 obj.save()
1037
1038 def delete_model(self, request, obj):
Tony Mack951dab42013-05-02 19:51:45 -04001039 # update openstack connection to use this site/tenant
1040 auth = request.session.get('auth', {})
Tony Mackf7f79a12014-08-11 11:21:42 -04001041 auth['tenant'] = obj.slice.slicename
Tony Mack951dab42013-05-02 19:51:45 -04001042 obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
Tony Mack9bcbe4f2013-04-29 08:13:27 -04001043 obj.delete()
1044
Siobhan Tully567e3e62013-06-21 18:03:16 -04001045
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001046class ImageAdmin(PlanetStackBaseAdmin):
1047
Scott Baker36f50872014-08-21 13:01:25 -07001048 fieldsets = [('Image Details',
Scott Baker40c00762014-08-21 16:55:59 -07001049 {'fields': ['backend_status_text', 'name', 'disk_format', 'container_format'],
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001050 'classes': ['suit-tab suit-tab-general']})
1051 ]
Scott Baker40c00762014-08-21 16:55:59 -07001052 readonly_fields = ('backend_status_text', )
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001053
Tony Mack336e0f92014-11-30 15:53:08 -05001054 suit_form_tabs =(('general','Image Details'),('slivers','Slivers'),('imagedeployments','Deployments'), ('controllerimages', 'Controllers'))
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001055
Tony Mack336e0f92014-11-30 15:53:08 -05001056 inlines = [SliverInline, ControllerImagesInline]
Scott Bakerb6f99242014-06-11 11:34:44 -07001057
Tony Mack32e1ce32014-05-07 13:29:41 -04001058 user_readonly_fields = ['name', 'disk_format', 'container_format']
Scott Bakerb27b62c2014-08-15 16:29:16 -07001059
Scott Baker63d1a552014-08-21 15:19:07 -07001060 list_display = ['backend_status_icon', 'name']
1061 list_display_links = ('backend_status_icon', 'name', )
1062
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001063class NodeForm(forms.ModelForm):
1064 class Meta:
1065 widgets = {
1066 'site': LinkedSelect,
1067 'deployment': LinkedSelect
1068 }
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001069
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001070class NodeAdmin(PlanetStackBaseAdmin):
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001071 form = NodeForm
Tony Mack93d1b032014-12-08 16:43:02 -05001072 list_display = ('backend_status_icon', 'name', 'site_deployment')
Scott Baker63d1a552014-08-21 15:19:07 -07001073 list_display_links = ('backend_status_icon', 'name', )
Tony Mack93d1b032014-12-08 16:43:02 -05001074 list_filter = ('site_deployment',)
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001075
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001076 inlines = [TagInline,SliverInline]
Tony Mack93d1b032014-12-08 16:43:02 -05001077 fieldsets = [('Node Details', {'fields': ['backend_status_text', 'name','site_deployment'], 'classes':['suit-tab suit-tab-details']})]
Scott Baker40c00762014-08-21 16:55:59 -07001078 readonly_fields = ('backend_status_text', )
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001079
Tony Mack93d1b032014-12-08 16:43:02 -05001080 user_readonly_fields = ['name','site_deployment']
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001081 user_readonly_inlines = [TagInline,SliverInline]
1082
Tony Mackab6538f2015-01-12 21:40:09 -05001083 suit_form_tabs =(('details','Node Details'),('slivers','Slivers'))
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001084
Siobhan Tully567e3e62013-06-21 18:03:16 -04001085
Tony Mackd90cdbf2013-04-16 22:48:40 -04001086class SliverForm(forms.ModelForm):
1087 class Meta:
Tony Mack1d6b85f2013-05-07 18:49:14 -04001088 model = Sliver
Tony Mackd90cdbf2013-04-16 22:48:40 -04001089 ip = forms.CharField(widget=PlainTextWidget)
Tony Mack18261812013-05-02 16:39:20 -04001090 instance_name = forms.CharField(widget=PlainTextWidget)
Tony Mackd90cdbf2013-04-16 22:48:40 -04001091 widgets = {
1092 'ip': PlainTextWidget(),
Tony Mack18261812013-05-02 16:39:20 -04001093 'instance_name': PlainTextWidget(),
Scott Baker887d4a82015-01-19 11:32:20 -08001094 'instance_id': PlainTextWidget(),
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001095 'slice': LinkedSelect,
Tony Mackbf6aa302014-12-26 13:38:02 -05001096 'deployment': LinkedSelect,
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001097 'node': LinkedSelect,
1098 'image': LinkedSelect
Siobhan Tully53437282013-04-26 19:30:27 -04001099 }
Tony Mackd90cdbf2013-04-16 22:48:40 -04001100
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001101class TagAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001102 list_display = ['backend_status_icon', 'service', 'name', 'value', 'content_type', 'content_object',]
1103 list_display_links = list_display
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001104 user_readonly_fields = ['service', 'name', 'value', 'content_type', 'content_object',]
1105 user_readonly_inlines = []
Siobhan Tullyd3515752013-06-21 16:34:53 -04001106
Tony Mack9bcbe4f2013-04-29 08:13:27 -04001107class SliverAdmin(PlanetStackBaseAdmin):
Tony Mackd90cdbf2013-04-16 22:48:40 -04001108 form = SliverForm
Tony Mackcdec0902013-04-15 00:38:49 -04001109 fieldsets = [
Scott Baker970314b2015-01-25 22:16:13 -08001110 ('Sliver Details', {'fields': ['backend_status_text', 'slice', 'deployment', 'node', 'ip', 'instance_id', 'instance_name', 'flavor', 'image', 'ssh_command'], 'classes': ['suit-tab suit-tab-general'], })
Tony Mackcdec0902013-04-15 00:38:49 -04001111 ]
Scott Baker970314b2015-01-25 22:16:13 -08001112 readonly_fields = ('backend_status_text', 'ssh_command', )
Scott Baker887d4a82015-01-19 11:32:20 -08001113 list_display = ['backend_status_icon', 'ip', 'instance_id', 'instance_name', 'slice', 'flavor', 'image', 'node', 'deployment']
Scott Baker63d1a552014-08-21 15:19:07 -07001114 list_display_links = ('backend_status_icon', 'ip',)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001115
Scott Baker1eace6f2015-01-19 08:24:08 -08001116 suit_form_tabs =(('general', 'Sliver Details'),)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001117
Siobhan Tullyde5450d2013-06-21 11:35:33 -04001118 inlines = [TagInline]
Tony Mack53106f32013-04-27 16:43:01 -04001119
Tony Mackbf6aa302014-12-26 13:38:02 -05001120 user_readonly_fields = ['slice', 'deployment', 'node', 'ip', 'instance_name', 'flavor', 'image']
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001121
Scott Baker970314b2015-01-25 22:16:13 -08001122 def ssh_command(self, obj):
1123 ssh_command = obj.get_ssh_command()
1124 if ssh_command:
1125 return ssh_command
1126 else:
1127 return "(not available)"
1128
Tony Mackc2835a92013-05-28 09:18:49 -04001129 def formfield_for_foreignkey(self, db_field, request, **kwargs):
1130 if db_field.name == 'slice':
Tony Mack5b061472014-02-04 07:57:10 -05001131 kwargs['queryset'] = Slice.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -04001132
1133 return super(SliverAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
1134
Tony Mack04062832013-05-10 08:22:44 -04001135 def queryset(self, request):
Scott Baker36f50872014-08-21 13:01:25 -07001136 # admins can see all slivers. Users can only see slivers of
Tony Mack04062832013-05-10 08:22:44 -04001137 # the slices they belong to.
Tony Mack5b061472014-02-04 07:57:10 -05001138 return Sliver.select_by_user(request.user)
1139
Tony Mack04062832013-05-10 08:22:44 -04001140
Tony Mack1d6b85f2013-05-07 18:49:14 -04001141 def get_formsets(self, request, obj=None):
1142 # make some fields read only if we are updating an existing record
1143 if obj == None:
Scott Baker970314b2015-01-25 22:16:13 -08001144 self.readonly_fields = ('backend_status_text', 'ssh_command', )
Tony Mack1d6b85f2013-05-07 18:49:14 -04001145 else:
Scott Baker970314b2015-01-25 22:16:13 -08001146 self.readonly_fields = ('backend_status_text', 'ssh_command',)
Tony Mack1d6b85f2013-05-07 18:49:14 -04001147
1148 for inline in self.get_inline_instances(request, obj):
1149 # hide MyInline in the add view
1150 if obj is None:
1151 continue
Scott Baker526b71e2014-05-13 13:18:01 -07001152 if isinstance(inline, SliverInline):
1153 inline.model.caller = request.user
1154 yield inline.get_formset(request, obj)
Tony Mack53106f32013-04-27 16:43:01 -04001155
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001156 #def save_model(self, request, obj, form, change):
1157 # # update openstack connection to use this site/tenant
1158 # auth = request.session.get('auth', {})
1159 # auth['tenant'] = obj.slice.name
1160 # obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
1161 # obj.creator = request.user
1162 # obj.save()
Tony Mack53106f32013-04-27 16:43:01 -04001163
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001164 #def delete_model(self, request, obj):
1165 # # update openstack connection to use this site/tenant
1166 # auth = request.session.get('auth', {})
1167 # auth['tenant'] = obj.slice.name
1168 # obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
1169 # obj.delete()
Tony Mackcdec0902013-04-15 00:38:49 -04001170
Siobhan Tully53437282013-04-26 19:30:27 -04001171class UserCreationForm(forms.ModelForm):
1172 """A form for creating new users. Includes all the required
1173 fields, plus a repeated password."""
1174 password1 = forms.CharField(label='Password', widget=forms.PasswordInput)
1175 password2 = forms.CharField(label='Password confirmation', widget=forms.PasswordInput)
1176
1177 class Meta:
Siobhan Tully30fd4292013-05-10 08:59:56 -04001178 model = User
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001179 fields = ('email', 'firstname', 'lastname', 'phone', 'public_key')
Siobhan Tully53437282013-04-26 19:30:27 -04001180
1181 def clean_password2(self):
1182 # Check that the two password entries match
1183 password1 = self.cleaned_data.get("password1")
1184 password2 = self.cleaned_data.get("password2")
1185 if password1 and password2 and password1 != password2:
1186 raise forms.ValidationError("Passwords don't match")
1187 return password2
1188
1189 def save(self, commit=True):
1190 # Save the provided password in hashed format
1191 user = super(UserCreationForm, self).save(commit=False)
Tony Mackf9f4afb2013-05-01 21:02:12 -04001192 user.password = self.cleaned_data["password1"]
1193 #user.set_password(self.cleaned_data["password1"])
Siobhan Tully53437282013-04-26 19:30:27 -04001194 if commit:
1195 user.save()
1196 return user
1197
Siobhan Tully567e3e62013-06-21 18:03:16 -04001198
Siobhan Tully53437282013-04-26 19:30:27 -04001199class UserChangeForm(forms.ModelForm):
1200 """A form for updating users. Includes all the fields on
1201 the user, but replaces the password field with admin's
1202 password hash display field.
1203 """
Siobhan Tully63b7ba42014-01-12 10:35:11 -05001204 password = ReadOnlyPasswordHashField(label='Password',
1205 help_text= '<a href=\"password/\">Change Password</a>.')
Siobhan Tully53437282013-04-26 19:30:27 -04001206
1207 class Meta:
Siobhan Tully30fd4292013-05-10 08:59:56 -04001208 model = User
Scott Baker9f6b8ed2014-11-17 23:44:03 -08001209 widgets = { 'public_key': UploadTextareaWidget, }
Siobhan Tully53437282013-04-26 19:30:27 -04001210
1211 def clean_password(self):
1212 # Regardless of what the user provides, return the initial value.
1213 # This is done here, rather than on the field, because the
1214 # field does not have access to the initial value
1215 return self.initial["password"]
1216
Scott Baker2c3cb642014-05-19 17:55:56 -07001217class UserDashboardViewInline(PlStackTabularInline):
1218 model = UserDashboardView
1219 extra = 0
1220 suit_classes = 'suit-tab suit-tab-dashboards'
1221 fields = ['user', 'dashboardView', 'order']
1222
Tony Mack3c01ff92015-01-10 23:08:10 -05001223class ControllerUserInline(PlStackTabularInline):
1224 model = ControllerUser
1225 extra = 0
1226 suit_classes = 'suit-tab suit-tab-admin-only'
1227 fields = ['controller', 'user', 'kuser_id']
1228 readonly_fields=['controller']
1229
1230
Scott Baker86c83ab2014-10-03 13:10:47 -07001231class UserAdmin(PermissionCheckingAdminMixin, UserAdmin):
1232 # Note: Make sure PermissionCheckingAdminMixin is listed before
1233 # admin.ModelAdmin in the class declaration.
1234
Siobhan Tully53437282013-04-26 19:30:27 -04001235 class Meta:
1236 app_label = "core"
1237
1238 # The forms to add and change user instances
1239 form = UserChangeForm
1240 add_form = UserCreationForm
1241
1242 # The fields to be used in displaying the User model.
1243 # These override the definitions on the base UserAdmin
1244 # that reference specific fields on auth.User.
Scott Bakerf587f442015-01-24 13:33:26 -08001245 list_display = ('backend_status_icon', 'email', 'firstname', 'lastname', 'site', 'last_login')
1246 list_display_links = ("email",)
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001247 list_filter = ('site',)
Scott Baker6e9027f2015-01-29 10:55:53 -08001248 inlines = [SlicePrivilegeInline,SitePrivilegeInline]
Tony Mack3c01ff92015-01-10 23:08:10 -05001249 admin_inlines = [ControllerUserInline]
Scott Bakercbfb6002014-10-03 00:32:37 -07001250 fieldListLoginDetails = ['backend_status_text', 'email','site','password','is_active','is_readonly','is_admin','public_key']
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001251 fieldListContactInfo = ['firstname','lastname','phone','timezone']
1252
Siobhan Tully53437282013-04-26 19:30:27 -04001253 fieldsets = (
Scott Baker40c00762014-08-21 16:55:59 -07001254 ('Login Details', {'fields': ['backend_status_text', 'email', 'site','password', 'is_active', 'is_readonly', 'is_admin', 'public_key'], 'classes':['suit-tab suit-tab-general']}),
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001255 ('Contact Information', {'fields': ('firstname','lastname','phone', 'timezone'), 'classes':['suit-tab suit-tab-contact']}),
Siobhan Tully53437282013-04-26 19:30:27 -04001256 #('Important dates', {'fields': ('last_login',)}),
1257 )
1258 add_fieldsets = (
1259 (None, {
1260 'classes': ('wide',),
Tony Mackb195b5f2015-01-28 12:03:15 -05001261 'fields': ('site', 'email', 'firstname', 'lastname', 'is_admin', 'is_readonly', 'phone', 'public_key','password1', 'password2')},
Siobhan Tully53437282013-04-26 19:30:27 -04001262 ),
1263 )
Scott Baker40c00762014-08-21 16:55:59 -07001264 readonly_fields = ('backend_status_text', )
Siobhan Tully53437282013-04-26 19:30:27 -04001265 search_fields = ('email',)
1266 ordering = ('email',)
1267 filter_horizontal = ()
1268
Scott Baker3ca51f62014-05-23 12:05:11 -07001269 user_readonly_fields = fieldListLoginDetails + fieldListContactInfo
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001270
Scott Baker0a5633b2014-10-06 17:51:20 -07001271 @property
1272 def suit_form_tabs(self):
1273 if getattr(_thread_locals, "obj", None) is None:
1274 return []
1275 else:
Tony Mack3c01ff92015-01-10 23:08:10 -05001276 tabs = [('general','Login Details'),
Scott Baker0a5633b2014-10-06 17:51:20 -07001277 ('contact','Contact Information'),
1278 ('sliceprivileges','Slice Privileges'),
Scott Baker6e9027f2015-01-29 10:55:53 -08001279 ('siteprivileges','Site Privileges')]
Tony Mack3c01ff92015-01-10 23:08:10 -05001280
1281 request=getattr(_thread_locals, "request", None)
1282 if request and request.user.is_admin:
1283 tabs.append( ('admin-only', 'Admin-Only') )
1284
1285 return tabs
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001286
Tony Mackc2835a92013-05-28 09:18:49 -04001287 def formfield_for_foreignkey(self, db_field, request, **kwargs):
1288 if db_field.name == 'site':
Tony Mack5b061472014-02-04 07:57:10 -05001289 kwargs['queryset'] = Site.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -04001290
1291 return super(UserAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
1292
Tony Mack5b061472014-02-04 07:57:10 -05001293 def queryset(self, request):
1294 return User.select_by_user(request.user)
1295
Tony Mack6235fc82015-01-25 21:58:30 -05001296 def get_form(self, request, obj=None, **kwargs):
Tony Mackb2c407f2015-01-28 12:37:12 -05001297 # copy login details list
1298 login_details_fields = list(self.fieldListLoginDetails)
Tony Mack92b12052015-01-28 12:49:58 -05001299 if not request.user.is_admin:
Scott Baker6e9027f2015-01-29 10:55:53 -08001300 # only admins can see 'is_admin' and 'is_readonly' fields
Tony Mackb2c407f2015-01-28 12:37:12 -05001301 if 'is_admin' in login_details_fields:
1302 login_details_fields.remove('is_admin')
1303 if 'is_readonly' in login_details_fields:
1304 login_details_fields.remove('is_readonly')
Tony Mack92b12052015-01-28 12:49:58 -05001305 #if len(request.user.siteprivileges.filter(role__role = 'pi')) > 0:
Tony Mackb2c407f2015-01-28 12:37:12 -05001306 # only admins and pis can change a user's site
Tony Mack92b12052015-01-28 12:49:58 -05001307 # self.readonly_fields = ('backend_status_text', 'site')
Tony Mackb2c407f2015-01-28 12:37:12 -05001308 self.fieldsets = (
1309 ('Login Details', {'fields': login_details_fields, 'classes':['suit-tab suit-tab-general']}),
1310 ('Contact Information', {'fields': self.fieldListContactInfo, 'classes':['suit-tab suit-tab-contact']}),
1311 )
Tony Mack6235fc82015-01-25 21:58:30 -05001312 return super(UserAdmin, self).get_form(request, obj, **kwargs)
1313
Scott Bakera6a0c772014-12-22 17:35:34 -08001314class ControllerDashboardViewInline(PlStackTabularInline):
1315 model = ControllerDashboardView
Scott Bakereef5a6b2014-12-19 16:41:12 -08001316 extra = 0
1317 fields = ["controller", "url"]
1318 suit_classes = 'suit-tab suit-tab-controllers'
1319
Scott Baker2c3cb642014-05-19 17:55:56 -07001320class DashboardViewAdmin(PlanetStackBaseAdmin):
1321 fieldsets = [('Dashboard View Details',
Scott Bakerb8f3cab2015-01-18 16:33:30 -08001322 {'fields': ['backend_status_text', 'name', 'url', 'enabled'],
Scott Baker2c3cb642014-05-19 17:55:56 -07001323 'classes': ['suit-tab suit-tab-general']})
1324 ]
Scott Baker9daf19c2015-01-18 16:46:26 -08001325 list_display = ["name", "enabled", "url"]
Scott Baker40c00762014-08-21 16:55:59 -07001326 readonly_fields = ('backend_status_text', )
Scott Bakera6a0c772014-12-22 17:35:34 -08001327 inlines = [ControllerDashboardViewInline]
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001328
Scott Bakereef5a6b2014-12-19 16:41:12 -08001329 suit_form_tabs =(('general','Dashboard View Details'),
1330 ('controllers', 'Per-controller Dashboard Details'))
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001331
Scott Baker0165fac2014-01-13 11:49:26 -08001332class ServiceResourceInline(PlStackTabularInline):
Scott Baker3de3e372013-05-10 16:50:44 -07001333 model = ServiceResource
1334 extra = 0
1335
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001336class ServiceClassAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001337 list_display = ('backend_status_icon', 'name', 'commitment', 'membershipFee')
1338 list_display_links = ('backend_status_icon', 'name', )
Scott Baker3de3e372013-05-10 16:50:44 -07001339 inlines = [ServiceResourceInline]
1340
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001341 user_readonly_fields = ['name', 'commitment', 'membershipFee']
1342 user_readonly_inlines = []
1343
Scott Baker0165fac2014-01-13 11:49:26 -08001344class ReservedResourceInline(PlStackTabularInline):
Scott Baker133c9212013-05-17 09:09:11 -07001345 model = ReservedResource
1346 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001347 suit_classes = 'suit-tab suit-tab-reservedresources'
Scott Baker133c9212013-05-17 09:09:11 -07001348
1349 def formfield_for_foreignkey(self, db_field, request=None, **kwargs):
1350 field = super(ReservedResourceInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
1351
1352 if db_field.name == 'resource':
1353 # restrict resources to those that the slice's service class allows
1354 if request._slice is not None:
1355 field.queryset = field.queryset.filter(serviceClass = request._slice.serviceClass, calendarReservable=True)
1356 if len(field.queryset) > 0:
1357 field.initial = field.queryset.all()[0]
1358 else:
1359 field.queryset = field.queryset.none()
1360 elif db_field.name == 'sliver':
1361 # restrict slivers to those that belong to the slice
1362 if request._slice is not None:
1363 field.queryset = field.queryset.filter(slice = request._slice)
1364 else:
1365 field.queryset = field.queryset.none()
1366
1367 return field
1368
Tony Mack5b061472014-02-04 07:57:10 -05001369 def queryset(self, request):
1370 return ReservedResource.select_by_user(request.user)
1371
Scott Baker133c9212013-05-17 09:09:11 -07001372class ReservationChangeForm(forms.ModelForm):
1373 class Meta:
1374 model = Reservation
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001375 widgets = {
1376 'slice' : LinkedSelect
1377 }
Scott Baker133c9212013-05-17 09:09:11 -07001378
1379class ReservationAddForm(forms.ModelForm):
1380 slice = forms.ModelChoiceField(queryset=Slice.objects.all(), widget=forms.Select(attrs={"onChange":"document.getElementById('id_refresh').value=1; submit()"}))
1381 refresh = forms.CharField(widget=forms.HiddenInput())
1382
1383 class Media:
1384 css = {'all': ('planetstack.css',)} # .field-refresh { display: none; }
1385
1386 def clean_slice(self):
1387 slice = self.cleaned_data.get("slice")
1388 x = ServiceResource.objects.filter(serviceClass = slice.serviceClass, calendarReservable=True)
1389 if len(x) == 0:
1390 raise forms.ValidationError("The slice you selected does not have a service class that allows reservations")
1391 return slice
1392
1393 class Meta:
1394 model = Reservation
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001395 widgets = {
1396 'slice' : LinkedSelect
1397 }
1398
Scott Baker133c9212013-05-17 09:09:11 -07001399
1400class ReservationAddRefreshForm(ReservationAddForm):
1401 """ This form is displayed when the Reservation Form receives an update
1402 from the Slice dropdown onChange handler. It doesn't validate the
1403 data and doesn't save the data. This will cause the form to be
1404 redrawn.
1405 """
1406
Scott Baker8737e5f2013-05-17 09:35:32 -07001407 """ don't validate anything other than slice """
1408 dont_validate_fields = ("startTime", "duration")
1409
Scott Baker133c9212013-05-17 09:09:11 -07001410 def full_clean(self):
1411 result = super(ReservationAddForm, self).full_clean()
Scott Baker8737e5f2013-05-17 09:35:32 -07001412
1413 for fieldname in self.dont_validate_fields:
1414 if fieldname in self._errors:
1415 del self._errors[fieldname]
1416
Scott Baker133c9212013-05-17 09:09:11 -07001417 return result
1418
1419 """ don't save anything """
1420 def is_valid(self):
1421 return False
1422
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001423class ReservationAdmin(PlanetStackBaseAdmin):
Scott Baker40c00762014-08-21 16:55:59 -07001424 fieldList = ['backend_status_text', 'slice', 'startTime', 'duration']
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001425 fieldsets = [('Reservation Details', {'fields': fieldList, 'classes': ['suit-tab suit-tab-general']})]
Scott Baker40c00762014-08-21 16:55:59 -07001426 readonly_fields = ('backend_status_text', )
Scott Baker133c9212013-05-17 09:09:11 -07001427 list_display = ('startTime', 'duration')
Scott Baker133c9212013-05-17 09:09:11 -07001428 form = ReservationAddForm
1429
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001430 suit_form_tabs = (('general','Reservation Details'), ('reservedresources','Reserved Resources'))
1431
1432 inlines = [ReservedResourceInline]
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001433 user_readonly_fields = fieldList
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001434
Scott Baker133c9212013-05-17 09:09:11 -07001435 def add_view(self, request, form_url='', extra_context=None):
Scott Bakeracd45142013-05-19 16:19:16 -07001436 timezone.activate(request.user.timezone)
Scott Baker133c9212013-05-17 09:09:11 -07001437 request._refresh = False
1438 request._slice = None
1439 if request.method == 'POST':
Scott Baker8737e5f2013-05-17 09:35:32 -07001440 # "refresh" will be set to "1" if the form was submitted due to
1441 # a change in the Slice dropdown.
Scott Baker133c9212013-05-17 09:09:11 -07001442 if request.POST.get("refresh","1") == "1":
1443 request._refresh = True
1444 request.POST["refresh"] = "0"
Scott Baker8737e5f2013-05-17 09:35:32 -07001445
1446 # Keep track of the slice that was selected, so the
1447 # reservedResource inline can filter items for the slice.
Scott Baker133c9212013-05-17 09:09:11 -07001448 request._slice = request.POST.get("slice",None)
1449 if (request._slice is not None):
1450 request._slice = Slice.objects.get(id=request._slice)
1451
1452 result = super(ReservationAdmin, self).add_view(request, form_url, extra_context)
1453 return result
1454
Scott Bakeracd45142013-05-19 16:19:16 -07001455 def changelist_view(self, request, extra_context = None):
1456 timezone.activate(request.user.timezone)
1457 return super(ReservationAdmin, self).changelist_view(request, extra_context)
1458
Scott Baker133c9212013-05-17 09:09:11 -07001459 def get_form(self, request, obj=None, **kwargs):
Siobhan Tullyd3515752013-06-21 16:34:53 -04001460 request._obj_ = obj
1461 if obj is not None:
1462 # For changes, set request._slice to the slice already set in the
1463 # object.
1464 request._slice = obj.slice
1465 self.form = ReservationChangeForm
1466 else:
1467 if getattr(request, "_refresh", False):
1468 self.form = ReservationAddRefreshForm
1469 else:
1470 self.form = ReservationAddForm
1471 return super(ReservationAdmin, self).get_form(request, obj, **kwargs)
1472
Scott Baker133c9212013-05-17 09:09:11 -07001473 def get_readonly_fields(self, request, obj=None):
Siobhan Tullyd3515752013-06-21 16:34:53 -04001474 if (obj is not None):
1475 # Prevent slice from being changed after the reservation has been
1476 # created.
1477 return ['slice']
1478 else:
Scott Baker133c9212013-05-17 09:09:11 -07001479 return []
Scott Baker3de3e372013-05-10 16:50:44 -07001480
Tony Mack5b061472014-02-04 07:57:10 -05001481 def queryset(self, request):
1482 return Reservation.select_by_user(request.user)
1483
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001484class NetworkParameterTypeAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001485 list_display = ("backend_status_icon", "name", )
1486 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001487 user_readonly_fields = ['name']
1488 user_readonly_inlines = []
Scott Baker74d8e622013-07-29 16:04:22 -07001489
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001490class RouterAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001491 list_display = ("backend_status_icon", "name", )
1492 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001493 user_readonly_fields = ['name']
1494 user_readonly_inlines = []
1495
Scott Baker0165fac2014-01-13 11:49:26 -08001496class RouterInline(PlStackTabularInline):
Scott Baker74d8e622013-07-29 16:04:22 -07001497 model = Router.networks.through
1498 extra = 0
1499 verbose_name_plural = "Routers"
1500 verbose_name = "Router"
Siobhan Tully2d95e482013-09-06 10:56:06 -04001501 suit_classes = 'suit-tab suit-tab-routers'
Scott Baker74d8e622013-07-29 16:04:22 -07001502
Scott Bakerb27b62c2014-08-15 16:29:16 -07001503class NetworkParameterInline(PlStackGenericTabularInline):
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001504 model = NetworkParameter
Scott Baker618e3792014-08-15 13:42:29 -07001505 extra = 0
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001506 verbose_name_plural = "Parameters"
1507 verbose_name = "Parameter"
1508 suit_classes = 'suit-tab suit-tab-netparams'
Scott Baker40c00762014-08-21 16:55:59 -07001509 fields = ['backend_status_icon', 'parameter', 'value']
1510 readonly_fields = ('backend_status_icon', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001511
Scott Baker0165fac2014-01-13 11:49:26 -08001512class NetworkSliversInline(PlStackTabularInline):
Scott Baker40c00762014-08-21 16:55:59 -07001513 fields = ['backend_status_icon', 'network','sliver','ip']
1514 readonly_fields = ("backend_status_icon", "ip", )
Scott Baker74d8e622013-07-29 16:04:22 -07001515 model = NetworkSliver
Scott Baker874936e2014-01-13 18:15:34 -08001516 selflink_fieldname = "sliver"
Scott Baker74d8e622013-07-29 16:04:22 -07001517 extra = 0
1518 verbose_name_plural = "Slivers"
1519 verbose_name = "Sliver"
Siobhan Tully2d95e482013-09-06 10:56:06 -04001520 suit_classes = 'suit-tab suit-tab-networkslivers'
Scott Baker74d8e622013-07-29 16:04:22 -07001521
Scott Baker0165fac2014-01-13 11:49:26 -08001522class NetworkSlicesInline(PlStackTabularInline):
Scott Bakerd7d2a392013-08-06 08:57:30 -07001523 model = NetworkSlice
Scott Baker874936e2014-01-13 18:15:34 -08001524 selflink_fieldname = "slice"
Scott Bakerd7d2a392013-08-06 08:57:30 -07001525 extra = 0
1526 verbose_name_plural = "Slices"
1527 verbose_name = "Slice"
Siobhan Tully2d95e482013-09-06 10:56:06 -04001528 suit_classes = 'suit-tab suit-tab-networkslices'
Scott Baker40c00762014-08-21 16:55:59 -07001529 fields = ['backend_status_icon', 'network','slice']
1530 readonly_fields = ('backend_status_icon', )
Scott Bakerd7d2a392013-08-06 08:57:30 -07001531
Tony Macka7dbd422015-01-05 22:48:11 -05001532class ControllerNetworkInline(PlStackTabularInline):
1533 model = ControllerNetwork
Scott Bakerfbb45862014-10-17 16:27:23 -07001534 extra = 0
Tony Mack336e0f92014-11-30 15:53:08 -05001535 verbose_name_plural = "Controller Networks"
1536 verbose_name = "Controller Network"
Scott Bakerfbb45862014-10-17 16:27:23 -07001537 suit_classes = 'suit-tab suit-tab-admin-only'
Tony Mack336e0f92014-11-30 15:53:08 -05001538 fields = ['backend_status_icon', 'controller','net_id','subnet_id']
Scott Bakerfbb45862014-10-17 16:27:23 -07001539 readonly_fields = ('backend_status_icon', )
1540
Scott Baker9f6b8ed2014-11-17 23:44:03 -08001541class NetworkForm(forms.ModelForm):
1542 class Meta:
1543 model = Network
1544 widgets = {
1545 'topologyParameters': UploadTextareaWidget,
1546 'controllerParameters': UploadTextareaWidget,
1547 }
1548
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001549class NetworkAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001550 list_display = ("backend_status_icon", "name", "subnet", "ports", "labels")
1551 list_display_links = ('backend_status_icon', 'name', )
Scott Baker74d8e622013-07-29 16:04:22 -07001552 readonly_fields = ("subnet", )
Siobhan Tully2d95e482013-09-06 10:56:06 -04001553
Scott Bakerd7d2a392013-08-06 08:57:30 -07001554 inlines = [NetworkParameterInline, NetworkSliversInline, NetworkSlicesInline, RouterInline]
Tony Macka7dbd422015-01-05 22:48:11 -05001555 admin_inlines = [ControllerNetworkInline]
Scott Baker74d8e622013-07-29 16:04:22 -07001556
Scott Baker9f6b8ed2014-11-17 23:44:03 -08001557 form=NetworkForm
1558
Siobhan Tully2d95e482013-09-06 10:56:06 -04001559 fieldsets = [
Scott Baker0451fb62015-01-03 12:29:29 -08001560 (None, {'fields': ['backend_status_text', 'name','template','ports','labels','owner','guaranteed_bandwidth', 'permit_all_slices','permitted_slices','network_id','router_id','subnet_id','subnet'],
Scott Baker40248712014-11-17 16:04:45 -08001561 'classes':['suit-tab suit-tab-general']}),
Scott Baker0451fb62015-01-03 12:29:29 -08001562 (None, {'fields': ['topology_parameters', 'controller_url', 'controller_parameters'],
Scott Baker40248712014-11-17 16:04:45 -08001563 'classes':['suit-tab suit-tab-sdn']}),
1564 ]
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001565
Scott Baker40c00762014-08-21 16:55:59 -07001566 readonly_fields = ('backend_status_text', )
Scott Baker0451fb62015-01-03 12:29:29 -08001567 user_readonly_fields = ['name','template','ports','labels','owner','guaranteed_bandwidth', 'permit_all_slices','permitted_slices','network_id','router_id','subnet_id','subnet']
Siobhan Tully2d95e482013-09-06 10:56:06 -04001568
Scott Bakerfbb45862014-10-17 16:27:23 -07001569 @property
1570 def suit_form_tabs(self):
1571 tabs=[('general','Network Details'),
Scott Baker40248712014-11-17 16:04:45 -08001572 ('sdn', 'SDN Configuration'),
Scott Bakerfbb45862014-10-17 16:27:23 -07001573 ('netparams', 'Parameters'),
1574 ('networkslivers','Slivers'),
1575 ('networkslices','Slices'),
1576 ('routers','Routers'),
1577 ]
1578
1579 request=getattr(_thread_locals, "request", None)
1580 if request and request.user.is_admin:
1581 tabs.append( ('admin-only', 'Admin-Only') )
1582
1583 return tabs
1584
1585
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001586class NetworkTemplateAdmin(PlanetStackBaseAdmin):
Scott Baker81fa17f2015-01-03 12:03:38 -08001587 list_display = ("backend_status_icon", "name", "guaranteed_bandwidth", "visibility")
Scott Baker63d1a552014-08-21 15:19:07 -07001588 list_display_links = ('backend_status_icon', 'name', )
Scott Baker81fa17f2015-01-03 12:03:38 -08001589 user_readonly_fields = ["name", "guaranteed_bandwidth", "visibility"]
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001590 user_readonly_inlines = []
Scott Baker40248712014-11-17 16:04:45 -08001591 fieldsets = [
Scott Baker81fa17f2015-01-03 12:03:38 -08001592 (None, {'fields': ['name', 'description', 'guaranteed_bandwidth', 'visibility', 'translation', 'shared_network_name', 'shared_network_id', 'topology_kind', 'controller_kind'],
Scott Baker40248712014-11-17 16:04:45 -08001593 'classes':['suit-tab suit-tab-general']}),]
1594 suit_form_tabs = (('general','Network Template Details'), )
Scott Baker74d8e622013-07-29 16:04:22 -07001595
Scott Baker37b47902014-09-02 14:37:41 -07001596class FlavorAdmin(PlanetStackBaseAdmin):
1597 list_display = ("backend_status_icon", "name", "flavor", "order", "default")
1598 list_display_links = ("backend_status_icon", "name")
1599 user_readonly_fields = ("name", "flavor")
1600 fields = ("name", "description", "flavor", "order", "default")
1601
Tony Mack31c2b8f2013-04-26 20:01:42 -04001602# register a signal that caches the user's credentials when they log in
1603def cache_credentials(sender, user, request, **kwds):
1604 auth = {'username': request.POST['username'],
1605 'password': request.POST['password']}
1606 request.session['auth'] = auth
1607user_logged_in.connect(cache_credentials)
1608
Scott Baker15cddfa2013-12-09 13:45:19 -08001609def dollar_field(fieldName, short_description):
1610 def newFunc(self, obj):
1611 try:
1612 x= "$ %0.2f" % float(getattr(obj, fieldName, 0.0))
1613 except:
1614 x=getattr(obj, fieldName, 0.0)
1615 return x
1616 newFunc.short_description = short_description
1617 return newFunc
1618
1619def right_dollar_field(fieldName, short_description):
1620 def newFunc(self, obj):
1621 try:
1622 #x= '<div align=right style="width:6em">$ %0.2f</div>' % float(getattr(obj, fieldName, 0.0))
1623 x= '<div align=right>$ %0.2f</div>' % float(getattr(obj, fieldName, 0.0))
1624 except:
1625 x=getattr(obj, fieldName, 0.0)
1626 return x
1627 newFunc.short_description = short_description
1628 newFunc.allow_tags = True
1629 return newFunc
Scott Baker43105042013-12-06 23:23:36 -08001630
Scott Baker0165fac2014-01-13 11:49:26 -08001631class InvoiceChargeInline(PlStackTabularInline):
Scott Baker43105042013-12-06 23:23:36 -08001632 model = Charge
Scott Baker15cddfa2013-12-09 13:45:19 -08001633 extra = 0
Scott Baker43105042013-12-06 23:23:36 -08001634 verbose_name_plural = "Charges"
1635 verbose_name = "Charge"
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001636 exclude = ['account']
Scott Baker9cb88a22013-12-09 18:56:00 -08001637 fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
1638 readonly_fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
1639 can_delete = False
1640 max_num = 0
1641
1642 dollar_amount = right_dollar_field("amount", "Amount")
Scott Baker43105042013-12-06 23:23:36 -08001643
1644class InvoiceAdmin(admin.ModelAdmin):
1645 list_display = ("date", "account")
1646
1647 inlines = [InvoiceChargeInline]
1648
Scott Baker9cb88a22013-12-09 18:56:00 -08001649 fields = ["date", "account", "dollar_amount"]
1650 readonly_fields = ["date", "account", "dollar_amount"]
1651
1652 dollar_amount = dollar_field("amount", "Amount")
Scott Baker43105042013-12-06 23:23:36 -08001653
Scott Baker0165fac2014-01-13 11:49:26 -08001654class InvoiceInline(PlStackTabularInline):
Scott Baker15cddfa2013-12-09 13:45:19 -08001655 model = Invoice
1656 extra = 0
1657 verbose_name_plural = "Invoices"
1658 verbose_name = "Invoice"
Scott Baker0165fac2014-01-13 11:49:26 -08001659 fields = ["date", "dollar_amount"]
1660 readonly_fields = ["date", "dollar_amount"]
Scott Baker15cddfa2013-12-09 13:45:19 -08001661 suit_classes = 'suit-tab suit-tab-accountinvoice'
1662 can_delete=False
1663 max_num=0
1664
1665 dollar_amount = right_dollar_field("amount", "Amount")
1666
Scott Baker0165fac2014-01-13 11:49:26 -08001667class PendingChargeInline(PlStackTabularInline):
Scott Baker43105042013-12-06 23:23:36 -08001668 model = Charge
Scott Baker15cddfa2013-12-09 13:45:19 -08001669 extra = 0
Scott Baker43105042013-12-06 23:23:36 -08001670 verbose_name_plural = "Charges"
1671 verbose_name = "Charge"
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001672 exclude = ["invoice"]
Scott Baker15cddfa2013-12-09 13:45:19 -08001673 fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
1674 readonly_fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
Scott Baker43105042013-12-06 23:23:36 -08001675 suit_classes = 'suit-tab suit-tab-accountpendingcharges'
Scott Baker15cddfa2013-12-09 13:45:19 -08001676 can_delete=False
1677 max_num=0
Scott Baker43105042013-12-06 23:23:36 -08001678
1679 def queryset(self, request):
1680 qs = super(PendingChargeInline, self).queryset(request)
1681 qs = qs.filter(state="pending")
1682 return qs
1683
Scott Baker15cddfa2013-12-09 13:45:19 -08001684 dollar_amount = right_dollar_field("amount", "Amount")
1685
Scott Baker0165fac2014-01-13 11:49:26 -08001686class PaymentInline(PlStackTabularInline):
Scott Baker43105042013-12-06 23:23:36 -08001687 model=Payment
1688 extra = 1
1689 verbose_name_plural = "Payments"
1690 verbose_name = "Payment"
Scott Baker15cddfa2013-12-09 13:45:19 -08001691 fields = ["date", "dollar_amount"]
1692 readonly_fields = ["date", "dollar_amount"]
Scott Baker43105042013-12-06 23:23:36 -08001693 suit_classes = 'suit-tab suit-tab-accountpayments'
Scott Baker15cddfa2013-12-09 13:45:19 -08001694 can_delete=False
1695 max_num=0
1696
1697 dollar_amount = right_dollar_field("amount", "Amount")
1698
Scott Baker43105042013-12-06 23:23:36 -08001699class AccountAdmin(admin.ModelAdmin):
1700 list_display = ("site", "balance_due")
1701
1702 inlines = [InvoiceInline, PaymentInline, PendingChargeInline]
1703
1704 fieldsets = [
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001705 (None, {'fields': ['site', 'dollar_balance_due', 'dollar_total_invoices', 'dollar_total_payments'],'classes':['suit-tab suit-tab-general']}),]
Scott Baker43105042013-12-06 23:23:36 -08001706
Scott Baker15cddfa2013-12-09 13:45:19 -08001707 readonly_fields = ['site', 'dollar_balance_due', 'dollar_total_invoices', 'dollar_total_payments']
Scott Baker43105042013-12-06 23:23:36 -08001708
1709 suit_form_tabs =(
1710 ('general','Account Details'),
1711 ('accountinvoice', 'Invoices'),
1712 ('accountpayments', 'Payments'),
1713 ('accountpendingcharges','Pending Charges'),
1714 )
1715
Scott Baker15cddfa2013-12-09 13:45:19 -08001716 dollar_balance_due = dollar_field("balance_due", "Balance Due")
1717 dollar_total_invoices = dollar_field("total_invoices", "Total Invoices")
1718 dollar_total_payments = dollar_field("total_payments", "Total Payments")
1719
Siobhan Tully53437282013-04-26 19:30:27 -04001720# Now register the new UserAdmin...
Siobhan Tully30fd4292013-05-10 08:59:56 -04001721admin.site.register(User, UserAdmin)
Siobhan Tully53437282013-04-26 19:30:27 -04001722# ... and, since we're not using Django's builtin permissions,
1723# unregister the Group model from admin.
Siobhan Tullyce652d02013-10-08 21:52:35 -04001724#admin.site.unregister(Group)
Siobhan Tully53437282013-04-26 19:30:27 -04001725
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001726#Do not show django evolution in the admin interface
1727from django_evolution.models import Version, Evolution
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001728#admin.site.unregister(Version)
1729#admin.site.unregister(Evolution)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001730
1731
1732# When debugging it is often easier to see all the classes, but for regular use
1733# only the top-levels should be displayed
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001734showAll = False
Scott Baker43105042013-12-06 23:23:36 -08001735
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001736admin.site.register(Deployment, DeploymentAdmin)
Tony Mack336e0f92014-11-30 15:53:08 -05001737admin.site.register(Controller, ControllerAdmin)
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001738admin.site.register(Site, SiteAdmin)
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001739admin.site.register(Slice, SliceAdmin)
Siobhan Tullyce652d02013-10-08 21:52:35 -04001740admin.site.register(Service, ServiceAdmin)
Tony Mack598eaf22015-01-25 12:35:29 -05001741#admin.site.register(Reservation, ReservationAdmin)
Scott Baker74d8e622013-07-29 16:04:22 -07001742admin.site.register(Network, NetworkAdmin)
1743admin.site.register(Router, RouterAdmin)
Scott Baker74d8e622013-07-29 16:04:22 -07001744admin.site.register(NetworkTemplate, NetworkTemplateAdmin)
Tony Mack598eaf22015-01-25 12:35:29 -05001745#admin.site.register(Account, AccountAdmin)
1746#admin.site.register(Invoice, InvoiceAdmin)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001747
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001748if True:
1749 admin.site.register(NetworkParameterType, NetworkParameterTypeAdmin)
1750 admin.site.register(ServiceClass, ServiceClassAdmin)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001751 #admin.site.register(PlanetStack)
Siobhan Tullyd3515752013-06-21 16:34:53 -04001752 admin.site.register(Tag, TagAdmin)
Tony Mack336e0f92014-11-30 15:53:08 -05001753 admin.site.register(ControllerRole)
Siobhan Tullyce652d02013-10-08 21:52:35 -04001754 admin.site.register(SiteRole)
1755 admin.site.register(SliceRole)
1756 admin.site.register(PlanetStackRole)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001757 admin.site.register(Node, NodeAdmin)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001758 #admin.site.register(SlicePrivilege, SlicePrivilegeAdmin)
1759 #admin.site.register(SitePrivilege, SitePrivilegeAdmin)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001760 admin.site.register(Sliver, SliverAdmin)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001761 admin.site.register(Image, ImageAdmin)
Scott Baker2c3cb642014-05-19 17:55:56 -07001762 admin.site.register(DashboardView, DashboardViewAdmin)
Scott Baker37b47902014-09-02 14:37:41 -07001763 admin.site.register(Flavor, FlavorAdmin)
Tony Mack7130ac32013-03-22 21:58:00 -04001764