From ec49e08b64bec3327c5a6c537f9bae242dfaef9a Mon Sep 17 00:00:00 2001 From: Chihiro Kaneko Date: Fri, 21 Sep 2018 18:42:29 +0900 Subject: [PATCH 1/4] Add new django versions and Drop EOL versions. --- .travis.yml | 81 +++++--------------- README.rst | 4 +- setup.py | 7 +- tox.ini | 212 ++++++++++++++++------------------------------------ 4 files changed, 90 insertions(+), 214 deletions(-) diff --git a/.travis.yml b/.travis.yml index 4dac53f..4a93bb4 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,80 +1,39 @@ language: python python: - - "2.6" - "2.7" - - "3.3" - "3.4" - "3.5" - "3.6" env: - - DJANGO_VERSION='Django>=1.4,<1.5' - - DJANGO_VERSION='Django>=1.5,<1.6' - - DJANGO_VERSION='Django>=1.6,<1.7' - - DJANGO_VERSION='Django>=1.7,<1.8' - - DJANGO_VERSION='Django>=1.8,<1.9' - - DJANGO_VERSION='Django>=1.9,<1.10' - - DJANGO_VERSION='Django>=1.10,<1.11' - DJANGO_VERSION='Django>=1.11,<2.0' + - DJANGO_VERSION='Django>=2.0,<2.1' + - DJANGO_VERSION='Django>=2.1,<2.2' - DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' matrix: exclude: - - python: "2.6" - env: DJANGO_VERSION='Django>=1.7,<1.8' - - python: "2.6" - env: DJANGO_VERSION='Django>=1.8,<1.9' - - python: "2.6" - env: DJANGO_VERSION='Django>=1.9,<1.10' - - python: "2.6" - env: DJANGO_VERSION='Django>=1.10,<1.11' - - python: "2.6" - env: DJANGO_VERSION='Django>=1.11,<2.0' - - python: "2.6" - env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' - python: "2.7" - env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' - - python: "3.3" - env: DJANGO_VERSION='Django>=1.4,<1.5' - - python: "3.3" - env: DJANGO_VERSION='Django>=1.5,<1.6' - - python: "3.3" - env: DJANGO_VERSION='Django>=1.9,<1.10' - - python: "3.3" - env: DJANGO_VERSION='Django>=1.10,<1.11' - - python: "3.3" env: DJANGO_VERSION='Django>=1.11,<2.0' - - python: "3.3" + - python: "3.4" + env: DJANGO_VERSION='Django>=1.11,<2.0' + - python: "3.4" + env: DJANGO_VERSION='Django>=2.0,<2.1' + - python: "3.5" + env: DJANGO_VERSION='Django>=1.11,<2.0' + - python: "3.5" + env: DJANGO_VERSION='Django>=2.0,<2.1' + - python: "3.5" + env: DJANGO_VERSION='Django>=2.1,<2.2' + - python: "3.5" env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' - - python: "3.4" - env: DJANGO_VERSION='Django>=1.4,<1.5' - - python: "3.4" - env: DJANGO_VERSION='Django>=1.5,<1.6' - - python: "3.4" - env: DJANGO_VERSION='Django>=1.6,<1.7' - - python: "3.4" + - python: "3.6" + env: DJANGO_VERSION='Django>=1.11,<2.0' + - python: "3.6" + env: DJANGO_VERSION='Django>=2.0,<2.1' + - python: "3.6" + env: DJANGO_VERSION='Django>=2.1,<2.2' + - python: "3.6" env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' - - python: "3.5" - env: DJANGO_VERSION='Django>=1.4,<1.5' - - python: "3.5" - env: DJANGO_VERSION='Django>=1.5,<1.6' - - python: "3.5" - env: DJANGO_VERSION='Django>=1.6,<1.7' - - python: "3.5" - env: DJANGO_VERSION='Django>=1.7,<1.8' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.4,<1.5' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.5,<1.6' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.6,<1.7' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.7,<1.8' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.8,<1.9' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.9,<1.10' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.10,<1.11' allow_failures: - env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' diff --git a/README.rst b/README.rst index f1daad0..628dbad 100644 --- a/README.rst +++ b/README.rst @@ -14,9 +14,9 @@ A new model field and form field. With this you can get a multiple select from a This egg is inspired by this `snippet `_. -Supported Python versions: 2.6, 2.7, 3.3+ +Supported Python versions: 2.7, 3.4+ -Supported Django versions: 1.4-1.11+ +Supported Django versions: 1.11-2.0+ Installation ============ diff --git a/setup.py b/setup.py index 6a8c581..acf9632 100644 --- a/setup.py +++ b/setup.py @@ -41,13 +41,12 @@ setup( 'Framework :: Django', 'License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL)', 'Programming Language :: Python :: 2', - 'Programming Language :: Python :: 2.6', 'Programming Language :: Python :: 2.7', 'Programming Language :: Python :: 3', - 'Programming Language :: Python :: 3.3', 'Programming Language :: Python :: 3.4', 'Programming Language :: Python :: 3.5', 'Programming Language :: Python :: 3.6', + 'Programming Language :: Python :: 3.7', ], license="LGPL 3", keywords="django,multiple,select,field,choices", @@ -55,13 +54,13 @@ setup( packages=find_packages(), include_package_data=True, tests_require=[ - 'django>=1.4', + 'django>=1.11', 'tox', 'coverage', 'flake8', ], install_requires=[ - 'django>=1.4', + 'django>=1.11', ], zip_safe=False, ) diff --git a/tox.ini b/tox.ini index 72bbf3f..08b8f0b 100644 --- a/tox.ini +++ b/tox.ini @@ -1,5 +1,5 @@ [tox] -envlist = py26-dj14,py26-dj15,py26-dj16,py27-dj14,py27-dj15,py27-dj16,py27-dj17,py27-dj18,py27-dj19,py27-dj110,py33-dj16,py33-dj17,py33-dj18,py34-dj17,py34-dj18,py34-dj19,py34-dj110,py35-dj18,py35-dj19,py35-dj110 +envlist = py{27,34,35,36}-dj111,py{34,35,36,37}-dj20,py{35,36,37}-dj21 [testenv] usedevelop = True @@ -12,186 +12,104 @@ commands = install_command = pip install {opts} {packages} -[testenv:py26-dj14] -basepython = python2.6 -deps = - django==1.4.22 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py26-dj15] -basepython = python2.6 -deps = - django==1.5.12 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py26-dj16] -basepython = python2.6 -deps = - django==1.6.11 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - - -[testenv:py27-dj14] +[testenv:py27-dj111] basepython = python2.7 deps = - django==1.4.22 + django==1.11.15 pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py27-dj15] -basepython = python2.7 -deps = - django==1.5.12 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py27-dj16] -basepython = python2.7 -deps = - django==1.6.11 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py27-dj17] -basepython = python2.7 -deps = - django==1.7.11 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py27-dj18] -basepython = python2.7 -deps = - django==1.8.17 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py27-dj19] -basepython = python2.7 -deps = - django==1.9.12 - pillow==1.7.8 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py27-dj110] -basepython = python2.7 -deps = - django==1.10.4 - pillow==1.7.8 - PyYAML==3.10 + PyYAML==3.13 coveralls==0.3 flake8 -[testenv:py33-dj16] -basepython = python3.3 -deps = - django==1.6.11 - pillow==2.1.0 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py33-dj17] -basepython = python3.3 -deps = - django==1.7.11 - pillow==2.1.0 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py33-dj18] -basepython = python3.3 -deps = - django==1.8.17 - pillow==2.1.0 - PyYAML==3.10 - coveralls==0.3 - flake8 - - -[testenv:py34-dj17] +[testenv:py34-dj111] basepython = python3.4 deps = - django==1.7.11 + django==1.11.15 pillow==2.1.0 - PyYAML==3.10 + PyYAML==3.13 coveralls==0.3 flake8 -[testenv:py34-dj18] +[testenv:py34-dj20] basepython = python3.4 deps = - django==1.8.17 + django==2.0.8 pillow==2.1.0 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py34-dj19] -basepython = python3.4 -deps = - django==1.9.12 - pillow==2.1.0 - PyYAML==3.10 - coveralls==0.3 - flake8 - -[testenv:py34-dj110] -basepython = python3.4 -deps = - django==1.10.4 - pillow==2.1.0 - PyYAML==3.10 + PyYAML==3.13 coveralls==0.3 flake8 -[testenv:py35-dj18] +[testenv:py35-dj111] basepython = python3.5 deps = - django==1.8.17 + django==1.11.15 pillow==2.1.0 - PyYAML==3.10 + PyYAML==3.13 coveralls==0.3 flake8 -[testenv:py35-dj19] +[testenv:py35-dj20] basepython = python3.5 deps = - django==1.9.12 + django==2.0.8 pillow==2.1.0 - PyYAML==3.10 + PyYAML==3.13 coveralls==0.3 flake8 -[testenv:py35-dj110] +[testenv:py35-dj21] basepython = python3.5 deps = - django==1.10.4 + django==2.1.1 pillow==2.1.0 - PyYAML==3.10 + PyYAML==3.13 coveralls==0.3 flake8 + +[testenv:py36-dj111] +basepython = python3.6 +deps = + django==1.11.15 + pillow==2.1.0 + PyYAML==3.13 + coveralls==0.3 + flake8 + +[testenv:py36-dj20] +basepython = python3.6 +deps = + django==2.0.8 + pillow==2.1.0 + PyYAML==3.13 + coveralls==0.3 + flake8 + +[testenv:py36-dj21] +basepython = python3.6 +deps = + django==2.1.1 + pillow==2.1.0 + PyYAML==3.13 + coveralls==0.3 + flake8 + + +[testenv:py37-dj20] +basepython = python3.7 +deps = + django==2.0.8 + pillow==2.1.0 + PyYAML==3.13 + coveralls==0.3 + flake8 + +[testenv:py37-dj21] +basepython = python3.7 +deps = + django==2.1.1 + pillow==2.1.0 + PyYAML==3.13 + coveralls==0.3 + flake8 \ No newline at end of file From 944f2b51ee20b36ee793f30e6d4a2a6b55653c39 Mon Sep 17 00:00:00 2001 From: Chihiro Kaneko Date: Fri, 21 Sep 2018 18:44:47 +0900 Subject: [PATCH 2/4] Drop django 1.10 or lower and Django 1.11 or higher support. --- example/app/test_msf.py | 32 +++++++--------------------- example/app/views.py | 2 +- example/example/urls.py | 40 +++++++++++++---------------------- example/run_tests.py | 7 ++---- multiselectfield/db/fields.py | 10 ++++----- 5 files changed, 31 insertions(+), 60 deletions(-) diff --git a/example/app/test_msf.py b/example/app/test_msf.py index b5e3932..dc057ee 100644 --- a/example/app/test_msf.py +++ b/example/app/test_msf.py @@ -32,12 +32,8 @@ else: u = str -if VERSION < (1, 9): - def get_field(model, name): - return model._meta.get_field_by_name(name)[0] -else: - def get_field(model, name): - return model._meta.get_field(name) +def get_field(model, name): + return model._meta.get_field(name) class MultiSelectTestCase(TestCase): @@ -77,12 +73,8 @@ class MultiSelectTestCase(TestCase): # call Field.from_db_field, it simply returns a Python representation # of the data in the database (which in our case is a string of # comma-separated values). The bug was fixed in Django 1.8+. - if VERSION >= (1, 6) and VERSION < (1, 8): - self.assertStringEqual(tag_list_list, [u('sex,work,happy')]) - self.assertStringEqual(categories_list_list, [u('1,3,5')]) - else: - self.assertListEqual(tag_list_list, [['sex', 'work', 'happy']]) - self.assertListEqual(categories_list_list, [['1', '3', '5']]) + self.assertListEqual(tag_list_list, [['sex', 'work', 'happy']]) + self.assertListEqual(categories_list_list, [['1', '3', '5']]) def test_form(self): form_class = modelform_factory(Book, fields=('title', 'tags', 'categories')) @@ -139,7 +131,7 @@ class MultiSelectTestCase(TestCase): self.assertEqual(len(form_class.base_fields), 1) form = form_class(initial={'published_in': ['BC', 'AK']}) - expected_html = u("""

  • Canada - Provinces
    • \n""" + expected_html = u("""

      • Canada - Provinces
        • \n""" """
      • \n""" """
      • USA - States
        • \n""" """
        • \n""" @@ -147,18 +139,10 @@ class MultiSelectTestCase(TestCase): actual_html = form.as_p() - if (1, 11) <= VERSION < (2, 0): - # Django 1.11+ does not assign 'for' attributes on labels if they - # are group labels - expected_html = expected_html.replace('label for="id_published_in_0"', 'label') + if VERSION >= (2, 0): + expected_html = expected_html.replace('input checked="checked"', 'input checked') - if VERSION < (1, 6): - # Django 1.6 renders the Python repr() for each group (eg: tuples - # with HTML entities), so we skip the test for that version - self.assertEqual(expected_html.replace('\n', ''), actual_html.replace('\n', '')) - - if VERSION >= (1, 7): - self.assertHTMLEqual(expected_html, actual_html) + self.assertHTMLEqual(expected_html, actual_html) class MultiSelectUtilsTestCase(TestCase): diff --git a/example/app/views.py b/example/app/views.py index 3b0bc22..6142ef1 100644 --- a/example/app/views.py +++ b/example/app/views.py @@ -19,7 +19,7 @@ from django.conf import settings from django.contrib.auth import login from django.contrib.auth import get_user_model -from django.core.urlresolvers import reverse +from django.urls import reverse from django.http import HttpResponseRedirect diff --git a/example/example/urls.py b/example/example/urls.py index efa4a7b..0e60278 100644 --- a/example/example/urls.py +++ b/example/example/urls.py @@ -14,43 +14,33 @@ # You should have received a copy of the GNU Lesser General Public License # along with this software. If not, see . -from django import VERSION from django.conf import settings try: from django.conf.urls import include, url - - # Compatibility for Django > 1.8 - def patterns(prefix, *args): - if VERSION < (1, 9): - from django.conf.urls import patterns as django_patterns - return django_patterns(prefix, *args) - elif prefix != '': - raise NotImplementedError("You need to update your URLConf for " - "Django 1.10, or tweak it to remove the " - "prefix parameter") - else: - return list(args) -except ImportError: # Django < 1.4 - from django.conf.urls.defaults import include, patterns, url - +except ImportError: + from django.urls import include, url from django.contrib import admin from django.views.static import serve + admin.autodiscover() js_info_dict = { 'packages': ('django.conf',), } -urlpatterns = patterns( - '', +urlpatterns = [ url(r'^', include('app.urls')), - url(r'^admin/', include(admin.site.urls)), -) + url(r'^admin/', admin.site.urls), +] -urlpatterns += patterns( - '', - url(r'^%s(?P.*)$' % settings.MEDIA_URL[1:], +urlpatterns += [ + url( + r'^%s(?P.*)$' % settings.MEDIA_URL[1:], serve, - {'document_root': settings.MEDIA_ROOT, 'show_indexes': True}), -) + { + 'document_root': settings.MEDIA_ROOT, + 'show_indexes': True, + }, + ), +] diff --git a/example/run_tests.py b/example/run_tests.py index 9cc12c4..ec3f39a 100644 --- a/example/run_tests.py +++ b/example/run_tests.py @@ -19,10 +19,9 @@ import os import sys -import django - from django.conf import ENVIRONMENT_VARIABLE from django.core import management +from django.core.wsgi import get_wsgi_application if len(sys.argv) == 1: @@ -30,8 +29,6 @@ if len(sys.argv) == 1: else: os.environ[ENVIRONMENT_VARIABLE] = sys.argv[1] -if django.VERSION[0] == 1 and django.VERSION[1] >= 7: - from django.core.wsgi import get_wsgi_application - application = get_wsgi_application() +application = get_wsgi_application() management.call_command('test', 'app') diff --git a/multiselectfield/db/fields.py b/multiselectfield/db/fields.py index 57b053f..e10832d 100644 --- a/multiselectfield/db/fields.py +++ b/multiselectfield/db/fields.py @@ -55,8 +55,8 @@ class MSFList(list): super(MSFList, self).__init__(*args, **kwargs) def __str__(msgl): - l = [msgl.choices.get(int(i)) if i.isdigit() else msgl.choices.get(i) for i in msgl] - return u', '.join([string_type(s) for s in l]) + list = [msgl.choices.get(int(i)) if i.isdigit() else msgl.choices.get(i) for i in msgl] + return u', '.join([string_type(s) for s in list]) class MultiSelectField(models.CharField): @@ -74,7 +74,7 @@ class MultiSelectField(models.CharField): self.validators.append(MaxChoicesValidator(self.max_choices)) def _get_flatchoices(self): - l = super(MultiSelectField, self)._get_flatchoices() + choices = super(MultiSelectField, self)._get_flatchoices() class MSFFlatchoices(list): # Used to trick django.contrib.admin.utils.display_for_field into @@ -83,7 +83,7 @@ class MultiSelectField(models.CharField): def __bool__(self): return False __nonzero__ = __bool__ - return MSFFlatchoices(l) + return MSFFlatchoices(choices) flatchoices = property(_get_flatchoices) def get_choices_default(self): @@ -102,7 +102,7 @@ class MultiSelectField(models.CharField): return choices_selected def value_to_string(self, obj): - value = self._get_val_from_obj(obj) + value = super(MultiSelectField, self).value_from_object(obj) return self.get_prep_value(value) def validate(self, value, model_instance): From a0b438280ca2f21e973c2b56e9e0ff6e0a15113d Mon Sep 17 00:00:00 2001 From: Chihiro Kaneko Date: Fri, 21 Sep 2018 18:47:57 +0900 Subject: [PATCH 3/4] Fix deprecation warnings. --- multiselectfield/db/fields.py | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/multiselectfield/db/fields.py b/multiselectfield/db/fields.py index e10832d..bd6ada2 100644 --- a/multiselectfield/db/fields.py +++ b/multiselectfield/db/fields.py @@ -147,10 +147,16 @@ class MultiSelectField(models.CharField): return value if isinstance(value, list) else MSFList(choices, value.split(',')) return MSFList(choices, []) - def from_db_value(self, value, expression, connection, context): - if value is None: - return value - return self.to_python(value) + if VERSION < (2, ): + def from_db_value(self, value, expression, connection, context): + if value is None: + return value + return self.to_python(value) + else: + def from_db_value(self, value, expression, connection): + if value is None: + return value + return self.to_python(value) def contribute_to_class(self, cls, name): super(MultiSelectField, self).contribute_to_class(cls, name) From 679a6e446b25dea2160d5bed082436ee87bec894 Mon Sep 17 00:00:00 2001 From: hirokinko Date: Mon, 24 Sep 2018 10:54:26 +0900 Subject: [PATCH 4/4] Fix .travis.yml --- .travis.yml | 20 ++++---------------- 1 file changed, 4 insertions(+), 16 deletions(-) diff --git a/.travis.yml b/.travis.yml index 4a93bb4..491b04b 100644 --- a/.travis.yml +++ b/.travis.yml @@ -13,26 +13,14 @@ env: matrix: exclude: - python: "2.7" - env: DJANGO_VERSION='Django>=1.11,<2.0' - - python: "3.4" - env: DJANGO_VERSION='Django>=1.11,<2.0' - - python: "3.4" env: DJANGO_VERSION='Django>=2.0,<2.1' - - python: "3.5" - env: DJANGO_VERSION='Django>=1.11,<2.0' - - python: "3.5" - env: DJANGO_VERSION='Django>=2.0,<2.1' - - python: "3.5" + - python: "2.7" env: DJANGO_VERSION='Django>=2.1,<2.2' - - python: "3.5" + - python: "2.7" env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' - - python: "3.6" - env: DJANGO_VERSION='Django>=1.11,<2.0' - - python: "3.6" - env: DJANGO_VERSION='Django>=2.0,<2.1' - - python: "3.6" + - python: "3.4" env: DJANGO_VERSION='Django>=2.1,<2.2' - - python: "3.6" + - python: "2.7" env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz' allow_failures: - env: DJANGO_VERSION='https://github.com/django/django/archive/master.tar.gz'