Statistics
| Branch: | Tag: | Revision:

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

History | View | Annotate | Download (11.3 kB)

1 6ef51e9f Giorgos Verigakis
# Copyright 2011-2012 GRNET S.A. All rights reserved.
2 cd2b0bf5 Christos Stavrakakis
#
3 adee02b8 Giorgos Verigakis
# Redistribution and use in source and binary forms, with or
4 adee02b8 Giorgos Verigakis
# without modification, are permitted provided that the following
5 adee02b8 Giorgos Verigakis
# conditions are met:
6 cd2b0bf5 Christos Stavrakakis
#
7 adee02b8 Giorgos Verigakis
#   1. Redistributions of source code must retain the above
8 adee02b8 Giorgos Verigakis
#      copyright notice, this list of conditions and the following
9 adee02b8 Giorgos Verigakis
#      disclaimer.
10 cd2b0bf5 Christos Stavrakakis
#
11 adee02b8 Giorgos Verigakis
#   2. Redistributions in binary form must reproduce the above
12 adee02b8 Giorgos Verigakis
#      copyright notice, this list of conditions and the following
13 adee02b8 Giorgos Verigakis
#      disclaimer in the documentation and/or other materials
14 adee02b8 Giorgos Verigakis
#      provided with the distribution.
15 cd2b0bf5 Christos Stavrakakis
#
16 adee02b8 Giorgos Verigakis
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
17 adee02b8 Giorgos Verigakis
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18 adee02b8 Giorgos Verigakis
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19 adee02b8 Giorgos Verigakis
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
20 adee02b8 Giorgos Verigakis
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 adee02b8 Giorgos Verigakis
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 adee02b8 Giorgos Verigakis
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
23 adee02b8 Giorgos Verigakis
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
24 adee02b8 Giorgos Verigakis
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 adee02b8 Giorgos Verigakis
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
26 adee02b8 Giorgos Verigakis
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 adee02b8 Giorgos Verigakis
# POSSIBILITY OF SUCH DAMAGE.
28 cd2b0bf5 Christos Stavrakakis
#
29 adee02b8 Giorgos Verigakis
# The views and conclusions contained in the software and
30 adee02b8 Giorgos Verigakis
# documentation are those of the authors and should not be
31 adee02b8 Giorgos Verigakis
# interpreted as representing official policies, either expressed
32 adee02b8 Giorgos Verigakis
# or implied, of GRNET S.A.
33 c36934a7 Giorgos Verigakis
34 2035039b Giorgos Verigakis
from logging import getLogger
35 2035039b Giorgos Verigakis
36 6ef51e9f Giorgos Verigakis
import dateutil.parser
37 6ef51e9f Giorgos Verigakis
38 d8e50a39 Giorgos Verigakis
from django.conf.urls.defaults import patterns
39 c36934a7 Giorgos Verigakis
from django.http import HttpResponse
40 c36934a7 Giorgos Verigakis
from django.template.loader import render_to_string
41 29a59bc1 Giorgos Verigakis
from django.utils import simplejson as json
42 c36934a7 Giorgos Verigakis
43 b36f78fa Giorgos Verigakis
from synnefo.api import util
44 b36f78fa Giorgos Verigakis
from synnefo.api.common import method_not_allowed
45 6ef51e9f Giorgos Verigakis
from synnefo.api.faults import BadRequest, ItemNotFound, ServiceUnavailable
46 6ef51e9f Giorgos Verigakis
from synnefo.api.util import api_method, isoformat, isoparse
47 6ef51e9f Giorgos Verigakis
from synnefo.plankton.backend import ImageBackend
48 b36f78fa Giorgos Verigakis
49 c36934a7 Giorgos Verigakis
50 9e98ba3c Giorgos Verigakis
log = getLogger('synnefo.api')
51 9e98ba3c Giorgos Verigakis
52 c36934a7 Giorgos Verigakis
urlpatterns = patterns('synnefo.api.images',
53 c36934a7 Giorgos Verigakis
    (r'^(?:/|.json|.xml)?$', 'demux'),
54 c36934a7 Giorgos Verigakis
    (r'^/detail(?:.json|.xml)?$', 'list_images', {'detail': True}),
55 6ef51e9f Giorgos Verigakis
    (r'^/([\w-]+)(?:.json|.xml)?$', 'image_demux'),
56 6ef51e9f Giorgos Verigakis
    (r'^/([\w-]+)/meta(?:.json|.xml)?$', 'metadata_demux'),
57 6ef51e9f Giorgos Verigakis
    (r'^/([\w-]+)/meta/(.+?)(?:.json|.xml)?$', 'metadata_item_demux')
58 c36934a7 Giorgos Verigakis
)
59 c36934a7 Giorgos Verigakis
60 c36934a7 Giorgos Verigakis
def demux(request):
61 c36934a7 Giorgos Verigakis
    if request.method == 'GET':
