blob: df207af4d463f3b539737ffaa3a4050f97a4e6e2 [file] [log] [blame]
Wei-Yu Chenba043592017-10-18 17:08:51 +08001# 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
15import os
16import sys
17from django.db.models import Q, F
Wei-Yu Chenba043592017-10-18 17:08:51 +080018from synchronizers.new_base.modelaccessor import *
19from synchronizers.new_base.SyncInstanceUsingAnsible import SyncInstanceUsingAnsible
20
21parentdir = os.path.join(os.path.dirname(__file__), "..")
22sys.path.insert(0, parentdir)
23
Sapan Bhatia15721932017-12-08 15:36:33 -050024
Sapan Bhatiafc9f6b32017-11-28 14:13:31 -050025class ServiceGraphException(Exception):
26 pass
Sapan Bhatia968f28f2017-11-22 12:00:01 -050027
Sapan Bhatia15721932017-12-08 15:36:33 -050028
Wei-Yu Chenba043592017-10-18 17:08:51 +080029class SyncVSPGWUTenant(SyncInstanceUsingAnsible):
Wei-Yu Chenba043592017-10-18 17:08:51 +080030 observes = VSPGWUTenant
Wei-Yu Chenba043592017-10-18 17:08:51 +080031 template_name = "vspgwutenant_playbook.yaml"
Wei-Yu Chenba043592017-10-18 17:08:51 +080032 service_key_name = "/opt/xos/configurations/mcord/mcord_private_key"
33
34 def __init__(self, *args, **kwargs):
35 super(SyncVSPGWUTenant, self).__init__(*args, **kwargs)
36
Woojoong Kimf002aec2017-10-25 14:43:37 -070037 def get_extra_attributes(self, o):
38
Sapan Bhatia968f28f2017-11-22 12:00:01 -050039 scenario = self.get_scenario(o)
Woojoong Kimf002aec2017-10-25 14:43:37 -070040
Woojoong Kim2f03e442018-01-18 20:02:22 -080041 if scenario == 'cord_4_1_scenario':
42 return self.get_values_for_CORD_4_1(o)
43 elif scenario == 'cord_5_0_scenario':
44 return self.get_values_for_CORD_5_0(o)
Woojoong Kimf002aec2017-10-25 14:43:37 -070045 else:
Sapan Bhatia968f28f2017-11-22 12:00:01 -050046 return self.get_extra_attributes_for_manual(o)
Woojoong Kimf002aec2017-10-25 14:43:37 -070047
48 # fields for manual case
Sapan Bhatia968f28f2017-11-22 12:00:01 -050049 def get_extra_attributes_for_manual(self, o):
Woojoong Kimf002aec2017-10-25 14:43:37 -070050 fields = {}
Woojoong Kim2f03e442018-01-18 20:02:22 -080051 fields['scenario'] = "manual"
52 fields['cord_version'] = "manual"
Woojoong Kimf002aec2017-10-25 14:43:37 -070053 # for interface.cfg file
54 fields['zmq_sub_ip'] = "manual"
55 fields['zmq_pub_ip'] = "manual"
56 fields['dp_comm_ip'] = "manual"
57 fields['cp_comm_ip'] = "manual"
58 fields['fpc_ip'] = "manual"
59 fields['cp_nb_server_ip'] = "manual"
60
61 # for dp_config.cfg file
62 fields['s1u_ip'] = "manual"
63 fields['sgi_ip'] = "manual"
64
Woojoong Kim04d12162017-11-27 10:59:08 -080065 # for static_arp.cfg file
66 fields['as_sgi_ip'] = "manual"
67 fields['as_sgi_mac'] = "manual"
68 fields['enb_s1u_ip'] = "manual"
69 fields['enb_s1u_mac'] = "manual"
70
Woojoong Kimf002aec2017-10-25 14:43:37 -070071 return fields
72
Woojoong Kim2f03e442018-01-18 20:02:22 -080073 def get_values_for_CORD_4_1(self, o):
Woojoong Kimf002aec2017-10-25 14:43:37 -070074 fields = {}
Woojoong Kim2f03e442018-01-18 20:02:22 -080075 fields['cord_version'] = "4.1"
76 fields['scenario'] = "cord_4_1_scenario"
Woojoong Kimf002aec2017-10-25 14:43:37 -070077 # for interface.cfg file
78 fields['zmq_sub_ip'] = "127.0.0.1"
79 fields['zmq_pub_ip'] = "127.0.0.1"
Sapan Bhatia968f28f2017-11-22 12:00:01 -050080 fields['dp_comm_ip'] = self.get_ip_address_from_peer_service_instance_instance(
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -050081 'spgw_network', o, o, 'dp_comm_ip')
Sapan Bhatia968f28f2017-11-22 12:00:01 -050082 fields['cp_comm_ip'] = self.get_ip_address_from_peer_service_instance(
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -050083 'spgw_network', "VSPGWCTenant", o, 'cp_comm_ip')
Woojoong Kimf002aec2017-10-25 14:43:37 -070084 fields['fpc_ip'] = "127.0.0.1"
85 fields['cp_nb_server_ip'] = "127.0.0.1"
86
87 # for cp_config.cfg file
Sapan Bhatia968f28f2017-11-22 12:00:01 -050088 fields['s1u_ip'] = self.get_ip_address_from_peer_service_instance_instance(
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -050089 's1u_network', o, o, 's1u_ip')
Sapan Bhatia968f28f2017-11-22 12:00:01 -050090 fields['sgi_ip'] = self.get_ip_address_from_peer_service_instance_instance(
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -050091 'sgi_network', o, o, 'sgi_ip')
Woojoong Kimf002aec2017-10-25 14:43:37 -070092
Woojoong Kim04d12162017-11-27 10:59:08 -080093 # for static_arp.cfg file
Sapan Bhatia15721932017-12-08 15:36:33 -050094 fields['as_sgi_ip'] = self.get_ip_address_from_peer_service_instance(
95 'sgi_network', "VENBServiceInstance", o, 'as_sgi_ip')
96 fields['as_sgi_mac'] = self.get_mac_address_from_peer_service_instance(
97 'sgi_network', "VENBServiceInstance", o, 'as_sgi_mac')
98 fields['enb_s1u_ip'] = self.get_ip_address_from_peer_service_instance(
99 's1u_network', "VENBServiceInstance", o, 'enb_s1u_ip')
100 fields['enb_s1u_mac'] = self.get_mac_address_from_peer_service_instance(
101 's1u_network', "VENBServiceInstance", o, 'enb_s1u_mac')
Woojoong Kim04d12162017-11-27 10:59:08 -0800102
Woojoong Kimf002aec2017-10-25 14:43:37 -0700103 return fields
104
Woojoong Kim2f03e442018-01-18 20:02:22 -0800105 def get_values_for_CORD_5_0(self, o):
Woojoong Kim4a15d412017-12-18 17:17:40 -0800106 fields = {}
Woojoong Kim2f03e442018-01-18 20:02:22 -0800107 fields['cord_version'] = "5.0"
108 fields['scenario'] = "cord_5_0_scenario"
Woojoong Kim4a15d412017-12-18 17:17:40 -0800109
Woojoong Kim4a15d412017-12-18 17:17:40 -0800110 # for interface.cfg file
111 fields['zmq_sub_ip'] = "127.0.0.1"
112 fields['zmq_pub_ip'] = "127.0.0.1"
113 fields['dp_comm_ip'] = self.get_ip_address_from_peer_service_instance_instance(
114 'spgw_network', o, o, 'dp_comm_ip')
115 fields['cp_comm_ip'] = self.get_ip_address_from_peer_service_instance(
116 'spgw_network', "VSPGWCTenant", o, 'cp_comm_ip')
117 fields['fpc_ip'] = "127.0.0.1"
118 fields['cp_nb_server_ip'] = "127.0.0.1"
119
Woojoong Kim2f03e442018-01-18 20:02:22 -0800120 # for cp_config.cfg file
Woojoong Kim4a15d412017-12-18 17:17:40 -0800121 fields['s1u_ip'] = self.get_ip_address_from_peer_service_instance_instance(
Woojoong Kim4f3ea432018-01-22 17:35:52 -0800122 'flat_network_s1u', o, o, 's1u_ip')
Woojoong Kim4a15d412017-12-18 17:17:40 -0800123 fields['sgi_ip'] = self.get_ip_address_from_peer_service_instance_instance(
124 'sgi_network', o, o, 'sgi_ip')
125
126 # for static_arp.cfg file
Woojoong Kim2f03e442018-01-18 20:02:22 -0800127 internetemulator_flag = self.has_instance("InternetEmulatorServiceInstance", o)
128 if (internetemulator_flag):
129 fields['as_sgi_ip'] = self.get_ip_address_from_peer_service_instance('sgi_network', "InternetEmulatorServiceInstance", o, 'as_sgi_ip')
130 fields['as_sgi_mac'] = self.get_mac_address_from_peer_service_instance('sgi_network', "InternetEmulatorServiceInstance", o, 'as_sgi_mac')
131 else:
132 fields['as_sgi_ip'] = o.appserver_ip_addr
133 fields['as_sgi_mac'] = o.appserver_mac_addr
134 fields['enb_s1u_ip'] = o.enodeb_ip_addr # write down eNB IP address manually (S1U)
135 fields['enb_s1u_mac'] = o.enodeb_mac_addr # write down eNB MAC address manually (S1U)
Woojoong Kim4a15d412017-12-18 17:17:40 -0800136
137 return fields
138
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500139 def has_instance(self, sitype, o):
Sapan Bhatiafc9f6b32017-11-28 14:13:31 -0500140 try:
141 i = self.get_peer_serviceinstance_of_type(sitype, o)
142 except ServiceGraphException:
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500143 self.log.info("Missing in ServiceInstance graph",
144 serviceinstance=sitype)
Woojoong Kimf002aec2017-10-25 14:43:37 -0700145 return False
146
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500147 return i.leaf_model.instance_id
Woojoong Kimf002aec2017-10-25 14:43:37 -0700148
149 # Which scenario does it use among Spirent or NG4T?
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500150 def get_scenario(self, o):
Woojoong Kimf002aec2017-10-25 14:43:37 -0700151 # try get vENB instance: one of both Spirent and NG4T
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500152 venb_flag = self.has_instance("VENBServiceInstance", o)
153 vmme_flag = self.has_instance("VMMETenant", o)
154 sdncontroller_flag = self.has_instance(
155 "SDNControllerServiceInstance", o)
Sapan Bhatia15721932017-12-08 15:36:33 -0500156 vspgwc_flag = self.has_instance("VSPGWCTenant", o)
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500157 internetemulator_flag = self.has_instance(
Woojoong Kim2f03e442018-01-18 20:02:22 -0800158 "InternetEmulatorServiceInstance", o)
159 vhss_flag = self.has_instance("VHSSTenant", o)
160 hssdb_flag = self.has_instance("HSSDBServiceInstance", o)
Woojoong Kimf002aec2017-10-25 14:43:37 -0700161
Woojoong Kim2f03e442018-01-18 20:02:22 -0800162 if (o.blueprint == "build") or (o.blueprint == "MCORD 4.1"):
163 if not venb_flag:
164 self.defer_sync(o, "Waiting for eNB image to become available")
165 if not vspgwc_flag:
166 self.defer_sync(o, "Waiting for SPGWC image to become available")
167 return 'cord_4_1_scenario'
Woojoong Kimf002aec2017-10-25 14:43:37 -0700168
Woojoong Kim2f03e442018-01-18 20:02:22 -0800169 if (o.blueprint == "mcord_5") or (o.blueprint == "MCORD 5"):
170 if not hssdb_flag:
171 self.defer_sync(o, "Waiting for HSS_DB image to become available")
172 if not vhss_flag:
173 self.defer_sync(o, "Waiting for vHSS image to become available")
174 if not vmme_flag:
175 self.defer_sync(o, "Waiting for vMME image to become available")
176 if not vspgwc_flag:
177 self.defer_sync(o, "Waiting for vSPGWC image to become available")
178 return 'cord_5_0_scenario'
Woojoong Kim4a15d412017-12-18 17:17:40 -0800179
Woojoong Kimf002aec2017-10-25 14:43:37 -0700180 return 'manual'
181
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500182 def get_peer_serviceinstance_of_type(self, sitype, o):
Woojoong Kim4f3ea432018-01-22 17:35:52 -0800183
184 global_list = self.get_all_instances_in_graph(o)
Woojoong Kimf002aec2017-10-25 14:43:37 -0700185
186 try:
Woojoong Kim4f3ea432018-01-22 17:35:52 -0800187 peer_service = next(p for p in global_list if p.leaf_model_name == sitype)
188
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500189 except StopIteration:
Woojoong Kim4f3ea432018-01-22 17:35:52 -0800190 self.log.error(
191 'Could not find service type in service graph', service_type=sitype, object=o)
192 raise ServiceGraphException(
193 "Synchronization failed due to incomplete service graph")
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500194
195 return peer_service
196
Woojoong Kim4f3ea432018-01-22 17:35:52 -0800197 def has_instance_in_list(self, list, o):
198 for instance in list:
199 if instance.leaf_model_name == o.leaf_model_name:
200 return True
201
202 return False
203
204 def get_all_instances_in_graph(self, o):
205
206 to_search_list = self.get_one_hop_instances_in_graph(o)
207 result_list = []
208
209 while len(to_search_list) > 0:
210 tmp_obj = to_search_list[0]
211 to_search_list.remove(tmp_obj)
212 tmp_list = self.get_one_hop_instances_in_graph(tmp_obj)
213
214 for index_obj in tmp_list:
215 if (not self.has_instance_in_list(to_search_list, index_obj)) and (not self.has_instance_in_list(result_list, index_obj)):
216 to_search_list.append(index_obj)
217
218 result_list.append(tmp_obj)
219 return result_list
220
221 def get_one_hop_instances_in_graph(self, o):
222 instance_list = []
223
224 # 1 hop forward and backward
225 prov_links = ServiceInstanceLink.objects.filter(subscriber_service_instance_id=o.id)
226 subs_links = ServiceInstanceLink.objects.filter(provider_service_instance_id=o.id)
227
228 # add instances located in 1 hop into instance_list
229 for tmp_link1 in prov_links:
230 if not self.has_instance_in_list(instance_list, tmp_link1.provider_service_instance):
231 instance_list.append(tmp_link1.provider_service_instance)
232
233 for tmp_link1 in subs_links:
234 if not self.has_instance_in_list(instance_list, tmp_link1.subscriber_service_instance):
235 instance_list.append(tmp_link1.subscriber_service_instance)
236
237 return instance_list
238
239
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500240 # Maybe merge the two pairs of functions into one, with an address type "mac" or "ip" - SB
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500241 def get_ip_address_from_peer_service_instance(self, network_name, sitype, o, parameter=None):
242 peer_si = self.get_peer_serviceinstance_of_type(sitype, o)
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500243 return self.get_ip_address_from_peer_service_instance_instance(network_name, peer_si, o, parameter)
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500244
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500245 def get_mac_address_from_peer_service_instance(self, network_name, sitype, o, parameter=None):
246 peer_si = self.get_peer_serviceinstance_of_type(sitype, o)
247 return self.get_mac_address_from_peer_service_instance_instance(network_name, peer_si, o, parameter)
248
249 def get_ip_address_from_peer_service_instance_instance(self, network_name, peer_si, o, parameter=None):
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500250 try:
Woojoong Kimf002aec2017-10-25 14:43:37 -0700251 net_id = self.get_network_id(network_name)
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500252 ins_id = peer_si.leaf_model.instance_id
Sapan Bhatia968f28f2017-11-22 12:00:01 -0500253 ip_address = Port.objects.get(
254 network_id=net_id, instance_id=ins_id).ip
Woojoong Kimf002aec2017-10-25 14:43:37 -0700255 except Exception:
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500256 self.log.error("Failed to fetch parameter",
257 parameter=parameter,
258 network_name=network_name)
259 self.defer_sync(o, "Waiting for parameters to become available")
Woojoong Kimf002aec2017-10-25 14:43:37 -0700260
261 return ip_address
262
Sapan Bhatia15721932017-12-08 15:36:33 -0500263 def get_mac_address_from_peer_service_instance_instance(self, network_name, peer_si, o, parameter):
Woojoong Kim04d12162017-11-27 10:59:08 -0800264 try:
265 net_id = self.get_network_id(network_name)
Sapan Bhatia15721932017-12-08 15:36:33 -0500266 ins_id = peer_si.leaf_model.instance_id
267 mac_address = Port.objects.get(
268 network_id=net_id, instance_id=ins_id).mac
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500269
Woojoong Kim04d12162017-11-27 10:59:08 -0800270 except Exception:
Sapan Bhatia15721932017-12-08 15:36:33 -0500271 self.log.error("Failed to fetch parameter to get MAC",
272 parameter=parameter, network_name=network_name)
Sapan Bhatia2f27dbe2017-11-27 21:17:32 -0500273 self.defer_sync(o, "Waiting for parameters to become available")
Woojoong Kim04d12162017-11-27 10:59:08 -0800274
275 return mac_address
276
Woojoong Kimf002aec2017-10-25 14:43:37 -0700277 # To get each network id
278 def get_network_id(self, network_name):
279 return Network.objects.get(name=network_name).id