blob: 711a28bc5473fc84d3eb8180d534c6dadf2e0aef [file] [log] [blame]
Sapan Bhatia2810e032017-11-03 00:31:00 -04001# Copyright 2017-present Open Networking Foundation
2#
3# Licensed under the Apache License, Version 2.0 (the "License");
4# you may not use this file except in compliance with the License.
5# You may obtain a copy of the License at
6#
7# http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS,
11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12# See the License for the specific language governing permissions and
13# limitations under the License.
14
15from synchronizers.new_base.modelaccessor import *
Sapan Bhatia503ad902017-11-13 13:06:17 -050016from synchronizers.new_base.model_policies.model_policy_tenantwithcontainer import Policy
Sapan Bhatia2810e032017-11-03 00:31:00 -040017from synchronizers.new_base.exceptions import *
18
19from xosconfig import Config
20from multistructlog import create_logger
21
22log = create_logger(Config().get('logging'))
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050023
Sapan Bhatia2810e032017-11-03 00:31:00 -040024blueprints = Config().get('blueprints')
25
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050026
Sapan Bhatia2810e032017-11-03 00:31:00 -040027def service_of_service_instance(si):
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050028 if si.endswith('Tenant'):
Sapan Bhatia2810e032017-11-03 00:31:00 -040029 return si[:-len('Tenant')] + 'Service'
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050030 elif si.endswith('ServiceInstance'):
Sapan Bhatia2810e032017-11-03 00:31:00 -040031 return si[:-len('ServiceInstance')] + 'Service'
32 else:
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050033 raise Exception(
34 'Could not translate service instance into service: %s' % si)
35
Sapan Bhatia2810e032017-11-03 00:31:00 -040036
Sapan Bhatia503ad902017-11-13 13:06:17 -050037class VEPCServiceInstancePolicy(Policy):
Sapan Bhatia2810e032017-11-03 00:31:00 -040038 model_name = "VEPCServiceInstance"
39
Sapan Bhatia503ad902017-11-13 13:06:17 -050040 def __init__(self):
41 self.in_memory_instances = []
Sapan Bhatia491a0802017-11-22 19:37:28 -050042 self.network_map = {}
43
Sapan Bhatia503ad902017-11-13 13:06:17 -050044 super(VEPCServiceInstancePolicy, self).__init__()
45
Sapan Bhatia2810e032017-11-03 00:31:00 -040046 """TODO: Update the following to not be service-specific
Andy Bavier52b5e0f2017-11-27 15:58:21 -070047 This code assumes there is only one vendor installed
Sapan Bhatia2810e032017-11-03 00:31:00 -040048 """
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050049
Sapan Bhatia2810e032017-11-03 00:31:00 -040050 def configure_service_instance(self, service_instance):
51 if service_instance.leaf_model_name == 'VSPGWUTenant':
52 vendor = VSPGWUVendor.objects.first()
53 if not vendor:
54 raise Exception('No VSPGWU vendors')
55 service_instance.vspgwu_vendor = vendor
Andy Bavier52b5e0f2017-11-27 15:58:21 -070056 service_instance.invalidate_cache('vspgwu_vendor')
Sapan Bhatia2810e032017-11-03 00:31:00 -040057 elif service_instance.leaf_model_name == 'VSPGWCTenant':
58 vendor = VSPGWCVendor.objects.first()
59 if not vendor:
60 raise Exception('No VSPGWC vendors')
61 service_instance.vspgwc_vendor = vendor
Andy Bavier52b5e0f2017-11-27 15:58:21 -070062 service_instance.invalidate_cache('vspgwc_vendor')
Woojoong Kim5d8fb6d2017-12-18 13:58:49 -080063 elif service_instance.leaf_model_name == 'VMMETenant':
64 vendor = VMMEVendor.objects.first()
65 if not vendor:
66 raise Exception('No VMME vendors')
67 service_instance.vmme_vendor = vendor
68 service_instance.invalidate_cache('vmme_vendor')
Andy Bavier8721f5e2018-01-02 14:36:23 -070069 elif service_instance.leaf_model_name == 'VHSSTenant':
70 vendor = VHSSVendor.objects.first()
71 if not vendor:
72 raise Exception('No VHSS vendors')
73 service_instance.vhss_vendor = vendor
74 service_instance.invalidate_cache('vhss_vendor')
Woojoong Kim4b5612b2018-01-11 10:08:02 -080075 elif service_instance.leaf_model_name == 'HSSDBServiceInstance':
76 vendor = HSSDBVendor.objects.first()
77 if not vendor:
78 raise Exception('No HSSDB vendors')
79 service_instance.hssdb_vendor = vendor
80 service_instance.invalidate_cache('hssdb_vendor')
Sapan Bhatia2810e032017-11-03 00:31:00 -040081
82 def child_service_instance_from_name(self, name):
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050083 service_instances = self.obj.child_serviceinstances.all()
Sapan Bhatia503ad902017-11-13 13:06:17 -050084 service_instances.extend(self.in_memory_instances)
Sapan Bhatia2810e032017-11-03 00:31:00 -040085
86 try:
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050087 service_instance = next(
88 s for s in service_instances if s.leaf_model_name == name)
Sapan Bhatia2810e032017-11-03 00:31:00 -040089 except StopIteration:
90 service_instance = None
91
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050092 return service_instance
Sapan Bhatia2810e032017-11-03 00:31:00 -040093
94 def get_service_for_service_instance(self, si):
95 service = service_of_service_instance(si)
96 service_class = getattr(Slice().stub, service)
Sapan Bhatia6eeb6942017-11-27 00:27:56 -050097 service_obj = service_class.objects.first() # There's only one service object
Sapan Bhatia2810e032017-11-03 00:31:00 -040098 return service_obj
99
Sapan Bhatia11d0c9a2017-12-10 21:16:34 -0500100 def create_service_instance(self, si, node_label = None):
Sapan Bhatia2810e032017-11-03 00:31:00 -0400101 service = self.get_service_for_service_instance(si)
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500102 if not service:
103 raise Exception('No service object for %s' % service)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400104
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500105 si_class = getattr(Slice().stub, si)
106 s = si_class(owner=service, name='epc-%s-%d' %
107 (si.lower(), self.obj.id))
108 s.master_serviceinstance = self.obj
Sapan Bhatia11d0c9a2017-12-10 21:16:34 -0500109
110 if node_label:
111 s.node_label = '%s-%d'%(node_label, self.obj.id)
112
Sapan Bhatiadce1b8e2017-12-18 11:00:41 -0500113 s.no_sync = True
Sapan Bhatiaac9bd312017-12-19 11:17:48 -0500114 s.no_policy = True
Andy Bavier52b5e0f2017-11-27 15:58:21 -0700115 s.save()
Sapan Bhatia2810e032017-11-03 00:31:00 -0400116
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500117 self.configure_service_instance(s)
118 s.save()
Sapan Bhatia503ad902017-11-13 13:06:17 -0500119
120 self.in_memory_instances.append(s)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400121 return s
122
Andy Bavier6e37c832017-12-01 14:29:44 -0700123 def add_networks_to_service(self, service, networks):
124 for n in networks:
125 net = Network.objects.filter(name=n)[0]
126 one_and_only_slice_hopefully = service.slices.all()[0]
127 ns_object = NetworkSlice.objects.filter(
128 network=net.id, slice=one_and_only_slice_hopefully.id)
129 if not ns_object:
130 ns_object = NetworkSlice(
131 network=net, slice=one_and_only_slice_hopefully)
132 ns_object.save()
133
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500134 def add_networks_to_service_instance(self, instance, networks):
135 for n in networks:
136 net = Network.objects.filter(name=n)[0]
137 one_and_only_slice_hopefully = instance.owner.slices.all()[0]
138 ns_object = NetworkSlice.objects.filter(
139 network=net.id, slice=one_and_only_slice_hopefully.id)
140 if not ns_object:
141 ns_object = NetworkSlice(
142 network=net, slice=one_and_only_slice_hopefully)
143 ns_object.save()
144
Sapan Bhatia11d0c9a2017-12-10 21:16:34 -0500145 def create_service_instance_with_networks(self, si_name, networks, node_label = None):
Andy Bavier6e37c832017-12-01 14:29:44 -0700146 service = self.get_service_for_service_instance(si_name)
147 self.add_networks_to_service(service, networks)
148
Sapan Bhatia491a0802017-11-22 19:37:28 -0500149 instance = self.child_service_instance_from_name(si_name)
Sapan Bhatia11d0c9a2017-12-10 21:16:34 -0500150
Sapan Bhatia491a0802017-11-22 19:37:28 -0500151 if not instance:
Sapan Bhatia11d0c9a2017-12-10 21:16:34 -0500152 instance = self.create_service_instance(si_name, node_label = node_label)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400153
Sapan Bhatia491a0802017-11-22 19:37:28 -0500154 return instance
155
156 def create_link(self, src_instance, dst_instance):
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500157 src_service = self.get_service_for_service_instance(
158 src_instance.leaf_model_name)
159 dst_service = self.get_service_for_service_instance(
160 dst_instance.leaf_model_name)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400161
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500162 service_dependency = ServiceDependency.objects.filter(
163 provider_service_id=dst_service.id, subscriber_service_id=src_service.id)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400164 if not service_dependency:
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500165 service_dependency = ServiceDependency(
166 provider_service=dst_service, subscriber_service=src_service)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400167 service_dependency.save()
168
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500169 service_instance_link = ServiceInstanceLink.objects.filter(
170 provider_service_instance_id=dst_instance.id, subscriber_service_instance_id=src_instance.id)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400171 if not service_instance_link:
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500172 service_instance_link = ServiceInstanceLink(
173 provider_service_instance=dst_instance, subscriber_service_instance=src_instance)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400174 service_instance_link.save()
175
Sapan Bhatia8af7b882017-12-08 15:46:07 -0500176 def recursive_create_instances_and_links(self, blueprint, src, service_instances):
Sapan Bhatia491a0802017-11-22 19:37:28 -0500177 for node in blueprint:
178 k = node['name']
179 networks = node.get('networks', [])
Sapan Bhatia11d0c9a2017-12-10 21:16:34 -0500180 links = node.get('links', [])
Sapan Bhatia8af7b882017-12-08 15:46:07 -0500181
Sapan Bhatia11d0c9a2017-12-10 21:16:34 -0500182 try:
183 node_label = node['node_label']
184 except KeyError:
185 try:
186 node_label = next(l['node_label'] for l in links if l.get('node_label', None))
187 except StopIteration:
188 node_label = None
189
190 instance = self.create_service_instance_with_networks(k, networks, node_label = node_label)
Sapan Bhatia8af7b882017-12-08 15:46:07 -0500191 service_instances.append(instance)
Sapan Bhatia491a0802017-11-22 19:37:28 -0500192
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500193 if src:
194 self.add_networks_to_service_instance(src, networks)
195 self.create_link(src, instance)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400196
Sapan Bhatia8af7b882017-12-08 15:46:07 -0500197 service_instances = self.recursive_create_instances_and_links(links, instance, service_instances)
198
199 return service_instances
200
Sapan Bhatia491a0802017-11-22 19:37:28 -0500201 def create_epc_network(self, n):
202 network_name = n['name']
Andy Bavier52b5e0f2017-11-27 15:58:21 -0700203 site_name = self.obj.site.login_base
Sapan Bhatia491a0802017-11-22 19:37:28 -0500204
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500205 nets = Network.objects.filter(name=network_name)
206 if not nets:
207 template_name = n.get('template', 'public')
Andy Bavier8721f5e2018-01-02 14:36:23 -0700208 try:
209 template = NetworkTemplate.objects.filter(name=template_name)[0]
210 except:
211 raise Exception('Template %s for network %s not found' % (template_name, network_name))
212
213 slice_name = '%s_%s' % (site_name, n['owner'])
214 try:
215 slice = Slice.objects.filter(name=slice_name)[0]
216 except:
217 raise Exception('Owner slice %s for network %s not found' % (slice_name, network_name))
218
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500219
220 net = Network(name=network_name, subnet=n['subnet'], permit_all_slices=n.get(
Andy Bavier8721f5e2018-01-02 14:36:23 -0700221 'permit_all_slices', False), template=template, owner=slice)
Sapan Bhatia491a0802017-11-22 19:37:28 -0500222 net.save()
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500223 else:
224 net = nets[0]
225 if net.subnet != n['subnet']:
226 net.subnet = n['subnet']
227 net.save()
Sapan Bhatia491a0802017-11-22 19:37:28 -0500228
229 self.network_map[network_name] = net
230
231 def create_networks(self, networks):
232 for n in networks:
233 self.create_epc_network(n)
234
235 def create_networks_and_child_services(self, service_instance):
Sapan Bhatia2810e032017-11-03 00:31:00 -0400236 self.obj = service_instance
237 # Create service graph based on blueprint
238 chosen_blueprint = service_instance.blueprint
239 try:
Sapan Bhatia6eeb6942017-11-27 00:27:56 -0500240 blueprint = next(
241 b for b in blueprints if b['name'] == chosen_blueprint)
Sapan Bhatia2810e032017-11-03 00:31:00 -0400242 except StopIteration:
243 log.error('Chosen blueprint (%s) not found' % chosen_blueprint)
244
Sapan Bhatia491a0802017-11-22 19:37:28 -0500245 self.create_networks(blueprint['networks'])
Sapan Bhatia8af7b882017-12-08 15:46:07 -0500246
247 service_instances = self.recursive_create_instances_and_links(blueprint['graph'], None, [])
248
249 for si in service_instances:
Sapan Bhatiaac9bd312017-12-19 11:17:48 -0500250 si.no_policy = False
Sapan Bhatia8af7b882017-12-08 15:46:07 -0500251 si.no_sync = False
252 si.save()
Sapan Bhatia2810e032017-11-03 00:31:00 -0400253
Sapan Bhatia503ad902017-11-13 13:06:17 -0500254 def handle_create(self, service_instance):
255 self.handle_update(service_instance)
256
Sapan Bhatia2810e032017-11-03 00:31:00 -0400257 def handle_update(self, service_instance):
Sapan Bhatia491a0802017-11-22 19:37:28 -0500258 self.create_networks_and_child_services(service_instance)