62 c36934a7 Giorgos Verigakis
        return list_images(request)
63 c36934a7 Giorgos Verigakis
    elif request.method == 'POST':
64 c36934a7 Giorgos Verigakis
        return create_image(request)
65 c36934a7 Giorgos Verigakis
    else:
66 d8e50a39 Giorgos Verigakis
        return method_not_allowed(request)
67 c36934a7 Giorgos Verigakis
68 c36934a7 Giorgos Verigakis
def image_demux(request, image_id):
69 c36934a7 Giorgos Verigakis
    if request.method == 'GET':
70 c36934a7 Giorgos Verigakis
        return get_image_details(request, image_id)
71 c36934a7 Giorgos Verigakis
    elif request.method == 'DELETE':
72 c36934a7 Giorgos Verigakis
        return delete_image(request, image_id)
73 c36934a7 Giorgos Verigakis
    else:
74 d8e50a39 Giorgos Verigakis
        return method_not_allowed(request)
75 c36934a7 Giorgos Verigakis
76 432fc8c3 Giorgos Verigakis
def metadata_demux(request, image_id):
77 432fc8c3 Giorgos Verigakis
    if request.method == 'GET':
78 432fc8c3 Giorgos Verigakis
        return list_metadata(request, image_id)
79 432fc8c3 Giorgos Verigakis
    elif request.method == 'POST':
80 432fc8c3 Giorgos Verigakis
        return update_metadata(request, image_id)
81 432fc8c3 Giorgos Verigakis
    else:
82 432fc8c3 Giorgos Verigakis
        return method_not_allowed(request)
83 432fc8c3 Giorgos Verigakis
84 432fc8c3 Giorgos Verigakis
def metadata_item_demux(request, image_id, key):
85 432fc8c3 Giorgos Verigakis
    if request.method == 'GET':
86 432fc8c3 Giorgos Verigakis
        return get_metadata_item(request, image_id, key)
87 432fc8c3 Giorgos Verigakis
    elif request.method == 'PUT':
88 432fc8c3 Giorgos Verigakis
        return create_metadata_item(request, image_id, key)
89 432fc8c3 Giorgos Verigakis
    elif request.method == 'DELETE':
90 432fc8c3 Giorgos Verigakis
        return delete_metadata_item(request, image_id, key)
91 432fc8c3 Giorgos Verigakis
    else:
92 432fc8c3 Giorgos Verigakis
        return method_not_allowed(request)
93 432fc8c3 Giorgos Verigakis
94 c36934a7 Giorgos Verigakis
95 c36934a7 Giorgos Verigakis
def image_to_dict(image, detail=True):
96 6ef51e9f Giorgos Verigakis
    d = dict(id=image['id'], name=image['name'])
97 c36934a7 Giorgos Verigakis
    if detail:
98 6ef51e9f Giorgos Verigakis
        d['updated'] = isoformat(dateutil.parser.parse(image['updated_at']))
99 6ef51e9f Giorgos Verigakis
        d['created'] = isoformat(dateutil.parser.parse(image['created_at']))
100 6ef51e9f Giorgos Verigakis
        d['status'] = 'DELETED' if image['deleted_at'] else 'ACTIVE'
101 6ef51e9f Giorgos Verigakis
        d['progress'] = 100 if image['status'] == 'available' else 0
102 6ef51e9f Giorgos Verigakis
        if image['properties']:
103 6ef51e9f Giorgos Verigakis
            d['metadata'] = {'values': image['properties']}
