blob: 35d9f912fc9d0b71abc2f8af4d1a78fdf655988d [file] [log] [blame]
Siobhan Tully4bc09f22013-04-10 21:15:21 -04001import os
2from django.db import models
Siobhan Tully30fd4292013-05-10 08:59:56 -04003from core.models import PlCoreBase
4from core.models import Site
Tony Mackc195d722014-08-20 11:15:17 -04005from core.models.site import SitePrivilege
Siobhan Tully30fd4292013-05-10 08:59:56 -04006from core.models import User
7from core.models import Role
Tony Mack51c4a7d2014-11-30 15:33:35 -05008from core.models import Controller,ControllerLinkManager,ControllerLinkDeletionManager
Scott Bakere8d596f2013-05-13 23:17:13 -07009from core.models import ServiceClass
Scott Bakerf5a19062016-05-02 09:37:37 -070010#from core.models.serviceclass import get_default_serviceclass
Siobhan Tullyde5450d2013-06-21 11:35:33 -040011from core.models import Tag
12from django.contrib.contenttypes import generic
Siobhan Tullyce652d02013-10-08 21:52:35 -040013from core.models import Service
Tony Mack51c4a7d2014-11-30 15:33:35 -050014from core.models import Controller
Scott Baker3c2b5cd2015-01-12 13:13:05 -080015from core.models import Flavor, Image
Tony Mack50e12212015-03-09 13:03:56 -040016from core.models.plcorebase import StrippedCharField
Tony Mackd0cf6122015-02-01 19:51:39 -050017from django.core.exceptions import PermissionDenied, ValidationError
Matteo Scandolo53ee3f02016-01-13 14:47:38 -080018from xos.exceptions import *
Siobhan Tully4bc09f22013-04-10 21:15:21 -040019
20# Create your models here.
21
22class Slice(PlCoreBase):
Scott Baker84e15b62015-11-16 19:12:25 -080023 ISOLATION_CHOICES = (('vm', 'Virtual Machine'), ('container', 'Container'), ('container_vm', 'Container In VM'))
Scott Baker53c449d2016-02-03 18:29:25 -080024 NETWORK_CHOICES = ((None, 'Default'), ('host', 'Host'), ('bridged', 'Bridged'), ('noauto', 'No Automatic Networks'))
Scott Baker84e15b62015-11-16 19:12:25 -080025
Tony Mack50e12212015-03-09 13:03:56 -040026 name = StrippedCharField(unique=True, help_text="The Name of the Slice", max_length=80)
Siobhan Tully4bc09f22013-04-10 21:15:21 -040027 enabled = models.BooleanField(default=True, help_text="Status for this Slice")
Scott Bakerd6dc5052014-09-12 12:24:24 -070028 omf_friendly = models.BooleanField(default=False)
Siobhan Tully4bc09f22013-04-10 21:15:21 -040029 description=models.TextField(blank=True,help_text="High level description of the slice and expected activities", max_length=1024)
30 slice_url = models.URLField(blank=True, max_length=512)
Tony Mack360afb82013-12-17 13:32:45 -050031 site = models.ForeignKey(Site, related_name='slices', help_text="The Site this Slice belongs to")
Tony Mackd8515472015-08-19 11:58:18 -040032 max_instances = models.IntegerField(default=10)
Scott Bakerfb530d82015-03-09 16:21:36 -070033 service = models.ForeignKey(Service, related_name='slices', null=True, blank=True)
Scott Baker956eca62015-12-14 16:21:33 -080034 network = models.CharField(null=True, blank=True, max_length=256, choices=NETWORK_CHOICES)
Scott Baker1c0312e2016-02-04 15:58:19 -080035 exposed_ports = models.CharField(null=True, blank=True, max_length=256)
Siobhan Tullyde5450d2013-06-21 11:35:33 -040036 tags = generic.GenericRelation(Tag)
Scott Bakerf5a19062016-05-02 09:37:37 -070037 serviceClass = models.ForeignKey(ServiceClass, related_name = "slices", null=True, blank=True) # DEPRECATED
Tony Mack2bd5b412013-06-11 21:05:06 -040038 creator = models.ForeignKey(User, related_name='slices', blank=True, null=True)
Siobhan Tully231f4c82013-05-02 05:47:24 -040039
Scott Bakerffce7852015-01-03 16:26:38 -080040 # for tenant view
Scott Baker3c2b5cd2015-01-12 13:13:05 -080041 default_flavor = models.ForeignKey(Flavor, related_name = "slices", null=True, blank=True)
42 default_image = models.ForeignKey(Image, related_name = "slices", null=True, blank=True);
Tony Mack50e12212015-03-09 13:03:56 -040043 mount_data_sets = StrippedCharField(default="GenBank",null=True, blank=True, max_length=256)
Scott Bakerffce7852015-01-03 16:26:38 -080044
Scott Baker84e15b62015-11-16 19:12:25 -080045 default_isolation = models.CharField(null=False, blank=False, max_length=30, choices=ISOLATION_CHOICES, default="vm")
46
Siobhan Tully4bc09f22013-04-10 21:15:21 -040047 def __unicode__(self): return u'%s' % (self.name)
48
Tony Mack2721d6f2014-08-11 11:14:58 -040049 @property
50 def slicename(self):
51 return "%s_%s" % (self.site.login_base, self.name)
52
Tony Mack62bc59a2013-04-14 23:27:12 -040053 def save(self, *args, **kwds):
Tony Mackdac85762014-09-03 13:19:42 -040054 site = Site.objects.get(id=self.site.id)
Tony Mack78070352014-09-24 12:44:22 -040055 # allow preexisting slices to keep their original name for now
56 if not self.id and not self.name.startswith(site.login_base):
Matteo Scandolo53ee3f02016-01-13 14:47:38 -080057 raise XOSValidationError('slice name must begin with %s' % site.login_base)
Scott Baker8c520512015-01-29 17:16:10 -080058
59 if self.name == site.login_base+"_":
Matteo Scandolo53ee3f02016-01-13 14:47:38 -080060 raise XOSValidationError('slice name is too short')
Scott Baker8c520512015-01-29 17:16:10 -080061
62 if " " in self.name:
Matteo Scandolo53ee3f02016-01-13 14:47:38 -080063 raise XOSValidationError('slice name must not contain spaces')
Scott Baker8c520512015-01-29 17:16:10 -080064
Scott Bakere8d596f2013-05-13 23:17:13 -070065 if self.serviceClass is None:
66 # We allowed None=True for serviceClass because Django evolution
67 # will fail unless it is allowed. But, we we really don't want it to
68 # ever save None, so fix it up here.
69 self.serviceClass = ServiceClass.get_default()
Tony Mackd0cf6122015-02-01 19:51:39 -050070
71 # set creator on first save
Tony Mack2bd5b412013-06-11 21:05:06 -040072 if not self.creator and hasattr(self, 'caller'):
73 self.creator = self.caller
Tony Mackd0cf6122015-02-01 19:51:39 -050074
75 # only admins change a slice's creator
76 if 'creator' in self.changed_fields and \
77 (not hasattr(self, 'caller') or not self.caller.is_admin):
Scott Baker8cfe3d22015-02-02 11:00:53 -080078
Scott Bakerbd46a922015-02-03 15:02:17 -080079 if (self._initial["creator"]==None) and (self.creator==getattr(self,"caller",None)):
Scott Baker8cfe3d22015-02-02 11:00:53 -080080 # it's okay if the creator is being set by the caller to
81 # himeself on a new slice object.
Scott Baker8cfe3d22015-02-02 11:00:53 -080082 pass
83 else:
84 raise PermissionDenied("Insufficient privileges to change slice creator")
Tony Mackd0cf6122015-02-01 19:51:39 -050085
Scott Baker9a9a9ab2015-01-29 17:53:26 -080086 if not self.creator:
Matteo Scandolo53ee3f02016-01-13 14:47:38 -080087 raise XOSValidationError('slice has no creator')
Tony Mackd0cf6122015-02-01 19:51:39 -050088
Scott Baker956eca62015-12-14 16:21:33 -080089 if self.network=="Private Only":
90 # "Private Only" was the default from the old Tenant View
91 self.network=None
92 self.enforce_choices(self.network, self.NETWORK_CHOICES)
93
Tony Mack62bc59a2013-04-14 23:27:12 -040094 super(Slice, self).save(*args, **kwds)
95
Tony Mack5b061472014-02-04 07:57:10 -050096 def can_update(self, user):
Tony Mack3428e6e2015-02-08 21:38:41 -050097 return user.can_update_slice(self)
98
Tony Mack5b061472014-02-04 07:57:10 -050099
Tony Mack5b061472014-02-04 07:57:10 -0500100 @staticmethod
101 def select_by_user(user):
102 if user.is_admin:
103 qs = Slice.objects.all()
104 else:
Tony Mack33df82b2014-08-20 11:29:40 -0400105 # users can see slices they belong to
Tony Mack5b061472014-02-04 07:57:10 -0500106 slice_ids = [sp.slice.id for sp in SlicePrivilege.objects.filter(user=user)]
Scott Baker48f83c52015-07-15 14:01:57 -0700107 # pis and admins can see slices at their sites
Tony Mack33df82b2014-08-20 11:29:40 -0400108 sites = [sp.site for sp in SitePrivilege.objects.filter(user=user)\
Scott Baker9ebd38c2015-07-14 18:29:44 -0700109 if (sp.role.role == 'pi') or (sp.role.role == 'admin')]
Scott Baker48f83c52015-07-15 14:01:57 -0700110 slice_ids.extend([s.id for s in Slice.objects.filter(site__in=sites)])
Tony Mack5b061472014-02-04 07:57:10 -0500111 qs = Slice.objects.filter(id__in=slice_ids)
112 return qs
113
Sapan Bhatia50de6be2015-05-09 18:07:29 +0200114 """
Tony Mack82d901b2014-10-22 13:30:41 -0400115 def delete(self, *args, **kwds):
116 # delete networks associated with this slice
117 from core.models.network import Network
118 nets = Network.objects.filter(slices=self)
119 nets.delete()
Tony Mack51c4a7d2014-11-30 15:33:35 -0500120 # delete slice controllers
Tony Mack3066a952015-01-05 22:48:11 -0500121 slice_controllers = ControllerSlice.objects.filter(slice=self)
Tony Mack51c4a7d2014-11-30 15:33:35 -0500122 slice_controllers.delete()
Tony Macka8812242014-11-12 10:01:26 -0500123 # delete slice privilege
124 slice_privileges = SlicePrivilege.objects.filter(slice=self)
125 slice_privileges.delete()
126 # continue with normal delete
Sapana283f502014-11-24 16:44:48 -0500127 super(Slice, self).delete(*args, **kwds)
Sapan Bhatia50de6be2015-05-09 18:07:29 +0200128 """
Tony Macka8812242014-11-12 10:01:26 -0500129
Tony Mack82d901b2014-10-22 13:30:41 -0400130
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400131class SliceRole(PlCoreBase):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500132 ROLE_CHOICES = (('admin','Admin'),('default','Default'))
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400133
Tony Mack50e12212015-03-09 13:03:56 -0400134 role = StrippedCharField(choices=ROLE_CHOICES, unique=True, max_length=30)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400135
136 def __unicode__(self): return u'%s' % (self.role)
137
138class SlicePrivilege(PlCoreBase):
Sapan Bhatia6bfa2ca2014-11-11 21:47:45 -0500139 user = models.ForeignKey('User', related_name='sliceprivileges')
140 slice = models.ForeignKey('Slice', related_name='sliceprivileges')
141 role = models.ForeignKey('SliceRole',related_name='sliceprivileges')
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400142
Tony Mack5d93a9e2015-04-11 12:17:59 -0400143 class Meta:
144 unique_together = ('user', 'slice', 'role')
Tony Mackc8836df2015-03-09 17:13:14 -0400145
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400146 def __unicode__(self): return u'%s %s %s' % (self.slice, self.user, self.role)
Tony Mack5b061472014-02-04 07:57:10 -0500147
Tony Macka6928d62015-03-29 09:29:12 -0400148 def save(self, *args, **kwds):
149 if not self.user.is_active:
150 raise PermissionDenied, "Cannot modify role(s) of a disabled user"
Scott Baker77fc1732015-04-17 15:44:30 -0700151 super(SlicePrivilege, self).save(*args, **kwds)
Tony Macka6928d62015-03-29 09:29:12 -0400152
Tony Mack5b061472014-02-04 07:57:10 -0500153 def can_update(self, user):
Tony Mack3428e6e2015-02-08 21:38:41 -0500154 return user.can_update_slice(self.slice)
Tony Mack5b061472014-02-04 07:57:10 -0500155
Tony Mack5b061472014-02-04 07:57:10 -0500156 @staticmethod
157 def select_by_user(user):
158 if user.is_admin:
159 qs = SlicePrivilege.objects.all()
160 else:
Scott Baker31ba9ca2015-07-16 12:40:07 -0700161 # You can see your own SlicePrivileges
Tony Mack5b061472014-02-04 07:57:10 -0500162 sp_ids = [sp.id for sp in SlicePrivilege.objects.filter(user=user)]
Scott Baker31ba9ca2015-07-16 12:40:07 -0700163
164 # A site pi or site admin can see the SlicePrivileges for all slices in his Site
165 for priv in SitePrivilege.objects.filter(user=user):
166 if priv.role.role in ['pi', 'admin']:
167 sp_ids.extend( [sp.id for sp in SlicePrivilege.objects.filter(slice__site = priv.site)] )
168
169 # A slice admin can see the SlicePrivileges for his Slice
170 for priv in SlicePrivilege.objects.filter(user=user, role__role="admin"):
171 sp_ids.extend( [sp.id for sp in SlicePrivilege.objects.filter(slice=priv.slice)] )
172
Tony Mack5b061472014-02-04 07:57:10 -0500173 qs = SlicePrivilege.objects.filter(id__in=sp_ids)
174 return qs
Tony Macke4be32f2014-03-11 20:45:25 -0400175
Tony Mack3066a952015-01-05 22:48:11 -0500176class ControllerSlice(PlCoreBase):
Tony Mack51c4a7d2014-11-30 15:33:35 -0500177 objects = ControllerLinkManager()
178 deleted_objects = ControllerLinkDeletionManager()
Sapan Bhatiaa6609032014-09-22 14:55:08 -0400179
Tony Mack51c4a7d2014-11-30 15:33:35 -0500180 controller = models.ForeignKey(Controller, related_name='controllerslices')
181 slice = models.ForeignKey(Slice, related_name='controllerslices')
Tony Mack50e12212015-03-09 13:03:56 -0400182 tenant_id = StrippedCharField(null=True, blank=True, max_length=200, help_text="Keystone tenant id")
Tony Macke4be32f2014-03-11 20:45:25 -0400183
Tony Mack5d93a9e2015-04-11 12:17:59 -0400184 class Meta:
185 unique_together = ('controller', 'slice')
Tony Mackc8836df2015-03-09 17:13:14 -0400186
Sapan Bhatiad0e8ed82016-04-06 19:30:24 +0200187 def tologdict(self):
188 d=super(ControllerSlice,self).tologdict()
189 try:
190 d['slice_name']=self.slice.name
191 d['controller_name']=self.controller.name
192 except:
193 pass
194 return d
195
Tony Mack51c4a7d2014-11-30 15:33:35 -0500196 def __unicode__(self): return u'%s %s' % (self.slice, self.controller)
Tony Macke4be32f2014-03-11 20:45:25 -0400197
198 @staticmethod
199 def select_by_user(user):
200 if user.is_admin:
Tony Mack3066a952015-01-05 22:48:11 -0500201 qs = ControllerSlice.objects.all()
Tony Macke4be32f2014-03-11 20:45:25 -0400202 else:
203 slices = Slice.select_by_user(user)
Tony Mack3066a952015-01-05 22:48:11 -0500204 qs = ControllerSlice.objects.filter(slice__in=slices)
Tony Macke4be32f2014-03-11 20:45:25 -0400205 return qs
Sapan Bhatia772c7c22014-12-16 01:09:04 -0500206
207 def get_cpu_stats(self):
208 filter = 'project_id=%s'%self.tenant_id
209 return monitor.get_meter('cpu',filter,None)
210
211 def get_bw_stats(self):
212 filter = 'project_id=%s'%self.tenant_id
213 return monitor.get_meter('network.outgoing.bytes',filter,None)
214
215 def get_node_stats(self):
Tony Mackd8515472015-08-19 11:58:18 -0400216 return len(self.slice.instances)