Statistics
| Branch: | Tag: | Revision:

root / snf-tools / synnefo_tools / burnin / images_tests.py @ 3e5bbd85

History | View | Annotate | Download (8.8 kB)

1

    
2
# Copyright 2013 GRNET S.A. All rights reserved.
3
#
4
# Redistribution and use in source and binary forms, with or
5
# without modification, are permitted provided that the following
6
# conditions are met:
7
#
8
#   1. Redistributions of source code must retain the above
9
#      copyright notice, this list of conditions and the following
10
#      disclaimer.
11
#
12
#   2. Redistributions in binary form must reproduce the above
13
#      copyright notice, this list of conditions and the following
14
#      disclaimer in the documentation and/or other materials
15
#      provided with the distribution.
16
#
17
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
18
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
20
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
21
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
24
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
25
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
27
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28
# POSSIBILITY OF SUCH DAMAGE.
29
#
30
# The views and conclusions contained in the software and
31
# documentation are those of the authors and should not be
32
# interpreted as representing official policies, either expressed
33
# or implied, of GRNET S.A.
34

    
35
"""
36
This is the burnin class that tests the Flavors/Images functionality
37

38
"""
39

    
40
import os
41
import shutil
42

    
43
from kamaki.clients import ClientError
44

    
45
from synnefo_tools.burnin.common import BurninTests, Proper
46

    
47

    
48
# Too many public methods. pylint: disable-msg=R0904
49
class FlavorsTestSuite(BurninTests):
50
    """Test flavor lists for consistency"""
51
    simple_flavors = Proper(value=None)
52
    detailed_flavors = Proper(value=None)
53
    simple_names = Proper(value=None)
54

    
55
    def test_001_simple_flavors(self):
56
        """Test flavor list actually returns flavors"""
57
        self.simple_flavors = self._get_list_of_flavors(detail=False)
58
        self.assertGreater(len(self.simple_flavors), 0)
59

    
60
    def test_002_get_detailed_flavors(self):
61
        """Test detailed flavor list is the same length as list"""
62
        self.detailed_flavors = self._get_list_of_flavors(detail=True)
63
        self.assertEquals(len(self.simple_flavors), len(self.detailed_flavors))
64

    
65
    def test_003_same_flavor_names(self):
66
        """Test detailed and simple flavor list contain same names"""
67
        names = sorted([flv['name'] for flv in self.simple_flavors])
68
        self.simple_names = names
69
        detailed_names = sorted([flv['name'] for flv in self.detailed_flavors])
70
        self.assertEqual(self.simple_names, detailed_names)
71

    
72
    def test_004_unique_flavor_names(self):
73
        """Test flavors have unique names"""
74
        self.assertEqual(sorted(list(set(self.simple_names))),
75
                         self.simple_names)
76

    
77
    def test_005_well_formed_names(self):
78
        """Test flavors have well formed names
79

80
        Test flavors have names of the form CxxRyyDzz, where xx is vCPU count,
81
        yy is RAM in MiB, zz is Disk in GiB
82

83
        """
84
        for flv in self.detailed_flavors:
85
            flavor = (flv['vcpus'], flv['ram'], flv['disk'],
86
                      flv['SNF:disk_template'])
87
            self.assertEqual("C%dR%dD%d%s" % flavor, flv['name'],
88
                             "Flavor %s doesn't match its specs" % flv['name'])
89

    
90

    
91
# --------------------------------------------------------------------
92
# Too many public methods. pylint: disable-msg=R0904
93
class ImagesTestSuite(BurninTests):
94
    """Test image lists for consistency"""
95
    simple_images = Proper(value=None)
96
    detailed_images = Proper(value=None)
97
    system_images = Proper(value=None)
98
    temp_dir = Proper(value=None)
99
    temp_image_name = Proper(value=None)
100
    temp_image_file = Proper(value=None)
101

    
102
    def test_001_list_images(self):
103
        """Test simple image list actually returns images"""
104
        self.simple_images = self._get_list_of_images(detail=False)
105
        self.assertGreater(len(self.simple_images), 0)
106

    
107
    def test_002_list_images_detailed(self):
108
        """Test detailed image list is the same length as simple list"""
109
        self.detailed_images = self._get_list_of_images(detail=True)
110
        self.assertEqual(len(self.simple_images), len(self.detailed_images))
111

    
112
    def test_003_same_image_names(self):
113
        """Test detailed and simple image list contain the same names"""
114
        snames = sorted([i['name'] for i in self.simple_images])