104 c36934a7 Giorgos Verigakis
    return d
105 c36934a7 Giorgos Verigakis
106 c36934a7 Giorgos Verigakis
107 6ef51e9f Giorgos Verigakis
@api_method('GET')
108 c36934a7 Giorgos Verigakis
def list_images(request, detail=False):
109 c36934a7 Giorgos Verigakis
    # Normal Response Codes: 200, 203
110 c36934a7 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
111 c36934a7 Giorgos Verigakis
    #                       serviceUnavailable (503),
112 c36934a7 Giorgos Verigakis
    #                       unauthorized (401),
113 c36934a7 Giorgos Verigakis
    #                       badRequest (400),
114 c36934a7 Giorgos Verigakis
    #                       overLimit (413)
115 cd2b0bf5 Christos Stavrakakis
116 0c37a721 Christos Stavrakakis
    log.debug('list_images detail=%s', detail)
117 4b3b8688 Giorgos Verigakis
    backend = ImageBackend(request.user_uniq)
118 cd2b0bf5 Christos Stavrakakis
119 6ef51e9f Giorgos Verigakis
    since = isoparse(request.GET.get('changes-since'))
120 d8e50a39 Giorgos Verigakis
    if since:
121 6ef51e9f Giorgos Verigakis
        images = []
122 6ef51e9f Giorgos Verigakis
        for image in backend.iter():
123 6ef51e9f Giorgos Verigakis
            updated = dateutil.parser.parse(image['updated_at'])
124 6ef51e9f Giorgos Verigakis
            if updated >= since:
125 6ef51e9f Giorgos Verigakis
                images.append(image)
126 6ef51e9f Giorgos Verigakis
        if not images:
127 d8e50a39 Giorgos Verigakis
            return HttpResponse(status=304)
128 93677203 Giorgos Verigakis
    else:
129 6ef51e9f Giorgos Verigakis
        images = backend.list()
130 cd2b0bf5 Christos Stavrakakis
131 cd2b0bf5 Christos Stavrakakis
    images = sorted(images, key=lambda x: x['id'])
132 6ef51e9f Giorgos Verigakis
    reply = [image_to_dict(image, detail) for image in images]
133 cd2b0bf5 Christos Stavrakakis
134 d8e50a39 Giorgos Verigakis
    if request.serialization == 'xml':
135 6ef51e9f Giorgos Verigakis
        data = render_to_string('list_images.xml',
136 6ef51e9f Giorgos Verigakis
                                dict(images=reply, detail=detail))
137 c36934a7 Giorgos Verigakis
    else:
138 6ef51e9f Giorgos Verigakis
        data = json.dumps(dict(images={'values': reply}))
139 cd2b0bf5 Christos Stavrakakis
140 d8e50a39 Giorgos Verigakis
    return HttpResponse(data, status=200)
141 c36934a7 Giorgos Verigakis
142 6ef51e9f Giorgos Verigakis
143 6ef51e9f Giorgos Verigakis
@api_method('POST')
144 c36934a7 Giorgos Verigakis
def create_image(request):
145 c36934a7 Giorgos Verigakis
    # Normal Response Code: 202
146 c36934a7 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
147 c36934a7 Giorgos Verigakis
    #                       serviceUnavailable (503),
148 c36934a7 Giorgos Verigakis
    #                       unauthorized (401),
149 c36934a7 Giorgos Verigakis
    #                       badMediaType(415),
150 c36934a7 Giorgos Verigakis
    #                       itemNotFound (404),
151 c36934a7 Giorgos Verigakis
    #                       badRequest (400),
152 c36934a7 Giorgos Verigakis
    #                       serverCapacityUnavailable (503),
153 c36934a7 Giorgos Verigakis
    #                       buildInProgress (409),
154 c36934a7 Giorgos Verigakis
    #                       resizeNotAllowed (403),
155 c36934a7 Giorgos Verigakis
    #                       backupOrResizeInProgress (409),
156 c36934a7 Giorgos Verigakis
    #                       overLimit (413)
157 cd2b0bf5 Christos Stavrakakis
158 6ef51e9f Giorgos Verigakis
    raise ServiceUnavailable('Not supported.')
