Statistics
| Branch: | Tag: | Revision:

root / snf-cyclades-app / synnefo / api / test / images.py @ d2b8ec7b

History | View | Annotate | Download (12.7 kB)

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

    
34
import json
35

    
36
from snf_django.lib.api import faults
37
from snf_django.utils.testing import BaseAPITest
38

    
39
from mock import patch
40
from functools import wraps
41

    
42

    
43
def assert_backend_closed(func):
44
    """Decorator for ensuring that ImageBackend is returned to pool."""
45
    @wraps(func)
46
    def wrapper(self, backend):
47
        result = func(self, backend)
48
        if backend.called is True:
49
            num = len(backend.mock_calls) / 2
50
            assert(len(backend.return_value.close.mock_calls), num)
51
        return result
52
    return wrapper
53

    
54

    
55
@patch('synnefo.plankton.utils.ImageBackend')
56
class ImageAPITest(BaseAPITest):
57
    @assert_backend_closed
58
    def test_create_image(self, mimage):
59
        """Test that create image is not implemented"""
60
        response = self.post('/api/v1.1/images/', 'user', json.dumps(''),
61
                             'json')
62
        self.assertEqual(response.status_code, 501)
63

    
64
    @assert_backend_closed
65
    def test_list_images(self, mimage):
66
        """Test that expected list of images is returned"""
67
        images = [{'id': 1, 'name': 'image-1'},
68
                  {'id': 2, 'name': 'image-2'},
69
                  {'id': 3, 'name': 'image-3'}]
70
        mimage().list.return_value = images
71
        response = self.get('/api/v1.1/images/', 'user')
72
        self.assertSuccess(response)
73
        api_images = json.loads(response.content)['images']['values']
74
        self.assertEqual(images, api_images)
75

    
76
    @assert_backend_closed
77
    def test_list_images_detail(self, mimage):
78
        images = [{'id': 1,
79
                   'name': 'image-1',
80
                   'status':'available',
81
                   'created_at': '2012-11-26 11:52:54',
82
                   'updated_at': '2012-12-26 11:52:54',
83
                   'deleted_at': '',
84
                   'properties': {'foo':'bar'}},
85
                  {'id': 2,
86
                   'name': 'image-2',
87
                   'status': 'deleted',
88
                   'created_at': '2012-11-26 11:52:54',
89
                   'updated_at': '2012-12-26 11:52:54',
90
                   'deleted_at': '2012-12-27 11:52:54',
91
                   'properties': ''},
92
                  {'id': 3,
93
                   'name': 'image-3',
94
                   'status': 'available',
95
                   'created_at': '2012-11-26 11:52:54',
96
                   'deleted_at': '',
97
                   'updated_at': '2012-12-26 11:52:54',
98
                   'properties': ''}]
99
        result_images = [
100
                  {'id': 1,
101
                   'name': 'image-1',
102
                   'status':'ACTIVE',
103
                   'progress': 100,
104
                   'created': '2012-11-26T11:52:54+00:00',
105
                   'updated': '2012-12-26T11:52:54+00:00',
106
                   'metadata': {'values': {'foo':'bar'}}},
107
                  {'id': 2,
108
                   'name': 'image-2',
109
                   'status': 'DELETED',
110
                   'progress': 0,
111
                   'created': '2012-11-26T11:52:54+00:00',
112
                   'updated': '2012-12-26T11:52:54+00:00'},
113
                  {'id': 3,
114
                   'name': 'image-3',
115
                   'status': 'ACTIVE',
116
                   'progress': 100,
117
                   'created': '2012-11-26T11:52:54+00:00',
118
                   'updated': '2012-12-26T11:52:54+00:00'}]
119
        mimage().list.return_value = images
120
        response = self.get('/api/v1.1/images/detail', 'user')
121
        self.assertSuccess(response)
122
        api_images = json.loads(response.content)['images']['values']
123
        self.assertEqual(len(result_images), len(api_images))
124
        self.assertEqual(result_images, api_images)
125

    
126
    @assert_backend_closed
127
    def test_list_images_detail_since(self, mimage):
128
        from datetime import datetime, timedelta
