From 337470f9f04d7b7373ac0895fa6c081fa7fa9a96 Mon Sep 17 00:00:00 2001 From: Simon Charette Date: Fri, 5 Jun 2020 09:55:58 -0400 Subject: [PATCH 1/4] Use force_str instead of force_text. The latter is pending deprecation since it's an alias of the former on Python 3. --- picklefield/fields.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/picklefield/fields.py b/picklefield/fields.py index 1e87690..2adb043 100644 --- a/picklefield/fields.py +++ b/picklefield/fields.py @@ -7,7 +7,7 @@ from django import VERSION as DJANGO_VERSION from django.core import checks from django.db import models -from django.utils.encoding import force_text +from django.utils.encoding import force_str from .constants import DEFAULT_PROTOCOL @@ -189,13 +189,13 @@ def get_db_prep_value(self, value, connection=None, prepared=False): """ if value is not None and not isinstance(value, PickledObject): - # We call force_text here explicitly, so that the encoded string + # We call force_str here explicitly, so that the encoded string # isn't rejected by the postgresql_psycopg2 backend. Alternatively, # we could have just registered PickledObject with the psycopg # marshaller (telling it to store it like it would a string), but # since both of these methods result in the same value being stored, # doing things this way is much easier. - value = force_text(dbsafe_encode(value, self.compress, self.protocol, self.copy)) + value = force_str(dbsafe_encode(value, self.compress, self.protocol, self.copy)) return value def value_to_string(self, obj): From ceee08cc5623745e15f8c4218b9c0043d0f35267 Mon Sep 17 00:00:00 2001 From: Simon Charette Date: Fri, 5 Jun 2020 10:13:13 -0400 Subject: [PATCH 2/4] Remove remnants of Python 2 support code. --- picklefield/constants.py | 2 -- picklefield/fields.py | 8 +------- tests/models.py | 2 -- tests/settings.py | 2 -- tests/tests.py | 8 ++------ 5 files changed, 3 insertions(+), 19 deletions(-) diff --git a/picklefield/constants.py b/picklefield/constants.py index e8232d2..fd8f71e 100644 --- a/picklefield/constants.py +++ b/picklefield/constants.py @@ -1,3 +1 @@ -from __future__ import unicode_literals - DEFAULT_PROTOCOL = 3 diff --git a/picklefield/fields.py b/picklefield/fields.py index 2adb043..23fe7d9 100644 --- a/picklefield/fields.py +++ b/picklefield/fields.py @@ -1,7 +1,6 @@ -from __future__ import unicode_literals - from base64 import b64decode, b64encode from copy import deepcopy +from pickle import dumps, loads from zlib import compress, decompress from django import VERSION as DJANGO_VERSION @@ -11,11 +10,6 @@ from .constants import DEFAULT_PROTOCOL -try: - from cPickle import loads, dumps # pragma: no cover -except ImportError: - from pickle import loads, dumps # pragma: no cover - class PickledObject(str): """ diff --git a/tests/models.py b/tests/models.py index 3c80b6e..eca1dbf 100644 --- a/tests/models.py +++ b/tests/models.py @@ -1,5 +1,3 @@ -from __future__ import unicode_literals - from datetime import date from django.db import models diff --git a/tests/settings.py b/tests/settings.py index e2a67fa..6911ad4 100644 --- a/tests/settings.py +++ b/tests/settings.py @@ -1,5 +1,3 @@ -from __future__ import unicode_literals - SECRET_KEY = 'not-anymore' DATABASES = { diff --git a/tests/tests.py b/tests/tests.py index 1bb42f8..5fc5844 100644 --- a/tests/tests.py +++ b/tests/tests.py @@ -1,5 +1,6 @@ import json from datetime import date +from unittest.mock import patch from django.core import checks, serializers from django.db import IntegrityError, models @@ -14,18 +15,13 @@ TestCustomDataType, TestingModel, ) -try: - from unittest.mock import patch # pragma: no cover -except ImportError: - from mock import patch # pragma: no cover - class PickledObjectFieldTests(TestCase): def setUp(self): self.testing_data = (D2, S1, T1, L1, TestCustomDataType(S1), MinimalTestingModel) - return super(PickledObjectFieldTests, self).setUp() + return super().setUp() def test_data_integrity(self): """ From fade70d7649cb1a1b73977714292e06ec9aa6878 Mon Sep 17 00:00:00 2001 From: Simon Charette Date: Fri, 5 Jun 2020 10:16:30 -0400 Subject: [PATCH 3/4] Remove Django < 2 support code. --- picklefield/fields.py | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/picklefield/fields.py b/picklefield/fields.py index 23fe7d9..cebfd61 100644 --- a/picklefield/fields.py +++ b/picklefield/fields.py @@ -3,7 +3,6 @@ from pickle import dumps, loads from zlib import compress, decompress -from django import VERSION as DJANGO_VERSION from django.core import checks from django.db import models from django.utils.encoding import force_str @@ -164,12 +163,8 @@ def pre_save(self, model_instance, add): value = super().pre_save(model_instance, add) return wrap_conflictual_object(value) - if DJANGO_VERSION < (2, 0): - def from_db_value(self, value, expression, connection, context): # pragma: no cover - return self.to_python(value) # pragma: no cover - else: - def from_db_value(self, value, expression, connection): # pragma: no cover - return self.to_python(value) # pragma: no cover + def from_db_value(self, value, expression, connection): + return self.to_python(value) def get_db_prep_value(self, value, connection=None, prepared=False): """ From 0d8484067afd50c83362373a68bf6ba68f7ad2a7 Mon Sep 17 00:00:00 2001 From: Simon Charette Date: Fri, 5 Jun 2020 11:27:03 -0400 Subject: [PATCH 4/4] Allow default pickle protocol to be specified using a setting instead. Bumping this value without allowing an override breaks backward compatibility for lookups such as __exact because the string representation of the pickles change. --- README.rst | 3 ++- picklefield/constants.py | 2 +- picklefield/fields.py | 22 ++++++++++++++++++++-- tests/tests.py | 11 +++++++++++ 4 files changed, 34 insertions(+), 4 deletions(-) diff --git a/README.rst b/README.rst index 828f64c..d68f300 100644 --- a/README.rst +++ b/README.rst @@ -165,8 +165,9 @@ Changes UNRELEASED ========== +* Allowed default pickle protocol to be overriden using the + `PICKLEFIELD_DEFAULT_PROTOCOL` setting. * Dropped support for Python 2. -* Updated default pickle protocol to version 3. * Added testing against Django 3.0. * Dropped support for Django 1.11. diff --git a/picklefield/constants.py b/picklefield/constants.py index fd8f71e..dc6d5c1 100644 --- a/picklefield/constants.py +++ b/picklefield/constants.py @@ -1 +1 @@ -DEFAULT_PROTOCOL = 3 +DEFAULT_PROTOCOL = 2 diff --git a/picklefield/fields.py b/picklefield/fields.py index cebfd61..10f0bf1 100644 --- a/picklefield/fields.py +++ b/picklefield/fields.py @@ -3,6 +3,7 @@ from pickle import dumps, loads from zlib import compress, decompress +from django.conf import settings from django.core import checks from django.db import models from django.utils.encoding import force_str @@ -45,13 +46,19 @@ def wrap_conflictual_object(obj): return obj -def dbsafe_encode(value, compress_object=False, pickle_protocol=DEFAULT_PROTOCOL, copy=True): +def get_default_protocol(): + return getattr(settings, 'PICKLEFIELD_DEFAULT_PROTOCOL', DEFAULT_PROTOCOL) + + +def dbsafe_encode(value, compress_object=False, pickle_protocol=None, copy=True): # We use deepcopy() here to avoid a problem with cPickle, where dumps # can generate different character streams for same lookup value if # they are referenced differently. # The reason this is important is because we do all of our lookups as # simple string matches, thus the character streams must be the same # for the lookups to work properly. See tests.py for more information. + if pickle_protocol is None: + pickle_protocol = get_default_protocol() if copy: # Copy can be very expensive if users aren't going to perform lookups # on the value anyway. @@ -85,7 +92,10 @@ class PickledObjectField(models.Field): def __init__(self, *args, **kwargs): self.compress = kwargs.pop('compress', False) - self.protocol = kwargs.pop('protocol', DEFAULT_PROTOCOL) + protocol = kwargs.pop('protocol', None) + if protocol is None: + protocol = get_default_protocol() + self.protocol = protocol self.copy = kwargs.pop('copy', True) kwargs.setdefault('editable', False) super().__init__(*args, **kwargs) @@ -136,6 +146,14 @@ def check(self, **kwargs): errors.extend(self._check_default()) return errors + def deconstruct(self): + name, path, args, kwargs = super().deconstruct() + if self.compress: + kwargs['compress'] = True + if self.protocol != get_default_protocol(): + kwargs['protocol'] = self.protocol + return name, path, args, kwargs + def to_python(self, value): """ B64decode and unpickle the object, optionally decompressing it. diff --git a/tests/tests.py b/tests/tests.py index 5fc5844..88be048 100644 --- a/tests/tests.py +++ b/tests/tests.py @@ -199,6 +199,17 @@ def mock_decode_error(*args, **kwargs): self.assertEqual(encoded_value, MinimalTestingModel.objects.get(pk=model.pk).pickle_field) +class PickledObjectFieldDeconstructTests(SimpleTestCase): + def test_protocol(self): + field = PickledObjectField() + self.assertNotIn('protocol', field.deconstruct()[3]) + with self.settings(PICKLEFIELD_DEFAULT_PROTOCOL=3): + field = PickledObjectField(protocol=4) + self.assertEqual(field.deconstruct()[3].get('protocol'), 4) + field = PickledObjectField(protocol=3) + self.assertNotIn('protocol', field.deconstruct()[3]) + + @isolate_apps('tests') class PickledObjectFieldCheckTests(SimpleTestCase): def test_mutable_default_check(self):