159 aa197ee4 Vangelis Koukis
160 aa197ee4 Vangelis Koukis
161 6ef51e9f Giorgos Verigakis
@api_method('GET')
162 c36934a7 Giorgos Verigakis
def get_image_details(request, image_id):
163 c36934a7 Giorgos Verigakis
    # Normal Response Codes: 200, 203
164 c36934a7 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
165 c36934a7 Giorgos Verigakis
    #                       serviceUnavailable (503),
166 c36934a7 Giorgos Verigakis
    #                       unauthorized (401),
167 c36934a7 Giorgos Verigakis
    #                       badRequest (400),
168 c36934a7 Giorgos Verigakis
    #                       itemNotFound (404),
169 c36934a7 Giorgos Verigakis
    #                       overLimit (413)
170 cd2b0bf5 Christos Stavrakakis
171 0c37a721 Christos Stavrakakis
    log.debug('get_image_details %s', image_id)
172 4b3b8688 Giorgos Verigakis
    image = util.get_image(image_id, request.user_uniq)
173 6ef51e9f Giorgos Verigakis
    reply = image_to_dict(image)
174 cd2b0bf5 Christos Stavrakakis
175 d8e50a39 Giorgos Verigakis
    if request.serialization == 'xml':
176 6ef51e9f Giorgos Verigakis
        data = render_to_string('image.xml', dict(image=reply))
177 c36934a7 Giorgos Verigakis
    else:
178 6ef51e9f Giorgos Verigakis
        data = json.dumps(dict(image=reply))
179 cd2b0bf5 Christos Stavrakakis
180 c36934a7 Giorgos Verigakis
    return HttpResponse(data, status=200)
181 c36934a7 Giorgos Verigakis
182 6ef51e9f Giorgos Verigakis
183 6ef51e9f Giorgos Verigakis
@api_method('DELETE')
184 c36934a7 Giorgos Verigakis
def delete_image(request, image_id):
185 c36934a7 Giorgos Verigakis
    # Normal Response Code: 204
186 c36934a7 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
187 c36934a7 Giorgos Verigakis
    #                       serviceUnavailable (503),
188 c36934a7 Giorgos Verigakis
    #                       unauthorized (401),
189 c36934a7 Giorgos Verigakis
    #                       itemNotFound (404),
190 c36934a7 Giorgos Verigakis
    #                       overLimit (413)
191 cd2b0bf5 Christos Stavrakakis
192 bf5c82dc Christos Stavrakakis
    log.info('delete_image %s', image_id)
193 4b3b8688 Giorgos Verigakis
    backend = ImageBackend(request.user_uniq)
194 6ef51e9f Giorgos Verigakis
    backend.delete(image_id)
195 6ef51e9f Giorgos Verigakis
    backend.close()
196 4b3b8688 Giorgos Verigakis
    log.info('User %s deleted image %s', request.user_uniq, image_id)
197 c36934a7 Giorgos Verigakis
    return HttpResponse(status=204)
198 432fc8c3 Giorgos Verigakis
199 6ef51e9f Giorgos Verigakis
200 6ef51e9f Giorgos Verigakis
@api_method('GET')
201 432fc8c3 Giorgos Verigakis
def list_metadata(request, image_id):
202 432fc8c3 Giorgos Verigakis
    # Normal Response Codes: 200, 203
203 432fc8c3 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
204 432fc8c3 Giorgos Verigakis
    #                       serviceUnavailable (503),
205 432fc8c3 Giorgos Verigakis
    #                       unauthorized (401),
206 432fc8c3 Giorgos Verigakis
    #                       badRequest (400),
207 432fc8c3 Giorgos Verigakis
    #                       overLimit (413)
208 cd2b0bf5 Christos Stavrakakis
209 0c37a721 Christos Stavrakakis
    log.debug('list_image_metadata %s', image_id)
210 4b3b8688 Giorgos Verigakis
    image = util.get_image(image_id, request.user_uniq)
211 6ef51e9f Giorgos Verigakis
    metadata = image['properties']
212 b36f78fa Giorgos Verigakis
    return util.render_metadata(request, metadata, use_values=True, status=200)
