@mytec: pushed back before 1.1
This commit is contained in:
436
backend/venv/lib/python3.12/site-packages/pymongo/errors.py
Normal file
436
backend/venv/lib/python3.12/site-packages/pymongo/errors.py
Normal file
@@ -0,0 +1,436 @@
|
||||
# Copyright 2009-present MongoDB, Inc.
|
||||
#
|
||||
# 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
|
||||
#
|
||||
# https://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.
|
||||
|
||||
"""Exceptions raised by PyMongo.
|
||||
|
||||
.. seealso:: This module is compatible with both the synchronous and asynchronous PyMongo APIs.
|
||||
"""
|
||||
from __future__ import annotations
|
||||
|
||||
from ssl import SSLCertVerificationError as _CertificateError # noqa: F401
|
||||
from typing import TYPE_CHECKING, Any, Iterable, Mapping, Optional, Sequence, Union
|
||||
|
||||
from bson.errors import InvalidDocument
|
||||
|
||||
if TYPE_CHECKING:
|
||||
from pymongo.results import ClientBulkWriteResult
|
||||
from pymongo.typings import _DocumentOut
|
||||
|
||||
|
||||
class PyMongoError(Exception):
|
||||
"""Base class for all PyMongo exceptions."""
|
||||
|
||||
def __init__(self, message: str = "", error_labels: Optional[Iterable[str]] = None) -> None:
|
||||
super().__init__(message)
|
||||
self._message = message
|
||||
self._error_labels = set(error_labels or [])
|
||||
|
||||
def has_error_label(self, label: str) -> bool:
|
||||
"""Return True if this error contains the given label.
|
||||
|
||||
.. versionadded:: 3.7
|
||||
"""
|
||||
return label in self._error_labels
|
||||
|
||||
def _add_error_label(self, label: str) -> None:
|
||||
"""Add the given label to this error."""
|
||||
self._error_labels.add(label)
|
||||
|
||||
def _remove_error_label(self, label: str) -> None:
|
||||
"""Remove the given label from this error."""
|
||||
self._error_labels.discard(label)
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
"""True if this error was caused by a timeout.
|
||||
|
||||
.. versionadded:: 4.2
|
||||
"""
|
||||
return False
|
||||
|
||||
|
||||
class ProtocolError(PyMongoError):
|
||||
"""Raised for failures related to the wire protocol."""
|
||||
|
||||
|
||||
class ConnectionFailure(PyMongoError):
|
||||
"""Raised when a connection to the database cannot be made or is lost."""
|
||||
|
||||
|
||||
class WaitQueueTimeoutError(ConnectionFailure):
|
||||
"""Raised when an operation times out waiting to checkout a connection from the pool.
|
||||
|
||||
Subclass of :exc:`~pymongo.errors.ConnectionFailure`.
|
||||
|
||||
.. versionadded:: 4.2
|
||||
"""
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
return True
|
||||
|
||||
|
||||
class AutoReconnect(ConnectionFailure):
|
||||
"""Raised when a connection to the database is lost and an attempt to
|
||||
auto-reconnect will be made.
|
||||
|
||||
In order to auto-reconnect you must handle this exception, recognizing that
|
||||
the operation which caused it has not necessarily succeeded. Future
|
||||
operations will attempt to open a new connection to the database (and
|
||||
will continue to raise this exception until the first successful
|
||||
connection is made).
|
||||
|
||||
Subclass of :exc:`~pymongo.errors.ConnectionFailure`.
|
||||
"""
|
||||
|
||||
errors: Union[Mapping[str, Any], Sequence[Any]]
|
||||
details: Union[Mapping[str, Any], Sequence[Any]]
|
||||
|
||||
def __init__(
|
||||
self, message: str = "", errors: Optional[Union[Mapping[str, Any], Sequence[Any]]] = None
|
||||
) -> None:
|
||||
error_labels = None
|
||||
if errors is not None:
|
||||
if isinstance(errors, dict):
|
||||
error_labels = errors.get("errorLabels")
|
||||
super().__init__(message, error_labels)
|
||||
self.errors = self.details = errors or []
|
||||
|
||||
|
||||
class NetworkTimeout(AutoReconnect):
|
||||
"""An operation on an open connection exceeded socketTimeoutMS.
|
||||
|
||||
The remaining connections in the pool stay open. In the case of a write
|
||||
operation, you cannot know whether it succeeded or failed.
|
||||
|
||||
Subclass of :exc:`~pymongo.errors.AutoReconnect`.
|
||||
"""
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
return True
|
||||
|
||||
|
||||
def _format_detailed_error(
|
||||
message: str, details: Optional[Union[Mapping[str, Any], list[Any]]]
|
||||
) -> str:
|
||||
if details is not None:
|
||||
message = f"{message}, full error: {details}"
|
||||
return message
|
||||
|
||||
|
||||
class NotPrimaryError(AutoReconnect):
|
||||
"""The server responded "not primary" or "node is recovering".
|
||||
|
||||
These errors result from a query, write, or command. The operation failed
|
||||
because the client thought it was using the primary but the primary has
|
||||
stepped down, or the client thought it was using a healthy secondary but
|
||||
the secondary is stale and trying to recover.
|
||||
|
||||
The client launches a refresh operation on a background thread, to update
|
||||
its view of the server as soon as possible after throwing this exception.
|
||||
|
||||
Subclass of :exc:`~pymongo.errors.AutoReconnect`.
|
||||
|
||||
.. versionadded:: 3.12
|
||||
"""
|
||||
|
||||
def __init__(
|
||||
self, message: str = "", errors: Optional[Union[Mapping[str, Any], list[Any]]] = None
|
||||
) -> None:
|
||||
super().__init__(_format_detailed_error(message, errors), errors=errors)
|
||||
|
||||
|
||||
class ServerSelectionTimeoutError(AutoReconnect):
|
||||
"""Thrown when no MongoDB server is available for an operation
|
||||
|
||||
If there is no suitable server for an operation PyMongo tries for
|
||||
``serverSelectionTimeoutMS`` (default 30 seconds) to find one, then
|
||||
throws this exception. For example, it is thrown after attempting an
|
||||
operation when PyMongo cannot connect to any server, or if you attempt
|
||||
an insert into a replica set that has no primary and does not elect one
|
||||
within the timeout window, or if you attempt to query with a Read
|
||||
Preference that the replica set cannot satisfy.
|
||||
"""
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
return True
|
||||
|
||||
|
||||
class ConfigurationError(PyMongoError):
|
||||
"""Raised when something is incorrectly configured."""
|
||||
|
||||
|
||||
class OperationFailure(PyMongoError):
|
||||
"""Raised when a database operation fails.
|
||||
|
||||
.. versionadded:: 2.7
|
||||
The :attr:`details` attribute.
|
||||
"""
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
error: str,
|
||||
code: Optional[int] = None,
|
||||
details: Optional[Mapping[str, Any]] = None,
|
||||
max_wire_version: Optional[int] = None,
|
||||
) -> None:
|
||||
error_labels = None
|
||||
if details is not None:
|
||||
error_labels = details.get("errorLabels")
|
||||
super().__init__(_format_detailed_error(error, details), error_labels=error_labels)
|
||||
self.__code = code
|
||||
self.__details = details
|
||||
self.__max_wire_version = max_wire_version
|
||||
|
||||
@property
|
||||
def _max_wire_version(self) -> Optional[int]:
|
||||
return self.__max_wire_version
|
||||
|
||||
@property
|
||||
def code(self) -> Optional[int]:
|
||||
"""The error code returned by the server, if any."""
|
||||
return self.__code
|
||||
|
||||
@property
|
||||
def details(self) -> Optional[Mapping[str, Any]]:
|
||||
"""The complete error document returned by the server.
|
||||
|
||||
Depending on the error that occurred, the error document
|
||||
may include useful information beyond just the error
|
||||
message. When connected to a mongos the error document
|
||||
may contain one or more subdocuments if errors occurred
|
||||
on multiple shards.
|
||||
"""
|
||||
return self.__details
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
return self.__code in (50,)
|
||||
|
||||
|
||||
class CursorNotFound(OperationFailure):
|
||||
"""Raised while iterating query results if the cursor is
|
||||
invalidated on the server.
|
||||
|
||||
.. versionadded:: 2.7
|
||||
"""
|
||||
|
||||
|
||||
class ExecutionTimeout(OperationFailure):
|
||||
"""Raised when a database operation times out, exceeding the $maxTimeMS
|
||||
set in the query or command option.
|
||||
|
||||
.. note:: Requires server version **>= 2.6.0**
|
||||
|
||||
.. versionadded:: 2.7
|
||||
"""
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
return True
|
||||
|
||||
|
||||
class WriteConcernError(OperationFailure):
|
||||
"""Base exception type for errors raised due to write concern.
|
||||
|
||||
.. versionadded:: 3.0
|
||||
"""
|
||||
|
||||
|
||||
class WriteError(OperationFailure):
|
||||
"""Base exception type for errors raised during write operations.
|
||||
|
||||
.. versionadded:: 3.0
|
||||
"""
|
||||
|
||||
|
||||
class WTimeoutError(WriteConcernError):
|
||||
"""Raised when a database operation times out (i.e. wtimeout expires)
|
||||
before replication completes.
|
||||
|
||||
With newer versions of MongoDB the `details` attribute may include
|
||||
write concern fields like 'n', 'updatedExisting', or 'writtenTo'.
|
||||
|
||||
.. versionadded:: 2.7
|
||||
"""
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
return True
|
||||
|
||||
|
||||
class DuplicateKeyError(WriteError):
|
||||
"""Raised when an insert or update fails due to a duplicate key error."""
|
||||
|
||||
|
||||
def _wtimeout_error(error: Any) -> bool:
|
||||
"""Return True if this writeConcernError doc is a caused by a timeout."""
|
||||
return error.get("code") == 50 or ("errInfo" in error and error["errInfo"].get("wtimeout"))
|
||||
|
||||
|
||||
class BulkWriteError(OperationFailure):
|
||||
"""Exception class for bulk write errors.
|
||||
|
||||
.. versionadded:: 2.7
|
||||
"""
|
||||
|
||||
details: _DocumentOut
|
||||
|
||||
def __init__(self, results: _DocumentOut) -> None:
|
||||
super().__init__("batch op errors occurred", 65, results)
|
||||
|
||||
def __reduce__(self) -> tuple[Any, Any]:
|
||||
return self.__class__, (self.details,)
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
# Check the last writeConcernError and last writeError to determine if this
|
||||
# BulkWriteError was caused by a timeout.
|
||||
wces = self.details.get("writeConcernErrors", [])
|
||||
if wces and _wtimeout_error(wces[-1]):
|
||||
return True
|
||||
|
||||
werrs = self.details.get("writeErrors", [])
|
||||
if werrs and werrs[-1].get("code") == 50:
|
||||
return True
|
||||
return False
|
||||
|
||||
|
||||
class ClientBulkWriteException(OperationFailure):
|
||||
"""Exception class for client-level bulk write errors."""
|
||||
|
||||
details: _DocumentOut
|
||||
verbose: bool
|
||||
|
||||
def __init__(self, results: _DocumentOut, verbose: bool) -> None:
|
||||
super().__init__("batch op errors occurred", 65, results)
|
||||
self.verbose = verbose
|
||||
|
||||
def __reduce__(self) -> tuple[Any, Any]:
|
||||
return self.__class__, (self.details,)
|
||||
|
||||
@property
|
||||
def error(self) -> Optional[Any]:
|
||||
"""A top-level error that occurred when attempting to
|
||||
communicate with the server or execute the bulk write.
|
||||
|
||||
This value may not be populated if the exception was
|
||||
thrown due to errors occurring on individual writes.
|
||||
"""
|
||||
return self.details.get("error", None)
|
||||
|
||||
@property
|
||||
def write_concern_errors(self) -> Optional[list[WriteConcernError]]:
|
||||
"""Write concern errors that occurred during the bulk write.
|
||||
|
||||
This list may have multiple items if more than one
|
||||
server command was required to execute the bulk write.
|
||||
"""
|
||||
return self.details.get("writeConcernErrors", [])
|
||||
|
||||
@property
|
||||
def write_errors(self) -> Optional[list[WriteError]]:
|
||||
"""Errors that occurred during the execution of individual write operations.
|
||||
|
||||
This list will contain at most one entry if the bulk write was ordered.
|
||||
"""
|
||||
return self.details.get("writeErrors", {})
|
||||
|
||||
@property
|
||||
def partial_result(self) -> Optional[ClientBulkWriteResult]:
|
||||
"""The results of any successful operations that were
|
||||
performed before the error was encountered.
|
||||
"""
|
||||
from pymongo.results import ClientBulkWriteResult
|
||||
|
||||
if self.details.get("anySuccessful"):
|
||||
return ClientBulkWriteResult(
|
||||
self.details, # type: ignore[arg-type]
|
||||
acknowledged=True,
|
||||
has_verbose_results=self.verbose,
|
||||
)
|
||||
return None
|
||||
|
||||
|
||||
class InvalidOperation(PyMongoError):
|
||||
"""Raised when a client attempts to perform an invalid operation."""
|
||||
|
||||
|
||||
class InvalidName(PyMongoError):
|
||||
"""Raised when an invalid name is used."""
|
||||
|
||||
|
||||
class CollectionInvalid(PyMongoError):
|
||||
"""Raised when collection validation fails."""
|
||||
|
||||
|
||||
class InvalidURI(ConfigurationError):
|
||||
"""Raised when trying to parse an invalid mongodb URI."""
|
||||
|
||||
|
||||
class DocumentTooLarge(InvalidDocument):
|
||||
"""Raised when an encoded document is too large for the connected server."""
|
||||
|
||||
|
||||
class EncryptionError(PyMongoError):
|
||||
"""Raised when encryption or decryption fails.
|
||||
|
||||
This error always wraps another exception which can be retrieved via the
|
||||
:attr:`cause` property.
|
||||
|
||||
.. versionadded:: 3.9
|
||||
"""
|
||||
|
||||
def __init__(self, cause: Exception) -> None:
|
||||
super().__init__(str(cause))
|
||||
self.__cause = cause
|
||||
|
||||
@property
|
||||
def cause(self) -> Exception:
|
||||
"""The exception that caused this encryption or decryption error."""
|
||||
return self.__cause
|
||||
|
||||
@property
|
||||
def timeout(self) -> bool:
|
||||
if isinstance(self.__cause, PyMongoError):
|
||||
return self.__cause.timeout
|
||||
return False
|
||||
|
||||
|
||||
class EncryptedCollectionError(EncryptionError):
|
||||
"""Raised when creating a collection with encrypted_fields fails.
|
||||
|
||||
.. versionadded:: 4.4
|
||||
"""
|
||||
|
||||
def __init__(self, cause: Exception, encrypted_fields: Mapping[str, Any]) -> None:
|
||||
super().__init__(cause)
|
||||
self.__encrypted_fields = encrypted_fields
|
||||
|
||||
@property
|
||||
def encrypted_fields(self) -> Mapping[str, Any]:
|
||||
"""The encrypted_fields document that allows inferring which data keys are *known* to be created.
|
||||
|
||||
Note that the returned document is not guaranteed to contain information about *all* of the data keys that
|
||||
were created, for example in the case of an indefinite error like a timeout. Use the `cause` property to
|
||||
determine whether a definite or indefinite error caused this error, and only rely on the accuracy of the
|
||||
encrypted_fields if the error is definite.
|
||||
"""
|
||||
return self.__encrypted_fields
|
||||
|
||||
|
||||
class _OperationCancelled(AutoReconnect):
|
||||
"""Internal error raised when a socket operation is cancelled."""
|
||||
Reference in New Issue
Block a user