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)
|