213 432fc8c3 Giorgos Verigakis
214 6ef51e9f Giorgos Verigakis
215 6ef51e9f Giorgos Verigakis
@api_method('POST')
216 432fc8c3 Giorgos Verigakis
def update_metadata(request, image_id):
217 432fc8c3 Giorgos Verigakis
    # Normal Response Code: 201
218 432fc8c3 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
219 432fc8c3 Giorgos Verigakis
    #                       serviceUnavailable (503),
220 432fc8c3 Giorgos Verigakis
    #                       unauthorized (401),
221 432fc8c3 Giorgos Verigakis
    #                       badRequest (400),
222 432fc8c3 Giorgos Verigakis
    #                       buildInProgress (409),
223 432fc8c3 Giorgos Verigakis
    #                       badMediaType(415),
224 432fc8c3 Giorgos Verigakis
    #                       overLimit (413)
225 cd2b0bf5 Christos Stavrakakis
226 b36f78fa Giorgos Verigakis
    req = util.get_request_dict(request)
227 bf5c82dc Christos Stavrakakis
    log.info('update_image_metadata %s %s', image_id, req)
228 4b3b8688 Giorgos Verigakis
    image = util.get_image(image_id, request.user_uniq)
229 432fc8c3 Giorgos Verigakis
    try:
230 432fc8c3 Giorgos Verigakis
        metadata = req['metadata']
231 432fc8c3 Giorgos Verigakis
        assert isinstance(metadata, dict)
232 432fc8c3 Giorgos Verigakis
    except (KeyError, AssertionError):
233 432fc8c3 Giorgos Verigakis
        raise BadRequest('Malformed request.')
234 cd2b0bf5 Christos Stavrakakis
235 6ef51e9f Giorgos Verigakis
    properties = image['properties']
236 6ef51e9f Giorgos Verigakis
    properties.update(metadata)
237 cd2b0bf5 Christos Stavrakakis
238 4b3b8688 Giorgos Verigakis
    backend = ImageBackend(request.user_uniq)
239 6ef51e9f Giorgos Verigakis
    backend.update(image_id, dict(properties=properties))
240 6ef51e9f Giorgos Verigakis
    backend.close()
241 cd2b0bf5 Christos Stavrakakis
242 6ef51e9f Giorgos Verigakis
    return util.render_metadata(request, properties, status=201)
243 432fc8c3 Giorgos Verigakis
244 6ef51e9f Giorgos Verigakis
245 6ef51e9f Giorgos Verigakis
@api_method('GET')
246 432fc8c3 Giorgos Verigakis
def get_metadata_item(request, image_id, key):
247 432fc8c3 Giorgos Verigakis
    # Normal Response Codes: 200, 203
248 432fc8c3 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
249 432fc8c3 Giorgos Verigakis
    #                       serviceUnavailable (503),
250 432fc8c3 Giorgos Verigakis
    #                       unauthorized (401),
251 432fc8c3 Giorgos Verigakis
    #                       itemNotFound (404),
252 432fc8c3 Giorgos Verigakis
    #                       badRequest (400),
253 432fc8c3 Giorgos Verigakis
    #                       overLimit (413)
254 cd2b0bf5 Christos Stavrakakis
255 0c37a721 Christos Stavrakakis
    log.debug('get_image_metadata_item %s %s', image_id, key)
256 4b3b8688 Giorgos Verigakis
    image = util.get_image(image_id, request.user_uniq)
257 6ef51e9f Giorgos Verigakis
    val = image['properties'].get(key)
258 6ef51e9f Giorgos Verigakis
    if val is None:
259 6ef51e9f Giorgos Verigakis
        raise ItemNotFound('Metadata key not found.')
260 6ef51e9f Giorgos Verigakis
    return util.render_meta(request, {key: val}, status=200)
261 6ef51e9f Giorgos Verigakis
262 432fc8c3 Giorgos Verigakis
263 6ef51e9f Giorgos Verigakis
@api_method('PUT')
264 432fc8c3 Giorgos Verigakis
def create_metadata_item(request, image_id, key):
265 432fc8c3 Giorgos Verigakis
    # Normal Response Code: 201
266 432fc8c3 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
267 432fc8c3 Giorgos Verigakis
    #                       serviceUnavailable (503),
