diff --git a/django-stubs/contrib/admin/models.pyi b/django-stubs/contrib/admin/models.pyi index 3915e02a7..a3fd44a66 100644 --- a/django-stubs/contrib/admin/models.pyi +++ b/django-stubs/contrib/admin/models.pyi @@ -4,6 +4,7 @@ from uuid import UUID from django.db import models from django.db.models import QuerySet from django.db.models.base import Model +from typing_extensions import deprecated ADDITION: int CHANGE: int @@ -11,6 +12,7 @@ DELETION: int ACTION_FLAG_CHOICES: Any class LogEntryManager(models.Manager[LogEntry]): + @deprecated("log_action() is deprecated and will be removed in Django 6.0. Use log_action_new() instead.") def log_action( self, user_id: int, diff --git a/django-stubs/contrib/admin/options.pyi b/django-stubs/contrib/admin/options.pyi index ce6eaed75..02e172dc0 100644 --- a/django-stubs/contrib/admin/options.pyi +++ b/django-stubs/contrib/admin/options.pyi @@ -33,7 +33,7 @@ from django.urls.resolvers import URLPattern from django.utils.datastructures import _ListOrTuple from django.utils.functional import _StrOrPromise from django.utils.safestring import SafeString -from typing_extensions import Self, TypeAlias, TypedDict +from typing_extensions import Self, TypeAlias, TypedDict, deprecated IS_POPUP_VAR: str TO_FIELD_VAR: str @@ -129,6 +129,7 @@ class BaseModelAdmin(Generic[_ModelT]): def get_prepopulated_fields(self, request: HttpRequest, obj: _ModelT | None = ...) -> dict[str, Sequence[str]]: ... def get_queryset(self, request: HttpRequest) -> QuerySet[_ModelT]: ... def get_sortable_by(self, request: HttpRequest) -> _DisplayT[_ModelT]: ... + @deprecated("The None value for the request parameter will be removed in Django 6.0.") def lookup_allowed(self, lookup: str, value: str, request: HttpRequest | None = ...) -> bool: ... def to_field_allowed(self, request: HttpRequest, to_field: str) -> bool: ... def has_add_permission(self, request: HttpRequest) -> bool: ... @@ -205,7 +206,9 @@ class ModelAdmin(BaseModelAdmin[_ModelT]): ) -> Paginator: ... def log_addition(self, request: HttpRequest, obj: _ModelT, message: Any) -> LogEntry: ... def log_change(self, request: HttpRequest, obj: _ModelT, message: Any) -> LogEntry: ... + @deprecated("log_deletion() is deprecated and will be removed in Django 6.0. Use log_deletions() instead.") def log_deletion(self, request: HttpRequest, obj: _ModelT, object_repr: str) -> LogEntry: ... + def log_deletions(self, request: HttpRequest, queryset: QuerySet[_ModelT]) -> list[LogEntry] | LogEntry: ... def action_checkbox(self, obj: _ModelT) -> SafeString: ... def get_actions(self, request: HttpRequest) -> dict[str, tuple[Callable[..., str], str, str] | None]: ... def get_action_choices( diff --git a/django-stubs/contrib/contenttypes/fields.pyi b/django-stubs/contrib/contenttypes/fields.pyi index 27661d6f7..27d666715 100644 --- a/django-stubs/contrib/contenttypes/fields.pyi +++ b/django-stubs/contrib/contenttypes/fields.pyi @@ -13,6 +13,7 @@ from django.db.models.fields.reverse_related import ForeignObjectRel from django.db.models.query import QuerySet from django.db.models.query_utils import FilteredRelation, PathInfo from django.db.models.sql.where import WhereNode +from typing_extensions import deprecated class GenericForeignKey(FieldCacheMixin): # django-stubs implementation only fields @@ -45,6 +46,9 @@ class GenericForeignKey(FieldCacheMixin): def get_content_type( self, obj: Model | None = ..., id: int | None = ..., using: str | None = ..., model: type[Model] | None = ... ) -> ContentType: ... + @deprecated( + "get_prefetch_queryset() is deprecated and will be removed in Django 6.0. Use get_prefetch_querysets() instead." + ) def get_prefetch_queryset( self, instances: list[Model] | QuerySet, queryset: QuerySet | None = ... ) -> tuple[list[Model], Callable, Callable, bool, str, bool]: ... diff --git a/django-stubs/contrib/gis/gdal/geometries.pyi b/django-stubs/contrib/gis/gdal/geometries.pyi index dc6d2a12f..23aaeb245 100644 --- a/django-stubs/contrib/gis/gdal/geometries.pyi +++ b/django-stubs/contrib/gis/gdal/geometries.pyi @@ -45,6 +45,12 @@ class OGRGeometry(GDALBase): def empty(self) -> bool: ... @property def extent(self) -> tuple[float, float, float, float]: ... + @property + def is_3d(self) -> bool: ... + def set_3d(self, value: bool) -> None: ... + @property + def is_measured(self) -> bool: ... + def set_measured(self, value: bool) -> None: ... srid: int | None @property def geos(self) -> GEOSGeometry: ... @@ -84,6 +90,8 @@ class OGRGeometry(GDALBase): def intersection(self, other: OGRGeometry) -> OGRGeometry: ... def sym_difference(self, other: OGRGeometry) -> OGRGeometry: ... def union(self, other: OGRGeometry) -> OGRGeometry: ... + @property + def centroid(self) -> Point: ... class Point(OGRGeometry): @property diff --git a/django-stubs/contrib/gis/geoip2/base.pyi b/django-stubs/contrib/gis/geoip2/base.pyi index 56546e504..3f75eb18c 100644 --- a/django-stubs/contrib/gis/geoip2/base.pyi +++ b/django-stubs/contrib/gis/geoip2/base.pyi @@ -4,6 +4,7 @@ from typing import Any from django.contrib.gis.geos import Point from django.utils.functional import cached_property +from typing_extensions import deprecated GEOIP_SETTINGS: dict[str, Any] @@ -28,6 +29,7 @@ class GeoIP2: def is_city(self) -> bool: ... @cached_property def is_country(self) -> bool: ... + @deprecated("The coords() is deprecated and will be removed in Django 6.0. Use lon_lat() instead.") def coords(self, query: str, ordering: Sequence[str] = ...) -> tuple[float, float] | tuple[None, None]: ... def lon_lat(self, query: str) -> tuple[float, float] | tuple[None, None]: ... def lat_lon(self, query: str) -> tuple[float, float] | tuple[None, None]: ... @@ -35,4 +37,5 @@ class GeoIP2: @property def info(self) -> str: ... @classmethod + @deprecated("The open() is deprecated and will be removed in Django 6.0. Use GeoIP2() instead.") def open(cls, full_path: Path | str | None, cache: int) -> Any: ... diff --git a/django-stubs/core/files/storage/filesystem.pyi b/django-stubs/core/files/storage/filesystem.pyi index 31448c7bf..214eebf24 100644 --- a/django-stubs/core/files/storage/filesystem.pyi +++ b/django-stubs/core/files/storage/filesystem.pyi @@ -6,6 +6,7 @@ from .base import Storage from .mixins import StorageSettingsMixin class FileSystemStorage(_Deconstructible, Storage, StorageSettingsMixin): + # RemovedInDjango60Warning OS_OPEN_FLAGS: int def __init__( diff --git a/django-stubs/db/backends/base/operations.pyi b/django-stubs/db/backends/base/operations.pyi index c07186645..fc5eac0f8 100644 --- a/django-stubs/db/backends/base/operations.pyi +++ b/django-stubs/db/backends/base/operations.pyi @@ -13,6 +13,7 @@ from django.db.models.constants import OnConflict from django.db.models.expressions import Case, Col, Expression from django.db.models.fields import Field from django.db.models.sql.compiler import SQLCompiler +from typing_extensions import deprecated class BaseDatabaseOperations: compiler_module: str @@ -44,6 +45,7 @@ class BaseDatabaseOperations: def deferrable_sql(self) -> str: ... def distinct_sql(self, fields: list[str], params: list[Any] | None) -> tuple[list[str], list[str]]: ... def fetch_returned_insert_columns(self, cursor: Any, returning_params: Any) -> Any: ... + @deprecated("The field_cast_sql() is deprecated and will be removed in Django 6.0. Use lookup_cast() instead.") def field_cast_sql(self, db_type: str | None, internal_type: str) -> str: ... def force_no_ordering(self) -> list[Any]: ... def for_update_sql( diff --git a/django-stubs/db/models/enums.pyi b/django-stubs/db/models/enums.pyi index 1393c4517..a13dc7c81 100644 --- a/django-stubs/db/models/enums.pyi +++ b/django-stubs/db/models/enums.pyi @@ -4,7 +4,7 @@ from typing import Any, TypeVar, overload, type_check_only from _typeshed import ConvertibleToInt from django.utils.functional import _StrOrPromise -from typing_extensions import TypeAlias +from typing_extensions import TypeAlias, deprecated _Self = TypeVar("_Self") @@ -21,6 +21,7 @@ else: class IntEnum(int, ReprEnum): ... class StrEnum(str, ReprEnum): ... +@deprecated("ChoicesMeta is deprecated in favor of ChoicesType and will be removed in Django 6.0.") class ChoicesMeta(EnumType): # There's a contradiction between mypy and PYI019 regarding metaclasses. Where mypy # disallows 'typing_extensions.Self' on metaclasses, while PYI019 try to enforce diff --git a/django-stubs/db/models/fields/mixins.pyi b/django-stubs/db/models/fields/mixins.pyi index a77d09fb5..fe6aaa2af 100644 --- a/django-stubs/db/models/fields/mixins.pyi +++ b/django-stubs/db/models/fields/mixins.pyi @@ -2,12 +2,14 @@ from typing import Any from django.db.models.base import Model from django.utils.functional import cached_property +from typing_extensions import deprecated NOT_PROVIDED: Any class FieldCacheMixin: def get_cache_name(self) -> str: ... @cached_property + @deprecated("The cache_name() is deprecated and will be removed in Django 6.0. Use get_cache_name() instead.") def cache_name(self) -> str: ... def get_cached_value(self, instance: Model, default: Any = ...) -> Model | None: ... def is_cached(self, instance: Model) -> bool: ... diff --git a/django-stubs/db/models/fields/related_descriptors.pyi b/django-stubs/db/models/fields/related_descriptors.pyi index 0d5782a32..94a970c1f 100644 --- a/django-stubs/db/models/fields/related_descriptors.pyi +++ b/django-stubs/db/models/fields/related_descriptors.pyi @@ -10,7 +10,7 @@ from django.db.models.manager import BaseManager, Manager from django.db.models.query import QuerySet from django.db.models.query_utils import DeferredAttribute from django.utils.functional import cached_property -from typing_extensions import Self +from typing_extensions import Self, deprecated _M = TypeVar("_M", bound=Model) _F = TypeVar("_F", bound=Field) @@ -28,6 +28,9 @@ class ForwardManyToOneDescriptor(Generic[_F]): def RelatedObjectDoesNotExist(self) -> type[ObjectDoesNotExist]: ... def is_cached(self, instance: Model) -> bool: ... def get_queryset(self, **hints: Any) -> QuerySet[Any]: ... + @deprecated( + "get_prefetch_queryset() is deprecated and will be removed in Django 6.0. Use get_prefetch_querysets() instead." + ) def get_prefetch_queryset( self, instances: list[Model], queryset: QuerySet[Any] | None = None ) -> tuple[QuerySet[Any], Callable[..., Any], Callable[..., Any], bool, str, bool]: ... @@ -60,6 +63,9 @@ class ReverseOneToOneDescriptor(Generic[_From, _To]): def RelatedObjectDoesNotExist(self) -> type[ObjectDoesNotExist]: ... def is_cached(self, instance: _From) -> bool: ... def get_queryset(self, **hints: Any) -> QuerySet[_To]: ... + @deprecated( + "get_prefetch_queryset() is deprecated and will be removed in Django 6.0. Use get_prefetch_querysets() instead." + ) def get_prefetch_queryset( self, instances: list[_From], queryset: QuerySet[_To] | None = None ) -> tuple[QuerySet[_To], Callable[..., Any], Callable[..., Any], bool, str, bool]: ... diff --git a/django-stubs/db/models/fields/reverse_related.pyi b/django-stubs/db/models/fields/reverse_related.pyi index 3db9e24df..2a7961de1 100644 --- a/django-stubs/db/models/fields/reverse_related.pyi +++ b/django-stubs/db/models/fields/reverse_related.pyi @@ -8,6 +8,7 @@ from django.db.models.lookups import Lookup, StartsWith, Transform from django.db.models.query_utils import FilteredRelation, PathInfo from django.db.models.sql.where import WhereNode from django.utils.functional import cached_property +from typing_extensions import deprecated from .mixins import FieldCacheMixin @@ -77,6 +78,9 @@ class ForeignObjectRel(FieldCacheMixin): ordering: Sequence[str] = (), ) -> _ChoicesList: ... def is_hidden(self) -> bool: ... + @deprecated( + "The get_joining_columns() is deprecated and will be removed in Django 6.0. Use get_joining_fields() instead." + ) def get_joining_columns(self) -> tuple: ... def get_joining_fields(self) -> tuple[tuple[Field, Field], ...]: ... def get_extra_restriction( diff --git a/django-stubs/db/models/query.pyi b/django-stubs/db/models/query.pyi index 7822e021f..d6d27f213 100644 --- a/django-stubs/db/models/query.pyi +++ b/django-stubs/db/models/query.pyi @@ -8,7 +8,7 @@ from django.db.models.base import Model from django.db.models.expressions import Combinable, OrderBy from django.db.models.sql.query import Query, RawQuery from django.utils.functional import cached_property -from typing_extensions import Self, TypeAlias, TypeVar +from typing_extensions import Self, TypeAlias, TypeVar, deprecated _T = TypeVar("_T", covariant=True) _Model = TypeVar("_Model", bound=Model, covariant=True) @@ -240,6 +240,9 @@ class Prefetch: def add_prefix(self, prefix: str) -> None: ... def get_current_prefetch_to(self, level: int) -> str: ... def get_current_to_attr(self, level: int) -> tuple[str, str]: ... + @deprecated( + "The get_current_queryset() is deprecated and will be removed in Django 6.0. Use get_current_querysets() instead." + ) def get_current_queryset(self, level: int) -> QuerySet | None: ... def get_current_querysets(self, level: int) -> list[QuerySet] | None: ... diff --git a/django-stubs/forms/renderers.pyi b/django-stubs/forms/renderers.pyi index 9b6d4ee46..a4ed0a296 100644 --- a/django-stubs/forms/renderers.pyi +++ b/django-stubs/forms/renderers.pyi @@ -6,6 +6,7 @@ from django.template.backends.django import DjangoTemplates as DjangoTemplatesR from django.template.backends.jinja2 import Jinja2 as Jinja2R from django.template.base import Template from django.utils.functional import cached_property +from typing_extensions import deprecated def get_default_renderer() -> BaseRenderer: ... @@ -28,6 +29,9 @@ class Jinja2(EngineMixin, BaseRenderer): @cached_property def backend(self) -> type[Jinja2R]: ... +@deprecated( + "The Jinja2DivFormRenderer transitional form renderer is deprecated and will be removed in Django 6.0. Use Jinja2 instead." +) class Jinja2DivFormRenderer(Jinja2): form_template_name: str formset_template_name: str diff --git a/django-stubs/urls/base.pyi b/django-stubs/urls/base.pyi index e2b8dd904..7d472a722 100644 --- a/django-stubs/urls/base.pyi +++ b/django-stubs/urls/base.pyi @@ -1,12 +1,12 @@ from collections.abc import Callable, Sequence from typing import Any, Literal -from django.http.response import HttpResponse +from django.http.response import HttpResponseBase from django.urls.resolvers import ResolverMatch def resolve(path: str, urlconf: str | None = ...) -> ResolverMatch: ... def reverse( - viewname: Callable[..., HttpResponse] | str | None, + viewname: Callable[..., HttpResponseBase] | str | None, urlconf: str | None = ..., args: Sequence[Any] | None = ..., kwargs: dict[str, Any] | None = ..., diff --git a/django-stubs/utils/html.pyi b/django-stubs/utils/html.pyi index c5ce522e5..38e059cff 100644 --- a/django-stubs/utils/html.pyi +++ b/django-stubs/utils/html.pyi @@ -8,6 +8,7 @@ from typing import Any from _typeshed import Incomplete from django.utils.functional import SimpleLazyObject, _StrOrPromise from django.utils.safestring import SafeData, SafeString +from typing_extensions import deprecated VOID_ELEMENTS: frozenset[str] MAX_URL_LENGTH: int @@ -19,6 +20,7 @@ def json_script(value: Any, element_id: str | None = None, encoder: type[JSONEnc # conditional_escape could use a protocol to be more precise, see https://github.com/typeddjango/django-stubs/issues/1474 def conditional_escape(text: _StrOrPromise | SafeData) -> SafeString: ... +@deprecated("Calling format_html() without passing args or kwargs is deprecated.") def format_html(format_string: str, *args: Any, **kwargs: Any) -> SafeString: ... def format_html_join(sep: str, format_string: str, args_generator: Iterable[Iterable[Any]]) -> SafeString: ... def linebreaks(value: Any, autoescape: bool = False) -> str: ... diff --git a/django-stubs/utils/itercompat.pyi b/django-stubs/utils/itercompat.pyi index 5c15fee5d..379e86037 100644 --- a/django-stubs/utils/itercompat.pyi +++ b/django-stubs/utils/itercompat.pyi @@ -1,3 +1,8 @@ from typing import Any +from typing_extensions import deprecated + +@deprecated( + "The django.utils.itercompat.is_iterable() is deprecated and will be removed in Django 6.0. Use isinstance(..., collections.abc.Iterable) instead." +) def is_iterable(x: Any) -> bool: ... diff --git a/scripts/stubtest/allowlist_todo_django51.txt b/scripts/stubtest/allowlist_todo_django51.txt index 4a9261c42..595cac4ab 100644 --- a/scripts/stubtest/allowlist_todo_django51.txt +++ b/scripts/stubtest/allowlist_todo_django51.txt @@ -3,10 +3,8 @@ # Only discrepancies that appeared after Django 5.0 -> 5.1 update. # Unsorted: there are real problems and things we can really ignore: -django.contrib.admin.ModelAdmin.log_deletions django.contrib.admin.helpers.Fieldset.is_collapsible django.contrib.admin.helpers.InlineAdminFormSet.is_collapsible -django.contrib.admin.options.ModelAdmin.log_deletions django.contrib.auth.base_user.BaseUserManager.make_random_password django.contrib.auth.checks.check_middleware django.contrib.auth.forms.AdminPasswordChangeForm.clean_password2 @@ -24,7 +22,6 @@ django.contrib.auth.models.BaseUserManager.make_random_password django.contrib.contenttypes.fields.GenericForeignKey.cache_name django.contrib.contenttypes.fields.GenericForeignKey.get_attname_column django.contrib.contenttypes.models.ContentType.get_object_for_this_type -django.contrib.gis.admin.ModelAdmin.log_deletions django.contrib.gis.db.backends.mysql.operations.MySQLOperations.collect django.contrib.gis.db.models.CharField.slice_expression django.contrib.gis.db.models.CheckConstraint.__init__ @@ -45,17 +42,7 @@ django.contrib.gis.db.models.functions.FromWKB.output_field django.contrib.gis.db.models.functions.FromWKT.output_field django.contrib.gis.forms.ModelChoiceField.validate_no_null_characters django.contrib.gis.forms.RadioSelect.__slotnames__ -django.contrib.gis.gdal.OGRGeometry.centroid -django.contrib.gis.gdal.OGRGeometry.is_3d -django.contrib.gis.gdal.OGRGeometry.is_measured -django.contrib.gis.gdal.OGRGeometry.set_3d -django.contrib.gis.gdal.OGRGeometry.set_measured django.contrib.gis.gdal.geometries.LineString.m -django.contrib.gis.gdal.geometries.OGRGeometry.centroid -django.contrib.gis.gdal.geometries.OGRGeometry.is_3d -django.contrib.gis.gdal.geometries.OGRGeometry.is_measured -django.contrib.gis.gdal.geometries.OGRGeometry.set_3d -django.contrib.gis.gdal.geometries.OGRGeometry.set_measured django.contrib.gis.gdal.geometries.Point.m django.contrib.gis.gdal.prototypes.ds.GDAL_OF_ALL django.contrib.gis.gdal.prototypes.ds.GDAL_OF_RASTER