115
        dnames = sorted([i['name'] for i in self.detailed_images])
116
        self.assertEqual(snames, dnames)
117

    
118
    def test_004_system_images(self):
119
        """Test that there are system images registered"""
120
        images = self._get_list_of_sys_images(images=self.detailed_images)
121
        self.system_images = images
122
        self.assertGreater(len(images), 0)
123

    
124
    def test_005_unique_image_names(self):
125
        """Test system images have unique names"""
126
        names = sorted([i['name'] for i in self.system_images])
127
        self.assertEqual(sorted(list(set(names))), names)
128

    
129
    def test_006_image_metadata(self):
130
        """Test every system image has specific metadata defined"""
131
        keys = frozenset(["osfamily", "root_partition"])
132
        for i in self.system_images:
133
            self.assertTrue(keys.issubset(i['properties'].keys()))
134

    
135
    def test_007_download_image(self):
136
        """Download image from Pithos"""
137
        # Find the 'Debian Base' image
138
        images = self._find_images(["name:^Debian Base$"],
139
                                   images=self.system_images)
140
        image = images[0]
141
        self.info("Will use %s with id %s", image['name'], image['id'])
142
        image_location = \
143
            image['location'].replace("://", " ").replace("/", " ").split()
144
        image_owner = image_location[1]
145
        self.info("Image's owner is %s", image_owner)
146
        image_container = image_location[2]
147
        self.info("Image's container is %s", image_container)
148
        image_name = image_location[3]
149
        self.info("Image's name is %s", image_name)
150
        self.temp_image_name = image_name
151

    
152
        self._set_pithos_account(image_owner)
153
        self._set_pithos_container(image_container)
154

    
155
        # Create temp directory
156
        self.temp_dir = self._create_tmp_directory()
157
        self.temp_image_file = \
158
            os.path.join(self.temp_dir, self.temp_image_name)
159

    
160
        # Write to file
161
        self.info("Downloading image to %s", self.temp_image_file)
162
        with open(self.temp_image_file, "w+b") as fout:
163
            self.clients.pithos.download_object(image_name, fout)
164

    
165
    def test_008_upload_image(self):
166
        """Upload the image to Pithos"""
167
        self._set_pithos_account(self._get_uuid())
168
        self._create_pithos_container("burnin-images")
169
        file_size = os.path.getsize(self.temp_image_file)
170
        with open(self.temp_image_file, "r+b") as fin:
171
            self.clients.pithos.upload_object(self.temp_image_name, fin)
172

    
173
        # Verify quotas
174
        self._check_quotas(diskspace=file_size)
175

    
176
    def test_009_register_image(self):
177
        """Register image to Plankton"""
178
        location = "pithos://" + self._get_uuid() + \
179
            "/burnin-images/" + self.temp_image_name
180
        self.info("Registering image %s", location)
181

    
182
        params = {'is_public': False}
183
        properties = {'OSFAMILY': "linux", 'ROOT_PARTITION': 1}
184
        self.clients.image.register(self.temp_image_name, location,
185
                                    params, properties)
186

    
187
        # Check that image is registered
188
        self.info("Checking that image has been registered")
189
        images = self._get_list_of_images(detail=True)
190
        images = [i for i in images if i['location'] == location]
191
        self.assertEqual(len(images), 1)
192
        self.info("Image registered with id %s", images[0]['id'])
193

    
194
    def test_010_cleanup_image(self):
195
        """Remove uploaded image from Pithos"""
196
        # Remove uploaded image
197
        self.info("Deleting uploaded image %s", self.temp_image_name)
198
        self.clients.pithos.del_object(self.temp_image_name)
199
        # Verify quotas
200
        file_size = os.path.getsize(self.temp_image_file)
201
        self._check_quotas(diskspace=-file_size)
202
        self.temp_image_name = None
203
        # Remove temp directory
204
        self.info("Deleting temp directory %s", self.temp_dir)
205
        self._remove_tmp_directory(self.temp_dir)
206
        self.temp_dir = None
207

    
208
    @classmethod
209
    def tearDownClass(cls):  # noqa
210
        """Clean up"""
211
        if cls.temp_image_name is not None:
212
            try:
213
                cls.clients.pithos.del_object(cls.temp_image_name)
214
            except ClientError:
215
                pass
216

    
217
        if cls.temp_dir is not None:
218
            try:
219
                shutil.rmtree(cls.temp_dir)
220
            except OSError:
221
                pass