268 432fc8c3 Giorgos Verigakis
    #                       unauthorized (401),
269 432fc8c3 Giorgos Verigakis
    #                       itemNotFound (404),
270 432fc8c3 Giorgos Verigakis
    #                       badRequest (400),
271 432fc8c3 Giorgos Verigakis
    #                       buildInProgress (409),
272 432fc8c3 Giorgos Verigakis
    #                       badMediaType(415),
273 432fc8c3 Giorgos Verigakis
    #                       overLimit (413)
274 cd2b0bf5 Christos Stavrakakis
275 b36f78fa Giorgos Verigakis
    req = util.get_request_dict(request)
276 bf5c82dc Christos Stavrakakis
    log.info('create_image_metadata_item %s %s %s', image_id, key, req)
277 432fc8c3 Giorgos Verigakis
    try:
278 432fc8c3 Giorgos Verigakis
        metadict = req['meta']
279 432fc8c3 Giorgos Verigakis
        assert isinstance(metadict, dict)
280 432fc8c3 Giorgos Verigakis
        assert len(metadict) == 1
281 432fc8c3 Giorgos Verigakis
        assert key in metadict
282 432fc8c3 Giorgos Verigakis
    except (KeyError, AssertionError):
283 432fc8c3 Giorgos Verigakis
        raise BadRequest('Malformed request.')
284 6ef51e9f Giorgos Verigakis
285 6ef51e9f Giorgos Verigakis
    val = metadict[key]
286 4b3b8688 Giorgos Verigakis
    image = util.get_image(image_id, request.user_uniq)
287 6ef51e9f Giorgos Verigakis
    properties = image['properties']
288 6ef51e9f Giorgos Verigakis
    properties[key] = val
289 cd2b0bf5 Christos Stavrakakis
290 4b3b8688 Giorgos Verigakis
    backend = ImageBackend(request.user_uniq)
291 6ef51e9f Giorgos Verigakis
    backend.update(image_id, dict(properties=properties))
292 6ef51e9f Giorgos Verigakis
    backend.close()
293 cd2b0bf5 Christos Stavrakakis
294 6ef51e9f Giorgos Verigakis
    return util.render_meta(request, {key: val}, status=201)
295 6ef51e9f Giorgos Verigakis
296 432fc8c3 Giorgos Verigakis
297 6ef51e9f Giorgos Verigakis
@api_method('DELETE')
298 432fc8c3 Giorgos Verigakis
def delete_metadata_item(request, image_id, key):
299 432fc8c3 Giorgos Verigakis
    # Normal Response Code: 204
300 432fc8c3 Giorgos Verigakis
    # Error Response Codes: computeFault (400, 500),
301 432fc8c3 Giorgos Verigakis
    #                       serviceUnavailable (503),
302 432fc8c3 Giorgos Verigakis
    #                       unauthorized (401),
303 432fc8c3 Giorgos Verigakis
    #                       itemNotFound (404),
304 432fc8c3 Giorgos Verigakis
    #                       badRequest (400),
305 432fc8c3 Giorgos Verigakis
    #                       buildInProgress (409),
306 432fc8c3 Giorgos Verigakis
    #                       badMediaType(415),
307 432fc8c3 Giorgos Verigakis
    #                       overLimit (413),
308 cd2b0bf5 Christos Stavrakakis
309 bf5c82dc Christos Stavrakakis
    log.info('delete_image_metadata_item %s %s', image_id, key)
310 4b3b8688 Giorgos Verigakis
    image = util.get_image(image_id, request.user_uniq)
311 6ef51e9f Giorgos Verigakis
    properties = image['properties']
312 6ef51e9f Giorgos Verigakis
    properties.pop(key, None)
313 cd2b0bf5 Christos Stavrakakis
314 4b3b8688 Giorgos Verigakis
    backend = ImageBackend(request.user_uniq)
315 6ef51e9f Giorgos Verigakis
    backend.update(image_id, dict(properties=properties))
316 6ef51e9f Giorgos Verigakis
    backend.close()
317 cd2b0bf5 Christos Stavrakakis
318 432fc8c3 Giorgos Verigakis
    return HttpResponse(status=204)