Skip to content

Commit e716a6d

Browse files
committedJan 27, 2015
Adds support for python >= 3.2. Version bumped
1 parent d05c3d8 commit e716a6d

20 files changed

+124
-72
lines changed
 

‎.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -52,3 +52,6 @@ docs/_build/
5252

5353
# PyBuilder
5454
target/
55+
56+
# Pyenv
57+
.python-version

‎.travis.yml

+15-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,24 @@
11
language: python
22

33
install:
4-
- pip install tox
4+
- pip install tox
55

66
script:
77
- tox
88

99
env:
10-
- TOXENV=django_1.5.X
11-
- TOXENV=django_1.6.X
12-
- TOXENV=django_1.7.X
10+
- TOXENV=py26-django15
11+
- TOXENV=py26-django16
12+
- TOXENV=py27-django15
13+
- TOXENV=py27-django16
14+
- TOXENV=py32-django15
15+
- TOXENV=py32-django16
16+
- TOXENV=py33-django15
17+
- TOXENV=py33-django16
18+
- TOXENV=py34-django15
19+
- TOXENV=py34-django16
20+
- TOXENV=py27-django17
21+
- TOXENV=py32-django17
22+
- TOXENV=py33-django17
23+
- TOXENV=py34-django17
1324
- TOXENV=coverage

‎LICENSE

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,4 +16,4 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
1616
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1717
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
1818
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
19-
THE SOFTWARE.
19+
THE SOFTWARE.

‎README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,4 +6,4 @@ Django Bricks
66

77
A Django application that sorts lists of heterogeneous models.
88

9-
Documentation can be found at: http://django-bricks.readthedocs.org/en/latest/
9+
Documentation can be found at: http://django-bricks.readthedocs.org/en/latest/

‎djangobricks/__init__.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,9 @@
22
When you need a wall.
33
"""
44

5-
VERSION = (1, 0, 0)
5+
VERSION = (1, 1, 0)
66

77

88
def get_version():
99
"""Returns the version as a string."""
10-
return '.'.join(map(str, VERSION))
10+
return '.'.join(map(str, VERSION))

‎djangobricks/exceptions.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
class BricksException(Exception): pass
22

3-
class TemplateNameNotFound(BricksException): pass
3+
class TemplateNameNotFound(BricksException): pass

‎djangobricks/models.py

+22-7
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,16 @@
1+
from __future__ import unicode_literals
2+
13
import copy
24
from operator import attrgetter
35
from itertools import chain
46

7+
from django.utils import six
8+
from django.utils.six.moves import range
9+
10+
if six.PY3:
11+
def cmp(a, b):
12+
return (a > b) - (a < b)
13+
514
SORTING_ASC = 1
615
SORTING_DESC = -1
716

@@ -140,7 +149,7 @@ def get_bricks_for_queryset(cls, queryset):
140149
# Execute the query once to avoid several OFFSET LIMIT
141150
items = list(queryset)
142151
return [cls(i) for i in (items[i:i+cls.chunk_size]
143-
for i in xrange(0, count, cls.chunk_size))]
152+
for i in range(0, count, cls.chunk_size))]
144153

145154
def get_context(self, **kwargs):
146155
"""
@@ -161,7 +170,7 @@ class BaseWall(object):
161170
162171
It orders a list of bricks using the given criteria and can be sliced or
163172
iterated.
164-
173+
165174
:param bricks: the list of bricks to sort.
166175
:param criteria: the list of criteria to sort the bricks by.
167176
"""
@@ -185,7 +194,7 @@ def __getstate__(self):
185194
# as those might not be pickable
186195
obj_dict = self.__dict__.copy()
187196
obj_dict['_sorted'] = self.sorted
188-
if obj_dict.has_key('criteria'):
197+
if 'criteria' in obj_dict:
189198
del obj_dict['criteria']
190199
return obj_dict
191200

@@ -208,7 +217,13 @@ def sorted(self):
208217
Lazy property that returns the list of bricks sorted by the criteria.
209218
"""
210219
if not self._sorted:
211-
self._sorted = sorted(self.bricks, cmp=self._cmp)
220+
try:
221+
# Python > 2.6
222+
from functools import cmp_to_key
223+
self._sorted = sorted(self.bricks, key=cmp_to_key(self._cmp))
224+
except ImportError:
225+
# Python <= 2.6
226+
self._sorted = sorted(self.bricks, cmp=self._cmp)
212227
return self._sorted
213228

214229
def filter(self, callback, operator='AND'):
@@ -282,8 +297,8 @@ def content_iterator():
282297
# Do some sanity check just to help the user
283298
for brick, queryset in self.get_content():
284299
if not issubclass(brick, BaseBrick):
285-
raise TypeError(u"Expected a BaseBrick subclass, "
286-
"got %r instead" % brick)
300+
raise TypeError("Expected a BaseBrick subclass, "
301+
"got %r instead" % brick)
287302
yield brick, queryset
288303

