blob: 60632821710cfcba0016bac9f697f33ec82037c6 [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
20from models_decl import RCORDService_decl, RCORDSubscriber_decl
Matteo Scandolo62a83f02018-03-01 15:59:18 -080021
Matteo Scandolod1707b32018-05-04 12:42:53 -070022class RCORDService(RCORDService_decl):
23 class Meta:
24 proxy = True
25
26class RCORDSubscriber(RCORDSubscriber_decl):
Matteo Scandolo520217f2018-05-16 14:15:56 -070027
Matteo Scandolo62a83f02018-03-01 15:59:18 -080028 class Meta:
29 proxy = True
30
31 def invalidate_related_objects(self):
32 # Dirty all vSGs related to this subscriber, so the vSG synchronizer
33 # will run.
34
35 # FIXME: This should be reimplemented when multiple-objects-per-synchronizer is implemented.
36
37 for link in self.subscribed_links.all():
38 outer_service_instance = link.provider_service_instance
39 # TODO: We may need to invalide the vOLT too...
40 for link in outer_service_instance.subscribed_links.all():
41 inner_service_instance = link.provider_service_instance
42 inner_service_instance.save(update_fields=["updated"])
43
Matteo Scandolocc94e902018-05-22 15:25:25 -070044 def generate_tag(self):
45 # NOTE this method will loop if available c_tags are ended
46 tag = random.randint(16, 4096)
47 if tag in self.get_used_c_tags():
48 return self.generate_tag()
49 return tag
50
51 def get_used_c_tags(self):
52 same_onu_subscribers = RCORDSubscriber.objects.filter(onu_device=self.onu_device)
53 return [s.c_tag for s in same_onu_subscribers]
54
Matteo Scandolo62a83f02018-03-01 15:59:18 -080055 def save(self, *args, **kwargs):
Matteo Scandolocc94e902018-05-22 15:25:25 -070056
Matteo Scandolo62a83f02018-03-01 15:59:18 -080057 self.validate_unique_service_specific_id(none_okay=True)
58
Scott Baker9d9ddf62018-03-20 20:44:27 -070059 # VSGServiceInstance will extract the creator from the Subscriber, as it needs a creator to create its
60 # Instance.
61 if not self.creator:
62 # If we weren't passed an explicit creator, then we will assume the caller is the creator.
63 if not getattr(self, "caller", None):
Matteo Scandolod1707b32018-05-04 12:42:53 -070064 raise XOSProgrammingError("RCORDSubscriber's self.caller was not set")
Scott Baker9d9ddf62018-03-20 20:44:27 -070065 self.creator = self.caller
66
Matteo Scandolo62a83f02018-03-01 15:59:18 -080067 if (not hasattr(self, 'caller') or not self.caller.is_admin):
68 if (self.has_field_changed("service_specific_id")):
69 raise XOSPermissionDenied("You do not have permission to change service_specific_id")
70
Matteo Scandolod1707b32018-05-04 12:42:53 -070071 # validate IP Address
72 if hasattr(self, 'ip_address') and self.ip_address is not None:
73 try:
74 socket.inet_aton(self.ip_address)
75 except socket.error:
76 raise XOSValidationError("The ip_address you specified (%s) is not valid" % self.ip_address)
77
78 # validate MAC Address
79 if hasattr(self, 'mac_address') and self.mac_address is not None:
80 if not re.match("[0-9a-f]{2}([-:]?)[0-9a-f]{2}(\\1[0-9a-f]{2}){4}$", self.mac_address.lower()):
81 raise XOSValidationError("The mac_address you specified (%s) is not valid" % self.mac_address)
82
Matteo Scandolocc94e902018-05-22 15:25:25 -070083 # validate c_tag
84 if hasattr(self, 'c_tag') and self.c_tag is not None:
Matteo Scandoloaa37b422018-05-23 15:36:56 -070085 is_update_with_same_tag = False
86
87 if not self.is_new:
88 # if it is an update, but the tag is the same, skip validation
89 existing = RCORDSubscriber.objects.filter(c_tag=self.c_tag)
90
91 if len(existing) > 0 and existing[0].c_tag == self.c_tag and existing[0].id == self.id:
92 is_update_with_same_tag = True
93
94 if self.c_tag in self.get_used_c_tags() and not is_update_with_same_tag:
Matteo Scandolocc94e902018-05-22 15:25:25 -070095 raise XOSValidationError("The c_tag you specified (%s) has already been used on device %s" % (self.c_tag, self.onu_device))
96
97 if not hasattr(self, "c_tag") or self.c_tag is None:
98 self.c_tag = self.generate_tag()
99
Matteo Scandolo520217f2018-05-16 14:15:56 -0700100 self.set_owner()
101
Matteo Scandolof9698702018-05-17 13:26:19 -0700102 if hasattr(self.owner.leaf_model, "access") and self.owner.leaf_model.access == "voltha":
Matteo Scandolocc94e902018-05-22 15:25:25 -0700103
Matteo Scandolob6d67fd2018-05-18 16:28:51 -0700104 # if the access network is managed by voltha, validate that onu_device actually exist
Matteo Scandolod1707b32018-05-04 12:42:53 -0700105 volt_service = self.owner.provider_services[0].leaf_model # we assume RCORDService is connected only to the vOLTService
106
Matteo Scandolob6d67fd2018-05-18 16:28:51 -0700107 if not volt_service.has_access_device(self.onu_device):
108 raise XOSValidationError("The onu_device you specified (%s) does not exists" % self.onu_device)
Matteo Scandolod1707b32018-05-04 12:42:53 -0700109
110 super(RCORDSubscriber, self).save(*args, **kwargs)
Matteo Scandolo62a83f02018-03-01 15:59:18 -0800111 self.invalidate_related_objects()
112 return