1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
|
# This file is part of cloud-init. See LICENSE file for license information.
from abc import ABC, abstractmethod
import logging
import os
from tempfile import NamedTemporaryFile
from pycloudlib import EC2, GCE, Azure, OCI, LXD
from pycloudlib.cloud import BaseCloud
from pycloudlib.instance import BaseInstance
import cloudinit
from cloudinit.subp import subp
from tests.integration_tests import integration_settings
try:
from typing import Callable, Optional
except ImportError:
pass
log = logging.getLogger('integration_testing')
class IntegrationClient(ABC):
client = None # type: Optional[BaseCloud]
instance = None # type: Optional[BaseInstance]
datasource = None # type: Optional[str]
use_sudo = True
current_image = None
def __init__(self, user_data=None, instance_type=None, wait=True,
settings=integration_settings, launch_kwargs=None):
self.user_data = user_data
self.instance_type = settings.INSTANCE_TYPE if \
instance_type is None else instance_type
self.wait = wait
self.settings = settings
self.launch_kwargs = launch_kwargs if launch_kwargs else {}
self.client = self._get_client()
@abstractmethod
def _get_client(self):
raise NotImplementedError
def _get_image(self):
if self.current_image:
return self.current_image
image_id = self.settings.OS_IMAGE
try:
image_id = self.client.released_image(self.settings.OS_IMAGE)
except (ValueError, IndexError):
pass
return image_id
def launch(self):
if self.settings.EXISTING_INSTANCE_ID:
log.info(
'Not launching instance due to EXISTING_INSTANCE_ID. '
'Instance id: %s', self.settings.EXISTING_INSTANCE_ID)
self.instance = self.client.get_instance(
self.settings.EXISTING_INSTANCE_ID
)
return
image_id = self._get_image()
launch_args = {
'image_id': image_id,
'user_data': self.user_data,
'wait': self.wait,
}
if self.instance_type:
launch_args['instance_type'] = self.instance_type
launch_args.update(self.launch_kwargs)
self.instance = self.client.launch(**launch_args)
log.info('Launched instance: %s', self.instance)
def destroy(self):
self.instance.delete()
def execute(self, command):
return self.instance.execute(command)
def pull_file(self, remote_file, local_file):
self.instance.pull_file(remote_file, local_file)
def push_file(self, local_path, remote_path):
self.instance.push_file(local_path, remote_path)
def read_from_file(self, remote_path) -> str:
tmp_file = NamedTemporaryFile('r')
self.pull_file(remote_path, tmp_file.name)
with tmp_file as f:
contents = f.read()
return contents
def write_to_file(self, remote_path, contents: str):
# Writes file locally and then pushes it rather
# than writing the file directly on the instance
with NamedTemporaryFile('w', delete=False) as tmp_file:
tmp_file.write(contents)
try:
self.push_file(tmp_file.name, remote_path)
finally:
os.unlink(tmp_file.name)
def snapshot(self):
return self.client.snapshot(self.instance, clean=True)
def _install_new_cloud_init(self, remote_script):
self.execute(remote_script)
version = self.execute('cloud-init -v').split()[-1]
log.info('Installed cloud-init version: %s', version)
self.instance.clean()
image_id = self.snapshot()
log.info('Created new image: %s', image_id)
IntegrationClient.current_image = image_id
def install_proposed_image(self):
log.info('Installing proposed image')
remote_script = (
'{sudo} echo deb "http://archive.ubuntu.com/ubuntu '
'$(lsb_release -sc)-proposed main" | '
'{sudo} tee /etc/apt/sources.list.d/proposed.list\n'
'{sudo} apt-get update -q\n'
'{sudo} apt-get install -qy cloud-init'
).format(sudo='sudo' if self.use_sudo else '')
self._install_new_cloud_init(remote_script)
def install_ppa(self, repo):
log.info('Installing PPA')
remote_script = (
'{sudo} add-apt-repository {repo} -y && '
'{sudo} apt-get update -q && '
'{sudo} apt-get install -qy cloud-init'
).format(sudo='sudo' if self.use_sudo else '', repo=repo)
self._install_new_cloud_init(remote_script)
def install_deb(self):
log.info('Installing deb package')
deb_path = integration_settings.CLOUD_INIT_SOURCE
deb_name = os.path.basename(deb_path)
remote_path = '/var/tmp/{}'.format(deb_name)
self.push_file(
local_path=integration_settings.CLOUD_INIT_SOURCE,
remote_path=remote_path)
remote_script = '{sudo} dpkg -i {path}'.format(
sudo='sudo' if self.use_sudo else '', path=remote_path)
self._install_new_cloud_init(remote_script)
def __enter__(self):
self.launch()
return self
def __exit__(self, exc_type, exc_val, exc_tb):
if not self.settings.KEEP_INSTANCE:
self.destroy()
class Ec2Client(IntegrationClient):
datasource = 'ec2'
def _get_client(self):
return EC2(tag='ec2-integration-test')
class GceClient(IntegrationClient):
datasource = 'gce'
def _get_client(self):
return GCE(
tag='gce-integration-test',
project=self.settings.GCE_PROJECT,
region=self.settings.GCE_REGION,
zone=self.settings.GCE_ZONE,
)
class AzureClient(IntegrationClient):
datasource = 'azure'
def _get_client(self):
return Azure(tag='azure-integration-test')
class OciClient(IntegrationClient):
datasource = 'oci'
def _get_client(self):
return OCI(
tag='oci-integration-test',
compartment_id=self.settings.OCI_COMPARTMENT_ID
)
class LxdContainerClient(IntegrationClient):
datasource = 'lxd_container'
use_sudo = False
def _get_client(self):
return LXD(tag='lxd-integration-test')
def _mount_source(self):
command = (
'lxc config device add {name} host-cloud-init disk '
'source={cloudinit_path} '
'path=/usr/lib/python3/dist-packages/cloudinit'
).format(
name=self.instance.name, cloudinit_path=cloudinit.__path__[0])
subp(command.split())
def launch(self):
super().launch()
if self.settings.CLOUD_INIT_SOURCE == 'IN_PLACE':
self._mount_source()
client_name_to_class = {
'ec2': Ec2Client,
'gce': GceClient,
# 'azure': AzureClient, # Not supported yet
'oci': OciClient,
'lxd_container': LxdContainerClient
}
try:
dynamic_client = client_name_to_class[
integration_settings.PLATFORM
] # type: Callable[..., IntegrationClient]
except KeyError:
raise ValueError(
"{} is an invalid PLATFORM specified in settings. "
"Must be one of {}".format(
integration_settings.PLATFORM, list(client_name_to_class.keys())
)
)
|