129
        from time import sleep
130
        old_time = datetime.now()
131
        new_time = old_time + timedelta(seconds=0.1)
132
        sleep(0.1)
133
        images = [
134
                  {'id': 1,
135
                   'name': 'image-1',
136
                   'status':'available',
137
                   'progress': 100,
138
                   'created_at': old_time.isoformat(),
139
                   'deleted_at': '',
140
                   'updated_at': old_time.isoformat(),
141
                   'properties': ''},
142
                  {'id': 2,
143
                   'name': 'image-2',
144
                   'status': 'deleted',
145
                   'progress': 0,
146
                   'created_at': new_time.isoformat(),
147
                   'updated_at': new_time.isoformat(),
148
                   'deleted_at': new_time.isoformat(),
149
                   'properties': ''}]
150
        mimage().iter.return_value = images
151
        response =\
152
            self.get('/api/v1.1/images/detail?changes-since=%sUTC' % new_time)
153
        self.assertSuccess(response)
154
        api_images = json.loads(response.content)['images']['values']
155
        self.assertEqual(1, len(api_images))
156

    
157
    @assert_backend_closed
158
    def test_get_image_details(self, mimage):
159
        image = {'id': 42,
160
                 'name': 'image-1',
161
                 'status': 'available',
162
                 'created_at': '2012-11-26 11:52:54',
163
                 'updated_at': '2012-12-26 11:52:54',
164
                 'deleted_at': '',
165
                 'properties': {'foo': 'bar'}}
166
        result_image = \
167
                  {'id': 42,
168
                   'name': 'image-1',
169
                   'status': 'ACTIVE',
170
                   'progress': 100,
171
                   'created': '2012-11-26T11:52:54+00:00',
172
                   'updated': '2012-12-26T11:52:54+00:00',
173
                   'metadata': {'values': {'foo': 'bar'}}}
174
        with patch('synnefo.api.util.get_image') as m:
175
            m.return_value = image
176
            response = self.get('/api/v1.1/images/42', 'user')
177
        self.assertSuccess(response)
178
        api_image = json.loads(response.content)['image']
179
        self.assertEqual(api_image, result_image)
180

    
181
    @assert_backend_closed
182
    def test_invalid_image(self, mimage):
183
        with patch('synnefo.api.util.get_image') as m:
184
            m.side_effect = faults.ItemNotFound('Image not found')
185
            response = self.get('/api/v1.1/images/42', 'user')
186
        self.assertItemNotFound(response)
187

    
188
    def test_delete_image(self, mimage):
189
        response = self.delete("/api/v1.1/images/42", "user")
190
        self.assertEqual(response.status_code, 204)
191
        mimage.return_value.unregister.assert_called_once_with('42')
192
        mimage.return_value._delete.assert_not_called('42')
193

    
194

    
195
@patch('synnefo.plankton.utils.ImageBackend')
196
class ImageMetadataAPITest(BaseAPITest):
197
    def setUp(self):
198
        self.image = {'id': 42,
199
                 'name': 'image-1',
200
                 'status': 'available',
201
                 'created_at': '2012-11-26 11:52:54',
202
                 'updated_at': '2012-12-26 11:52:54',
203
                 'deleted_at': '',
204
                 'properties': {'foo': 'bar', 'foo2': 'bar2'}}
205
        self.result_image = \
206
                  {'id': 42,
207
                   'name': 'image-1',
208
                   'status': 'ACTIVE',
209
                   'progress': 100,
210
                   'created': '2012-11-26T11:52:54+00:00',
211
                   'updated': '2012-12-26T11:52:54+00:00',
212
                   'metadata': {'values': {'foo': 'bar'}}}
213

    
214
    @assert_backend_closed
215
    def test_list_metadata(self, backend):
216
        backend.return_value.get_image.return_value = self.image
217
        response = self.get('/api/v1.1/images/42/meta', 'user')
218
        self.assertSuccess(response)
219
        meta = json.loads(response.content)['metadata']['values']
220
        self.assertEqual(meta, self.image['properties'])
221

    
222
    @assert_backend_closed
223
    def test_get_metadata(self, backend):
