Python
0.4.0
Source code for gcloud.exceptions
# Copyright 2014 Google Inc. All rights reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Custom exceptions for gcloud.storage package.
See: https://cloud.google.com/storage/docs/json_api/v1/status-codes
"""
import json
_HTTP_CODE_TO_EXCEPTION = {} # populated at end of module
[docs]class GCloudError(Exception):
"""Base error class for gcloud errors (abstract).
Each subclass represents a single type of HTTP error response.
"""
code = None
"""HTTP status code. Concrete subclasses *must* define.
See: http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html
"""
def __init__(self, message, errors=()):
super(GCloudError, self).__init__()
# suppress deprecation warning under 2.6.x
self.message = message
self._errors = [error.copy() for error in errors]
def __str__(self):
return '%d %s' % (self.code, self.message)
@property
[docs] def errors(self):
"""Detailed error information.
:rtype: list(dict)
:returns: a list of mappings describing each error.
"""
return [error.copy() for error in self._errors]
[docs]class MovedPermanently(Redirection):
"""Exception mapping a '301 Moved Permanently' response."""
code = 301
[docs]class NotModified(Redirection):
"""Exception mapping a '304 Not Modified' response."""
code = 304
[docs]class TemporaryRedirect(Redirection):
"""Exception mapping a '307 Temporary Redirect' response."""
code = 307
[docs]class ResumeIncomplete(Redirection):
"""Exception mapping a '308 Resume Incomplete' response."""
code = 308
[docs]class BadRequest(ClientError):
"""Exception mapping a '400 Bad Request' response."""
code = 400
[docs]class Unauthorized(ClientError):
"""Exception mapping a '401 Unauthorized' response."""
code = 401
[docs]class MethodNotAllowed(ClientError):
"""Exception mapping a '405 Method Not Allowed' response."""
code = 405
[docs]class LengthRequired(ClientError):
"""Exception mapping a '411 Length Required' response."""
code = 411
[docs]class PreconditionFailed(ClientError):
"""Exception mapping a '412 Precondition Failed' response."""
code = 412
[docs]class RequestRangeNotSatisfiable(ClientError):
"""Exception mapping a '416 Request Range Not Satisfiable' response."""
code = 416
[docs]class TooManyRequests(ClientError):
"""Exception mapping a '429 Too Many Requests' response."""
code = 429
[docs]class InternalServerError(ServerError):
"""Exception mapping a '500 Internal Server Error' response."""
code = 500
[docs]class NotImplemented(ServerError):
"""Exception mapping a '501 Not Implemented' response."""
code = 501
[docs]def make_exception(response, content):
"""Factory: create exception based on HTTP response code.
:rtype: instance of :class:`GCloudError`, or a concrete subclass.
"""
if isinstance(content, str):
content = json.loads(content)
message = content.get('message')
error = content.get('error', {})
errors = error.get('errors', ())
try:
klass = _HTTP_CODE_TO_EXCEPTION[response.status]
except KeyError:
error = GCloudError(message, errors)
error.code = response.status
else:
error = klass(message, errors)
return error
def _walk_subclasses(klass):
"""Recursively walk subclass tree."""
for sub in klass.__subclasses__():
yield sub
for subsub in _walk_subclasses(sub):
yield subsub
# Build the code->exception class mapping.
for eklass in _walk_subclasses(GCloudError):
code = getattr(eklass, 'code', None)
if code is not None:
_HTTP_CODE_TO_EXCEPTION[code] = eklass