관리-도구
편집 파일: models.py
"""Models.""" from astroid import Const from astroid.nodes import Assign, AssignName, ClassDef, FunctionDef from pylint.checkers import BaseChecker from pylint.checkers.utils import check_messages from pylint.interfaces import IAstroidChecker from pylint_django.__pkginfo__ import BASE_ID from pylint_django.utils import PY3, node_is_subclass MESSAGES = { f"E{BASE_ID}01": ( "__unicode__ on a model must be callable (%s)", "model-unicode-not-callable", "Django models require a callable __unicode__ method", ), f"W{BASE_ID}01": ( "No __unicode__ method on model (%s)", "model-missing-unicode", "Django models should implement a __unicode__ method for string representation", ), f"W{BASE_ID}02": ( "Found __unicode__ method on model (%s). Python3 uses __str__.", "model-has-unicode", "Django models should not implement a __unicode__ method for string representation when using Python3", ), f"W{BASE_ID}03": ( "Model does not explicitly define __unicode__ (%s)", "model-no-explicit-unicode", "Django models should implement a __unicode__ method for string representation. " "A parent class of this model does, but ideally all models should be explicit.", ), } def _is_meta_with_abstract(node): if isinstance(node, ClassDef) and node.name == "Meta": for meta_child in node.get_children(): if not isinstance(meta_child, Assign): continue if not meta_child.targets[0].name == "abstract": continue if not isinstance(meta_child.value, Const): continue # TODO: handle tuple assignment? # eg: # abstract, something_else = True, 1 if meta_child.value.value: # this class is abstract return True return False def _has_python_2_unicode_compatible_decorator(node): if node.decorators is None: return False for decorator in node.decorators.nodes: if getattr(decorator, "name", None) == "python_2_unicode_compatible": return True return False def _is_unicode_or_str_in_python_2_compatibility(method): if method.name == "__unicode__": return True if method.name == "__str__" and _has_python_2_unicode_compatible_decorator(method.parent): return True return False class ModelChecker(BaseChecker): """Django model checker.""" __implements__ = IAstroidChecker name = "django-model-checker" msgs = MESSAGES @check_messages("model-missing-unicode") def visit_classdef(self, node): """Class visitor.""" if not node_is_subclass(node, "django.db.models.base.Model", ".Model"): # we only care about models return for child in node.get_children(): if _is_meta_with_abstract(child): return if isinstance(child, Assign): grandchildren = list(child.get_children()) if not isinstance(grandchildren[0], AssignName): continue name = grandchildren[0].name if name != "__unicode__": continue grandchild = grandchildren[1] assigned = grandchild.inferred()[0] if assigned.callable(): return self.add_message(f"E{BASE_ID}01", args=node.name, node=node) return if isinstance(child, FunctionDef) and child.name == "__unicode__": if PY3: self.add_message(f"W{BASE_ID}02", args=node.name, node=node) return # if we get here, then we have no __unicode__ method directly on the class itself # a different warning is emitted if a parent declares __unicode__ for method in node.methods(): if method.parent != node and _is_unicode_or_str_in_python_2_compatibility(method): # this happens if a parent declares the unicode method but # this node does not self.add_message(f"W{BASE_ID}03", args=node.name, node=node) return # if the Django compatibility decorator is used then we don't emit a warning # see https://github.com/PyCQA/pylint-django/issues/10 if _has_python_2_unicode_compatible_decorator(node): return if PY3: return self.add_message(f"W{BASE_ID}01", args=node.name, node=node)