blob: 1f8875867c2e72a2ad69be4febd65fb58a2b08e1 [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
Siobhan Tully4bc09f22013-04-10 21:15:21 -040010from django.contrib.admin.widgets import FilteredSelectMultiple
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
Tony Mack7130ac32013-03-22 21:58:00 -040018
Scott Baker36f50872014-08-21 13:01:25 -070019import django_evolution
Scott Baker0a5633b2014-10-06 17:51:20 -070020import threading
21
22# thread locals necessary to work around a django-suit issue
23_thread_locals = threading.local()
Scott Baker36f50872014-08-21 13:01:25 -070024
Scott Baker40c00762014-08-21 16:55:59 -070025def backend_icon(obj): # backend_status, enacted, updated):
26 #return "%s %s %s" % (str(obj.updated), str(obj.enacted), str(obj.backend_status))
27 if (obj.enacted is not None) and obj.enacted >= obj.updated:
Scott Bakerfbf06642014-09-09 10:38:15 -070028 return '<span style="min-width:16px;"><img src="/static/admin/img/icon_success.gif"></span>'
Scott Baker40c00762014-08-21 16:55:59 -070029 else:
30 if obj.backend_status == "Provisioning in progress" or obj.backend_status=="":
Scott Bakerfbf06642014-09-09 10:38:15 -070031 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 -070032 else:
Scott Bakerfbf06642014-09-09 10:38:15 -070033 return '<span style="min-width:16px;" title="%s"><img src="/static/admin/img/icon_error.gif"></span>' % obj.backend_status
Scott Baker40c00762014-08-21 16:55:59 -070034
35def backend_text(obj):
36 icon = backend_icon(obj)
37 if (obj.enacted is not None) and obj.enacted >= obj.updated:
38 return "%s %s" % (icon, "successfully enacted") # enacted on %s" % str(obj.enacted))
39 else:
40 return "%s %s" % (icon, obj.backend_status)
Scott Baker63d1a552014-08-21 15:19:07 -070041
Scott Baker36f50872014-08-21 13:01:25 -070042class PlainTextWidget(forms.HiddenInput):
43 input_type = 'hidden'
44
45 def render(self, name, value, attrs=None):
46 if value is None:
47 value = ''
48 return mark_safe(str(value) + super(PlainTextWidget, self).render(name, value, attrs))
49
Scott Baker86c83ab2014-10-03 13:10:47 -070050class PermissionCheckingAdminMixin(object):
Scott Bakercbfb6002014-10-03 00:32:37 -070051 # call save_by_user and delete_by_user instead of save and delete
Siobhan Tullycf04fb62014-01-11 11:25:57 -050052
53 def has_add_permission(self, request, obj=None):
54 return (not self.__user_is_readonly(request))
Scott Baker36f50872014-08-21 13:01:25 -070055
Siobhan Tullycf04fb62014-01-11 11:25:57 -050056 def has_delete_permission(self, request, obj=None):
57 return (not self.__user_is_readonly(request))
58
59 def save_model(self, request, obj, form, change):
60 if self.__user_is_readonly(request):
Scott Bakercbfb6002014-10-03 00:32:37 -070061 # this 'if' might be redundant if save_by_user is implemented right
Siobhan Tullycf04fb62014-01-11 11:25:57 -050062 raise PermissionDenied
Scott Bakercbfb6002014-10-03 00:32:37 -070063
64 obj.caller = request.user
65 # update openstack connection to use this site/tenant
66 obj.save_by_user(request.user)
67
68 def delete_model(self, request, obj):
69 obj.delete_by_user(request.user)
70
71 def save_formset(self, request, form, formset, change):
72 instances = formset.save(commit=False)
73 for instance in instances:
74 instance.save_by_user(request.user)
75
76 # BUG in django 1.7? Objects are not deleted by formset.save if
77 # commit is False. So let's delete them ourselves.
78 #
79 # code from forms/models.py save_existing_objects()
80 try:
81 forms_to_delete = formset.deleted_forms
82 except AttributeError:
83 forms_to_delete = []
84 if formset.initial_forms:
85 for form in formset.initial_forms:
86 obj = form.instance
87 if form in forms_to_delete:
88 if obj.pk is None:
89 continue
90 formset.deleted_objects.append(obj)
91 obj.delete()
92
93 formset.save_m2m()
Siobhan Tullycf04fb62014-01-11 11:25:57 -050094
95 def get_actions(self,request):
Scott Baker86c83ab2014-10-03 13:10:47 -070096 actions = super(PermissionCheckingAdminMixin,self).get_actions(request)
Siobhan Tullycf04fb62014-01-11 11:25:57 -050097
98 if self.__user_is_readonly(request):
99 if 'delete_selected' in actions:
100 del actions['delete_selected']
101
102 return actions
103
104 def change_view(self,request,object_id, extra_context=None):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500105 if self.__user_is_readonly(request):
Scott Bakeraf73e102014-04-22 22:40:07 -0700106 if not hasattr(self, "readonly_save"):
107 # save the original readonly fields
108 self.readonly_save = self.readonly_fields
109 self.inlines_save = self.inlines
Scott Bakere8859f92014-05-23 12:42:40 -0700110 if hasattr(self, "user_readonly_fields"):
111 self.readonly_fields=self.user_readonly_fields
112 if hasattr(self, "user_readonly_inlines"):
113 self.inlines = self.user_readonly_inlines
Scott Bakeraf73e102014-04-22 22:40:07 -0700114 else:
115 if hasattr(self, "readonly_save"):
116 # restore the original readonly fields
117 self.readonly_fields = self.readonly_save
Scott Bakere8859f92014-05-23 12:42:40 -0700118 if hasattr(self, "inlines_save"):
Scott Bakeraf73e102014-04-22 22:40:07 -0700119 self.inlines = self.inlines_save
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500120
121 try:
Scott Baker86c83ab2014-10-03 13:10:47 -0700122 return super(PermissionCheckingAdminMixin, self).change_view(request, object_id, extra_context=extra_context)
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500123 except PermissionDenied:
124 pass
125 if request.method == 'POST':
126 raise PermissionDenied
127 request.readonly = True
Scott Baker86c83ab2014-10-03 13:10:47 -0700128 return super(PermissionCheckingAdminMixin, self).change_view(request, object_id, extra_context=extra_context)
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500129
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500130 def __user_is_readonly(self, request):
131 return request.user.isReadOnlyUser()
132
Scott Baker40c00762014-08-21 16:55:59 -0700133 def backend_status_text(self, obj):
134 return mark_safe(backend_text(obj))
Scott Baker36f50872014-08-21 13:01:25 -0700135
Scott Baker63d1a552014-08-21 15:19:07 -0700136 def backend_status_icon(self, obj):
Scott Baker40c00762014-08-21 16:55:59 -0700137 return mark_safe(backend_icon(obj))
Scott Baker63d1a552014-08-21 15:19:07 -0700138 backend_status_icon.short_description = ""
139
Scott Baker5c432692014-10-16 00:57:55 -0700140 def get_form(self, request, obj=None):
141 # Save obj and request in thread-local storage, so suit_form_tabs can
142 # use it to determine whether we're in edit or add mode, and can
143 # determine whether the user is an admin.
144 _thread_locals.request = request
145 _thread_locals.obj = obj
146 return super(PermissionCheckingAdminMixin, self).get_form(request, obj)
147
148 def get_inline_instances(self, request, obj=None):
149 inlines = super(PermissionCheckingAdminMixin, self).get_inline_instances(request, obj)
150
151 # inlines that should only be shown to an admin user
152 if request.user.is_admin:
153 for inline_class in getattr(self, "admin_inlines", []):
154 inlines.append(inline_class(self.model, self.admin_site))
155
156 return inlines
157
Scott Baker86c83ab2014-10-03 13:10:47 -0700158class ReadOnlyAwareAdmin(PermissionCheckingAdminMixin, admin.ModelAdmin):
159 # Note: Make sure PermissionCheckingAdminMixin is listed before
160 # admin.ModelAdmin in the class declaration.
161
Scott Bakercbfb6002014-10-03 00:32:37 -0700162 pass
163
164class PlanetStackBaseAdmin(ReadOnlyAwareAdmin):
165 save_on_top = False
Scott Baker36f50872014-08-21 13:01:25 -0700166
Scott Bakere8859f92014-05-23 12:42:40 -0700167class SingletonAdmin (ReadOnlyAwareAdmin):
Siobhan Tullyce652d02013-10-08 21:52:35 -0400168 def has_add_permission(self, request):
Scott Bakere8859f92014-05-23 12:42:40 -0700169 if not super(SingletonAdmin, self).has_add_permission(request):
170 return False
171
Siobhan Tullyce652d02013-10-08 21:52:35 -0400172 num_objects = self.model.objects.count()
173 if num_objects >= 1:
174 return False
175 else:
176 return True
177
Siobhan Tullyd3515752013-06-21 16:34:53 -0400178class PlStackTabularInline(admin.TabularInline):
Scott Baker86568322014-01-12 16:53:31 -0800179 def __init__(self, *args, **kwargs):
180 super(PlStackTabularInline, self).__init__(*args, **kwargs)
181
182 # InlineModelAdmin as no get_fields() method, so in order to add
183 # the selflink field, we override __init__ to modify self.fields and
184 # self.readonly_fields.
185
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800186 self.setup_selflink()
187
Scott Baker874936e2014-01-13 18:15:34 -0800188 def get_change_url(self, model, id):
189 """ Get the URL to a change form in the admin for this model """
190 reverse_path = "admin:%s_change" % (model._meta.db_table)
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800191 try:
Scott Baker874936e2014-01-13 18:15:34 -0800192 url = reverse(reverse_path, args=(id,))
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800193 except NoReverseMatch:
Scott Baker874936e2014-01-13 18:15:34 -0800194 return None
195
196 return url
197
198 def setup_selflink(self):
199 if hasattr(self, "selflink_fieldname"):
200 """ self.selflink_model can be defined to punch through a relation
201 to its target object. For example, in SliceNetworkInline, set
202 selflink_model = "network", and the URL will lead to the Network
203 object instead of trying to bring up a change view of the
204 SliceNetwork object.
205 """
206 self.selflink_model = getattr(self.model,self.selflink_fieldname).field.rel.to
207 else:
208 self.selflink_model = self.model
209
210 url = self.get_change_url(self.selflink_model, 0)
211
212 # We don't have an admin for this object, so don't create the
213 # selflink.
214 if (url == None):
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800215 return
216
Scott Baker874936e2014-01-13 18:15:34 -0800217 # Since we need to add "selflink" to the field list, we need to create
218 # self.fields if it is None.
Scott Baker0165fac2014-01-13 11:49:26 -0800219 if (self.fields is None):
220 self.fields = []
221 for f in self.model._meta.fields:
222 if f.editable and f.name != "id":
223 self.fields.append(f.name)
Scott Baker86568322014-01-12 16:53:31 -0800224
Scott Baker874936e2014-01-13 18:15:34 -0800225 self.fields = tuple(self.fields) + ("selflink", )
Scott Baker86568322014-01-12 16:53:31 -0800226
Scott Baker874936e2014-01-13 18:15:34 -0800227 if self.readonly_fields is None:
228 self.readonly_fields = ()
Scott Baker86568322014-01-12 16:53:31 -0800229
Scott Baker874936e2014-01-13 18:15:34 -0800230 self.readonly_fields = tuple(self.readonly_fields) + ("selflink", )
Scott Baker86568322014-01-12 16:53:31 -0800231
232 def selflink(self, obj):
Scott Baker874936e2014-01-13 18:15:34 -0800233 if hasattr(self, "selflink_fieldname"):
234 obj = getattr(obj, self.selflink_fieldname)
235
Scott Baker86568322014-01-12 16:53:31 -0800236 if obj.id:
Scott Baker874936e2014-01-13 18:15:34 -0800237 url = self.get_change_url(self.selflink_model, obj.id)
238 return "<a href='%s'>Details</a>" % str(url)
Scott Baker86568322014-01-12 16:53:31 -0800239 else:
Scott Bakere2bbf7e2014-01-13 12:09:31 -0800240 return "Not present"
Scott Baker86568322014-01-12 16:53:31 -0800241
242 selflink.allow_tags = True
243 selflink.short_description = "Details"
Siobhan Tullyd3515752013-06-21 16:34:53 -0400244
Scott Bakerb27b62c2014-08-15 16:29:16 -0700245 def has_add_permission(self, request):
246 return not request.user.isReadOnlyUser()
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500247
248 def get_readonly_fields(self, request, obj=None):
Scott Bakerb27b62c2014-08-15 16:29:16 -0700249 readonly_fields = list(self.readonly_fields)[:]
250 if request.user.isReadOnlyUser():
251 for field in self.fields:
252 if not field in readonly_fields:
253 readonly_fields.append(field)
254 return readonly_fields
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500255
Scott Baker40c00762014-08-21 16:55:59 -0700256 def backend_status_icon(self, obj):
257 return mark_safe(backend_icon(obj))
258 backend_status_icon.short_description = ""
Scott Baker36f50872014-08-21 13:01:25 -0700259
Scott Bakerb27b62c2014-08-15 16:29:16 -0700260class PlStackGenericTabularInline(generic.GenericTabularInline):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500261 def has_add_permission(self, request):
Scott Bakerb27b62c2014-08-15 16:29:16 -0700262 return not request.user.isReadOnlyUser()
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500263
Scott Bakerb27b62c2014-08-15 16:29:16 -0700264 def get_readonly_fields(self, request, obj=None):
265 readonly_fields = list(self.readonly_fields)[:]
266 if request.user.isReadOnlyUser():
267 for field in self.fields:
268 if not field in readonly_fields:
269 readonly_fields.append(field)
270 return readonly_fields
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500271
Scott Baker40c00762014-08-21 16:55:59 -0700272 def backend_status_icon(self, obj):
273 return mark_safe(backend_icon(obj))
274 backend_status_icon.short_description = ""
275
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400276class ReservationInline(PlStackTabularInline):
277 model = Reservation
278 extra = 0
279 suit_classes = 'suit-tab suit-tab-reservations'
Scott Baker36f50872014-08-21 13:01:25 -0700280
Tony Mack5b061472014-02-04 07:57:10 -0500281 def queryset(self, request):
282 return Reservation.select_by_user(request.user)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400283
Scott Bakerb27b62c2014-08-15 16:29:16 -0700284class TagInline(PlStackGenericTabularInline):
Siobhan Tullyde5450d2013-06-21 11:35:33 -0400285 model = Tag
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400286 extra = 0
287 suit_classes = 'suit-tab suit-tab-tags'
Tony Mack5b061472014-02-04 07:57:10 -0500288 fields = ['service', 'name', 'value']
289
290 def queryset(self, request):
291 return Tag.select_by_user(request.user)
Siobhan Tullyde5450d2013-06-21 11:35:33 -0400292
Scott Baker74d8e622013-07-29 16:04:22 -0700293class NetworkLookerUpper:
Siobhan Tully2c780ad2013-09-06 11:22:40 -0400294 """ This is a callable that looks up a network name in a sliver and returns
295 the ip address for that network.
296 """
297
Scott Baker434ca7e2014-08-15 12:29:20 -0700298 byNetworkName = {} # class variable
299
Siobhan Tully2c780ad2013-09-06 11:22:40 -0400300 def __init__(self, name):
301 self.short_description = name
302 self.__name__ = name
303 self.network_name = name
304
305 def __call__(self, obj):
306 if obj is not None:
307 for nbs in obj.networksliver_set.all():
308 if (nbs.network.name == self.network_name):
309 return nbs.ip
Scott Baker74d8e622013-07-29 16:04:22 -0700310 return ""
311
312 def __str__(self):
313 return self.network_name
314
Scott Baker434ca7e2014-08-15 12:29:20 -0700315 @staticmethod
316 def get(network_name):
317 """ We want to make sure we alwars return the same NetworkLookerUpper
318 because sometimes django will cause them to be instantiated multiple
319 times (and we don't want different ones in form.fields vs
320 SliverInline.readonly_fields).
321 """
322 if network_name not in NetworkLookerUpper.byNetworkName:
323 NetworkLookerUpper.byNetworkName[network_name] = NetworkLookerUpper(network_name)
324 return NetworkLookerUpper.byNetworkName[network_name]
325
Siobhan Tullyd3515752013-06-21 16:34:53 -0400326class SliverInline(PlStackTabularInline):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400327 model = Sliver
Scott Baker7a61dc42014-09-02 17:08:20 -0700328 fields = ['backend_status_icon', 'all_ips_string', 'instance_name', 'slice', 'deploymentNetwork', 'flavor', 'image', 'node']
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400329 extra = 0
Scott Baker40c00762014-08-21 16:55:59 -0700330 readonly_fields = ['backend_status_icon', 'all_ips_string', 'instance_name']
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400331 suit_classes = 'suit-tab suit-tab-slivers'
Scott Baker74d8e622013-07-29 16:04:22 -0700332
Tony Mack5b061472014-02-04 07:57:10 -0500333 def queryset(self, request):
334 return Sliver.select_by_user(request.user)
335
Scott Bakerb24cc932014-06-09 10:51:16 -0700336 def formfield_for_foreignkey(self, db_field, request=None, **kwargs):
Scott Bakerb24cc932014-06-09 10:51:16 -0700337 if db_field.name == 'deploymentNetwork':
Scott Baker3b678742014-06-09 13:11:54 -0700338 kwargs['queryset'] = Deployment.select_by_acl(request.user)
Scott Baker7a61dc42014-09-02 17:08:20 -0700339 kwargs['widget'] = forms.Select(attrs={'onChange': "sliver_deployment_changed(this);"})
Scott Baker32481312014-09-08 12:14:14 -0700340 elif db_field.name == 'flavor':
341 kwargs['widget'] = forms.Select(attrs={'onChange': "sliver_flavor_changed(this);"})
Scott Baker3b678742014-06-09 13:11:54 -0700342
343 field = super(SliverInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
Scott Bakerb24cc932014-06-09 10:51:16 -0700344
345 return field
346
Siobhan Tullyd3515752013-06-21 16:34:53 -0400347class SiteInline(PlStackTabularInline):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400348 model = Site
349 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400350 suit_classes = 'suit-tab suit-tab-sites'
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400351
Tony Mack5b061472014-02-04 07:57:10 -0500352 def queryset(self, request):
353 return Site.select_by_user(request.user)
354
Siobhan Tullyd3515752013-06-21 16:34:53 -0400355class UserInline(PlStackTabularInline):
Siobhan Tully30fd4292013-05-10 08:59:56 -0400356 model = User
Scott Baker40c00762014-08-21 16:55:59 -0700357 fields = ['backend_status_icon', 'email', 'firstname', 'lastname']
358 readonly_fields = ('backend_status_icon', )
Siobhan Tully30fd4292013-05-10 08:59:56 -0400359 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400360 suit_classes = 'suit-tab suit-tab-users'
Siobhan Tully30fd4292013-05-10 08:59:56 -0400361
Tony Mack5b061472014-02-04 07:57:10 -0500362 def queryset(self, request):
363 return User.select_by_user(request.user)
364
Siobhan Tullyd3515752013-06-21 16:34:53 -0400365class SliceInline(PlStackTabularInline):
Tony Mack00d361f2013-04-28 10:28:42 -0400366 model = Slice
Scott Baker40c00762014-08-21 16:55:59 -0700367 fields = ['backend_status_icon', 'name', 'site', 'serviceClass', 'service']
368 readonly_fields = ('backend_status_icon', )
Tony Mack00d361f2013-04-28 10:28:42 -0400369 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400370 suit_classes = 'suit-tab suit-tab-slices'
371
Tony Mack5b061472014-02-04 07:57:10 -0500372 def queryset(self, request):
373 return Slice.select_by_user(request.user)
374
Siobhan Tullyd3515752013-06-21 16:34:53 -0400375class NodeInline(PlStackTabularInline):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400376 model = Node
377 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400378 suit_classes = 'suit-tab suit-tab-nodes'
Scott Baker40c00762014-08-21 16:55:59 -0700379 fields = ['backend_status_icon', 'name','deployment','site']
380 readonly_fields = ('backend_status_icon', )
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400381
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400382class DeploymentPrivilegeInline(PlStackTabularInline):
383 model = DeploymentPrivilege
384 extra = 0
385 suit_classes = 'suit-tab suit-tab-deploymentprivileges'
Scott Baker40c00762014-08-21 16:55:59 -0700386 fields = ['backend_status_icon', 'user','role','deployment']
387 readonly_fields = ('backend_status_icon', )
Tony Mack5b061472014-02-04 07:57:10 -0500388
389 def queryset(self, request):
390 return DeploymentPrivilege.select_by_user(request.user)
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400391
Siobhan Tullyd3515752013-06-21 16:34:53 -0400392class SitePrivilegeInline(PlStackTabularInline):
Siobhan Tullyaa1bcd52013-05-10 12:43:09 -0400393 model = SitePrivilege
394 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400395 suit_classes = 'suit-tab suit-tab-siteprivileges'
Scott Baker40c00762014-08-21 16:55:59 -0700396 fields = ['backend_status_icon', 'user','site', 'role']
397 readonly_fields = ('backend_status_icon', )
Siobhan Tullyaa1bcd52013-05-10 12:43:09 -0400398
Tony Mackc2835a92013-05-28 09:18:49 -0400399 def formfield_for_foreignkey(self, db_field, request, **kwargs):
400 if db_field.name == 'site':
Tony Mack5b061472014-02-04 07:57:10 -0500401 kwargs['queryset'] = Site.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400402
403 if db_field.name == 'user':
Tony Mack5b061472014-02-04 07:57:10 -0500404 kwargs['queryset'] = User.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400405 return super(SitePrivilegeInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
406
Tony Mack5b061472014-02-04 07:57:10 -0500407 def queryset(self, request):
408 return SitePrivilege.select_by_user(request.user)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400409
Tony Macke4be32f2014-03-11 20:45:25 -0400410class SiteDeploymentInline(PlStackTabularInline):
411 model = SiteDeployments
Tony Macke4be32f2014-03-11 20:45:25 -0400412 extra = 0
413 suit_classes = 'suit-tab suit-tab-deployments'
Scott Baker40c00762014-08-21 16:55:59 -0700414 fields = ['backend_status_icon', 'deployment','site']
415 readonly_fields = ('backend_status_icon', )
Tony Macke4be32f2014-03-11 20:45:25 -0400416
417 def formfield_for_foreignkey(self, db_field, request, **kwargs):
418 if db_field.name == 'site':
419 kwargs['queryset'] = Site.select_by_user(request.user)
420
421 if db_field.name == 'deployment':
422 kwargs['queryset'] = Deployment.select_by_user(request.user)
423 return super(SiteDeploymentInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
424
425 def queryset(self, request):
426 return SiteDeployments.select_by_user(request.user)
427
428
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400429class SlicePrivilegeInline(PlStackTabularInline):
430 model = SlicePrivilege
431 suit_classes = 'suit-tab suit-tab-sliceprivileges'
432 extra = 0
Scott Baker40c00762014-08-21 16:55:59 -0700433 fields = ('backend_status_icon', 'user', 'slice', 'role')
434 readonly_fields = ('backend_status_icon', )
Siobhan Tullyaa1bcd52013-05-10 12:43:09 -0400435
Tony Mackc2835a92013-05-28 09:18:49 -0400436 def formfield_for_foreignkey(self, db_field, request, **kwargs):
437 if db_field.name == 'slice':
Scott Baker36f50872014-08-21 13:01:25 -0700438 kwargs['queryset'] = Slice.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400439 if db_field.name == 'user':
Scott Baker36f50872014-08-21 13:01:25 -0700440 kwargs['queryset'] = User.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400441
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400442 return super(SlicePrivilegeInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
Tony Mackc2835a92013-05-28 09:18:49 -0400443
Tony Mack5b061472014-02-04 07:57:10 -0500444 def queryset(self, request):
445 return SlicePrivilege.select_by_user(request.user)
446
Scott Bakera0015eb2013-08-14 17:28:14 -0700447class SliceNetworkInline(PlStackTabularInline):
Scott Baker74d8e622013-07-29 16:04:22 -0700448 model = Network.slices.through
Scott Baker874936e2014-01-13 18:15:34 -0800449 selflink_fieldname = "network"
Scott Baker74d8e622013-07-29 16:04:22 -0700450 extra = 0
451 verbose_name = "Network Connection"
452 verbose_name_plural = "Network Connections"
Siobhan Tully2d95e482013-09-06 10:56:06 -0400453 suit_classes = 'suit-tab suit-tab-slicenetworks'
Scott Baker40c00762014-08-21 16:55:59 -0700454 fields = ['backend_status_icon', 'network']
455 readonly_fields = ('backend_status_icon', )
Scott Baker2170b972014-06-03 12:14:07 -0700456
457class ImageDeploymentsInline(PlStackTabularInline):
458 model = ImageDeployments
459 extra = 0
460 verbose_name = "Image Deployments"
461 verbose_name_plural = "Image Deployments"
462 suit_classes = 'suit-tab suit-tab-imagedeployments'
Scott Baker40c00762014-08-21 16:55:59 -0700463 fields = ['backend_status_icon', 'image', 'deployment', 'glance_image_id']
464 readonly_fields = ['backend_status_icon', 'glance_image_id']
Scott Baker74d8e622013-07-29 16:04:22 -0700465
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400466class SliceRoleAdmin(PlanetStackBaseAdmin):
467 model = SliceRole
468 pass
469
470class SiteRoleAdmin(PlanetStackBaseAdmin):
471 model = SiteRole
472 pass
473
Siobhan Tullybf1153a2013-05-27 20:53:48 -0400474class DeploymentAdminForm(forms.ModelForm):
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400475 sites = forms.ModelMultipleChoiceField(
476 queryset=Site.objects.all(),
477 required=False,
Scott Baker70983182014-06-09 22:10:00 -0700478 help_text="Select which sites are allowed to host nodes in this deployment",
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400479 widget=FilteredSelectMultiple(
480 verbose_name=('Sites'), is_stacked=False
481 )
482 )
Scott Bakerde0f4412014-06-11 15:40:26 -0700483 images = forms.ModelMultipleChoiceField(
484 queryset=Image.objects.all(),
485 required=False,
486 help_text="Select which images should be deployed on this deployment",
487 widget=FilteredSelectMultiple(
488 verbose_name=('Images'), is_stacked=False
489 )
490 )
Scott Baker37b47902014-09-02 14:37:41 -0700491 flavors = forms.ModelMultipleChoiceField(
492 queryset=Flavor.objects.all(),
493 required=False,
494 help_text="Select which flavors should be usable on this deployment",
495 widget=FilteredSelectMultiple(
496 verbose_name=('Flavors'), is_stacked=False
497 )
498 )
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400499 class Meta:
Siobhan Tullybf1153a2013-05-27 20:53:48 -0400500 model = Deployment
Scott Baker37b47902014-09-02 14:37:41 -0700501 many_to_many = ["flavors",]
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400502
Siobhan Tully320b4622014-01-17 15:11:14 -0500503 def __init__(self, *args, **kwargs):
Scott Baker5380c522014-06-06 14:49:43 -0700504 request = kwargs.pop('request', None)
Siobhan Tully320b4622014-01-17 15:11:14 -0500505 super(DeploymentAdminForm, self).__init__(*args, **kwargs)
506
Scott Baker5380c522014-06-06 14:49:43 -0700507 self.fields['accessControl'].initial = "allow site " + request.user.site.name
508
Siobhan Tully320b4622014-01-17 15:11:14 -0500509 if self.instance and self.instance.pk:
Scott Bakerc9b14f72014-05-22 13:44:20 -0700510 self.fields['sites'].initial = [x.site for x in self.instance.sitedeployments_set.all()]
Scott Bakerde0f4412014-06-11 15:40:26 -0700511 self.fields['images'].initial = [x.image for x in self.instance.imagedeployments_set.all()]
Scott Baker37b47902014-09-02 14:37:41 -0700512 self.fields['flavors'].initial = self.instance.flavors.all()
Scott Bakerde0f4412014-06-11 15:40:26 -0700513
514 def manipulate_m2m_objs(self, this_obj, selected_objs, all_relations, relation_class, local_attrname, foreign_attrname):
515 """ helper function for handling m2m relations from the MultipleChoiceField
516
517 this_obj: the source object we want to link from
518
519 selected_objs: a list of destination objects we want to link to
520
521 all_relations: the full set of relations involving this_obj, including ones we don't want
522
523 relation_class: the class that implements the relation from source to dest
524
525 local_attrname: field name representing this_obj in relation_class
526
527 foreign_attrname: field name representing selected_objs in relation_class
528
529 This function will remove all newobjclass relations from this_obj
530 that are not contained in selected_objs, and add any relations that
531 are in selected_objs but don't exist in the data model yet.
532 """
533
534 existing_dest_objs = []
535 for relation in list(all_relations):
536 if getattr(relation, foreign_attrname) not in selected_objs:
537 #print "deleting site", sdp.site
538 relation.delete()
539 else:
540 existing_dest_objs.append(getattr(relation, foreign_attrname))
541
542 for dest_obj in selected_objs:
543 if dest_obj not in existing_dest_objs:
544 #print "adding site", site
545 kwargs = {foreign_attrname: dest_obj, local_attrname: this_obj}
546 relation = relation_class(**kwargs)
547 relation.save()
Siobhan Tully320b4622014-01-17 15:11:14 -0500548
549 def save(self, commit=True):
550 deployment = super(DeploymentAdminForm, self).save(commit=False)
551
552 if commit:
553 deployment.save()
Scott Baker61b6aec2014-10-06 17:17:40 -0700554 # this has to be done after save() if/when a deployment is first created
555 deployment.flavors = self.cleaned_data['flavors']
Siobhan Tully320b4622014-01-17 15:11:14 -0500556
557 if deployment.pk:
Scott Bakerc9b14f72014-05-22 13:44:20 -0700558 # save_m2m() doesn't seem to work with 'through' relations. So we
559 # create/destroy the through models ourselves. There has to be
560 # a better way...
561
Scott Bakerde0f4412014-06-11 15:40:26 -0700562 self.manipulate_m2m_objs(deployment, self.cleaned_data['sites'], deployment.sitedeployments_set.all(), SiteDeployments, "deployment", "site")
563 self.manipulate_m2m_objs(deployment, self.cleaned_data['images'], deployment.imagedeployments_set.all(), ImageDeployments, "deployment", "image")
Scott Bakerc9b14f72014-05-22 13:44:20 -0700564
Scott Baker37b47902014-09-02 14:37:41 -0700565 self.save_m2m()
Siobhan Tully320b4622014-01-17 15:11:14 -0500566
567 return deployment
568
Scott Bakerff5e0f32014-05-22 14:40:27 -0700569class DeploymentAdminROForm(DeploymentAdminForm):
570 def save(self, commit=True):
571 raise PermissionDenied
572
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500573class SiteAssocInline(PlStackTabularInline):
574 model = Site.deployments.through
575 extra = 0
576 suit_classes = 'suit-tab suit-tab-sites'
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400577
Siobhan Tullybf1153a2013-05-27 20:53:48 -0400578class DeploymentAdmin(PlanetStackBaseAdmin):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500579 model = Deployment
Tony Mack2cbd3802014-09-29 16:10:52 -0400580 fieldList = ['backend_status_text', 'name', 'availability_zone', 'sites', 'images', 'flavors', 'accessControl']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500581 fieldsets = [(None, {'fields': fieldList, 'classes':['suit-tab suit-tab-sites']})]
Scott Bakerde0f4412014-06-11 15:40:26 -0700582 inlines = [DeploymentPrivilegeInline,NodeInline,TagInline] # ,ImageDeploymentsInline]
Scott Baker63d1a552014-08-21 15:19:07 -0700583 list_display = ['backend_status_icon', 'name']
584 list_display_links = ('backend_status_icon', 'name', )
Scott Baker40c00762014-08-21 16:55:59 -0700585 readonly_fields = ('backend_status_text', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500586
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500587 user_readonly_fields = ['name']
588
Scott Bakerde0f4412014-06-11 15:40:26 -0700589 suit_form_tabs =(('sites','Deployment Details'),('nodes','Nodes'),('deploymentprivileges','Privileges'),('tags','Tags')) # ,('imagedeployments','Images'))
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500590
Scott Bakerff5e0f32014-05-22 14:40:27 -0700591 def get_form(self, request, obj=None, **kwargs):
592 if request.user.isReadOnlyUser():
593 kwargs["form"] = DeploymentAdminROForm
594 else:
595 kwargs["form"] = DeploymentAdminForm
Scott Baker5380c522014-06-06 14:49:43 -0700596 adminForm = super(DeploymentAdmin,self).get_form(request, obj, **kwargs)
597
598 # from stackexchange: pass the request object into the form
599
600 class AdminFormMetaClass(adminForm):
601 def __new__(cls, *args, **kwargs):
602 kwargs['request'] = request
603 return adminForm(*args, **kwargs)
604
605 return AdminFormMetaClass
606
Siobhan Tullyce652d02013-10-08 21:52:35 -0400607class ServiceAttrAsTabInline(PlStackTabularInline):
608 model = ServiceAttribute
609 fields = ['name','value']
610 extra = 0
611 suit_classes = 'suit-tab suit-tab-serviceattrs'
612
Siobhan Tullyce652d02013-10-08 21:52:35 -0400613class ServiceAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -0700614 list_display = ("backend_status_icon","name","description","versionNumber","enabled","published")
615 list_display_links = ('backend_status_icon', 'name', )
Scott Baker40c00762014-08-21 16:55:59 -0700616 fieldList = ["backend_status_text","name","description","versionNumber","enabled","published"]
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500617 fieldsets = [(None, {'fields': fieldList, 'classes':['suit-tab suit-tab-general']})]
618 inlines = [ServiceAttrAsTabInline,SliceInline]
Scott Baker40c00762014-08-21 16:55:59 -0700619 readonly_fields = ('backend_status_text', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500620
621 user_readonly_fields = fieldList
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500622
623 suit_form_tabs =(('general', 'Service Details'),
624 ('slices','Slices'),
625 ('serviceattrs','Additional Attributes'),
626 )
Siobhan Tullyce652d02013-10-08 21:52:35 -0400627
Tony Mack0553f282013-06-10 22:54:50 -0400628class SiteAdmin(PlanetStackBaseAdmin):
Scott Baker40c00762014-08-21 16:55:59 -0700629 fieldList = ['backend_status_text', 'name', 'site_url', 'enabled', 'is_public', 'login_base', 'accountLink','location']
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400630 fieldsets = [
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500631 (None, {'fields': fieldList, 'classes':['suit-tab suit-tab-general']}),
Tony Macke4be32f2014-03-11 20:45:25 -0400632 #('Deployment Networks', {'fields': ['deployments'], 'classes':['suit-tab suit-tab-deployments']}),
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400633 ]
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400634 suit_form_tabs =(('general', 'Site Details'),
635 ('users','Users'),
Siobhan Tully2d95e482013-09-06 10:56:06 -0400636 ('siteprivileges','Privileges'),
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400637 ('deployments','Deployments'),
638 ('slices','Slices'),
Scott Baker36f50872014-08-21 13:01:25 -0700639 ('nodes','Nodes'),
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400640 ('tags','Tags'),
641 )
Scott Baker40c00762014-08-21 16:55:59 -0700642 readonly_fields = ['backend_status_text', 'accountLink']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500643
644 user_readonly_fields = ['name', 'deployments','site_url', 'enabled', 'is_public', 'login_base', 'accountLink']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500645
Scott Baker63d1a552014-08-21 15:19:07 -0700646 list_display = ('backend_status_icon', 'name', 'login_base','site_url', 'enabled')
647 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400648 filter_horizontal = ('deployments',)
Tony Macke4be32f2014-03-11 20:45:25 -0400649 inlines = [SliceInline,UserInline,TagInline, NodeInline, SitePrivilegeInline, SiteDeploymentInline]
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400650 search_fields = ['name']
651
Tony Mack04062832013-05-10 08:22:44 -0400652 def queryset(self, request):
Tony Mack5b061472014-02-04 07:57:10 -0500653 return Site.select_by_user(request.user)
Tony Mack04062832013-05-10 08:22:44 -0400654
Tony Mack5cd13202013-05-01 21:48:38 -0400655 def get_formsets(self, request, obj=None):
656 for inline in self.get_inline_instances(request, obj):
657 # hide MyInline in the add view
658 if obj is None:
659 continue
Tony Mack2bd5b412013-06-11 21:05:06 -0400660 if isinstance(inline, SliverInline):
661 inline.model.caller = request.user
Tony Mack5cd13202013-05-01 21:48:38 -0400662 yield inline.get_formset(request, obj)
663
Scott Baker545db2a2013-12-09 18:44:43 -0800664 def accountLink(self, obj):
665 link_obj = obj.accounts.all()
666 if link_obj:
667 reverse_path = "admin:core_account_change"
668 url = reverse(reverse_path, args =(link_obj[0].id,))
669 return "<a href='%s'>%s</a>" % (url, "view billing details")
670 else:
671 return "no billing data for this site"
672 accountLink.allow_tags = True
673 accountLink.short_description = "Billing"
674
Tony Mack332ee1d2014-02-04 15:33:45 -0500675 def save_model(self, request, obj, form, change):
676 # update openstack connection to use this site/tenant
677 obj.save_by_user(request.user)
678
679 def delete_model(self, request, obj):
680 obj.delete_by_user(request.user)
681
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500682
Tony Mack9bcbe4f2013-04-29 08:13:27 -0400683class SitePrivilegeAdmin(PlanetStackBaseAdmin):
Scott Baker40c00762014-08-21 16:55:59 -0700684 fieldList = ['backend_status_text', 'user', 'site', 'role']
Tony Mack00d361f2013-04-28 10:28:42 -0400685 fieldsets = [
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500686 (None, {'fields': fieldList, 'classes':['collapse']})
Tony Mack00d361f2013-04-28 10:28:42 -0400687 ]
Scott Baker40c00762014-08-21 16:55:59 -0700688 readonly_fields = ('backend_status_text', )
Scott Baker63d1a552014-08-21 15:19:07 -0700689 list_display = ('backend_status_icon', 'user', 'site', 'role')
690 list_display_links = list_display
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500691 user_readonly_fields = fieldList
692 user_readonly_inlines = []
Tony Mack00d361f2013-04-28 10:28:42 -0400693
Tony Mackc2835a92013-05-28 09:18:49 -0400694 def formfield_for_foreignkey(self, db_field, request, **kwargs):
695 if db_field.name == 'site':
696 if not request.user.is_admin:
697 # only show sites where user is an admin or pi
698 sites = set()
699 for site_privilege in SitePrivilege.objects.filer(user=request.user):
700 if site_privilege.role.role_type in ['admin', 'pi']:
701 sites.add(site_privilege.site)
702 kwargs['queryset'] = Site.objects.filter(site__in=list(sites))
703
704 if db_field.name == 'user':
705 if not request.user.is_admin:
706 # only show users from sites where caller has admin or pi role
707 roles = Role.objects.filter(role_type__in=['admin', 'pi'])
708 site_privileges = SitePrivilege.objects.filter(user=request.user).filter(role__in=roles)
709 sites = [site_privilege.site for site_privilege in site_privileges]
710 site_privileges = SitePrivilege.objects.filter(site__in=sites)
711 emails = [site_privilege.user.email for site_privilege in site_privileges]
712 users = User.objects.filter(email__in=emails)
713 kwargs['queryset'] = users
714
715 return super(SitePrivilegeAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
716
Tony Mack04062832013-05-10 08:22:44 -0400717 def queryset(self, request):
718 # admins can see all privileges. Users can only see privileges at sites
Tony Mackc2835a92013-05-28 09:18:49 -0400719 # where they have the admin role or pi role.
Tony Mack04062832013-05-10 08:22:44 -0400720 qs = super(SitePrivilegeAdmin, self).queryset(request)
Tony Mack5b061472014-02-04 07:57:10 -0500721 #if not request.user.is_admin:
722 # roles = Role.objects.filter(role_type__in=['admin', 'pi'])
723 # site_privileges = SitePrivilege.objects.filter(user=request.user).filter(role__in=roles)
724 # login_bases = [site_privilege.site.login_base for site_privilege in site_privileges]
725 # sites = Site.objects.filter(login_base__in=login_bases)
726 # qs = qs.filter(site__in=sites)
Tony Mack04062832013-05-10 08:22:44 -0400727 return qs
728
Siobhan Tullyce652d02013-10-08 21:52:35 -0400729class SliceForm(forms.ModelForm):
730 class Meta:
731 model = Slice
732 widgets = {
Scott Baker36f50872014-08-21 13:01:25 -0700733 'service': LinkedSelect
Siobhan Tullyce652d02013-10-08 21:52:35 -0400734 }
735
Tony Mack2cbd3802014-09-29 16:10:52 -0400736 def clean(self):
737 cleaned_data = super(SliceForm, self).clean()
738 name = cleaned_data.get('name')
Scott Baker6efad462014-10-06 23:09:59 -0700739 site = cleaned_data.get('site')
740 if (not isinstance(site,Site)):
741 # previous code indicates 'site' could be a site_id and not a site?
742 site = Slice.objects.get(id=site.id)
Tony Mack2cbd3802014-09-29 16:10:52 -0400743 if not name.startswith(site.login_base):
744 raise forms.ValidationError('slice name must begin with %s' % site.login_base)
745 return cleaned_data
746
Scott Bakerc4efdc72014-10-15 16:54:04 -0700747class SliceDeploymentsInline(PlStackTabularInline):
748 model = SliceDeployments
749 extra = 0
750 verbose_name = "Slice Deployment"
751 verbose_name_plural = "Slice Deployments"
752 suit_classes = 'suit-tab suit-tab-admin-only'
753 fields = ['backend_status_icon', 'deployment', 'tenant_id']
754 readonly_fields = ('backend_status_icon', )
755
Tony Mack2bd5b412013-06-11 21:05:06 -0400756class SliceAdmin(PlanetStackBaseAdmin):
Siobhan Tullyce652d02013-10-08 21:52:35 -0400757 form = SliceForm
Tony Mackfbb26fc2014-09-02 07:03:27 -0400758 fieldList = ['backend_status_text', 'site', 'name', 'serviceClass', 'enabled','description', 'service', 'slice_url', 'max_slivers']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500759 fieldsets = [('Slice Details', {'fields': fieldList, 'classes':['suit-tab suit-tab-general']}),]
Scott Baker40c00762014-08-21 16:55:59 -0700760 readonly_fields = ('backend_status_text', )
Tony Mack7d459902014-09-03 13:18:57 -0400761 list_display = ('backend_status_icon', 'name', 'site','serviceClass', 'slice_url', 'max_slivers')
762 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tully2d95e482013-09-06 10:56:06 -0400763 inlines = [SlicePrivilegeInline,SliverInline, TagInline, ReservationInline,SliceNetworkInline]
Scott Bakerc4efdc72014-10-15 16:54:04 -0700764 admin_inlines = [SliceDeploymentsInline]
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400765
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500766 user_readonly_fields = fieldList
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400767
Scott Bakerc4efdc72014-10-15 16:54:04 -0700768# suit_form_tabs =(('general', 'Slice Details'),
769# ('slicenetworks','Networks'),
770# ('sliceprivileges','Privileges'),
771# ('slivers','Slivers'),
772# ('tags','Tags'),
773# ('reservations','Reservations'),
774# )
775
Scott Bakerc4efdc72014-10-15 16:54:04 -0700776 @property
777 def suit_form_tabs(self):
778 tabs =[('general', 'Slice Details'),
779 ('slicenetworks','Networks'),
780 ('sliceprivileges','Privileges'),
781 ('slivers','Slivers'),
782 ('tags','Tags'),
783 ('reservations','Reservations'),
784 ]
785
786 request=getattr(_thread_locals, "request", None)
787 if request and request.user.is_admin:
788 tabs.append( ('admin-only', 'Admin-Only') )
789
790 return tabs
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400791
Scott Baker510fdbb2014-08-05 17:19:24 -0700792 def render_change_form(self, request, context, add=False, change=False, form_url='', obj=None):
Scott Baker510fdbb2014-08-05 17:19:24 -0700793 deployment_nodes = []
794 for node in Node.objects.all():
795 deployment_nodes.append( (node.deployment.id, node.id, node.name) )
796
Scott Baker7a61dc42014-09-02 17:08:20 -0700797 deployment_flavors = []
798 for flavor in Flavor.objects.all():
799 for deployment in flavor.deployments.all():
800 deployment_flavors.append( (deployment.id, flavor.id, flavor.name) )
801
Scott Baker93e80cd2014-09-09 09:58:49 -0700802 deployment_images = []
803 for image in Image.objects.all():
804 for imageDeployment in image.imagedeployments_set.all():
805 deployment_images.append( (imageDeployment.deployment.id, image.id, image.name) )
806
Tony Mackec23b992014-09-02 21:18:45 -0400807 site_login_bases = []
808 for site in Site.objects.all():
Scott Baker93e80cd2014-09-09 09:58:49 -0700809 site_login_bases.append((site.id, site.login_base))
810
Scott Baker510fdbb2014-08-05 17:19:24 -0700811 context["deployment_nodes"] = deployment_nodes
Scott Baker7a61dc42014-09-02 17:08:20 -0700812 context["deployment_flavors"] = deployment_flavors
Scott Baker93e80cd2014-09-09 09:58:49 -0700813 context["deployment_images"] = deployment_images
Tony Mackec23b992014-09-02 21:18:45 -0400814 context["site_login_bases"] = site_login_bases
Scott Baker510fdbb2014-08-05 17:19:24 -0700815 return super(SliceAdmin, self).render_change_form(request, context, add, change, form_url, obj)
816
Tony Mackc2835a92013-05-28 09:18:49 -0400817 def formfield_for_foreignkey(self, db_field, request, **kwargs):
818 if db_field.name == 'site':
Tony Mack5b061472014-02-04 07:57:10 -0500819 kwargs['queryset'] = Site.select_by_user(request.user)
Tony Mackec23b992014-09-02 21:18:45 -0400820 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 -0700821
Tony Mackc2835a92013-05-28 09:18:49 -0400822 return super(SliceAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
823
Tony Mack04062832013-05-10 08:22:44 -0400824 def queryset(self, request):
825 # admins can see all keys. Users can only see slices they belong to.
Tony Mack5b061472014-02-04 07:57:10 -0500826 return Slice.select_by_user(request.user)
Tony Mack04062832013-05-10 08:22:44 -0400827
Tony Mack79748612013-05-01 14:52:03 -0400828 def get_formsets(self, request, obj=None):
829 for inline in self.get_inline_instances(request, obj):
830 # hide MyInline in the add view
831 if obj is None:
832 continue
Tony Mack2bd5b412013-06-11 21:05:06 -0400833 if isinstance(inline, SliverInline):
834 inline.model.caller = request.user
Tony Mack79748612013-05-01 14:52:03 -0400835 yield inline.get_formset(request, obj)
836
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400837class SlicePrivilegeAdmin(PlanetStackBaseAdmin):
Tony Mack00d361f2013-04-28 10:28:42 -0400838 fieldsets = [
Scott Baker40c00762014-08-21 16:55:59 -0700839 (None, {'fields': ['backend_status_text', 'user', 'slice', 'role']})
Tony Mack00d361f2013-04-28 10:28:42 -0400840 ]
Scott Baker40c00762014-08-21 16:55:59 -0700841 readonly_fields = ('backend_status_text', )
Scott Baker63d1a552014-08-21 15:19:07 -0700842 list_display = ('backend_status_icon', 'user', 'slice', 'role')
843 list_display_links = list_display
Tony Mack00d361f2013-04-28 10:28:42 -0400844
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500845 user_readonly_fields = ['user', 'slice', 'role']
846 user_readonly_inlines = []
847
Tony Mackc2835a92013-05-28 09:18:49 -0400848 def formfield_for_foreignkey(self, db_field, request, **kwargs):
849 if db_field.name == 'slice':
Tony Mack5b061472014-02-04 07:57:10 -0500850 kwargs['queryset'] = Slice.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400851
852 if db_field.name == 'user':
Tony Mack5b061472014-02-04 07:57:10 -0500853 kwargs['queryset'] = User.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400854
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400855 return super(SlicePrivilegeAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
Tony Mackc2835a92013-05-28 09:18:49 -0400856
Tony Mack04062832013-05-10 08:22:44 -0400857 def queryset(self, request):
858 # admins can see all memberships. Users can only see memberships of
859 # slices where they have the admin role.
Tony Mack5b061472014-02-04 07:57:10 -0500860 return SlicePrivilege.select_by_user(request.user)
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400861
Tony Mack9bcbe4f2013-04-29 08:13:27 -0400862 def save_model(self, request, obj, form, change):
Tony Mack951dab42013-05-02 19:51:45 -0400863 # update openstack connection to use this site/tenant
864 auth = request.session.get('auth', {})
Tony Mackf7f79a12014-08-11 11:21:42 -0400865 auth['tenant'] = obj.slice.slicename
Tony Mack951dab42013-05-02 19:51:45 -0400866 obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
Tony Mack9bcbe4f2013-04-29 08:13:27 -0400867 obj.save()
868
869 def delete_model(self, request, obj):
Tony Mack951dab42013-05-02 19:51:45 -0400870 # update openstack connection to use this site/tenant
871 auth = request.session.get('auth', {})
Tony Mackf7f79a12014-08-11 11:21:42 -0400872 auth['tenant'] = obj.slice.slicename
Tony Mack951dab42013-05-02 19:51:45 -0400873 obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
Tony Mack9bcbe4f2013-04-29 08:13:27 -0400874 obj.delete()
875
Siobhan Tully567e3e62013-06-21 18:03:16 -0400876
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400877class ImageAdmin(PlanetStackBaseAdmin):
878
Scott Baker36f50872014-08-21 13:01:25 -0700879 fieldsets = [('Image Details',
Scott Baker40c00762014-08-21 16:55:59 -0700880 {'fields': ['backend_status_text', 'name', 'disk_format', 'container_format'],
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400881 'classes': ['suit-tab suit-tab-general']})
882 ]
Scott Baker40c00762014-08-21 16:55:59 -0700883 readonly_fields = ('backend_status_text', )
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400884
Scott Baker2170b972014-06-03 12:14:07 -0700885 suit_form_tabs =(('general','Image Details'),('slivers','Slivers'),('imagedeployments','Deployments'))
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400886
Scott Baker2170b972014-06-03 12:14:07 -0700887 inlines = [SliverInline, ImageDeploymentsInline]
Scott Bakerb6f99242014-06-11 11:34:44 -0700888
Tony Mack32e1ce32014-05-07 13:29:41 -0400889 user_readonly_fields = ['name', 'disk_format', 'container_format']
Scott Bakerb27b62c2014-08-15 16:29:16 -0700890
Scott Baker63d1a552014-08-21 15:19:07 -0700891 list_display = ['backend_status_icon', 'name']
892 list_display_links = ('backend_status_icon', 'name', )
893
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400894class NodeForm(forms.ModelForm):
895 class Meta:
896 widgets = {
897 'site': LinkedSelect,
898 'deployment': LinkedSelect
899 }
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400900
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500901class NodeAdmin(PlanetStackBaseAdmin):
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400902 form = NodeForm
Scott Baker63d1a552014-08-21 15:19:07 -0700903 list_display = ('backend_status_icon', 'name', 'site', 'deployment')
904 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tullybf1153a2013-05-27 20:53:48 -0400905 list_filter = ('deployment',)
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500906
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400907 inlines = [TagInline,SliverInline]
Scott Baker40c00762014-08-21 16:55:59 -0700908 fieldsets = [('Node Details', {'fields': ['backend_status_text', 'name','site','deployment'], 'classes':['suit-tab suit-tab-details']})]
909 readonly_fields = ('backend_status_text', )
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400910
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500911 user_readonly_fields = ['name','site','deployment']
912 user_readonly_inlines = [TagInline,SliverInline]
913
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400914 suit_form_tabs =(('details','Node Details'),('slivers','Slivers'),('tags','Tags'))
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400915
Siobhan Tully567e3e62013-06-21 18:03:16 -0400916
Tony Mackd90cdbf2013-04-16 22:48:40 -0400917class SliverForm(forms.ModelForm):
918 class Meta:
Tony Mack1d6b85f2013-05-07 18:49:14 -0400919 model = Sliver
Tony Mackd90cdbf2013-04-16 22:48:40 -0400920 ip = forms.CharField(widget=PlainTextWidget)
Tony Mack18261812013-05-02 16:39:20 -0400921 instance_name = forms.CharField(widget=PlainTextWidget)
Tony Mackd90cdbf2013-04-16 22:48:40 -0400922 widgets = {
923 'ip': PlainTextWidget(),
Tony Mack18261812013-05-02 16:39:20 -0400924 'instance_name': PlainTextWidget(),
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400925 'slice': LinkedSelect,
926 'deploymentNetwork': LinkedSelect,
927 'node': LinkedSelect,
928 'image': LinkedSelect
Siobhan Tully53437282013-04-26 19:30:27 -0400929 }
Tony Mackd90cdbf2013-04-16 22:48:40 -0400930
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500931class TagAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -0700932 list_display = ['backend_status_icon', 'service', 'name', 'value', 'content_type', 'content_object',]
933 list_display_links = list_display
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500934 user_readonly_fields = ['service', 'name', 'value', 'content_type', 'content_object',]
935 user_readonly_inlines = []
Siobhan Tullyd3515752013-06-21 16:34:53 -0400936
Tony Mack9bcbe4f2013-04-29 08:13:27 -0400937class SliverAdmin(PlanetStackBaseAdmin):
Tony Mackd90cdbf2013-04-16 22:48:40 -0400938 form = SliverForm
Tony Mackcdec0902013-04-15 00:38:49 -0400939 fieldsets = [
Scott Baker7a61dc42014-09-02 17:08:20 -0700940 ('Sliver Details', {'fields': ['backend_status_text', 'slice', 'deploymentNetwork', 'node', 'ip', 'instance_name', 'flavor', 'image', ], 'classes': ['suit-tab suit-tab-general'], })
Tony Mackcdec0902013-04-15 00:38:49 -0400941 ]
Scott Baker40c00762014-08-21 16:55:59 -0700942 readonly_fields = ('backend_status_text', )
Scott Baker7a61dc42014-09-02 17:08:20 -0700943 list_display = ['backend_status_icon', 'ip', 'instance_name', 'slice', 'flavor', 'image', 'node', 'deploymentNetwork']
Scott Baker63d1a552014-08-21 15:19:07 -0700944 list_display_links = ('backend_status_icon', 'ip',)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400945
946 suit_form_tabs =(('general', 'Sliver Details'),
947 ('tags','Tags'),
948 )
949
Siobhan Tullyde5450d2013-06-21 11:35:33 -0400950 inlines = [TagInline]
Tony Mack53106f32013-04-27 16:43:01 -0400951
Scott Baker7a61dc42014-09-02 17:08:20 -0700952 user_readonly_fields = ['slice', 'deploymentNetwork', 'node', 'ip', 'instance_name', 'flavor', 'image']
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500953
Tony Mackc2835a92013-05-28 09:18:49 -0400954 def formfield_for_foreignkey(self, db_field, request, **kwargs):
955 if db_field.name == 'slice':
Tony Mack5b061472014-02-04 07:57:10 -0500956 kwargs['queryset'] = Slice.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -0400957
958 return super(SliverAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
959
Tony Mack04062832013-05-10 08:22:44 -0400960 def queryset(self, request):
Scott Baker36f50872014-08-21 13:01:25 -0700961 # admins can see all slivers. Users can only see slivers of
Tony Mack04062832013-05-10 08:22:44 -0400962 # the slices they belong to.
Tony Mack5b061472014-02-04 07:57:10 -0500963 return Sliver.select_by_user(request.user)
964
Tony Mack04062832013-05-10 08:22:44 -0400965
Tony Mack1d6b85f2013-05-07 18:49:14 -0400966 def get_formsets(self, request, obj=None):
967 # make some fields read only if we are updating an existing record
968 if obj == None:
Scott Baker36f50872014-08-21 13:01:25 -0700969 #self.readonly_fields = ('ip', 'instance_name')
Scott Baker40c00762014-08-21 16:55:59 -0700970 self.readonly_fields = ('backend_status_text')
Tony Mack1d6b85f2013-05-07 18:49:14 -0400971 else:
Scott Baker40c00762014-08-21 16:55:59 -0700972 self.readonly_fields = ('backend_status_text')
Scott Baker36f50872014-08-21 13:01:25 -0700973 #self.readonly_fields = ('ip', 'instance_name', 'slice', 'image', 'key')
Tony Mack1d6b85f2013-05-07 18:49:14 -0400974
975 for inline in self.get_inline_instances(request, obj):
976 # hide MyInline in the add view
977 if obj is None:
978 continue
Scott Baker526b71e2014-05-13 13:18:01 -0700979 if isinstance(inline, SliverInline):
980 inline.model.caller = request.user
981 yield inline.get_formset(request, obj)
Tony Mack53106f32013-04-27 16:43:01 -0400982
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500983 #def save_model(self, request, obj, form, change):
984 # # update openstack connection to use this site/tenant
985 # auth = request.session.get('auth', {})
986 # auth['tenant'] = obj.slice.name
987 # obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
988 # obj.creator = request.user
989 # obj.save()
Tony Mack53106f32013-04-27 16:43:01 -0400990
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500991 #def delete_model(self, request, obj):
992 # # update openstack connection to use this site/tenant
993 # auth = request.session.get('auth', {})
994 # auth['tenant'] = obj.slice.name
995 # obj.os_manager = OpenStackManager(auth=auth, caller=request.user)
996 # obj.delete()
Tony Mackcdec0902013-04-15 00:38:49 -0400997
Siobhan Tully53437282013-04-26 19:30:27 -0400998class UserCreationForm(forms.ModelForm):
999 """A form for creating new users. Includes all the required
1000 fields, plus a repeated password."""
1001 password1 = forms.CharField(label='Password', widget=forms.PasswordInput)
1002 password2 = forms.CharField(label='Password confirmation', widget=forms.PasswordInput)
1003
1004 class Meta:
Siobhan Tully30fd4292013-05-10 08:59:56 -04001005 model = User
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001006 fields = ('email', 'firstname', 'lastname', 'phone', 'public_key')
Siobhan Tully53437282013-04-26 19:30:27 -04001007
1008 def clean_password2(self):
1009 # Check that the two password entries match
1010 password1 = self.cleaned_data.get("password1")
1011 password2 = self.cleaned_data.get("password2")
1012 if password1 and password2 and password1 != password2:
1013 raise forms.ValidationError("Passwords don't match")
1014 return password2
1015
1016 def save(self, commit=True):
1017 # Save the provided password in hashed format
1018 user = super(UserCreationForm, self).save(commit=False)
Tony Mackf9f4afb2013-05-01 21:02:12 -04001019 user.password = self.cleaned_data["password1"]
1020 #user.set_password(self.cleaned_data["password1"])
Siobhan Tully53437282013-04-26 19:30:27 -04001021 if commit:
1022 user.save()
1023 return user
1024
Siobhan Tully567e3e62013-06-21 18:03:16 -04001025
Siobhan Tully53437282013-04-26 19:30:27 -04001026class UserChangeForm(forms.ModelForm):
1027 """A form for updating users. Includes all the fields on
1028 the user, but replaces the password field with admin's
1029 password hash display field.
1030 """
Siobhan Tully63b7ba42014-01-12 10:35:11 -05001031 password = ReadOnlyPasswordHashField(label='Password',
1032 help_text= '<a href=\"password/\">Change Password</a>.')
Siobhan Tully53437282013-04-26 19:30:27 -04001033
1034 class Meta:
Siobhan Tully30fd4292013-05-10 08:59:56 -04001035 model = User
Siobhan Tully53437282013-04-26 19:30:27 -04001036
1037 def clean_password(self):
1038 # Regardless of what the user provides, return the initial value.
1039 # This is done here, rather than on the field, because the
1040 # field does not have access to the initial value
1041 return self.initial["password"]
1042
Scott Baker2c3cb642014-05-19 17:55:56 -07001043class UserDashboardViewInline(PlStackTabularInline):
1044 model = UserDashboardView
1045 extra = 0
1046 suit_classes = 'suit-tab suit-tab-dashboards'
1047 fields = ['user', 'dashboardView', 'order']
1048
Scott Baker86c83ab2014-10-03 13:10:47 -07001049class UserAdmin(PermissionCheckingAdminMixin, UserAdmin):
1050 # Note: Make sure PermissionCheckingAdminMixin is listed before
1051 # admin.ModelAdmin in the class declaration.
1052
Siobhan Tully53437282013-04-26 19:30:27 -04001053 class Meta:
1054 app_label = "core"
1055
1056 # The forms to add and change user instances
1057 form = UserChangeForm
1058 add_form = UserCreationForm
1059
1060 # The fields to be used in displaying the User model.
1061 # These override the definitions on the base UserAdmin
1062 # that reference specific fields on auth.User.
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001063 list_display = ('email', 'firstname', 'lastname', 'site', 'last_login')
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001064 list_filter = ('site',)
Scott Baker2c3cb642014-05-19 17:55:56 -07001065 inlines = [SlicePrivilegeInline,SitePrivilegeInline,DeploymentPrivilegeInline,UserDashboardViewInline]
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001066
Scott Bakercbfb6002014-10-03 00:32:37 -07001067 fieldListLoginDetails = ['backend_status_text', 'email','site','password','is_active','is_readonly','is_admin','public_key']
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001068 fieldListContactInfo = ['firstname','lastname','phone','timezone']
1069
Siobhan Tully53437282013-04-26 19:30:27 -04001070 fieldsets = (
Scott Baker40c00762014-08-21 16:55:59 -07001071 ('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 -04001072 ('Contact Information', {'fields': ('firstname','lastname','phone', 'timezone'), 'classes':['suit-tab suit-tab-contact']}),
Scott Baker2c3cb642014-05-19 17:55:56 -07001073 #('Dashboard Views', {'fields': ('dashboards',), 'classes':['suit-tab suit-tab-dashboards']}),
Siobhan Tully53437282013-04-26 19:30:27 -04001074 #('Important dates', {'fields': ('last_login',)}),
1075 )
1076 add_fieldsets = (
1077 (None, {
1078 'classes': ('wide',),
Scott Baker0a5633b2014-10-06 17:51:20 -07001079 'fields': ('email', 'firstname', 'lastname', 'is_readonly', 'phone', 'public_key','password1', 'password2')},
Siobhan Tully53437282013-04-26 19:30:27 -04001080 ),
1081 )
Scott Baker40c00762014-08-21 16:55:59 -07001082 readonly_fields = ('backend_status_text', )
Siobhan Tully53437282013-04-26 19:30:27 -04001083 search_fields = ('email',)
1084 ordering = ('email',)
1085 filter_horizontal = ()
1086
Scott Baker3ca51f62014-05-23 12:05:11 -07001087 user_readonly_fields = fieldListLoginDetails + fieldListContactInfo
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001088
Scott Baker0a5633b2014-10-06 17:51:20 -07001089 @property
1090 def suit_form_tabs(self):
1091 if getattr(_thread_locals, "obj", None) is None:
1092 return []
1093 else:
1094 return (('general','Login Details'),
1095 ('contact','Contact Information'),
1096 ('sliceprivileges','Slice Privileges'),
1097 ('siteprivileges','Site Privileges'),
1098 ('deploymentprivileges','Deployment Privileges'),
1099 ('dashboards','Dashboard Views'))
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001100
Tony Mackc2835a92013-05-28 09:18:49 -04001101 def formfield_for_foreignkey(self, db_field, request, **kwargs):
1102 if db_field.name == 'site':
Tony Mack5b061472014-02-04 07:57:10 -05001103 kwargs['queryset'] = Site.select_by_user(request.user)
Tony Mackc2835a92013-05-28 09:18:49 -04001104
1105 return super(UserAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)
1106
Tony Mack5b061472014-02-04 07:57:10 -05001107 def queryset(self, request):
1108 return User.select_by_user(request.user)
1109
Scott Baker2c3cb642014-05-19 17:55:56 -07001110class DashboardViewAdmin(PlanetStackBaseAdmin):
1111 fieldsets = [('Dashboard View Details',
Scott Baker40c00762014-08-21 16:55:59 -07001112 {'fields': ['backend_status_text', 'name', 'url'],
Scott Baker2c3cb642014-05-19 17:55:56 -07001113 'classes': ['suit-tab suit-tab-general']})
1114 ]
Scott Baker40c00762014-08-21 16:55:59 -07001115 readonly_fields = ('backend_status_text', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001116
Scott Baker2c3cb642014-05-19 17:55:56 -07001117 suit_form_tabs =(('general','Dashboard View Details'),)
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001118
Scott Baker0165fac2014-01-13 11:49:26 -08001119class ServiceResourceInline(PlStackTabularInline):
Scott Baker3de3e372013-05-10 16:50:44 -07001120 model = ServiceResource
1121 extra = 0
1122
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001123class ServiceClassAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001124 list_display = ('backend_status_icon', 'name', 'commitment', 'membershipFee')
1125 list_display_links = ('backend_status_icon', 'name', )
Scott Baker3de3e372013-05-10 16:50:44 -07001126 inlines = [ServiceResourceInline]
1127
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001128 user_readonly_fields = ['name', 'commitment', 'membershipFee']
1129 user_readonly_inlines = []
1130
Scott Baker0165fac2014-01-13 11:49:26 -08001131class ReservedResourceInline(PlStackTabularInline):
Scott Baker133c9212013-05-17 09:09:11 -07001132 model = ReservedResource
1133 extra = 0
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001134 suit_classes = 'suit-tab suit-tab-reservedresources'
Scott Baker133c9212013-05-17 09:09:11 -07001135
1136 def formfield_for_foreignkey(self, db_field, request=None, **kwargs):
1137 field = super(ReservedResourceInline, self).formfield_for_foreignkey(db_field, request, **kwargs)
1138
1139 if db_field.name == 'resource':
1140 # restrict resources to those that the slice's service class allows
1141 if request._slice is not None:
1142 field.queryset = field.queryset.filter(serviceClass = request._slice.serviceClass, calendarReservable=True)
1143 if len(field.queryset) > 0:
1144 field.initial = field.queryset.all()[0]
1145 else:
1146 field.queryset = field.queryset.none()
1147 elif db_field.name == 'sliver':
1148 # restrict slivers to those that belong to the slice
1149 if request._slice is not None:
1150 field.queryset = field.queryset.filter(slice = request._slice)
1151 else:
1152 field.queryset = field.queryset.none()
1153
1154 return field
1155
Tony Mack5b061472014-02-04 07:57:10 -05001156 def queryset(self, request):
1157 return ReservedResource.select_by_user(request.user)
1158
Scott Baker133c9212013-05-17 09:09:11 -07001159class ReservationChangeForm(forms.ModelForm):
1160 class Meta:
1161 model = Reservation
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001162 widgets = {
1163 'slice' : LinkedSelect
1164 }
Scott Baker133c9212013-05-17 09:09:11 -07001165
1166class ReservationAddForm(forms.ModelForm):
1167 slice = forms.ModelChoiceField(queryset=Slice.objects.all(), widget=forms.Select(attrs={"onChange":"document.getElementById('id_refresh').value=1; submit()"}))
1168 refresh = forms.CharField(widget=forms.HiddenInput())
1169
1170 class Media:
1171 css = {'all': ('planetstack.css',)} # .field-refresh { display: none; }
1172
1173 def clean_slice(self):
1174 slice = self.cleaned_data.get("slice")
1175 x = ServiceResource.objects.filter(serviceClass = slice.serviceClass, calendarReservable=True)
1176 if len(x) == 0:
1177 raise forms.ValidationError("The slice you selected does not have a service class that allows reservations")
1178 return slice
1179
1180 class Meta:
1181 model = Reservation
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001182 widgets = {
1183 'slice' : LinkedSelect
1184 }
1185
Scott Baker133c9212013-05-17 09:09:11 -07001186
1187class ReservationAddRefreshForm(ReservationAddForm):
1188 """ This form is displayed when the Reservation Form receives an update
1189 from the Slice dropdown onChange handler. It doesn't validate the
1190 data and doesn't save the data. This will cause the form to be
1191 redrawn.
1192 """
1193
Scott Baker8737e5f2013-05-17 09:35:32 -07001194 """ don't validate anything other than slice """
1195 dont_validate_fields = ("startTime", "duration")
1196
Scott Baker133c9212013-05-17 09:09:11 -07001197 def full_clean(self):
1198 result = super(ReservationAddForm, self).full_clean()
Scott Baker8737e5f2013-05-17 09:35:32 -07001199
1200 for fieldname in self.dont_validate_fields:
1201 if fieldname in self._errors:
1202 del self._errors[fieldname]
1203
Scott Baker133c9212013-05-17 09:09:11 -07001204 return result
1205
1206 """ don't save anything """
1207 def is_valid(self):
1208 return False
1209
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001210class ReservationAdmin(PlanetStackBaseAdmin):
Scott Baker40c00762014-08-21 16:55:59 -07001211 fieldList = ['backend_status_text', 'slice', 'startTime', 'duration']
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001212 fieldsets = [('Reservation Details', {'fields': fieldList, 'classes': ['suit-tab suit-tab-general']})]
Scott Baker40c00762014-08-21 16:55:59 -07001213 readonly_fields = ('backend_status_text', )
Scott Baker133c9212013-05-17 09:09:11 -07001214 list_display = ('startTime', 'duration')
Scott Baker133c9212013-05-17 09:09:11 -07001215 form = ReservationAddForm
1216
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001217 suit_form_tabs = (('general','Reservation Details'), ('reservedresources','Reserved Resources'))
1218
1219 inlines = [ReservedResourceInline]
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001220 user_readonly_fields = fieldList
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001221
Scott Baker133c9212013-05-17 09:09:11 -07001222 def add_view(self, request, form_url='', extra_context=None):
Scott Bakeracd45142013-05-19 16:19:16 -07001223 timezone.activate(request.user.timezone)
Scott Baker133c9212013-05-17 09:09:11 -07001224 request._refresh = False
1225 request._slice = None
1226 if request.method == 'POST':
Scott Baker8737e5f2013-05-17 09:35:32 -07001227 # "refresh" will be set to "1" if the form was submitted due to
1228 # a change in the Slice dropdown.
Scott Baker133c9212013-05-17 09:09:11 -07001229 if request.POST.get("refresh","1") == "1":
1230 request._refresh = True
1231 request.POST["refresh"] = "0"
Scott Baker8737e5f2013-05-17 09:35:32 -07001232
1233 # Keep track of the slice that was selected, so the
1234 # reservedResource inline can filter items for the slice.
Scott Baker133c9212013-05-17 09:09:11 -07001235 request._slice = request.POST.get("slice",None)
1236 if (request._slice is not None):
1237 request._slice = Slice.objects.get(id=request._slice)
1238
1239 result = super(ReservationAdmin, self).add_view(request, form_url, extra_context)
1240 return result
1241
Scott Bakeracd45142013-05-19 16:19:16 -07001242 def changelist_view(self, request, extra_context = None):
1243 timezone.activate(request.user.timezone)
1244 return super(ReservationAdmin, self).changelist_view(request, extra_context)
1245
Scott Baker133c9212013-05-17 09:09:11 -07001246 def get_form(self, request, obj=None, **kwargs):
Siobhan Tullyd3515752013-06-21 16:34:53 -04001247 request._obj_ = obj
1248 if obj is not None:
1249 # For changes, set request._slice to the slice already set in the
1250 # object.
1251 request._slice = obj.slice
1252 self.form = ReservationChangeForm
1253 else:
1254 if getattr(request, "_refresh", False):
1255 self.form = ReservationAddRefreshForm
1256 else:
1257 self.form = ReservationAddForm
1258 return super(ReservationAdmin, self).get_form(request, obj, **kwargs)
1259
Scott Baker133c9212013-05-17 09:09:11 -07001260 def get_readonly_fields(self, request, obj=None):
Siobhan Tullyd3515752013-06-21 16:34:53 -04001261 if (obj is not None):
1262 # Prevent slice from being changed after the reservation has been
1263 # created.
1264 return ['slice']
1265 else:
Scott Baker133c9212013-05-17 09:09:11 -07001266 return []
Scott Baker3de3e372013-05-10 16:50:44 -07001267
Tony Mack5b061472014-02-04 07:57:10 -05001268 def queryset(self, request):
1269 return Reservation.select_by_user(request.user)
1270
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001271class NetworkParameterTypeAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001272 list_display = ("backend_status_icon", "name", )
1273 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001274 user_readonly_fields = ['name']
1275 user_readonly_inlines = []
Scott Baker74d8e622013-07-29 16:04:22 -07001276
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001277class RouterAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001278 list_display = ("backend_status_icon", "name", )
1279 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001280 user_readonly_fields = ['name']
1281 user_readonly_inlines = []
1282
Scott Baker0165fac2014-01-13 11:49:26 -08001283class RouterInline(PlStackTabularInline):
Scott Baker74d8e622013-07-29 16:04:22 -07001284 model = Router.networks.through
1285 extra = 0
1286 verbose_name_plural = "Routers"
1287 verbose_name = "Router"
Siobhan Tully2d95e482013-09-06 10:56:06 -04001288 suit_classes = 'suit-tab suit-tab-routers'
Scott Baker74d8e622013-07-29 16:04:22 -07001289
Scott Bakerb27b62c2014-08-15 16:29:16 -07001290class NetworkParameterInline(PlStackGenericTabularInline):
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001291 model = NetworkParameter
Scott Baker618e3792014-08-15 13:42:29 -07001292 extra = 0
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001293 verbose_name_plural = "Parameters"
1294 verbose_name = "Parameter"
1295 suit_classes = 'suit-tab suit-tab-netparams'
Scott Baker40c00762014-08-21 16:55:59 -07001296 fields = ['backend_status_icon', 'parameter', 'value']
1297 readonly_fields = ('backend_status_icon', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001298
Scott Baker0165fac2014-01-13 11:49:26 -08001299class NetworkSliversInline(PlStackTabularInline):
Scott Baker40c00762014-08-21 16:55:59 -07001300 fields = ['backend_status_icon', 'network','sliver','ip']
1301 readonly_fields = ("backend_status_icon", "ip", )
Scott Baker74d8e622013-07-29 16:04:22 -07001302 model = NetworkSliver
Scott Baker874936e2014-01-13 18:15:34 -08001303 selflink_fieldname = "sliver"
Scott Baker74d8e622013-07-29 16:04:22 -07001304 extra = 0
1305 verbose_name_plural = "Slivers"
1306 verbose_name = "Sliver"
Siobhan Tully2d95e482013-09-06 10:56:06 -04001307 suit_classes = 'suit-tab suit-tab-networkslivers'
Scott Baker74d8e622013-07-29 16:04:22 -07001308
Scott Baker0165fac2014-01-13 11:49:26 -08001309class NetworkSlicesInline(PlStackTabularInline):
Scott Bakerd7d2a392013-08-06 08:57:30 -07001310 model = NetworkSlice
Scott Baker874936e2014-01-13 18:15:34 -08001311 selflink_fieldname = "slice"
Scott Bakerd7d2a392013-08-06 08:57:30 -07001312 extra = 0
1313 verbose_name_plural = "Slices"
1314 verbose_name = "Slice"
Siobhan Tully2d95e482013-09-06 10:56:06 -04001315 suit_classes = 'suit-tab suit-tab-networkslices'
Scott Baker40c00762014-08-21 16:55:59 -07001316 fields = ['backend_status_icon', 'network','slice']
1317 readonly_fields = ('backend_status_icon', )
Scott Bakerd7d2a392013-08-06 08:57:30 -07001318
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001319class NetworkAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001320 list_display = ("backend_status_icon", "name", "subnet", "ports", "labels")
1321 list_display_links = ('backend_status_icon', 'name', )
Scott Baker74d8e622013-07-29 16:04:22 -07001322 readonly_fields = ("subnet", )
Siobhan Tully2d95e482013-09-06 10:56:06 -04001323
Scott Bakerd7d2a392013-08-06 08:57:30 -07001324 inlines = [NetworkParameterInline, NetworkSliversInline, NetworkSlicesInline, RouterInline]
Scott Baker74d8e622013-07-29 16:04:22 -07001325
Siobhan Tully2d95e482013-09-06 10:56:06 -04001326 fieldsets = [
Scott Baker40c00762014-08-21 16:55:59 -07001327 (None, {'fields': ['backend_status_text', 'name','template','ports','labels','owner','guaranteedBandwidth', 'permitAllSlices','permittedSlices','network_id','router_id','subnet_id','subnet'], 'classes':['suit-tab suit-tab-general']}),]
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001328
Scott Baker40c00762014-08-21 16:55:59 -07001329 readonly_fields = ('backend_status_text', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001330 user_readonly_fields = ['name','template','ports','labels','owner','guaranteedBandwidth', 'permitAllSlices','permittedSlices','network_id','router_id','subnet_id','subnet']
Siobhan Tully2d95e482013-09-06 10:56:06 -04001331
1332 suit_form_tabs =(
1333 ('general','Network Details'),
1334 ('netparams', 'Parameters'),
1335 ('networkslivers','Slivers'),
1336 ('networkslices','Slices'),
1337 ('routers','Routers'),
1338 )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001339class NetworkTemplateAdmin(PlanetStackBaseAdmin):
Scott Baker63d1a552014-08-21 15:19:07 -07001340 list_display = ("backend_status_icon", "name", "guaranteedBandwidth", "visibility")
1341 list_display_links = ('backend_status_icon', 'name', )
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001342 user_readonly_fields = ["name", "guaranteedBandwidth", "visibility"]
1343 user_readonly_inlines = []
Scott Baker74d8e622013-07-29 16:04:22 -07001344
Scott Baker37b47902014-09-02 14:37:41 -07001345class FlavorAdmin(PlanetStackBaseAdmin):
1346 list_display = ("backend_status_icon", "name", "flavor", "order", "default")
1347 list_display_links = ("backend_status_icon", "name")
1348 user_readonly_fields = ("name", "flavor")
1349 fields = ("name", "description", "flavor", "order", "default")
1350
Tony Mack31c2b8f2013-04-26 20:01:42 -04001351# register a signal that caches the user's credentials when they log in
1352def cache_credentials(sender, user, request, **kwds):
1353 auth = {'username': request.POST['username'],
1354 'password': request.POST['password']}
1355 request.session['auth'] = auth
1356user_logged_in.connect(cache_credentials)
1357
Scott Baker15cddfa2013-12-09 13:45:19 -08001358def dollar_field(fieldName, short_description):
1359 def newFunc(self, obj):
1360 try:
1361 x= "$ %0.2f" % float(getattr(obj, fieldName, 0.0))
1362 except:
1363 x=getattr(obj, fieldName, 0.0)
1364 return x
1365 newFunc.short_description = short_description
1366 return newFunc
1367
1368def right_dollar_field(fieldName, short_description):
1369 def newFunc(self, obj):
1370 try:
1371 #x= '<div align=right style="width:6em">$ %0.2f</div>' % float(getattr(obj, fieldName, 0.0))
1372 x= '<div align=right>$ %0.2f</div>' % float(getattr(obj, fieldName, 0.0))
1373 except:
1374 x=getattr(obj, fieldName, 0.0)
1375 return x
1376 newFunc.short_description = short_description
1377 newFunc.allow_tags = True
1378 return newFunc
Scott Baker43105042013-12-06 23:23:36 -08001379
Scott Baker0165fac2014-01-13 11:49:26 -08001380class InvoiceChargeInline(PlStackTabularInline):
Scott Baker43105042013-12-06 23:23:36 -08001381 model = Charge
Scott Baker15cddfa2013-12-09 13:45:19 -08001382 extra = 0
Scott Baker43105042013-12-06 23:23:36 -08001383 verbose_name_plural = "Charges"
1384 verbose_name = "Charge"
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001385 exclude = ['account']
Scott Baker9cb88a22013-12-09 18:56:00 -08001386 fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
1387 readonly_fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
1388 can_delete = False
1389 max_num = 0
1390
1391 dollar_amount = right_dollar_field("amount", "Amount")
Scott Baker43105042013-12-06 23:23:36 -08001392
1393class InvoiceAdmin(admin.ModelAdmin):
1394 list_display = ("date", "account")
1395
1396 inlines = [InvoiceChargeInline]
1397
Scott Baker9cb88a22013-12-09 18:56:00 -08001398 fields = ["date", "account", "dollar_amount"]
1399 readonly_fields = ["date", "account", "dollar_amount"]
1400
1401 dollar_amount = dollar_field("amount", "Amount")
Scott Baker43105042013-12-06 23:23:36 -08001402
Scott Baker0165fac2014-01-13 11:49:26 -08001403class InvoiceInline(PlStackTabularInline):
Scott Baker15cddfa2013-12-09 13:45:19 -08001404 model = Invoice
1405 extra = 0
1406 verbose_name_plural = "Invoices"
1407 verbose_name = "Invoice"
Scott Baker0165fac2014-01-13 11:49:26 -08001408 fields = ["date", "dollar_amount"]
1409 readonly_fields = ["date", "dollar_amount"]
Scott Baker15cddfa2013-12-09 13:45:19 -08001410 suit_classes = 'suit-tab suit-tab-accountinvoice'
1411 can_delete=False
1412 max_num=0
1413
1414 dollar_amount = right_dollar_field("amount", "Amount")
1415
Scott Baker0165fac2014-01-13 11:49:26 -08001416class PendingChargeInline(PlStackTabularInline):
Scott Baker43105042013-12-06 23:23:36 -08001417 model = Charge
Scott Baker15cddfa2013-12-09 13:45:19 -08001418 extra = 0
Scott Baker43105042013-12-06 23:23:36 -08001419 verbose_name_plural = "Charges"
1420 verbose_name = "Charge"
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001421 exclude = ["invoice"]
Scott Baker15cddfa2013-12-09 13:45:19 -08001422 fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
1423 readonly_fields = ["date", "kind", "state", "object", "coreHours", "dollar_amount", "slice"]
Scott Baker43105042013-12-06 23:23:36 -08001424 suit_classes = 'suit-tab suit-tab-accountpendingcharges'
Scott Baker15cddfa2013-12-09 13:45:19 -08001425 can_delete=False
1426 max_num=0
Scott Baker43105042013-12-06 23:23:36 -08001427
1428 def queryset(self, request):
1429 qs = super(PendingChargeInline, self).queryset(request)
1430 qs = qs.filter(state="pending")
1431 return qs
1432
Scott Baker15cddfa2013-12-09 13:45:19 -08001433 dollar_amount = right_dollar_field("amount", "Amount")
1434
Scott Baker0165fac2014-01-13 11:49:26 -08001435class PaymentInline(PlStackTabularInline):
Scott Baker43105042013-12-06 23:23:36 -08001436 model=Payment
1437 extra = 1
1438 verbose_name_plural = "Payments"
1439 verbose_name = "Payment"
Scott Baker15cddfa2013-12-09 13:45:19 -08001440 fields = ["date", "dollar_amount"]
1441 readonly_fields = ["date", "dollar_amount"]
Scott Baker43105042013-12-06 23:23:36 -08001442 suit_classes = 'suit-tab suit-tab-accountpayments'
Scott Baker15cddfa2013-12-09 13:45:19 -08001443 can_delete=False
1444 max_num=0
1445
1446 dollar_amount = right_dollar_field("amount", "Amount")
1447
Scott Baker43105042013-12-06 23:23:36 -08001448class AccountAdmin(admin.ModelAdmin):
1449 list_display = ("site", "balance_due")
1450
1451 inlines = [InvoiceInline, PaymentInline, PendingChargeInline]
1452
1453 fieldsets = [
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001454 (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 -08001455
Scott Baker15cddfa2013-12-09 13:45:19 -08001456 readonly_fields = ['site', 'dollar_balance_due', 'dollar_total_invoices', 'dollar_total_payments']
Scott Baker43105042013-12-06 23:23:36 -08001457
1458 suit_form_tabs =(
1459 ('general','Account Details'),
1460 ('accountinvoice', 'Invoices'),
1461 ('accountpayments', 'Payments'),
1462 ('accountpendingcharges','Pending Charges'),
1463 )
1464
Scott Baker15cddfa2013-12-09 13:45:19 -08001465 dollar_balance_due = dollar_field("balance_due", "Balance Due")
1466 dollar_total_invoices = dollar_field("total_invoices", "Total Invoices")
1467 dollar_total_payments = dollar_field("total_payments", "Total Payments")
1468
Siobhan Tully53437282013-04-26 19:30:27 -04001469# Now register the new UserAdmin...
Siobhan Tully30fd4292013-05-10 08:59:56 -04001470admin.site.register(User, UserAdmin)
Siobhan Tully53437282013-04-26 19:30:27 -04001471# ... and, since we're not using Django's builtin permissions,
1472# unregister the Group model from admin.
Siobhan Tullyce652d02013-10-08 21:52:35 -04001473#admin.site.unregister(Group)
Siobhan Tully53437282013-04-26 19:30:27 -04001474
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001475#Do not show django evolution in the admin interface
1476from django_evolution.models import Version, Evolution
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001477#admin.site.unregister(Version)
1478#admin.site.unregister(Evolution)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001479
1480
1481# When debugging it is often easier to see all the classes, but for regular use
1482# only the top-levels should be displayed
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001483showAll = False
Scott Baker43105042013-12-06 23:23:36 -08001484
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001485admin.site.register(Deployment, DeploymentAdmin)
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001486admin.site.register(Site, SiteAdmin)
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001487admin.site.register(Slice, SliceAdmin)
Siobhan Tullyce652d02013-10-08 21:52:35 -04001488admin.site.register(Service, ServiceAdmin)
smbakera3cf70c2013-06-27 02:01:41 -07001489admin.site.register(Reservation, ReservationAdmin)
Scott Baker74d8e622013-07-29 16:04:22 -07001490admin.site.register(Network, NetworkAdmin)
1491admin.site.register(Router, RouterAdmin)
Scott Baker74d8e622013-07-29 16:04:22 -07001492admin.site.register(NetworkTemplate, NetworkTemplateAdmin)
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001493admin.site.register(Account, AccountAdmin)
1494admin.site.register(Invoice, InvoiceAdmin)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001495
Siobhan Tullycf04fb62014-01-11 11:25:57 -05001496if True:
1497 admin.site.register(NetworkParameterType, NetworkParameterTypeAdmin)
1498 admin.site.register(ServiceClass, ServiceClassAdmin)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001499 #admin.site.register(PlanetStack)
Siobhan Tullyd3515752013-06-21 16:34:53 -04001500 admin.site.register(Tag, TagAdmin)
Siobhan Tullyce652d02013-10-08 21:52:35 -04001501 admin.site.register(DeploymentRole)
1502 admin.site.register(SiteRole)
1503 admin.site.register(SliceRole)
1504 admin.site.register(PlanetStackRole)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001505 admin.site.register(Node, NodeAdmin)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -04001506 #admin.site.register(SlicePrivilege, SlicePrivilegeAdmin)
1507 #admin.site.register(SitePrivilege, SitePrivilegeAdmin)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001508 admin.site.register(Sliver, SliverAdmin)
Siobhan Tullybf1153a2013-05-27 20:53:48 -04001509 admin.site.register(Image, ImageAdmin)
Scott Baker2c3cb642014-05-19 17:55:56 -07001510 admin.site.register(DashboardView, DashboardViewAdmin)
Scott Baker37b47902014-09-02 14:37:41 -07001511 admin.site.register(Flavor, FlavorAdmin)
Tony Mack7130ac32013-03-22 21:58:00 -04001512