Skip to content

Commit

Permalink
feat: Adding reason, domain, metadata & error_details fields in Custo…
Browse files Browse the repository at this point in the history
…m Exceptions for additional info (#804)

* feat: Adding reason, domain, metadata & error_details fields in DBAPI custom exceptions.

* linting

* docs: Updating function docs

Co-authored-by: Astha Mohta <[email protected]>
  • Loading branch information
gauravpurohit06 and asthamohta authored Sep 26, 2022
1 parent d88b37f commit 2a74060
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 4 deletions.
6 changes: 3 additions & 3 deletions google/cloud/spanner_dbapi/cursor.py
Original file line number Diff line number Diff line change
Expand Up @@ -281,11 +281,11 @@ def execute(self, sql, args=None):
self._do_execute_update, sql, args or None
)
except (AlreadyExists, FailedPrecondition, OutOfRange) as e:
raise IntegrityError(getattr(e, "details", e))
raise IntegrityError(getattr(e, "details", e)) from e
except InvalidArgument as e:
raise ProgrammingError(getattr(e, "details", e))
raise ProgrammingError(getattr(e, "details", e)) from e
except InternalServerError as e:
raise OperationalError(getattr(e, "details", e))
raise OperationalError(getattr(e, "details", e)) from e

@check_not_closed
def executemany(self, operation, seq_of_params):
Expand Down
62 changes: 61 additions & 1 deletion google/cloud/spanner_dbapi/exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,8 @@

"""Spanner DB API exceptions."""

from google.api_core.exceptions import GoogleAPICallError


class Warning(Exception):
"""Important DB API warning."""
Expand All @@ -27,7 +29,65 @@ class Error(Exception):
Does not include :class:`Warning`.
"""

pass
def _is_error_cause_instance_of_google_api_exception(self):
return isinstance(self.__cause__, GoogleAPICallError)

@property
def reason(self):
"""The reason of the error.
Reference:
https://cloud.google.com/apis/design/errors#error_info
Returns:
Union[str, None]: An optional string containing reason of the error.
"""
return (
self.__cause__.reason
if self._is_error_cause_instance_of_google_api_exception()
else None
)

@property
def domain(self):
"""The logical grouping to which the "reason" belongs.
Reference:
https://cloud.google.com/apis/design/errors#error_info
Returns:
Union[str, None]: An optional string containing a logical grouping to which the "reason" belongs.
"""
return (
self.__cause__.domain
if self._is_error_cause_instance_of_google_api_exception()
else None
)

@property
def metadata(self):
"""Additional structured details about this error.
Reference:
https://cloud.google.com/apis/design/errors#error_info
Returns:
Union[Dict[str, str], None]: An optional object containing structured details about the error.
"""
return (
self.__cause__.metadata
if self._is_error_cause_instance_of_google_api_exception()
else None
)

@property
def details(self):
"""Information contained in google.rpc.status.details.
Reference:
https://cloud.google.com/apis/design/errors#error_model
https://cloud.google.com/apis/design/errors#error_details
Returns:
Sequence[Any]: A list of structured objects from error_details.proto
"""
return (
self.__cause__.details
if self._is_error_cause_instance_of_google_api_exception()
else None
)


class InterfaceError(Error):
Expand Down

0 comments on commit 2a74060

Please sign in to comment.