added rbac for viewing objects
[plstackapi.git] / planetstack / core / models / network.py
1 import os
2 import socket
3 from django.db import models
4 from core.models import PlCoreBase, Site, Slice, Sliver
5 from django.contrib.contenttypes.models import ContentType
6 from django.contrib.contenttypes import generic
7
8 # If true, then IP addresses will be allocated by the model. If false, then
9 # we will assume the observer handles it.
10 NO_OBSERVER=False
11
12 class NetworkTemplate(PlCoreBase):
13     VISIBILITY_CHOICES = (('public', 'public'), ('private', 'private'))
14     TRANSLATION_CHOICES = (('none', 'none'), ('NAT', 'NAT'))
15
16     name = models.CharField(max_length=32)
17     description = models.CharField(max_length=1024, blank=True, null=True)
18     guaranteedBandwidth = models.IntegerField(default=0)
19     visibility = models.CharField(max_length=30, choices=VISIBILITY_CHOICES, default="private")
20     translation = models.CharField(max_length=30, choices=TRANSLATION_CHOICES, default="none")
21     sharedNetworkName = models.CharField(max_length=30, blank=True, null=True)
22     sharedNetworkId = models.CharField(null=True, blank=True, max_length=256, help_text="Quantum network")
23
24     def __unicode__(self):  return u'%s' % (self.name)
25
26 class Network(PlCoreBase):
27     name = models.CharField(max_length=32)
28     template = models.ForeignKey(NetworkTemplate)
29     subnet = models.CharField(max_length=32, blank=True)
30     ports = models.CharField(max_length=1024, blank=True, null=True)
31     labels = models.CharField(max_length=1024, blank=True, null=True)
32     owner = models.ForeignKey(Slice, related_name="ownedNetworks", help_text="Slice that owns control of this Network")
33
34     guaranteedBandwidth = models.IntegerField(default=0)
35     permitAllSlices = models.BooleanField(default=False)
36     permittedSlices = models.ManyToManyField(Slice, blank=True, related_name="availableNetworks")
37     slices = models.ManyToManyField(Slice, blank=True, related_name="networks", through="NetworkSlice")
38     slivers = models.ManyToManyField(Sliver, blank=True, related_name="networks", through="NetworkSliver")
39
40     # for observer/manager
41     network_id = models.CharField(null=True, blank=True, max_length=256, help_text="Quantum network")
42     router_id = models.CharField(null=True, blank=True, max_length=256, help_text="Quantum router id")
43     subnet_id = models.CharField(null=True, blank=True, max_length=256, help_text="Quantum subnet id")
44
45     def __unicode__(self):  return u'%s' % (self.name)
46
47     def save(self, *args, **kwds):
48         if (not self.subnet) and (NO_OBSERVER):
49             from util.network_subnet_allocator import find_unused_subnet
50             self.subnet = find_unused_subnet(existing_subnets=[x.subnet for x in Network.objects.all()])
51         super(Network, self).save(*args, **kwds)
52
53     def can_update(self, user):
54         return self.slice.can_update(user)
55
56     def save_by_user(self, user, *args, **kwds):
57         if self.slice.can_update(user):
58             super(Network, self).save(*args, **kwds)
59
60     @staticmethod
61     def select_by_user(user):
62         if user.is_admin:
63             qs = Network.objects.all()
64         else:
65             slice_ids = [s.id for s in Slice.select_by_user(user)]
66             qs = Network.objects.filter(id__in=slice_ids)
67         return qs
68
69 class NetworkSlice(PlCoreBase):
70     # This object exists solely so we can implement the permission check when
71     # adding slices to networks. It adds no additional fields to the relation.
72
73     network = models.ForeignKey(Network)
74     slice = models.ForeignKey(Slice)
75
76     def save(self, *args, **kwds):
77         slice = self.slice
78         if (slice not in self.network.permittedSlices.all()) and (slice != self.network.owner) and (not self.network.permitAllSlices):
79             # to add a sliver to the network, then one of the following must be true:
80             #   1) sliver's slice is in network's permittedSlices list,
81             #   2) sliver's slice is network's owner, or
82             #   3) network's permitAllSlices is true
83             raise ValueError("Slice %s is not allowed to connect to network %s" % (str(slice), str(self.network)))
84
85         super(NetworkSlice, self).save(*args, **kwds)
86
87     def __unicode__(self):  return u'%s-%s' % (self.network.name, self.slice.name)
88
89     def can_update(self, user):
90         return self.slice.can_update(user)
91
92     def save_by_user(self, user, *args, **kwds):
93         if self.slice.can_update(user):
94             super(NetworkSlice, self).save(*args, **kwds)
95
96     @staticmethod
97     def select_by_user(user):
98         if user.is_admin:
99             qs = NetworkSlice.objects.all()
100         else:
101             slice_ids = [s.id for s in Slice.select_by_user(user)]
102             qs = NetworkSlice.objects.filter(id__in=slice_ids)
103         return qs
104
105 class NetworkSliver(PlCoreBase):
106     network = models.ForeignKey(Network)
107     sliver = models.ForeignKey(Sliver)
108     ip = models.GenericIPAddressField(help_text="Sliver ip address", blank=True, null=True)
109     port_id = models.CharField(null=True, blank=True, max_length=256, help_text="Quantum port id")
110
111     def save(self, *args, **kwds):
112         slice = self.sliver.slice
113         if (slice not in self.network.permittedSlices.all()) and (slice != self.network.owner) and (not self.network.permitAllSlices):
114             # to add a sliver to the network, then one of the following must be true:
115             #   1) sliver's slice is in network's permittedSlices list,
116             #   2) sliver's slice is network's owner, or
117             #   3) network's permitAllSlices is true
118             raise ValueError("Slice %s is not allowed to connect to network %s" % (str(slice), str(self.network)))
119
120         if (not self.ip) and (NO_OBSERVER):
121             from util.network_subnet_allocator import find_unused_address
122             self.ip = find_unused_address(self.network.subnet,
123                                           [x.ip for x in self.network.networksliver_set.all()])
124         super(NetworkSliver, self).save(*args, **kwds)
125
126     def __unicode__(self):  return u'%s-%s' % (self.network.name, self.sliver.instance_name)
127
128     def can_update(self, user):
129         return self.sliver.can_update(user)
130
131     def save_by_user(self, user, *args, **kwds):
132         if self.sliver.can_update(user):
133             super(NetworkSliver, self).save(*args, **kwds)
134
135     @staticmethod
136     def select_by_user(user):
137         if user.is_admin:
138             qs = NetworkSliver.objects.all()
139         else:
140             sliver_ids = [s.id for s in NetworkSliver.select_by_user(user)]
141             qs = NetworkSliver.objects.filter(id__in=sliver_ids)
142         return qs
143
144 class Router(PlCoreBase):
145     name = models.CharField(max_length=32)
146     owner = models.ForeignKey(Slice, related_name="routers")
147     permittedNetworks = models.ManyToManyField(Network, blank=True, related_name="availableRouters")
148     networks = models.ManyToManyField(Network, blank=True, related_name="routers")
149
150     def __unicode__(self):  return u'%s' % (self.name)
151
152 class NetworkParameterType(PlCoreBase):
153     name = models.SlugField(help_text="The name of this parameter", max_length=128)
154     description = models.CharField(max_length=1024)
155
156     def __unicode__(self):  return u'%s' % (self.name)
157
158 class NetworkParameter(PlCoreBase):
159     parameter = models.ForeignKey(NetworkParameterType, related_name="parameters", help_text="The type of the parameter")
160     value = models.CharField(help_text="The value of this parameter", max_length=1024)
161
162     # The required fields to do a ObjectType lookup, and object_id assignment
163     content_type = models.ForeignKey(ContentType)
164     object_id = models.PositiveIntegerField()
165     content_object = generic.GenericForeignKey('content_type', 'object_id')
166
167     def __unicode__(self):
168         return self.parameter.name
169
170