Fixed #25730 -- Made Model.__str__() always return str instances.

Thanks to Kevin Turner for the report and Tim for the review.
This commit is contained in:
Simon Charette 2015-11-11 01:30:19 -05:00
parent 74365585e2
commit 4cd5d846d4
2 changed files with 22 additions and 1 deletions

View File

@ -463,7 +463,7 @@ class Model(six.with_metaclass(ModelBase)):
def __str__(self): def __str__(self):
if six.PY2 and hasattr(self, '__unicode__'): if six.PY2 and hasattr(self, '__unicode__'):
return force_text(self).encode('utf-8') return force_text(self).encode('utf-8')
return '%s object' % self.__class__.__name__ return str('%s object' % self.__class__.__name__)
def __eq__(self, other): def __eq__(self, other):
if not isinstance(other, Model): if not isinstance(other, Model):

View File

@ -4,6 +4,8 @@ from __future__ import unicode_literals
import datetime import datetime
from unittest import skipIf from unittest import skipIf
from django.apps.registry import Apps
from django.db import models
from django.test import TestCase from django.test import TestCase
from django.utils import six from django.utils import six
@ -33,3 +35,22 @@ class SimpleTests(TestCase):
# output of __unicode__() -- or __str__() when the # output of __unicode__() -- or __str__() when the
# python_2_unicode_compatible decorator is used. # python_2_unicode_compatible decorator is used.
self.assertEqual(str(a), b'Girl wins \xe2\x82\xac12.500 in lottery') self.assertEqual(str(a), b'Girl wins \xe2\x82\xac12.500 in lottery')
def test_defaults(self):
"""
The default implementation of __str__ and __repr__ should return
instances of str.
"""
test_apps = Apps(['str'])
class Default(models.Model):
class Meta:
apps = test_apps
obj = Default()
# Explicit call to __str__/__repr__ to make sure str()/repr() don't
# coerce the returned value.
self.assertIsInstance(obj.__str__(), str)
self.assertIsInstance(obj.__repr__(), str)
self.assertEqual(str(obj), str('Default object'))
self.assertEqual(repr(obj), str('<Default: Default object>'))