blob: aefe08cf90fcdc6a2eb6eb800aa2210e39346b62 [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 Mackf3bbe472014-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 Bakercfe0fd92014-09-12 12:24:24 -070010from 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 Mackf3bbe472014-11-30 15:33:35 -050014from core.models import Controller
Pingping Lin276c1b92016-04-20 18:02:10 -070015from core.models.node import Node
Scott Baker200181e2015-01-12 13:13:05 -080016from core.models import Flavor, Image
Tony Mackd84b1ff2015-03-09 13:03:56 -040017from core.models.plcorebase import StrippedCharField
Tony Mackb9925772015-02-01 19:51:39 -050018from django.core.exceptions import PermissionDenied, ValidationError
Matteo Scandolob12050e2016-01-13 14:47:38 -080019from xos.exceptions import *
Siobhan Tully4bc09f22013-04-10 21:15:21 -040020
21# Create your models here.
22
23class Slice(PlCoreBase):
Scott Baker3f237242015-11-16 19:12:25 -080024 ISOLATION_CHOICES = (('vm', 'Virtual Machine'), ('container', 'Container'), ('container_vm', 'Container In VM'))
Scott Baker951a81b2016-02-03 18:29:25 -080025 NETWORK_CHOICES = ((None, 'Default'), ('host', 'Host'), ('bridged', 'Bridged'), ('noauto', 'No Automatic Networks'))
Scott Baker3f237242015-11-16 19:12:25 -080026
Tony Mackd84b1ff2015-03-09 13:03:56 -040027 name = StrippedCharField(unique=True, help_text="The Name of the Slice", max_length=80)
Siobhan Tully4bc09f22013-04-10 21:15:21 -040028 enabled = models.BooleanField(default=True, help_text="Status for this Slice")
Scott Bakercfe0fd92014-09-12 12:24:24 -070029 omf_friendly = models.BooleanField(default=False)
Siobhan Tully4bc09f22013-04-10 21:15:21 -040030 description=models.TextField(blank=True,help_text="High level description of the slice and expected activities", max_length=1024)
31 slice_url = models.URLField(blank=True, max_length=512)
Tony Mack360afb82013-12-17 13:32:45 -050032 site = models.ForeignKey(Site, related_name='slices', help_text="The Site this Slice belongs to")
Tony Mack3de59e32015-08-19 11:58:18 -040033 max_instances = models.IntegerField(default=10)
Scott Bakerf24ca9d2015-03-09 16:21:36 -070034 service = models.ForeignKey(Service, related_name='slices', null=True, blank=True)
Scott Bakerddf80092015-12-14 16:21:33 -080035 network = models.CharField(null=True, blank=True, max_length=256, choices=NETWORK_CHOICES)
Scott Baker3fe056a2016-02-04 15:58:19 -080036 exposed_ports = models.CharField(null=True, blank=True, max_length=256)
Siobhan Tullyde5450d2013-06-21 11:35:33 -040037 tags = generic.GenericRelation(Tag)
Scott Bakercfe0fd92014-09-12 12:24:24 -070038 serviceClass = models.ForeignKey(ServiceClass, related_name = "slices", null=True, default=get_default_serviceclass)
Tony Mack2bd5b412013-06-11 21:05:06 -040039 creator = models.ForeignKey(User, related_name='slices', blank=True, null=True)
Siobhan Tully231f4c82013-05-02 05:47:24 -040040
Scott Bakeree9736d2015-01-03 16:26:38 -080041 # for tenant view
Scott Baker200181e2015-01-12 13:13:05 -080042 default_flavor = models.ForeignKey(Flavor, related_name = "slices", null=True, blank=True)
43 default_image = models.ForeignKey(Image, related_name = "slices", null=True, blank=True);
Pingping Lin276c1b92016-04-20 18:02:10 -070044 default_node = models.ForeignKey(Node, related_name = "slices", null=True, blank=True)
Tony Mackd84b1ff2015-03-09 13:03:56 -040045 mount_data_sets = StrippedCharField(default="GenBank",null=True, blank=True, max_length=256)
Scott Bakeree9736d2015-01-03 16:26:38 -080046
Scott Baker3f237242015-11-16 19:12:25 -080047 default_isolation = models.CharField(null=False, blank=False, max_length=30, choices=ISOLATION_CHOICES, default="vm")
48
Siobhan Tully4bc09f22013-04-10 21:15:21 -040049 def __unicode__(self): return u'%s' % (self.name)
50
Tony Mack2721d6f2014-08-11 11:14:58 -040051 @property
52 def slicename(self):
53 return "%s_%s" % (self.site.login_base, self.name)
54
Tony Mack62bc59a2013-04-14 23:27:12 -040055 def save(self, *args, **kwds):
Tony Mackdac85762014-09-03 13:19:42 -040056 site = Site.objects.get(id=self.site.id)
Tony Mackab5d8872014-09-24 12:44:22 -040057 # allow preexisting slices to keep their original name for now
58 if not self.id and not self.name.startswith(site.login_base):
Matteo Scandolob12050e2016-01-13 14:47:38 -080059 raise XOSValidationError('slice name must begin with %s' % site.login_base)
Scott Baker22732bb2015-01-29 17:16:10 -080060
61 if self.name == site.login_base+"_":
Matteo Scandolob12050e2016-01-13 14:47:38 -080062 raise XOSValidationError('slice name is too short')
Scott Baker22732bb2015-01-29 17:16:10 -080063
64 if " " in self.name:
Matteo Scandolob12050e2016-01-13 14:47:38 -080065 raise XOSValidationError('slice name must not contain spaces')
Scott Baker22732bb2015-01-29 17:16:10 -080066
Scott Bakere8d596f2013-05-13 23:17:13 -070067 if self.serviceClass is None:
68 # We allowed None=True for serviceClass because Django evolution
69 # will fail unless it is allowed. But, we we really don't want it to
70 # ever save None, so fix it up here.
71 self.serviceClass = ServiceClass.get_default()
Tony Mackb9925772015-02-01 19:51:39 -050072
73 # set creator on first save
Tony Mack2bd5b412013-06-11 21:05:06 -040074 if not self.creator and hasattr(self, 'caller'):
75 self.creator = self.caller
Tony Mackb9925772015-02-01 19:51:39 -050076
77 # only admins change a slice's creator
78 if 'creator' in self.changed_fields and \
79 (not hasattr(self, 'caller') or not self.caller.is_admin):
Scott Bakerca03b642015-02-02 11:00:53 -080080
Scott Bakercba0ffe2015-02-03 15:02:17 -080081 if (self._initial["creator"]==None) and (self.creator==getattr(self,"caller",None)):
Scott Bakerca03b642015-02-02 11:00:53 -080082 # it's okay if the creator is being set by the caller to
83 # himeself on a new slice object.
Scott Bakerca03b642015-02-02 11:00:53 -080084 pass
85 else:
86 raise PermissionDenied("Insufficient privileges to change slice creator")
Tony Mackb9925772015-02-01 19:51:39 -050087
Scott Bakerd168ae02015-01-29 17:53:26 -080088 if not self.creator:
Matteo Scandolob12050e2016-01-13 14:47:38 -080089 raise XOSValidationError('slice has no creator')
Tony Mackb9925772015-02-01 19:51:39 -050090
Scott Bakerddf80092015-12-14 16:21:33 -080091 if self.network=="Private Only":
92 # "Private Only" was the default from the old Tenant View
93 self.network=None
94 self.enforce_choices(self.network, self.NETWORK_CHOICES)
95
Tony Mack62bc59a2013-04-14 23:27:12 -040096 super(Slice, self).save(*args, **kwds)
97
Tony Mack5b061472014-02-04 07:57:10 -050098 def can_update(self, user):
Tony Mack5ff90fc2015-02-08 21:38:41 -050099 return user.can_update_slice(self)
100
Tony Mack5b061472014-02-04 07:57:10 -0500101
Tony Mack5b061472014-02-04 07:57:10 -0500102 @staticmethod
103 def select_by_user(user):
104 if user.is_admin:
105 qs = Slice.objects.all()
106 else:
Tony Mack33df82b2014-08-20 11:29:40 -0400107 # users can see slices they belong to
Tony Mack5b061472014-02-04 07:57:10 -0500108 slice_ids = [sp.slice.id for sp in SlicePrivilege.objects.filter(user=user)]
Scott Bakere96247a2015-07-15 14:01:57 -0700109 # pis and admins can see slices at their sites
Tony Mack33df82b2014-08-20 11:29:40 -0400110 sites = [sp.site for sp in SitePrivilege.objects.filter(user=user)\
Scott Baker888022a2015-07-14 18:29:44 -0700111 if (sp.role.role == 'pi') or (sp.role.role == 'admin')]
Scott Bakere96247a2015-07-15 14:01:57 -0700112 slice_ids.extend([s.id for s in Slice.objects.filter(site__in=sites)])
Tony Mack5b061472014-02-04 07:57:10 -0500113 qs = Slice.objects.filter(id__in=slice_ids)
114 return qs
115
Sapan Bhatiad84f3852015-05-09 18:07:29 +0200116 """
Tony Mackc767c982014-10-22 13:30:41 -0400117 def delete(self, *args, **kwds):
118 # delete networks associated with this slice
119 from core.models.network import Network
120 nets = Network.objects.filter(slices=self)
121 nets.delete()
Tony Mackf3bbe472014-11-30 15:33:35 -0500122 # delete slice controllers
Tony Macka7dbd422015-01-05 22:48:11 -0500123 slice_controllers = ControllerSlice.objects.filter(slice=self)
Tony Mackf3bbe472014-11-30 15:33:35 -0500124 slice_controllers.delete()
Tony Mack9b53a4c2014-11-12 10:01:26 -0500125 # delete slice privilege
126 slice_privileges = SlicePrivilege.objects.filter(slice=self)
127 slice_privileges.delete()
128 # continue with normal delete
Sapane0abef32014-11-24 16:44:48 -0500129 super(Slice, self).delete(*args, **kwds)
Sapan Bhatiad84f3852015-05-09 18:07:29 +0200130 """
Tony Mack9b53a4c2014-11-12 10:01:26 -0500131
Tony Mackc767c982014-10-22 13:30:41 -0400132
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400133class SliceRole(PlCoreBase):
Siobhan Tullycf04fb62014-01-11 11:25:57 -0500134 ROLE_CHOICES = (('admin','Admin'),('default','Default'))
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400135
Tony Mackd84b1ff2015-03-09 13:03:56 -0400136 role = StrippedCharField(choices=ROLE_CHOICES, unique=True, max_length=30)
Siobhan Tullybfd11dc2013-09-03 12:59:24 -0400137
138 def __unicode__(self): return u'%s' % (self.role)
139
140class SlicePrivilege(PlCoreBase):
Sapan Bhatia13d2db92014-11-11 21:47:45 -0500141 user = models.ForeignKey('User', related_name='sliceprivileges')
142 slice = models.ForeignKey('Slice', related_name='sliceprivileges')
143 role = models.ForeignKey('SliceRole',related_name='sliceprivileges')
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400144
Tony Mack549f7b12015-04-11 12:17:59 -0400145 class Meta:
146 unique_together = ('user', 'slice', 'role')
Tony Macka4736f52015-03-09 17:13:14 -0400147
Siobhan Tully4bc09f22013-04-10 21:15:21 -0400148 def __unicode__(self): return u'%s %s %s' % (self.slice, self.user, self.role)
Tony Mack5b061472014-02-04 07:57:10 -0500149
Tony Mackd43be4c2015-03-29 09:29:12 -0400150 def save(self, *args, **kwds):
151 if not self.user.is_active:
152 raise PermissionDenied, "Cannot modify role(s) of a disabled user"
Scott Bakerb261b902015-04-17 15:44:30 -0700153 super(SlicePrivilege, self).save(*args, **kwds)
Tony Mackd43be4c2015-03-29 09:29:12 -0400154
Tony Mack5b061472014-02-04 07:57:10 -0500155 def can_update(self, user):
Tony Mack5ff90fc2015-02-08 21:38:41 -0500156 return user.can_update_slice(self.slice)
Tony Mack5b061472014-02-04 07:57:10 -0500157
Tony Mack5b061472014-02-04 07:57:10 -0500158 @staticmethod
159 def select_by_user(user):
160 if user.is_admin:
161 qs = SlicePrivilege.objects.all()
162 else:
Scott Baker521abcc2015-07-16 12:40:07 -0700163 # You can see your own SlicePrivileges
Tony Mack5b061472014-02-04 07:57:10 -0500164 sp_ids = [sp.id for sp in SlicePrivilege.objects.filter(user=user)]
Scott Baker521abcc2015-07-16 12:40:07 -0700165
166 # A site pi or site admin can see the SlicePrivileges for all slices in his Site
167 for priv in SitePrivilege.objects.filter(user=user):
168 if priv.role.role in ['pi', 'admin']:
169 sp_ids.extend( [sp.id for sp in SlicePrivilege.objects.filter(slice__site = priv.site)] )
170
171 # A slice admin can see the SlicePrivileges for his Slice
172 for priv in SlicePrivilege.objects.filter(user=user, role__role="admin"):
173 sp_ids.extend( [sp.id for sp in SlicePrivilege.objects.filter(slice=priv.slice)] )
174
Tony Mack5b061472014-02-04 07:57:10 -0500175 qs = SlicePrivilege.objects.filter(id__in=sp_ids)
176 return qs
Tony Macke4be32f2014-03-11 20:45:25 -0400177
Tony Macka7dbd422015-01-05 22:48:11 -0500178class ControllerSlice(PlCoreBase):
Tony Mackf3bbe472014-11-30 15:33:35 -0500179 objects = ControllerLinkManager()
180 deleted_objects = ControllerLinkDeletionManager()
Sapan Bhatiaa16be692014-09-22 14:55:08 -0400181
Tony Mackf3bbe472014-11-30 15:33:35 -0500182 controller = models.ForeignKey(Controller, related_name='controllerslices')
183 slice = models.ForeignKey(Slice, related_name='controllerslices')
Tony Mackd84b1ff2015-03-09 13:03:56 -0400184 tenant_id = StrippedCharField(null=True, blank=True, max_length=200, help_text="Keystone tenant id")
Tony Macke4be32f2014-03-11 20:45:25 -0400185
Tony Mack549f7b12015-04-11 12:17:59 -0400186 class Meta:
187 unique_together = ('controller', 'slice')
Tony Macka4736f52015-03-09 17:13:14 -0400188
Sapan Bhatia8dc418a2016-04-06 19:30:24 +0200189 def tologdict(self):
190 d=super(ControllerSlice,self).tologdict()
191 try:
192 d['slice_name']=self.slice.name
193 d['controller_name']=self.controller.name
194 except:
195 pass
196 return d
197
Tony Mackf3bbe472014-11-30 15:33:35 -0500198 def __unicode__(self): return u'%s %s' % (self.slice, self.controller)
Tony Macke4be32f2014-03-11 20:45:25 -0400199
200 @staticmethod
201 def select_by_user(user):
202 if user.is_admin:
Tony Macka7dbd422015-01-05 22:48:11 -0500203 qs = ControllerSlice.objects.all()
Tony Macke4be32f2014-03-11 20:45:25 -0400204 else:
205 slices = Slice.select_by_user(user)
Tony Macka7dbd422015-01-05 22:48:11 -0500206 qs = ControllerSlice.objects.filter(slice__in=slices)
Tony Macke4be32f2014-03-11 20:45:25 -0400207 return qs
Sapan Bhatia4d7cdf22014-12-16 01:09:04 -0500208
209 def get_cpu_stats(self):
210 filter = 'project_id=%s'%self.tenant_id
211 return monitor.get_meter('cpu',filter,None)
212
213 def get_bw_stats(self):
214 filter = 'project_id=%s'%self.tenant_id
215 return monitor.get_meter('network.outgoing.bytes',filter,None)
216
217 def get_node_stats(self):
Tony Mack3de59e32015-08-19 11:58:18 -0400218 return len(self.slice.instances)