Skip to content

Commit

Permalink
Fixed #28626 -- Dropped support for PostgreSQL 9.3.
Browse files Browse the repository at this point in the history
Thanks Simon Charette for the introspection changes.
  • Loading branch information
timgraham committed Sep 27, 2017
1 parent ea7ca5d commit 1d8cfa3
Show file tree
Hide file tree
Showing 13 changed files with 18 additions and 53 deletions.
9 changes: 1 addition & 8 deletions django/db/backends/postgresql/features.py
Expand Up @@ -49,10 +49,7 @@ class DatabaseFeatures(BaseDatabaseFeatures):
END;
$$ LANGUAGE plpgsql;"""
supports_over_clause = True

@cached_property
def supports_aggregate_filter_clause(self):
return self.connection.pg_version >= 90400
supports_aggregate_filter_clause = True

@cached_property
def has_select_for_update_skip_locked(self):
Expand All @@ -62,10 +59,6 @@ def has_select_for_update_skip_locked(self):
def has_brin_index_support(self):
return self.connection.pg_version >= 90500

@cached_property
def has_jsonb_datatype(self):
return self.connection.pg_version >= 90400

@cached_property
def has_jsonb_agg(self):
return self.connection.pg_version >= 90500
Expand Down
22 changes: 6 additions & 16 deletions django/db/backends/postgresql/introspection.py
Expand Up @@ -145,17 +145,12 @@ def get_constraints(self, cursor, table_name):
# Loop over the key table, collecting things as constraints. The column
# array must return column names in the same order in which they were
# created.
# The subquery containing generate_series can be replaced with
# "WITH ORDINALITY" when support for PostgreSQL 9.3 is dropped.
cursor.execute("""
SELECT
c.conname,
array(
SELECT attname
FROM (
SELECT unnest(c.conkey) AS colid,
generate_series(1, array_length(c.conkey, 1)) AS arridx
) AS cols
FROM unnest(c.conkey) WITH ORDINALITY cols(colid, arridx)
JOIN pg_attribute AS ca ON cols.colid = ca.attnum
WHERE ca.attrelid = c.conrelid
ORDER BY cols.arridx
Expand Down Expand Up @@ -183,17 +178,13 @@ def get_constraints(self, cursor, table_name):
"options": options,
}
# Now get indexes
# The row_number() function for ordering the index fields can be
# replaced by WITH ORDINALITY in the unnest() functions when support
# for PostgreSQL 9.3 is dropped.
cursor.execute("""
SELECT
indexname, array_agg(attname ORDER BY rnum), indisunique, indisprimary,
array_agg(ordering ORDER BY rnum), amname, exprdef, s2.attoptions
indexname, array_agg(attname ORDER BY arridx), indisunique, indisprimary,
array_agg(ordering ORDER BY arridx), amname, exprdef, s2.attoptions
FROM (
SELECT
row_number() OVER () as rnum, c2.relname as indexname,
idx.*, attr.attname, am.amname,
c2.relname as indexname, idx.*, attr.attname, am.amname,
CASE
WHEN idx.indexprs IS NOT NULL THEN
pg_get_indexdef(idx.indexrelid)
Expand All @@ -206,9 +197,8 @@ def get_constraints(self, cursor, table_name):
END as ordering,
c2.reloptions as attoptions
FROM (
SELECT
*, unnest(i.indkey) as key, unnest(i.indoption) as option
FROM pg_index i
SELECT *
FROM pg_index i, unnest(i.indkey, i.indoption) WITH ORDINALITY koi(key, option, arridx)
) idx
LEFT JOIN pg_class c ON idx.indrelid = c.oid
LEFT JOIN pg_class c2 ON idx.indexrelid = c2.oid
Expand Down
2 changes: 1 addition & 1 deletion docs/ref/contrib/gis/install/index.txt
Expand Up @@ -58,7 +58,7 @@ supported versions, and any notes for each of the supported database backends:
================== ============================== ================== =========================================
Database Library Requirements Supported Versions Notes
================== ============================== ================== =========================================
PostgreSQL GEOS, GDAL, PROJ.4, PostGIS 9.3+ Requires PostGIS.
PostgreSQL GEOS, GDAL, PROJ.4, PostGIS 9.4+ Requires PostGIS.
MySQL GEOS, GDAL 5.6+ Not OGC-compliant; :ref:`limited functionality <mysql-spatial-limitations>`.
Oracle GEOS, GDAL 12.1+ XE not supported.
SQLite GEOS, GDAL, PROJ.4, SpatiaLite 3.6.+ Requires SpatiaLite 4.0+
Expand Down
2 changes: 0 additions & 2 deletions docs/ref/contrib/postgres/fields.txt
Expand Up @@ -509,8 +509,6 @@ using in conjunction with lookups on
of the JSON which allows indexing. The trade-off is a small additional cost
on writing to the ``jsonb`` field. ``JSONField`` uses ``jsonb``.

**As a result, this field requires PostgreSQL ≥ 9.4**.

Querying ``JSONField``
----------------------

Expand Down
2 changes: 0 additions & 2 deletions docs/ref/contrib/postgres/functions.txt
Expand Up @@ -16,8 +16,6 @@ All of these functions are available from the

Returns a version 4 UUID.

Requires PostgreSQL 9.4 or greater.

The `pgcrypto extension`_ must be installed. You can use the
:class:`~django.contrib.postgres.operations.CryptoExtension` migration
operation to install it.
Expand Down
2 changes: 1 addition & 1 deletion docs/ref/databases.txt
Expand Up @@ -92,7 +92,7 @@ below for information on how to set up your database correctly.
PostgreSQL notes
================

Django supports PostgreSQL 9.3 and higher. `psycopg2`_ 2.5.4 or higher is
Django supports PostgreSQL 9.4 and higher. `psycopg2`_ 2.5.4 or higher is
required, though the latest release is recommended.

.. _psycopg2: http://initd.org/psycopg/
Expand Down
6 changes: 6 additions & 0 deletions docs/releases/2.1.txt
Expand Up @@ -206,6 +206,12 @@ Dropped support for MySQL 5.5
The end of upstream support for MySQL 5.5 is December 2018. Django 2.1 supports
MySQL 5.6 and higher.

Dropped support for PostgreSQL 9.3
----------------------------------

The end of upstream support for PostgreSQL 9.3 is September 2018. Django 2.1
supports PostgreSQL 9.4 and higher.

Miscellaneous
-------------

Expand Down
10 changes: 0 additions & 10 deletions tests/postgres_tests/__init__.py
Expand Up @@ -7,16 +7,6 @@
from django.test import TestCase, modify_settings


def skipUnlessPG94(test):
try:
PG_VERSION = connection.pg_version
except AttributeError:
PG_VERSION = 0
if PG_VERSION < 90400:
return unittest.skip('PostgreSQL ≥ 9.4 required')(test)
return test


@unittest.skipUnless(connection.vendor == 'postgresql', "PostgreSQL specific tests")
class PostgreSQLTestCase(TestCase):
@classmethod
Expand Down
2 changes: 1 addition & 1 deletion tests/postgres_tests/migrations/0002_create_test_models.py
Expand Up @@ -242,7 +242,7 @@ class Migration(migrations.Migration):
('field_custom', JSONField(null=True, blank=True, encoder=DjangoJSONEncoder)),
],
options={
'required_db_features': {'has_jsonb_datatype'},
'required_db_vendor': 'postgresql',
},
bases=(models.Model,),
),
Expand Down
5 changes: 1 addition & 4 deletions tests/postgres_tests/models.py
Expand Up @@ -140,13 +140,10 @@ class RangeLookupsModel(PostgreSQLModel):
date = models.DateField(blank=True, null=True)


class JSONModel(models.Model):
class JSONModel(PostgreSQLModel):
field = JSONField(blank=True, null=True)
field_custom = JSONField(blank=True, null=True, encoder=DjangoJSONEncoder)

class Meta:
required_db_features = ['has_jsonb_datatype']


class ArrayFieldSubclass(ArrayField):
def __init__(self, *args, **kwargs):
Expand Down
3 changes: 1 addition & 2 deletions tests/postgres_tests/test_functions.py
Expand Up @@ -4,7 +4,7 @@

from django.contrib.postgres.functions import RandomUUID, TransactionNow

from . import PostgreSQLTestCase, skipUnlessPG94
from . import PostgreSQLTestCase
from .models import NowTestModel, UUIDTestModel


Expand All @@ -29,7 +29,6 @@ def test_transaction_now(self):
self.assertEqual(m1.when, m2.when)


@skipUnlessPG94
class TestRandomUUID(PostgreSQLTestCase):

def test_random_uuid(self):
Expand Down
2 changes: 0 additions & 2 deletions tests/postgres_tests/test_introspection.py
@@ -1,7 +1,6 @@
from io import StringIO

from django.core.management import call_command
from django.test import skipUnlessDBFeature
from django.test.utils import modify_settings

from . import PostgreSQLTestCase
Expand All @@ -20,7 +19,6 @@ def assertFieldsInModel(self, model, field_outputs):
for field_output in field_outputs:
self.assertIn(field_output, output)

@skipUnlessDBFeature('has_jsonb_datatype')
def test_json_field(self):
self.assertFieldsInModel(
'postgres_tests_jsonmodel',
Expand Down
4 changes: 0 additions & 4 deletions tests/postgres_tests/test_json.py
Expand Up @@ -5,7 +5,6 @@
from django.core import exceptions, serializers
from django.core.serializers.json import DjangoJSONEncoder
from django.forms import CharField, Form, widgets
from django.test import skipUnlessDBFeature
from django.utils.html import escape

from . import PostgreSQLTestCase
Expand All @@ -18,7 +17,6 @@
pass


@skipUnlessDBFeature('has_jsonb_datatype')
class TestSaveLoad(PostgreSQLTestCase):
def test_null(self):
instance = JSONModel()
Expand Down Expand Up @@ -92,7 +90,6 @@ def test_custom_encoding(self):
self.assertEqual(loaded.field_custom, obj_after)


@skipUnlessDBFeature('has_jsonb_datatype')
class TestQuerying(PostgreSQLTestCase):
@classmethod
def setUpTestData(cls):
Expand Down Expand Up @@ -262,7 +259,6 @@ def test_iregex(self):
self.assertTrue(JSONModel.objects.filter(field__foo__iregex=r'^bAr$').exists())


@skipUnlessDBFeature('has_jsonb_datatype')
class TestSerialization(PostgreSQLTestCase):
test_data = (
'[{"fields": {"field": {"a": "b", "c": null}, "field_custom": null}, '
Expand Down

0 comments on commit 1d8cfa3

Please sign in to comment.