224
        backend.return_value.get_image.return_value = self.image
225
        response = self.get('/api/v1.1/images/42/meta/foo', 'user')
226
        self.assertSuccess(response)
227
        meta = json.loads(response.content)['meta']
228
        self.assertEqual(meta['foo'], 'bar')
229

    
230
    @assert_backend_closed
231
    def test_get_invalid_metadata(self, backend):
232
        backend.return_value.get_image.return_value = self.image
233
        response = self.get('/api/v1.1/images/42/meta/not_found', 'user')
234
        self.assertItemNotFound(response)
235

    
236
    def test_delete_metadata_item(self, backend):
237
        backend.return_value.get_image.return_value = self.image
238
        response = self.delete('/api/v1.1/images/42/meta/foo', 'user')
239
        self.assertEqual(response.status_code, 204)
240
        backend.return_value.update.assert_called_once_with('42', {'properties': {'foo2':
241
                                                    'bar2'}})
242

    
243
    @assert_backend_closed
244
    def test_create_metadata_item(self, backend):
245
        backend.return_value.get_image.return_value = self.image
246
        request = {'meta': {'foo3': 'bar3'}}
247
        response = self.put('/api/v1.1/images/42/meta/foo3', 'user',
248
                            json.dumps(request), 'json')
249
        self.assertEqual(response.status_code, 201)
250
        backend.return_value.update.assert_called_once_with('42',
251
                {'properties':
252
                    {'foo': 'bar', 'foo2': 'bar2', 'foo3': 'bar3'}})
253

    
254
    @assert_backend_closed
255
    def test_create_metadata_malformed_1(self, backend):
256
        backend.return_value.get_image.return_value = self.image
257
        request = {'met': {'foo3': 'bar3'}}
258
        response = self.put('/api/v1.1/images/42/meta/foo3', 'user',
259
                            json.dumps(request), 'json')
260
        self.assertBadRequest(response)
261

    
262
    @assert_backend_closed
263
    def test_create_metadata_malformed_2(self, backend):
264
        backend.return_value.get_image.return_value = self.image
265
        request = {'meta': [('foo3', 'bar3')]}
266
        response = self.put('/api/v1.1/images/42/meta/foo3', 'user',
267
                            json.dumps(request), 'json')
268
        self.assertBadRequest(response)
269

    
270
    @assert_backend_closed
271
    def test_create_metadata_malformed_3(self, backend):
272
        backend.return_value.get_image.return_value = self.image
273
        request = {'met': {'foo3': 'bar3', 'foo4': 'bar4'}}
274
        response = self.put('/api/v1.1/images/42/meta/foo3', 'user',
275
                                json.dumps(request), 'json')
276
        self.assertBadRequest(response)
277

    
278
    @assert_backend_closed
279
    def test_create_metadata_malformed_4(self, backend):
280
        backend.return_value.get_image.return_value = self.image
281
        request = {'met': {'foo3': 'bar3'}}
282
        response = self.put('/api/v1.1/images/42/meta/foo4', 'user',
283
                                json.dumps(request), 'json')
284
        self.assertBadRequest(response)
285

    
286
    @assert_backend_closed
287
    def test_update_metadata_item(self, backend):
288
        backend.return_value.get_image.return_value = self.image
289
        request = {'metadata': {'foo': 'bar_new', 'foo4': 'bar4'}}
290
        response = self.post('/api/v1.1/images/42/meta', 'user',
291
                             json.dumps(request), 'json')
292
        self.assertEqual(response.status_code, 201)
293
        backend.return_value.update.assert_called_once_with('42',
294
                {'properties':
295
                    {'foo': 'bar_new', 'foo2': 'bar2', 'foo4': 'bar4'}
296
                })
297

    
298
    @assert_backend_closed
299
    def test_update_metadata_malformed(self, backend):
300
        backend.return_value.get_image.return_value = self.image
301
        request = {'meta': {'foo': 'bar_new', 'foo4': 'bar4'}}
302
        response = self.post('/api/v1.1/images/42/meta', 'user',
303
                            json.dumps(request), 'json')
304
        self.assertBadRequest(response)