1. ホーム
  2. python

[解決済み] この例外はどのようにキャッチするのですか?

2022-04-15 16:26:37

質問

このコードはdjango/db/models/fields.pyにあり、例外を作成/定義していますか?

class ReverseSingleRelatedObjectDescriptor(six.with_metaclass(RenameRelatedObjectDescriptorMethods)):
    # This class provides the functionality that makes the related-object
    # managers available as attributes on a model class, for fields that have
    # a single "remote" value, on the class that defines the related field.
    # In the example "choice.poll", the poll attribute is a
    # ReverseSingleRelatedObjectDescriptor instance.
    def __init__(self, field_with_rel):
        self.field = field_with_rel
        self.cache_name = self.field.get_cache_name()

    @cached_property
    def RelatedObjectDoesNotExist(self):
        # The exception can't be created at initialization time since the
        # related model might not be resolved yet; `rel.to` might still be
        # a string model reference.
        return type(
            str('RelatedObjectDoesNotExist'),
            (self.field.rel.to.DoesNotExist, AttributeError),
            {}
        )

これは django/db/models/fields/related.py にあり、上で述べた例外を発生させます。

def __get__(self, instance, instance_type=None):
    if instance is None:
        return self
    try:
        rel_obj = getattr(instance, self.cache_name)
    except AttributeError:
        val = self.field.get_local_related_value(instance)
        if None in val:
            rel_obj = None
        else:
            params = dict(
                (rh_field.attname, getattr(instance, lh_field.attname))
                for lh_field, rh_field in self.field.related_fields)
            qs = self.get_queryset(instance=instance)
            extra_filter = self.field.get_extra_descriptor_filter(instance)
            if isinstance(extra_filter, dict):
                params.update(extra_filter)
                qs = qs.filter(**params)
            else:
                qs = qs.filter(extra_filter, **params)
            # Assuming the database enforces foreign keys, this won't fail.
            rel_obj = qs.get()
            if not self.field.rel.multiple:
                setattr(rel_obj, self.field.related.get_cache_name(), instance)
        setattr(instance, self.cache_name, rel_obj)
    if rel_obj is None and not self.field.null:
        raise self.RelatedObjectDoesNotExist(
            "%s has no %s." % (self.field.model.__name__, self.field.name)
        )
    else:
        return rel_obj

問題は、このコードです。

    try:
        val = getattr(obj, attr_name)
    except related.ReverseSingleRelatedObjectDescriptor.RelatedObjectDoesNotExist:
        val = None  # Does not catch the thrown exception
    except Exception as foo:
        print type(foo)  # Catches here, not above

はその例外をキャッチしません。

>>>print type(foo)
<class 'django.db.models.fields.related.RelatedObjectDoesNotExist'>
>>>isinstance(foo, related.FieldDoesNotExist)
False

そして

except related.RelatedObjectDoesNotExist:

を発生させます。 AttributeError: 'module' object has no attribute 'RelatedObjectDoesNotExist'

>>>isinstance(foo, related.ReverseSingleRelatedObjectDescriptor.RelatedObjectDoesNotExist)
Traceback (most recent call last):
  File "<string>", line 1, in <fragment>
TypeError: isinstance() arg 2 must be a class, type, or tuple of classes and types

ということなのでしょう。

解決方法は?

関連するモデルの名前が Foo であれば、次のようにすればよい。

except Foo.DoesNotExist:

Djangoは、恐ろしくないときは素晴らしい。 RelatedObjectDoesNotExist は、実行時に動的に把握される型を返すプロパティです。その型は self.field.rel.to.DoesNotExist を基底クラスとする。

Django のドキュメントによると。

存在しない(DoesNotExist

例外 Model.DoesNotExist

この例外は、期待されたオブジェクトが存在しない場合に ORM によって発生します。 が見つかりました。例えば QuerySet.get() はオブジェクト は、与えられたルックアップのために見つかります。

Djangoでは DoesNotExist 例外の属性として 各モデルクラスで、見つからなかったオブジェクトのクラスを識別するために が見つかるので、特定のモデルクラスの例外をキャッチすることができます。

のサブクラスで、例外は django.core.exceptions.ObjectDoesNotExist .

それを実現するためのマジックです。モデルを作り上げたら self.field.rel.to.DoesNotExist はそのモデルに対するdoes-not-existの例外です。