289304
bricks = (b.get_bricks_for_queryset(qs) for b, qs in content_iterator())
@@ -307,4 +322,4 @@ def get_content():
307322
factory = BaseWallFactory(criteria, wall_class)
308323
# Monkey patch the factory instance
309324
factory.get_content = get_content
310-
return factory.wall()
325+
return factory.wall()

‎djangobricks/templatetags/bricks.py

+5-3
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
1+
from __future__ import unicode_literals
2+
13
from django import template
24
from django.template.loader import render_to_string
35

4-
from ..exceptions import TemplateNameNotFound
6+
from djangobricks.exceptions import TemplateNameNotFound
57

68
register = template.Library()
79

@@ -16,8 +18,8 @@ def render_brick(context, brick, **extra_context):
1618
to the brick.
1719
"""
1820
if brick.template_name is None:
19-
raise TemplateNameNotFound(u'%r does not define '
20-
'any template name.' % brick.__class__)
21+
raise TemplateNameNotFound('%r does not define '
22+
'any template name.' % brick.__class__)
2123

2224
request = context.get('request')
2325

‎djangobricks/tests.py

+25-18
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,16 @@
1+
from __future__ import unicode_literals
2+
13
import datetime
24
import os
35

46
from django import get_version
5-
from django.utils import unittest
6-
from django.test import SimpleTestCase
7-
from django.test.utils import override_settings
87
from django.db import models
98
from django.template import Template, Context
9+
from django.test import SimpleTestCase
10+
from django.test.utils import override_settings
11+
from django.utils import unittest
12+
from django.utils.encoding import python_2_unicode_compatible
13+
from django.utils.six.moves import range
1014

1115
from .models import (
1216
SingleBrick,
@@ -18,7 +22,7 @@
1822
BaseWallFactory,
1923
wall_factory,
2024
)
21-
from .exceptions import TemplateNameNotFound
25+
from djangobricks.exceptions import TemplateNameNotFound
2226

2327
CURRENT_DIR = os.path.abspath(os.path.dirname(__file__))
2428

@@ -51,40 +55,43 @@ class NotABrick(object): pass
5155
class TestBrickWall(BaseWall): pass
5256

5357

58+
@python_2_unicode_compatible
5459
class TestModelA(models.Model):
5560
name = models.CharField(max_length=8)
5661
popularity = models.PositiveIntegerField()
5762
pub_date = models.DateTimeField()
5863
is_sticky = models.BooleanField(default=False)
5964

60-
def __unicode__(self):
61-
return unicode(self.name)
65+
def __str__(self):
66+
return self.name
6267

6368
def callable_popularity(self):
6469
return self.popularity
6570

6671

72+
@python_2_unicode_compatible
6773
class TestModelB(models.Model):
6874
name = models.CharField(max_length=8)
6975
date_add = models.DateTimeField()
7076
popularity = models.PositiveIntegerField()
7177
is_sticky = models.BooleanField(default=False)
7278

73-
def __unicode__(self):
74-
return unicode(self.name)
79+
def __str__(self):
80+
return self.name
7581

7682
def pub_date(self):
7783
return self.date_add
7884

7985

86+
@python_2_unicode_compatible
8087
class TestModelC(models.Model):
8188
name = models.CharField(max_length=8)
8289
pub_date = models.DateTimeField()
8390
popularity = models.PositiveIntegerField()
8491
is_sticky = models.BooleanField(default=False)
8592

86-
def __unicode__(self):
87-
return unicode(self.name)
93+
def __str__(self):
94+
return self.name
8895

8996

9097
class TestWallFactory(BaseWallFactory):
@@ -131,7 +138,7 @@ def _create_model_a_objects_and_bricks(self):
131138
pub_date=datetime.datetime(2012, 1, 1, 12, 0), is_sticky=True)
132139
objectA4 = TestModelA.objects.create(name='objectA4', popularity=2,
133140
pub_date=datetime.datetime(2013, 1, 1, 12, 0), is_sticky=False)
134-
141+
135142
self.brickA1 = SingleBrick(objectA1)
136143
self.brickA2 = SingleBrick(objectA2)
137144
self.brickA3 = SingleBrick(objectA3)
@@ -148,7 +155,7 @@ def _create_model_b_objects_and_bricks(self):
148155
date_add=datetime.datetime(2008, 1, 1, 12, 0), is_sticky=True)
149156
objectB4 = TestModelB.objects.create(name='objectB4', popularity=7,
150157
date_add=datetime.datetime(2009, 1, 1, 12, 0), is_sticky=False)
151-
158+
152159
self.brickB1 = SingleBrick(objectB1)
153160
self.brickB2 = SingleBrick(objectB2)
154161
self.brickB3 = SingleBrick(objectB3)
@@ -165,7 +172,7 @@ def _create_model_c_objects_and_bricks(self):
165172
pub_date=datetime.datetime(2004, 1, 1, 12, 0), is_sticky=True)
166173
objectC4 = TestModelC.objects.create(name='objectC4', popularity=17,
167174
pub_date=datetime.datetime(2005, 1, 1, 12, 0), is_sticky=False)
168-
175+
169176
self.brickC1 = ListBrick([objectC1, objectC2])
170177
self.brickC2 = ListBrick([objectC3, objectC4])
171178
for i in range(1, 3):
@@ -203,7 +210,7 @@ def test_single_brick_init(self):
203210
pub_date=datetime.datetime(2012, 1, 1, 12, 0), is_sticky=True)
204211
objectA4 = TestModelA.objects.create(name='objectA4', popularity=2,
205212
pub_date=datetime.datetime(2013, 1, 1, 12, 0), is_sticky=False)
206-
213+
207214
bricks = SingleBrick.get_bricks_for_queryset(TestModelA.objects.all())
208215
wall = TestBrickWall(bricks)
209216
self.assertEqual(wall[0].item, objectA1)
@@ -220,7 +227,7 @@ def test_list_brick_init(self):
220227
pub_date=datetime.datetime(2012, 1, 1, 12, 0), is_sticky=True)
221228
objectA4 = TestModelA.objects.create(name='objectA4', popularity=2,
222229
pub_date=datetime.datetime(2013, 1, 1, 12, 0), is_sticky=False)
223-
230+
224231
bricks = ListBrick.get_bricks_for_queryset(TestModelA.objects.all())
225232
wall = TestBrickWall(bricks)
226233
self.assertEqual(wall[0].items, [objectA1, objectA2, objectA3, objectA4])
@@ -558,11 +565,11 @@ def test_list_brick_context(self):
558565
def test_list_brick_chunk(self):
559566
now = datetime.datetime.now()
560567
objects = []
561-
for i in xrange(12):
568+
for i in range(12):
562569
obj = TestModelC.objects.create(name=i, popularity=i, pub_date=now)
563570
objects.append(obj)
564571
bricks = TestListBrick.get_bricks_for_queryset(TestModelC.objects.all())
565-
for i in xrange(3):
572+
for i in range(3):
566573
start = i * TestListBrick.chunk_size
567574
stop = start + TestListBrick.chunk_size
568575
self.assertEqual(bricks[i].items, objects[start:stop])
@@ -748,4 +755,4 @@ def test_factory_method_multiple_queryset(self):
748755
expected = [self.brickB1.item, self.brickB2.item, self.brickB4.item,
749756
self.brickA1.item, self.brickA2.item, self.brickA4.item,
750757
self.brickB3.item, self.brickA3.item]
751-
self.assertEqual([b.item for b in wall], expected)
758+
self.assertEqual([b.item for b in wall], expected)

‎docs/advanced_usage.rst

+2-2
Original file line numberDiff line numberDiff line change
@@ -122,7 +122,7 @@ content. So let's write a brick class for our new model:
122122
def get_value_for_criterion(self, criterion):
123123
if criterion.attrname == 'pub_date':
124124
return self.item.public_from_date
125-
return super(PhotoGalleryBrick, self).get_value_for_criterion(criterion)
125+
return super(PhotoGalleryBrick, self).get_value_for_criterion(criterion)
126126
127127
And that's it! Unless you are sure to cover each possible criterion, it's a good
128128
practice to return the value from super at least, as shown above.
@@ -154,4 +154,4 @@ or you can add them using directly the templatetag
154154

155155
.. code-block:: html+django
156156

157-
{% render_brick brick color='red' %}
157+
{% render_brick brick color='red' %}

‎docs/api.rst

+1-1
Original file line numberDiff line numberDiff line change
@@ -37,4 +37,4 @@ Utilities
3737
.. >>>>>>>>>>>>
3838
.. .. automodule:: djangobricks.templatetags.bricks
3939
..
40-
.. .. autofunction:: render_brick
40+
.. .. autofunction:: render_brick

‎docs/changelog.rst

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
=========
2+
Changelog
3+
=========
4+
5+
Version 1.1
6+
===========
7+
* Added support for Python >= 3.2
8+
* Minor documentation fixes
9+
10+
Version 1.0
11+
===========
12+
* First version

‎docs/index.rst

+12-2
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,17 @@ tricky.
1818
The :doc:`usage` will guide you through a complete tutorial to explain the
1919
concepts behind Bricks and to show how simple can be to achieve that goal.
2020

21-
Contents:
21+
Requirements
22+
------------
23+
24+
======== ======
25+
Python 2 >= 2.6
26+
Python 3 >= 3.2
27+
Django >= 1.5
28+
======== ======
29+
30+
Contents
31+
--------
2232

2333
.. toctree::
2434
:maxdepth: 2
@@ -27,10 +37,10 @@ Contents:
2737
usage
2838
advanced_usage
2939
api
40+
changelog
3041

3142
Indices and tables
3243
==================
3344

3445
* :ref:`genindex`
3546
* :ref:`search`
36-

0 commit comments

Comments
 (0)
Please sign in to comment.