blob: ce2327152e2cab85422d0332877632b4c72cbd3a [file] [log] [blame]
A R Karthick41adfce2016-06-10 09:51:25 -07001#
Chetan Gaonkercfcce782016-05-10 10:10:42 -07002# Copyright 2016-present Ciena Corporation
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
A R Karthick41adfce2016-06-10 09:51:25 -07007#
Chetan Gaonkercfcce782016-05-10 10:10:42 -07008# http://www.apache.org/licenses/LICENSE-2.0
A R Karthick41adfce2016-06-10 09:51:25 -07009#
Chetan Gaonkercfcce782016-05-10 10:10:42 -070010# 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.
15#
Chetan Gaonker3533faa2016-04-25 17:50:14 -070016import os,time
17import io
18import json
A R Karthickd44cea12016-07-20 12:16:41 -070019import yaml
Chetan Gaonker3533faa2016-04-25 17:50:14 -070020from pyroute2 import IPRoute
21from itertools import chain
22from nsenter import Namespace
23from docker import Client
24from shutil import copy
A.R Karthick95d044e2016-06-10 18:44:36 -070025from OnosCtrl import OnosCtrl
Chetan Gaonker3533faa2016-04-25 17:50:14 -070026
27class docker_netns(object):
28
29 dckr = Client()
30 def __init__(self, name):
31 pid = int(self.dckr.inspect_container(name)['State']['Pid'])
32 if pid == 0:
33 raise Exception('no container named {0}'.format(name))
34 self.pid = pid
35
36 def __enter__(self):
37 pid = self.pid
38 if not os.path.exists('/var/run/netns'):
39 os.mkdir('/var/run/netns')
40 os.symlink('/proc/{0}/ns/net'.format(pid), '/var/run/netns/{0}'.format(pid))
41 return str(pid)
42
43 def __exit__(self, type, value, traceback):
44 pid = self.pid
45 os.unlink('/var/run/netns/{0}'.format(pid))
46
47flatten = lambda l: chain.from_iterable(l)
48
49class Container(object):
50 dckr = Client()
A R Karthick07608ef2016-08-23 16:51:19 -070051 IMAGE_PREFIX = '' ##for saving global prefix for all test classes
52
53 def __init__(self, name, image, prefix='', tag = 'candidate', command = 'bash', quagga_config = None):
Chetan Gaonker3533faa2016-04-25 17:50:14 -070054 self.name = name
A R Karthick07608ef2016-08-23 16:51:19 -070055 self.prefix = prefix
56 if prefix:
57 self.prefix += '/'
58 image = '{}{}'.format(self.prefix, image)
Chetan Gaonker3533faa2016-04-25 17:50:14 -070059 self.image = image
60 self.tag = tag
A R Karthickd44cea12016-07-20 12:16:41 -070061 if tag:
62 self.image_name = image + ':' + tag
63 else:
64 self.image_name = image
Chetan Gaonker3533faa2016-04-25 17:50:14 -070065 self.id = None
66 self.command = command
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -070067 self.quagga_config = quagga_config
Chetan Gaonker3533faa2016-04-25 17:50:14 -070068
69 @classmethod
70 def build_image(cls, dockerfile, tag, force=True, nocache=False):
71 f = io.BytesIO(dockerfile.encode('utf-8'))
72 if force or not cls.image_exists(tag):
73 print('Build {0}...'.format(tag))
74 for line in cls.dckr.build(fileobj=f, rm=True, tag=tag, decode=True, nocache=nocache):
75 if 'stream' in line:
76 print(line['stream'].strip())
77
78 @classmethod
79 def image_exists(cls, name):
80 return name in [ctn['RepoTags'][0] for ctn in cls.dckr.images()]
81
82 @classmethod
83 def create_host_config(cls, port_list = None, host_guest_map = None, privileged = False):
84 port_bindings = None
85 binds = None
86 if port_list:
87 port_bindings = {}
88 for p in port_list:
89 port_bindings[str(p)] = str(p)
90
91 if host_guest_map:
92 binds = []
93 for h, g in host_guest_map:
94 binds.append('{0}:{1}'.format(h, g))
95
96 return cls.dckr.create_host_config(binds = binds, port_bindings = port_bindings, privileged = privileged)
97
98 @classmethod
99 def cleanup(cls, image):
A R Karthick09b1f4e2016-05-12 14:31:50 -0700100 cnt_list = filter(lambda c: c['Image'] == image, cls.dckr.containers(all=True))
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700101 for cnt in cnt_list:
102 print('Cleaning container %s' %cnt['Id'])
A.R Karthick95d044e2016-06-10 18:44:36 -0700103 if cnt.has_key('State') and cnt['State'] == 'running':
A R Karthick09b1f4e2016-05-12 14:31:50 -0700104 cls.dckr.kill(cnt['Id'])
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700105 cls.dckr.remove_container(cnt['Id'], force=True)
106
107 @classmethod
108 def remove_container(cls, name, force=True):
109 try:
110 cls.dckr.remove_container(name, force = force)
111 except: pass
112
113 def exists(self):
114 return '/{0}'.format(self.name) in list(flatten(n['Names'] for n in self.dckr.containers()))
115
116 def img_exists(self):
A R Karthick6d98a592016-08-24 15:16:46 -0700117 return self.image_name in [ctn['RepoTags'][0] if ctn['RepoTags'] else '' for ctn in self.dckr.images()]
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700118
119 def ip(self):
A R Karthick2b93d6a2016-09-06 15:19:09 -0700120 cnt_list = filter(lambda c: c['Names'][0] == '/{}'.format(self.name), self.dckr.containers())
121 #if not cnt_list:
122 # cnt_list = filter(lambda c: c['Image'] == self.image_name, self.dckr.containers())
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700123 cnt_settings = cnt_list.pop()
124 return cnt_settings['NetworkSettings']['Networks']['bridge']['IPAddress']
125
A R Karthick2b93d6a2016-09-06 15:19:09 -0700126 @classmethod
127 def ips(cls, image_name):
128 cnt_list = filter(lambda c: c['Image'] == image_name, cls.dckr.containers())
129 ips = [ cnt['NetworkSettings']['Networks']['bridge']['IPAddress'] for cnt in cnt_list ]
130 return ips
131
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700132 def kill(self, remove = True):
133 self.dckr.kill(self.name)
134 self.dckr.remove_container(self.name, force=True)
135
A R Karthick41adfce2016-06-10 09:51:25 -0700136 def start(self, rm = True, ports = None, volumes = None, host_config = None,
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700137 environment = None, tty = False, stdin_open = True):
138
139 if rm and self.exists():
140 print('Removing container:', self.name)
141 self.dckr.remove_container(self.name, force=True)
142
A R Karthick41adfce2016-06-10 09:51:25 -0700143 ctn = self.dckr.create_container(image=self.image_name, ports = ports, command=self.command,
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700144 detach=True, name=self.name,
A R Karthick41adfce2016-06-10 09:51:25 -0700145 environment = environment,
146 volumes = volumes,
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700147 host_config = host_config, stdin_open=stdin_open, tty = tty)
148 self.dckr.start(container=self.name)
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700149 if self.quagga_config:
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700150 self.connect_to_br()
151 self.id = ctn['Id']
152 return ctn
153
154 def connect_to_br(self):
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700155 index = 0
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700156 with docker_netns(self.name) as pid:
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700157 for quagga_config in self.quagga_config:
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700158 ip = IPRoute()
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700159 br = ip.link_lookup(ifname=quagga_config['bridge'])
160 if len(br) == 0:
Chetan Gaonker5a0fda32016-05-10 14:09:07 -0700161 ip.link_create(ifname=quagga_config['bridge'], kind='bridge')
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700162 br = ip.link_lookup(ifname=quagga_config['bridge'])
163 br = br[0]
164 ip.link('set', index=br, state='up')
165 ifname = '{0}-{1}'.format(self.name, index)
166 ifs = ip.link_lookup(ifname=ifname)
167 if len(ifs) > 0:
168 ip.link_remove(ifs[0])
169 peer_ifname = '{0}-{1}'.format(pid, index)
Chetan Gaonker5a0fda32016-05-10 14:09:07 -0700170 ip.link_create(ifname=ifname, kind='veth', peer=peer_ifname)
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700171 host = ip.link_lookup(ifname=ifname)[0]
172 ip.link('set', index=host, master=br)
173 ip.link('set', index=host, state='up')
174 guest = ip.link_lookup(ifname=peer_ifname)[0]
175 ip.link('set', index=guest, net_ns_fd=pid)
176 with Namespace(pid, 'net'):
177 ip = IPRoute()
178 ip.link('set', index=guest, ifname='eth{}'.format(index+1))
179 ip.addr('add', index=guest, address=quagga_config['ip'], mask=quagga_config['mask'])
180 ip.link('set', index=guest, state='up')
181 index += 1
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700182
183 def execute(self, cmd, tty = True, stream = False, shell = False):
184 res = 0
185 if type(cmd) == str:
186 cmds = (cmd,)
187 else:
188 cmds = cmd
189 if shell:
190 for c in cmds:
191 res += os.system('docker exec {0} {1}'.format(self.name, c))
192 return res
193 for c in cmds:
194 i = self.dckr.exec_create(container=self.name, cmd=c, tty = tty, privileged = True)
195 self.dckr.exec_start(i['Id'], stream = stream, detach=True)
196 result = self.dckr.exec_inspect(i['Id'])
197 res += 0 if result['ExitCode'] == None else result['ExitCode']
198 return res
199
ChetanGaonker6138fcd2016-08-18 17:56:39 -0700200 def restart(self, timeout =10):
201 return self.dckr.restart(self.name, timeout)
202
Chetan Gaonker462d9fa2016-05-03 16:39:10 -0700203def get_mem():
204 with open('/proc/meminfo', 'r') as fd:
205 meminfo = fd.readlines()
206 mem = 0
207 for m in meminfo:
208 if m.startswith('MemTotal:') or m.startswith('SwapTotal:'):
209 mem += int(m.split(':')[1].strip().split()[0])
210
Chetan Gaonkerc0421e82016-05-04 17:23:08 -0700211 mem = max(mem/1024/1024/2, 1)
Chetan Gaonker6d0a7b02016-05-03 16:57:28 -0700212 mem = min(mem, 16)
Chetan Gaonker462d9fa2016-05-03 16:39:10 -0700213 return str(mem) + 'G'
214
A R Karthickd44cea12016-07-20 12:16:41 -0700215class OnosCord(Container):
216 """Use this when running the cord tester agent on the onos compute node"""
217 onos_cord_dir = os.path.join(os.getenv('HOME'), 'cord-tester-cord')
218 onos_config_dir_guest = '/root/onos/config'
219 onos_config_dir = os.path.join(onos_cord_dir, 'config')
220 docker_yaml = os.path.join(onos_cord_dir, 'docker-compose.yml')
221
A R Karthickbd9b8a32016-07-21 09:56:45 -0700222 def __init__(self, onos_ip, conf, boot_delay = 60):
223 self.onos_ip = onos_ip
A R Karthickd44cea12016-07-20 12:16:41 -0700224 self.cord_conf_dir = conf
A R Karthickbd9b8a32016-07-21 09:56:45 -0700225 self.boot_delay = boot_delay
A R Karthickd44cea12016-07-20 12:16:41 -0700226 if os.access(self.cord_conf_dir, os.F_OK) and not os.access(self.onos_cord_dir, os.F_OK):
227 os.mkdir(self.onos_cord_dir)
228 os.mkdir(self.onos_config_dir)
229 ##copy the config file from cord-tester-config
230 cmd = 'cp {}/* {}'.format(self.cord_conf_dir, self.onos_cord_dir)
231 os.system(cmd)
232
233 ##update the docker yaml with the config volume
234 with open(self.docker_yaml, 'r') as f:
235 yaml_config = yaml.load(f)
236 image = yaml_config['services'].keys()[0]
237 name = 'cordtestercord_{}_1'.format(image)
238 volumes = yaml_config['services'][image]['volumes']
239 config_volumes = filter(lambda e: e.find(self.onos_config_dir_guest) >= 0, volumes)
240 if not config_volumes:
241 config_volume = '{}:{}'.format(self.onos_config_dir, self.onos_config_dir_guest)
242 volumes.append(config_volume)
243 docker_yaml_changed = '{}-changed'.format(self.docker_yaml)
244 with open(docker_yaml_changed, 'w') as wf:
245 yaml.dump(yaml_config, wf)
246
247 os.rename(docker_yaml_changed, self.docker_yaml)
248 self.volumes = volumes
249
250 super(OnosCord, self).__init__(name, image, tag = '')
251 cord_conf_dir_basename = os.path.basename(self.cord_conf_dir.replace('-', ''))
252 self.xos_onos_name = '{}_{}_1'.format(cord_conf_dir_basename, image)
253 ##Create an container instance of xos onos
254 self.xos_onos = Container(self.xos_onos_name, image, tag = '')
255
256 def start(self, restart = False, network_cfg = None):
257 if restart is True:
258 if self.exists():
259 ##Kill the existing instance
260 print('Killing container %s' %self.name)
261 self.kill()
262 if self.xos_onos.exists():
263 print('Killing container %s' %self.xos_onos.name)
264 self.xos_onos.kill()
265
266 if network_cfg is not None:
267 json_data = json.dumps(network_cfg, indent=4)
268 with open('{}/network-cfg.json'.format(self.onos_config_dir), 'w') as f:
269 f.write(json_data)
270
271 #start the container using docker-compose
272 cmd = 'cd {} && docker-compose up -d'.format(self.onos_cord_dir)
273 os.system(cmd)
A R Karthickbd9b8a32016-07-21 09:56:45 -0700274 #Delay to make sure ONOS fully boots
275 time.sleep(self.boot_delay)
276 Onos.install_cord_apps(onos_ip = self.onos_ip)
A R Karthickd44cea12016-07-20 12:16:41 -0700277
278 def build_image(self):
279 build_cmd = 'cd {} && docker-compose build'.format(self.onos_cord_dir)
280 os.system(build_cmd)
281
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700282class Onos(Container):
283
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700284 quagga_config = ( { 'bridge' : 'quagga-br', 'ip': '10.10.0.4', 'mask' : 16 }, )
Chetan Gaonker462d9fa2016-05-03 16:39:10 -0700285 SYSTEM_MEMORY = (get_mem(),) * 2
286 JAVA_OPTS = '-Xms{} -Xmx{} -XX:+UseConcMarkSweepGC -XX:+CMSIncrementalMode'.format(*SYSTEM_MEMORY)#-XX:+PrintGCDetails -XX:+PrintGCTimeStamps'
A.R Karthick95d044e2016-06-10 18:44:36 -0700287 env = { 'ONOS_APPS' : 'drivers,openflow,proxyarp,vrouter', 'JAVA_OPTS' : JAVA_OPTS }
288 onos_cord_apps = ( ('cord-config', '1.0-SNAPSHOT'),
289 ('aaa', '1.0-SNAPSHOT'),
290 ('igmp', '1.0-SNAPSHOT'),
A R Karthickedab01c2016-09-08 14:05:44 -0700291 #('vtn', '1.0-SNAPSHOT'),
A.R Karthick95d044e2016-06-10 18:44:36 -0700292 )
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700293 ports = [ 8181, 8101, 9876, 6653, 6633, 2000, 2620 ]
A R Karthickf2f4ca62016-08-17 10:34:08 -0700294 setup_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), '..', 'setup')
295 host_config_dir = os.path.join(setup_dir, 'onos-config')
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700296 guest_config_dir = '/root/onos/config'
A R Karthickf2f4ca62016-08-17 10:34:08 -0700297 onos_gen_partitions = os.path.join(setup_dir, 'onos-gen-partitions')
A R Karthick2b93d6a2016-09-06 15:19:09 -0700298 onos_form_cluster = os.path.join(setup_dir, 'onos-form-cluster')
A.R Karthick95d044e2016-06-10 18:44:36 -0700299 cord_apps_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), '..', 'apps')
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700300 host_guest_map = ( (host_config_dir, guest_config_dir), )
A R Karthick2b93d6a2016-09-06 15:19:09 -0700301 cluster_cfg = os.path.join(host_config_dir, 'cluster.json')
302 cluster_mode = False
303 cluster_instances = []
Chetan Gaonker503032a2016-05-12 12:06:29 -0700304 NAME = 'cord-onos'
A R Karthickf2f4ca62016-08-17 10:34:08 -0700305 ##the ip of ONOS in default cluster.json in setup/onos-config
306 CLUSTER_CFG_IP = '172.17.0.2'
A R Karthick07608ef2016-08-23 16:51:19 -0700307 IMAGE = 'onosproject/onos'
308 TAG = 'latest'
309 PREFIX = ''
A R Karthickf2f4ca62016-08-17 10:34:08 -0700310
311 @classmethod
A R Karthick2b93d6a2016-09-06 15:19:09 -0700312 def generate_cluster_cfg(cls, ip):
313 if type(ip) in [ list, tuple ]:
314 ips = ' '.join(ip)
315 else:
316 ips = ip
A R Karthickf2f4ca62016-08-17 10:34:08 -0700317 try:
A R Karthick2b93d6a2016-09-06 15:19:09 -0700318 cmd = '{} {} {}'.format(cls.onos_gen_partitions, cls.cluster_cfg, ips)
319 os.system(cmd)
320 except: pass
321
322 @classmethod
323 def form_cluster(cls, ips):
324 nodes = ' '.join(ips)
325 try:
326 cmd = '{} {}'.format(cls.onos_form_cluster, nodes)
A R Karthickf2f4ca62016-08-17 10:34:08 -0700327 os.system(cmd)
328 except: pass
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700329
A R Karthick07608ef2016-08-23 16:51:19 -0700330 def __init__(self, name = NAME, image = 'onosproject/onos', prefix = '', tag = 'latest',
A R Karthick2b93d6a2016-09-06 15:19:09 -0700331 boot_delay = 60, restart = False, network_cfg = None, cluster = False):
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700332 if restart is True:
333 ##Find the right image to restart
334 running_image = filter(lambda c: c['Names'][0] == '/{}'.format(name), self.dckr.containers())
335 if running_image:
336 image_name = running_image[0]['Image']
337 try:
338 image = image_name.split(':')[0]
339 tag = image_name.split(':')[1]
340 except: pass
341
A R Karthick07608ef2016-08-23 16:51:19 -0700342 super(Onos, self).__init__(name, image, prefix = prefix, tag = tag, quagga_config = self.quagga_config)
A R Karthick2b93d6a2016-09-06 15:19:09 -0700343 self.boot_delay = boot_delay
344 if cluster is True:
345 self.ports = []
346 if os.access(self.cluster_cfg, os.F_OK):
347 try:
348 os.unlink(self.cluster_cfg)
349 except: pass
350
351 self.host_config = self.create_host_config(port_list = self.ports,
352 host_guest_map = self.host_guest_map)
353 self.volumes = []
354 for _,g in self.host_guest_map:
355 self.volumes.append(g)
356
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700357 if restart is True and self.exists():
358 self.kill()
A R Karthick2b93d6a2016-09-06 15:19:09 -0700359
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700360 if not self.exists():
361 self.remove_container(name, force=True)
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700362 if network_cfg is not None:
A R Karthick81acbff2016-06-17 14:45:16 -0700363 json_data = json.dumps(network_cfg, indent=4)
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700364 with open('{}/network-cfg.json'.format(self.host_config_dir), 'w') as f:
365 f.write(json_data)
366 print('Starting ONOS container %s' %self.name)
A R Karthick41adfce2016-06-10 09:51:25 -0700367 self.start(ports = self.ports, environment = self.env,
A R Karthick2b93d6a2016-09-06 15:19:09 -0700368 host_config = self.host_config, volumes = self.volumes, tty = True)
A R Karthickf2f4ca62016-08-17 10:34:08 -0700369 if not restart:
370 ##wait a bit before fetching IP to regenerate cluster cfg
371 time.sleep(5)
372 ip = self.ip()
373 ##Just a quick hack/check to ensure we don't regenerate in the common case.
374 ##As ONOS is usually the first test container that is started
A R Karthick2b93d6a2016-09-06 15:19:09 -0700375 if cluster is False:
376 if ip != self.CLUSTER_CFG_IP or not os.access(self.cluster_cfg, os.F_OK):
377 print('Regenerating ONOS cluster cfg for ip %s' %ip)
378 self.generate_cluster_cfg(ip)
379 self.kill()
380 self.remove_container(self.name, force=True)
381 print('Restarting ONOS container %s' %self.name)
382 self.start(ports = self.ports, environment = self.env,
383 host_config = self.host_config, volumes = self.volumes, tty = True)
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700384 print('Waiting %d seconds for ONOS to boot' %(boot_delay))
385 time.sleep(boot_delay)
A R Karthick2b93d6a2016-09-06 15:19:09 -0700386 self.ipaddr = self.ip()
387 if cluster is False:
388 self.install_cord_apps(self.ipaddr)
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700389
A R Karthick2b93d6a2016-09-06 15:19:09 -0700390 @classmethod
391 def setup_cluster_deprecated(cls, onos_instances, image_name = None):
392 if not onos_instances or len(onos_instances) < 2:
393 return
394 ips = []
395 if image_name is not None:
396 ips = Container.ips(image_name)
397 else:
398 for onos in onos_instances:
399 ips.append(onos.ipaddr)
400 Onos.cluster_instances = onos_instances
401 Onos.cluster_mode = True
402 ##regenerate the cluster json with the 3 instance ips before restarting them back
403 print('Generating cluster cfg for ONOS instances with ips %s' %ips)
404 Onos.generate_cluster_cfg(ips)
405 for onos in onos_instances:
406 onos.kill()
407 onos.remove_container(onos.name, force=True)
408 print('Restarting ONOS container %s for forming cluster' %onos.name)
409 onos.start(ports = onos.ports, environment = onos.env,
410 host_config = onos.host_config, volumes = onos.volumes, tty = True)
411 print('Waiting %d seconds for ONOS %s to boot' %(onos.boot_delay, onos.name))
412 time.sleep(onos.boot_delay)
413 onos.ipaddr = onos.ip()
414 onos.install_cord_apps(onos.ipaddr)
415
416 @classmethod
417 def setup_cluster(cls, onos_instances, image_name = None):
418 if not onos_instances or len(onos_instances) < 2:
419 return
420 ips = []
421 if image_name is not None:
422 ips = Container.ips(image_name)
423 else:
424 for onos in onos_instances:
425 ips.append(onos.ipaddr)
426 Onos.cluster_instances = onos_instances
427 Onos.cluster_mode = True
428 ##regenerate the cluster json with the 3 instance ips before restarting them back
429 print('Forming cluster for ONOS instances with ips %s' %ips)
430 Onos.form_cluster(ips)
431 ##wait for the cluster to be formed
432 print('Waiting for the cluster to be formed')
433 time.sleep(60)
434 for onos in onos_instances:
435 onos.install_cord_apps(onos.ipaddr)
436
437 @classmethod
438 def restart_cluster(cls, network_cfg = None):
439 if cls.cluster_mode is False:
440 return
441 if not cls.cluster_instances:
442 return
443
444 if network_cfg is not None:
445 json_data = json.dumps(network_cfg, indent=4)
446 with open('{}/network-cfg.json'.format(cls.host_config_dir), 'w') as f:
447 f.write(json_data)
448
449 for onos in cls.cluster_instances:
450 if onos.exists():
451 onos.kill()
452 onos.remove_container(onos.name, force=True)
453 print('Restarting ONOS container %s' %onos.name)
454 onos.start(ports = onos.ports, environment = onos.env,
455 host_config = onos.host_config, volumes = onos.volumes, tty = True)
456 print('Waiting %d seconds for ONOS %s to boot' %(onos.boot_delay, onos.name))
457 time.sleep(onos.boot_delay)
458 onos.ipaddr = onos.ip()
459
460 ##form the cluster
461 cls.setup_cluster(cls.cluster_instances)
462
463 @classmethod
464 def cluster_ips(cls):
465 if cls.cluster_mode is False:
466 return []
467 if not cls.cluster_instances:
468 return []
469 ips = [ onos.ipaddr for onos in cls.cluster_instances ]
470 return ips
471
472 @classmethod
473 def cleanup_cluster(cls):
474 if cls.cluster_mode is False:
475 return
476 if not cls.cluster_instances:
477 return
478 for onos in cls.cluster_instances:
479 if onos.exists():
480 onos.kill()
481 onos.remove_container(onos.name, force=True)
A R Karthickd44cea12016-07-20 12:16:41 -0700482
A.R Karthick95d044e2016-06-10 18:44:36 -0700483 @classmethod
A R Karthickeaf1c4e2016-07-19 12:22:35 -0700484 def install_cord_apps(cls, onos_ip = None):
A.R Karthick95d044e2016-06-10 18:44:36 -0700485 for app, version in cls.onos_cord_apps:
486 app_file = '{}/{}-{}.oar'.format(cls.cord_apps_dir, app, version)
A R Karthickeaf1c4e2016-07-19 12:22:35 -0700487 ok, code = OnosCtrl.install_app(app_file, onos_ip = onos_ip)
A.R Karthick95d044e2016-06-10 18:44:36 -0700488 ##app already installed (conflicts)
489 if code in [ 409 ]:
490 ok = True
491 print('ONOS app %s, version %s %s' %(app, version, 'installed' if ok else 'failed to install'))
492 time.sleep(2)
493
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700494class Radius(Container):
495 ports = [ 1812, 1813 ]
A R Karthick41adfce2016-06-10 09:51:25 -0700496 env = {'TIMEZONE':'America/Los_Angeles',
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700497 'DEBUG': 'true', 'cert_password':'whatever', 'primary_shared_secret':'radius_password'
498 }
Chetan Gaonker7f4bf742016-05-04 15:56:08 -0700499 host_db_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), '..', 'setup/radius-config/db')
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700500 guest_db_dir = os.path.join(os.path.sep, 'opt', 'db')
Chetan Gaonker7f4bf742016-05-04 15:56:08 -0700501 host_config_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), '..', 'setup/radius-config/freeradius')
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700502 guest_config_dir = os.path.join(os.path.sep, 'etc', 'freeradius')
Chetan Gaonker7f4bf742016-05-04 15:56:08 -0700503 start_command = os.path.join(guest_config_dir, 'start-radius.py')
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700504 host_guest_map = ( (host_db_dir, guest_db_dir),
505 (host_config_dir, guest_config_dir)
506 )
Chetan Gaonker503032a2016-05-12 12:06:29 -0700507 IMAGE = 'cord-test/radius'
508 NAME = 'cord-radius'
509
A R Karthick07608ef2016-08-23 16:51:19 -0700510 def __init__(self, name = NAME, image = IMAGE, prefix = '', tag = 'candidate',
Chetan Gaonker503032a2016-05-12 12:06:29 -0700511 boot_delay = 10, restart = False, update = False):
A R Karthick07608ef2016-08-23 16:51:19 -0700512 super(Radius, self).__init__(name, image, prefix = prefix, tag = tag, command = self.start_command)
Chetan Gaonker503032a2016-05-12 12:06:29 -0700513 if update is True or not self.img_exists():
A R Karthick07608ef2016-08-23 16:51:19 -0700514 self.build_image(self.image_name)
Chetan Gaonker7f4bf742016-05-04 15:56:08 -0700515 if restart is True and self.exists():
516 self.kill()
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700517 if not self.exists():
518 self.remove_container(name, force=True)
519 host_config = self.create_host_config(port_list = self.ports,
520 host_guest_map = self.host_guest_map)
521 volumes = []
522 for _,g in self.host_guest_map:
523 volumes.append(g)
A R Karthick41adfce2016-06-10 09:51:25 -0700524 self.start(ports = self.ports, environment = self.env,
525 volumes = volumes,
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700526 host_config = host_config, tty = True)
Chetan Gaonker7f4bf742016-05-04 15:56:08 -0700527 time.sleep(boot_delay)
528
529 @classmethod
530 def build_image(cls, image):
531 print('Building Radius image %s' %image)
532 dockerfile = '''
533FROM hbouvier/docker-radius
534MAINTAINER chetan@ciena.com
535LABEL RUN docker pull hbouvier/docker-radius
536LABEL RUN docker run -it --name cord-radius hbouvier/docker-radius
A R Karthickc762df42016-05-25 10:09:21 -0700537RUN apt-get update && \
538 apt-get -y install python python-pexpect strace
Chetan Gaonker7f4bf742016-05-04 15:56:08 -0700539WORKDIR /root
540CMD ["/etc/freeradius/start-radius.py"]
541'''
542 super(Radius, cls).build_image(dockerfile, image)
543 print('Done building image %s' %image)
Chetan Gaonker3533faa2016-04-25 17:50:14 -0700544
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700545class Quagga(Container):
A R Karthick41adfce2016-06-10 09:51:25 -0700546 quagga_config = ( { 'bridge' : 'quagga-br', 'ip': '10.10.0.3', 'mask' : 16 },
Chetan Gaonker8e25e1b2016-05-02 13:42:21 -0700547 { 'bridge' : 'quagga-br', 'ip': '192.168.10.3', 'mask': 16 },
548 )
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700549 ports = [ 179, 2601, 2602, 2603, 2604, 2605, 2606 ]
550 host_quagga_config = os.path.join(os.path.dirname(os.path.realpath(__file__)), '..', 'setup/quagga-config')
551 guest_quagga_config = '/root/config'
552 quagga_config_file = os.path.join(guest_quagga_config, 'testrib.conf')
553 host_guest_map = ( (host_quagga_config, guest_quagga_config), )
Chetan Gaonker503032a2016-05-12 12:06:29 -0700554 IMAGE = 'cord-test/quagga'
555 NAME = 'cord-quagga'
556
A R Karthick07608ef2016-08-23 16:51:19 -0700557 def __init__(self, name = NAME, image = IMAGE, prefix = '', tag = 'candidate',
Chetan Gaonker503032a2016-05-12 12:06:29 -0700558 boot_delay = 15, restart = False, config_file = quagga_config_file, update = False):
A R Karthick07608ef2016-08-23 16:51:19 -0700559 super(Quagga, self).__init__(name, image, prefix = prefix, tag = tag, quagga_config = self.quagga_config)
Chetan Gaonker503032a2016-05-12 12:06:29 -0700560 if update is True or not self.img_exists():
A R Karthick07608ef2016-08-23 16:51:19 -0700561 self.build_image(self.image_name)
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700562 if restart is True and self.exists():
563 self.kill()
564 if not self.exists():
565 self.remove_container(name, force=True)
A R Karthick41adfce2016-06-10 09:51:25 -0700566 host_config = self.create_host_config(port_list = self.ports,
567 host_guest_map = self.host_guest_map,
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700568 privileged = True)
569 volumes = []
570 for _,g in self.host_guest_map:
571 volumes.append(g)
572 self.start(ports = self.ports,
A R Karthick41adfce2016-06-10 09:51:25 -0700573 host_config = host_config,
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700574 volumes = volumes, tty = True)
575 print('Starting Quagga on container %s' %self.name)
576 self.execute('{0}/start.sh {1}'.format(self.guest_quagga_config, config_file))
577 time.sleep(boot_delay)
578
579 @classmethod
580 def build_image(cls, image):
Chetan Gaonker2a6601b2016-05-02 17:28:26 -0700581 onos_quagga_ip = Onos.quagga_config[0]['ip']
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700582 print('Building Quagga image %s' %image)
583 dockerfile = '''
A R Karthick41adfce2016-06-10 09:51:25 -0700584FROM ubuntu:14.04
585MAINTAINER chetan@ciena.com
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700586WORKDIR /root
587RUN useradd -M quagga
588RUN mkdir /var/log/quagga && chown quagga:quagga /var/log/quagga
589RUN mkdir /var/run/quagga && chown quagga:quagga /var/run/quagga
590RUN apt-get update && apt-get install -qy git autoconf libtool gawk make telnet libreadline6-dev
ChetanGaonkerb5b46c62016-08-16 12:02:53 -0700591RUN git clone git://git.savannah.nongnu.org/quagga.git quagga && \
Chetan Gaonker6cf6e472016-04-26 14:41:51 -0700592(cd quagga && git checkout HEAD && ./bootstrap.sh && \
593sed -i -r 's,htonl.*?\(INADDR_LOOPBACK\),inet_addr\("{0}"\),g' zebra/zebra_fpm.c && \
594./configure --enable-fpm --disable-doc --localstatedir=/var/run/quagga && make && make install)
595RUN ldconfig
596'''.format(onos_quagga_ip)
597 super(Quagga, cls).build_image(dockerfile, image)
598 print('Done building image %s' %image)
A R Karthick81acbff2016-06-17 14:45:16 -0700599
600def reinitContainerClients():
601 docker_netns.dckr = Client()
602 Container.dckr = Client()