blob: aa859ea96a441ca6820b3823ca5294fe60548d3b [file] [log] [blame]
Matteo Scandolo62a83f02018-03-01 15:59:18 -08001
2# Copyright 2017-present Open Networking Foundation
3#
4# Licensed under the Apache License, Version 2.0 (the "License");
5# you may not use this file except in compliance with the License.
6# You may obtain a copy of the License at
7#
8# http://www.apache.org/licenses/LICENSE-2.0
9#
10# Unless required by applicable law or agreed to in writing, software
11# distributed under the License is distributed on an "AS IS" BASIS,
12# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13# See the License for the specific language governing permissions and
14# limitations under the License.
Matteo Scandolod1707b32018-05-04 12:42:53 -070015import re
16import socket
Matteo Scandolocc94e902018-05-22 15:25:25 -070017import random
Matteo Scandolo62a83f02018-03-01 15:59:18 -080018
Matteo Scandolod1707b32018-05-04 12:42:53 -070019from xos.exceptions import XOSValidationError, XOSProgrammingError, XOSPermissionDenied
Matteo Scandoloa4a279a2019-03-14 15:56:22 -070020from models_decl import RCORDService_decl, RCORDSubscriber_decl, RCORDIpAddress_decl, BandwidthProfile_decl
21
22class BandwidthProfile(BandwidthProfile_decl):
23 class Meta:
24 proxy = True
Matteo Scandolo62a83f02018-03-01 15:59:18 -080025
Matteo Scandolod1707b32018-05-04 12:42:53 -070026class RCORDService(RCORDService_decl):
27 class Meta:
28 proxy = True
29
Luca Preted6700ba2018-09-12 16:40:49 -070030class RCORDIpAddress(RCORDIpAddress_decl):
31 class Meta:
32 proxy = True
33
34 def save(self, *args, **kwargs):
35 try:
36 if ":" in self.ip:
37 # it's an IPv6 address
38 socket.inet_pton(socket.AF_INET6, self.ip)
39 else:
40 # it's an IPv4 address
41 socket.inet_pton(socket.AF_INET, self.ip)
42 except socket.error:
43 raise XOSValidationError("The IP specified is not valid: %s" % self.ip)
44 super(RCORDIpAddress, self).save(*args, **kwargs)
45 return
46
Matteo Scandolod1707b32018-05-04 12:42:53 -070047class RCORDSubscriber(RCORDSubscriber_decl):
Matteo Scandolo520217f2018-05-16 14:15:56 -070048
Matteo Scandolo62a83f02018-03-01 15:59:18 -080049 class Meta:
50 proxy = True
51
52 def invalidate_related_objects(self):
53 # Dirty all vSGs related to this subscriber, so the vSG synchronizer
54 # will run.
55
56 # FIXME: This should be reimplemented when multiple-objects-per-synchronizer is implemented.
57
58 for link in self.subscribed_links.all():
59 outer_service_instance = link.provider_service_instance
Matteo Scandoloa1875602018-10-16 07:35:04 -070060 # TODO: We may need to invalidate the vOLT too...
Matteo Scandolo62a83f02018-03-01 15:59:18 -080061 for link in outer_service_instance.subscribed_links.all():
62 inner_service_instance = link.provider_service_instance
63 inner_service_instance.save(update_fields=["updated"])
64
Matteo Scandoloc348b3f2018-07-29 09:35:11 -070065 def generate_s_tag(self):
66 # NOTE what's the right way to generate an s_tag?
67 tag = random.randint(16, 4096)
68 return tag
69
70 def generate_c_tag(self):
Matteo Scandolocc94e902018-05-22 15:25:25 -070071 # NOTE this method will loop if available c_tags are ended
72 tag = random.randint(16, 4096)
73 if tag in self.get_used_c_tags():
Matteo Scandoloc348b3f2018-07-29 09:35:11 -070074 return self.generate_c_tag()
Matteo Scandolocc94e902018-05-22 15:25:25 -070075 return tag
76
77 def get_used_c_tags(self):
Matteo Scandoloa1875602018-10-16 07:35:04 -070078 # TODO add validation for no duplicate c_tag on the same s_tag
Matteo Scandolocc94e902018-05-22 15:25:25 -070079 same_onu_subscribers = RCORDSubscriber.objects.filter(onu_device=self.onu_device)
Matteo Scandoloa1875602018-10-16 07:35:04 -070080 same_onu_subscribers = [s for s in same_onu_subscribers if s.id != self.id]
81 used_tags = [s.c_tag for s in same_onu_subscribers]
82 return used_tags
Matteo Scandolocc94e902018-05-22 15:25:25 -070083
Matteo Scandolo62a83f02018-03-01 15:59:18 -080084 def save(self, *args, **kwargs):
85 self.validate_unique_service_specific_id(none_okay=True)
86
Scott Baker9d9ddf62018-03-20 20:44:27 -070087 # VSGServiceInstance will extract the creator from the Subscriber, as it needs a creator to create its
88 # Instance.
89 if not self.creator:
90 # If we weren't passed an explicit creator, then we will assume the caller is the creator.
91 if not getattr(self, "caller", None):
Matteo Scandolod1707b32018-05-04 12:42:53 -070092 raise XOSProgrammingError("RCORDSubscriber's self.caller was not set")
Scott Baker9d9ddf62018-03-20 20:44:27 -070093 self.creator = self.caller
94
Matteo Scandolod1707b32018-05-04 12:42:53 -070095 # validate MAC Address
Andy Bavier618f0ed2018-12-11 13:45:23 -070096 if hasattr(self, 'mac_address') and self.mac_address:
Matteo Scandolod1707b32018-05-04 12:42:53 -070097 if not re.match("[0-9a-f]{2}([-:]?)[0-9a-f]{2}(\\1[0-9a-f]{2}){4}$", self.mac_address.lower()):
Luca Preted6700ba2018-09-12 16:40:49 -070098 raise XOSValidationError("The MAC address specified is not valid: %s" % self.mac_address)
Matteo Scandolod1707b32018-05-04 12:42:53 -070099
Matteo Scandolocc94e902018-05-22 15:25:25 -0700100 # validate c_tag
Andy Bavier618f0ed2018-12-11 13:45:23 -0700101 if hasattr(self, 'c_tag') and self.c_tag:
Matteo Scandoloaa37b422018-05-23 15:36:56 -0700102 is_update_with_same_tag = False
103
104 if not self.is_new:
105 # if it is an update, but the tag is the same, skip validation
106 existing = RCORDSubscriber.objects.filter(c_tag=self.c_tag)
107
108 if len(existing) > 0 and existing[0].c_tag == self.c_tag and existing[0].id == self.id:
109 is_update_with_same_tag = True
110
111 if self.c_tag in self.get_used_c_tags() and not is_update_with_same_tag:
Matteo Scandolocc94e902018-05-22 15:25:25 -0700112 raise XOSValidationError("The c_tag you specified (%s) has already been used on device %s" % (self.c_tag, self.onu_device))
113
Andy Bavier618f0ed2018-12-11 13:45:23 -0700114 if not hasattr(self, "c_tag") or not self.c_tag:
Matteo Scandoloc348b3f2018-07-29 09:35:11 -0700115 self.c_tag = self.generate_c_tag()
116
Andy Bavier618f0ed2018-12-11 13:45:23 -0700117 if not hasattr(self, "s_tag") or not self.s_tag:
Matteo Scandoloc348b3f2018-07-29 09:35:11 -0700118 self.s_tag = self.generate_s_tag()
Matteo Scandolocc94e902018-05-22 15:25:25 -0700119
Matteo Scandolo520217f2018-05-16 14:15:56 -0700120 self.set_owner()
121
Matteo Scandolo10d94512019-03-04 15:54:43 -0800122 if self.status != "pre-provisioned" and hasattr(self.owner.leaf_model, "access") and self.owner.leaf_model.access == "voltha" and not self.deleted:
Matteo Scandolocc94e902018-05-22 15:25:25 -0700123
Matteo Scandolob6d67fd2018-05-18 16:28:51 -0700124 # if the access network is managed by voltha, validate that onu_device actually exist
Matteo Scandolod1707b32018-05-04 12:42:53 -0700125 volt_service = self.owner.provider_services[0].leaf_model # we assume RCORDService is connected only to the vOLTService
126
Matteo Scandolob6d67fd2018-05-18 16:28:51 -0700127 if not volt_service.has_access_device(self.onu_device):
128 raise XOSValidationError("The onu_device you specified (%s) does not exists" % self.onu_device)
Matteo Scandolod1707b32018-05-04 12:42:53 -0700129
130 super(RCORDSubscriber, self).save(*args, **kwargs)
Matteo Scandolo62a83f02018-03-01 15:59:18 -0800131 self.invalidate_related_objects()
132 return