blob: d3e618dcf071218bdcaafbd9bdfbdf6b3bc63903 [file] [log] [blame]
Scott Baker435c2c92015-01-14 00:34:45 -08001from core.models import Slice, SlicePrivilege, SliceRole, Sliver, Site, Node, User
Scott Bakere791dc62014-08-28 14:02:54 -07002from plus import PlusObjectMixin
Scott Bakera76f65d2015-01-13 16:22:57 -08003from operator import itemgetter, attrgetter
Scott Baker5f4770d2014-07-13 11:17:58 -07004
Scott Bakere791dc62014-08-28 14:02:54 -07005class SlicePlus(Slice, PlusObjectMixin):
Scott Baker88e34372014-07-13 11:46:36 -07006 class Meta:
7 proxy = True
8
Scott Bakera76f65d2015-01-13 16:22:57 -08009 def __init__(self, *args, **kwargs):
10 super(SlicePlus, self).__init__(*args, **kwargs)
Scott Bakera76f65d2015-01-13 16:22:57 -080011 self._update_users = None
Scott Baker7a76f322015-01-16 19:07:36 -080012 self._sliceInfo = None
Scott Baker55f6de62015-01-18 16:07:58 -080013 self.getSliceInfo()
14 self._site_allocation = self._sliceInfo["sitesUsed"]
15 self._initial_site_allocation = self._site_allocation
Scott Baker04ab7c82015-01-20 13:30:40 -080016 self._network_ports = self._sliceInfo["networkPorts"]
17 self._initial_network_ports = self._network_ports
Scott Bakera76f65d2015-01-13 16:22:57 -080018
Scott Baker5f4770d2014-07-13 11:17:58 -070019 def getSliceInfo(self, user=None):
Scott Baker7a76f322015-01-16 19:07:36 -080020 if not self._sliceInfo:
21 used_sites = {}
Scott Baker21909342015-01-22 15:21:24 -080022 ready_sites = {}
Scott Baker7a76f322015-01-16 19:07:36 -080023 used_deployments = {}
24 sliverCount = 0
Scott Bakerec930102015-01-20 01:02:08 -080025 sshCommands = []
Scott Baker7a76f322015-01-16 19:07:36 -080026 for sliver in self.slivers.all():
27 site = sliver.node.site_deployment.site
28 deployment = sliver.node.site_deployment.deployment
29 used_sites[site.name] = used_sites.get(site.name, 0) + 1
30 used_deployments[deployment.name] = used_deployments.get(deployment.name, 0) + 1
31 sliverCount = sliverCount + 1
Scott Baker5f4770d2014-07-13 11:17:58 -070032
Scott Bakerec930102015-01-20 01:02:08 -080033 if (sliver.instance_id and sliver.instance_name):
34 sshCommand = 'ssh -o "ProxyCommand ssh -q %s@%s" ubuntu@%s' % (sliver.instance_id, sliver.node.name, sliver.instance_name)
35 sshCommands.append(sshCommand);
36
Scott Baker21909342015-01-22 15:21:24 -080037 ready_sites[site.name] = ready_sites.get(site.name, 0) + 1
38
Scott Baker7a76f322015-01-16 19:07:36 -080039 users = {}
40 for priv in SlicePrivilege.objects.filter(slice=self):
41 if not (priv.user.id in users.keys()):
42 users[priv.user.id] = {"name": priv.user.email, "id": priv.user.id, "roles": []}
43 users[priv.user.id]["roles"].append(priv.role.role)
Scott Baker5f4770d2014-07-13 11:17:58 -070044
Scott Baker04ab7c82015-01-20 13:30:40 -080045 # XXX this assumes there is only one network that can have ports bound
46 # to it for a given slice. This is intended for the tenant view, which
47 # will obey this field.
48 networkPorts = ""
49 for networkSlice in self.networkslices.all():
50 network = networkSlice.network
51 if (network.owner.id != self.id):
52 continue
53 if network.ports:
54 networkPorts = network.ports
55
Scott Baker7a76f322015-01-16 19:07:36 -080056 self._sliceInfo= {"sitesUsed": used_sites,
Scott Baker21909342015-01-22 15:21:24 -080057 "sitesReady": ready_sites,
Scott Baker7a76f322015-01-16 19:07:36 -080058 "deploymentsUsed": used_deployments,
59 "sliverCount": sliverCount,
60 "siteCount": len(used_sites.keys()),
61 "users": users,
Scott Bakerec930102015-01-20 01:02:08 -080062 "roles": [],
Scott Baker04ab7c82015-01-20 13:30:40 -080063 "sshCommands": sshCommands,
64 "networkPorts": networkPorts}
Scott Baker7a76f322015-01-16 19:07:36 -080065
66 if user:
67 auser = self._sliceInfo["users"].get(user.id, None)
68 if (auser):
69 self._sliceInfo["roles"] = auser["roles"]
70
71 return self._sliceInfo
Scott Baker88e34372014-07-13 11:46:36 -070072
Scott Baker8b89d302015-01-08 22:34:51 -080073 @property
Scott Baker21909342015-01-22 15:21:24 -080074 def site_ready(self):
75 return self.getSliceInfo()["sitesReady"]
76
77 @site_ready.setter
78 def site_ready(self, value):
79 pass
80
81 @property
Scott Bakerdcf6fbf2015-01-11 13:45:19 -080082 def site_allocation(self):
Scott Baker55f6de62015-01-18 16:07:58 -080083 return self._site_allocation
Scott Bakerdcf6fbf2015-01-11 13:45:19 -080084
85 @site_allocation.setter
86 def site_allocation(self, value):
Scott Baker55f6de62015-01-18 16:07:58 -080087 self._site_allocation = value
Scott Bakerdcf6fbf2015-01-11 13:45:19 -080088
89 @property
Scott Baker7a76f322015-01-16 19:07:36 -080090 def user_names(self):
91 return [user["name"] for user in self.getSliceInfo()["users"].values()]
92
Scott Baker12154242015-01-16 19:26:54 -080093 @user_names.setter
94 def user_names(self, value):
95 pass # it's read-only
96
Scott Baker7a76f322015-01-16 19:07:36 -080097 @property
Scott Baker97acad92015-01-12 19:45:40 -080098 def users(self):
Scott Baker7a76f322015-01-16 19:07:36 -080099 return [user["id"] for user in self.getSliceInfo()["users"].values()]
Scott Baker97acad92015-01-12 19:45:40 -0800100
101 @users.setter
102 def users(self, value):
Scott Bakera76f65d2015-01-13 16:22:57 -0800103 self._update_users = value
104 #print "XXX set users to", value
Scott Baker97acad92015-01-12 19:45:40 -0800105
106 @property
Scott Bakerd3a6b2c2015-01-08 22:37:34 -0800107 def network_ports(self):
Scott Baker04ab7c82015-01-20 13:30:40 -0800108 return self._network_ports
Scott Baker8b89d302015-01-08 22:34:51 -0800109
Scott Bakerd3a6b2c2015-01-08 22:37:34 -0800110 @network_ports.setter
111 def network_ports(self, value):
Scott Baker04ab7c82015-01-20 13:30:40 -0800112 self._network_ports = value
113 #print "XXX set networkPorts to", value
Scott Baker8b89d302015-01-08 22:34:51 -0800114
Scott Baker88e34372014-07-13 11:46:36 -0700115 @staticmethod
116 def select_by_user(user):
Scott Baker88e34372014-07-13 11:46:36 -0700117 if user.is_admin:
118 qs = SlicePlus.objects.all()
119 else:
120 slice_ids = [sp.slice.id for sp in SlicePrivilege.objects.filter(user=user)]
121 qs = SlicePlus.objects.filter(id__in=slice_ids)
Scott Baker88e34372014-07-13 11:46:36 -0700122 return qs
Scott Bakera76f65d2015-01-13 16:22:57 -0800123
Scott Baker435c2c92015-01-14 00:34:45 -0800124 def get_node_allocation(self, siteList):
Scott Bakera76f65d2015-01-13 16:22:57 -0800125 siteIDList = [site.id for site in siteList]
126 nodeList = []
127 for node in Node.objects.all():
128 if (node.site_deployment.site.id in siteIDList):
129 node.sliverCount = 0
130 for sliver in node.slivers.all():
131 if sliver.slice.id == self.id:
132 node.sliverCount = node.sliverCount + 1
133 nodeList.append(node)
134 return nodeList
135
136 def save(self, *args, **kwargs):
Scott Baker55f6de62015-01-18 16:07:58 -0800137 updated_image = self.has_field_changed("default_image")
138 updated_flavor = self.has_field_changed("default_flavor")
139
Scott Bakera76f65d2015-01-13 16:22:57 -0800140 super(SlicePlus, self).save(*args, **kwargs)
141
Scott Baker04ab7c82015-01-20 13:30:40 -0800142 # try things out first
143
Scott Baker55f6de62015-01-18 16:07:58 -0800144 updated_sites = (self._site_allocation != self._initial_site_allocation) or updated_image or updated_flavor
145 if updated_sites:
146 self.save_site_allocation(noAct=True, reset=(updated_image or updated_flavor))
Scott Baker435c2c92015-01-14 00:34:45 -0800147
148 if self._update_users:
149 self.save_users(noAct=True)
150
Scott Baker04ab7c82015-01-20 13:30:40 -0800151 if (self._network_ports != self._initial_network_ports):
152 self.save_network_ports(noAct=True)
153
154 # now actually save them
155
Scott Baker55f6de62015-01-18 16:07:58 -0800156 if updated_sites:
157 self.save_site_allocation(reset=(updated_image or updated_flavor))
Scott Bakera76f65d2015-01-13 16:22:57 -0800158
Scott Baker435c2c92015-01-14 00:34:45 -0800159 if self._update_users:
160 self.save_users()
161
Scott Baker04ab7c82015-01-20 13:30:40 -0800162 if (self._network_ports != self._initial_network_ports):
163 self.save_network_ports()
164
Scott Baker55f6de62015-01-18 16:07:58 -0800165 def save_site_allocation(self, noAct = False, reset=False):
166 print "save_site_allocation, reset=",reset
Scott Bakera76f65d2015-01-13 16:22:57 -0800167
Scott Baker75081422015-01-19 08:43:50 -0800168 if (not self._site_allocation):
169 # Must be a sliver that was just created, and has not site_allocation
170 # field.
171 return
172
Scott Baker435c2c92015-01-14 00:34:45 -0800173 all_slice_slivers = self.slivers.all()
Scott Baker55f6de62015-01-18 16:07:58 -0800174 for site_name in self._site_allocation.keys():
175 desired_allocation = self._site_allocation[site_name]
Scott Bakera76f65d2015-01-13 16:22:57 -0800176
177 # make a list of the slivers for this site
178 slivers = []
179 for sliver in all_slice_slivers:
180 if sliver.node.site_deployment.site.name == site_name:
181 slivers.append(sliver)
182
183 # delete extra slivers
Scott Baker55f6de62015-01-18 16:07:58 -0800184 while (reset and len(slivers)>0) or (len(slivers) > desired_allocation):
Scott Bakera76f65d2015-01-13 16:22:57 -0800185 sliver = slivers.pop()
Scott Bakera76f65d2015-01-13 16:22:57 -0800186 if (not noAct):
Scott Baker55f6de62015-01-18 16:07:58 -0800187 print "deleting sliver", sliver
Scott Bakera76f65d2015-01-13 16:22:57 -0800188 sliver.delete()
Scott Baker55f6de62015-01-18 16:07:58 -0800189 else:
190 print "would delete sliver", sliver
Scott Bakera76f65d2015-01-13 16:22:57 -0800191
192 # add more slivers
193 if (len(slivers) < desired_allocation):
194 site = Site.objects.get(name = site_name)
Scott Baker435c2c92015-01-14 00:34:45 -0800195 nodes = self.get_node_allocation([site])
Scott Bakera76f65d2015-01-13 16:22:57 -0800196
197 if (not nodes):
Scott Baker1f7901b2015-01-13 16:28:05 -0800198 raise ValueError("no nodes in site %s" % site_name)
Scott Bakera76f65d2015-01-13 16:22:57 -0800199
200 while (len(slivers) < desired_allocation):
201 # pick the least allocated node
202 nodes = sorted(nodes, key=attrgetter("sliverCount"))
203 node = nodes[0]
204
205 sliver = Sliver(name=node.name,
206 slice=self,
207 node=node,
208 image = self.default_image,
209 flavor = self.default_flavor,
210 creator = self.creator,
211 deployment = node.site_deployment.deployment)
212 slivers.append(sliver)
213 if (not noAct):
Scott Baker55f6de62015-01-18 16:07:58 -0800214 print "added sliver", sliver
Scott Bakera76f65d2015-01-13 16:22:57 -0800215 sliver.save()
Scott Baker55f6de62015-01-18 16:07:58 -0800216 else:
217 print "would add sliver", sliver
Scott Bakera76f65d2015-01-13 16:22:57 -0800218
219 node.sliverCount = node.sliverCount + 1
220
Scott Baker435c2c92015-01-14 00:34:45 -0800221 def save_users(self, noAct = False):
222 new_users = self._update_users
223
224 default_role = SliceRole.objects.get(role="default")
225
226 slice_privs = self.sliceprivileges.all()
227 slice_user_ids = [priv.user.id for priv in slice_privs]
228
229 for user_id in new_users:
230 if (user_id not in slice_user_ids):
Scott Baker435c2c92015-01-14 00:34:45 -0800231 priv = SlicePrivilege(slice=self, user=User.objects.get(id=user_id), role=default_role)
232 if (not noAct):
233 priv.save()
234
235 print "added user id", user_id
236
237 for priv in slice_privs:
238 if (priv.role.id != default_role.id):
239 # only mess with 'default' users; don't kill an admin
240 continue
241
242 if (priv.user.id not in new_users):
243 if (not noAct):
244 priv.delete()
245
246 print "deleted user id", user_id
247
Scott Baker04ab7c82015-01-20 13:30:40 -0800248 def save_network_ports(self, noAct=False):
249 # First search for any network that already has a filled in 'ports'
250 # field. We'll assume there can only be one, so it must be the right
251 # one.
252 for networkSlice in self.networkslices.all():
253 network = networkSlice.network
254 if (network.owner.id != self.id):
255 continue
256 if network.ports:
257 network.ports = self._network_ports
258 if (not noAct):
259 network.save()
260 return
261
262 # Now try a network that is a "NAT", since setting ports on a non-NAT
263 # network doesn't make much sense.
264 for networkSlice in self.networkslices.all():
265 network = networkSlice.network
266 if (network.owner.id != self.id):
267 continue
268 if network.template.translation=="NAT":
269 network.ports = self._network_ports
270 if (not noAct):
271 network.save()
272 return
273
274 # uh oh, we didn't find a network
275
276 raise ValueError("No network was found that ports could be set on")
277
Scott Baker435c2c92015-01-14 00:34:45 -0800278
279
280
Scott Bakera76f65d2015-01-13 16:22:57 -0800281