Fixed #5789 -- Changed LocaleMiddleware session variable to '_language'.

The old 'django_language' variable will still be read from in order
to migrate users. The backwards-compatability shim will be removed in
Django 1.8.

Thanks to jdunck for the report and stugots for the initial patch.
This commit is contained in:
Bouke Haarsma 2013-10-18 13:56:32 +02:00 committed by Tim Graham
parent 8e2029f8dd
commit 0d0f4f020a
8 changed files with 58 additions and 14 deletions

View File

@ -56,8 +56,14 @@ class LocaleMiddleware(object):
return self.response_redirect_class(language_url) return self.response_redirect_class(language_url)
# Store language back into session if it is not present # Store language back into session if it is not present
if hasattr(request, 'session'): if hasattr(request, 'session') and '_language' not in request.session:
request.session.setdefault('django_language', language) # Backwards compatibility check on django_language (remove in 1.8);
# revert to: `request.session.setdefault('_language', language)`.
if 'django_language' in request.session:
request.session['_language'] = request.session['django_language']
del request.session['django_language']
else:
request.session['_language'] = language
if not (self.is_language_prefix_patterns_used() if not (self.is_language_prefix_patterns_used()
and language_from_path): and language_from_path):

View File

@ -426,7 +426,8 @@ def get_language_from_request(request, check_path=False):
return lang_code return lang_code
if hasattr(request, 'session'): if hasattr(request, 'session'):
lang_code = request.session.get('django_language', None) # for backwards compatibility django_language is also checked (remove in 1.8)
lang_code = request.session.get('_language', request.session.get('django_language'))
if lang_code in supported and lang_code is not None and check_for_language(lang_code): if lang_code in supported and lang_code is not None and check_for_language(lang_code):
return lang_code return lang_code

View File

@ -34,7 +34,7 @@ def set_language(request):
lang_code = request.POST.get('language', None) lang_code = request.POST.get('language', None)
if lang_code and check_for_language(lang_code): if lang_code and check_for_language(lang_code):
if hasattr(request, 'session'): if hasattr(request, 'session'):
request.session['django_language'] = lang_code request.session['_language'] = lang_code
else: else:
response.set_cookie(settings.LANGUAGE_COOKIE_NAME, lang_code) response.set_cookie(settings.LANGUAGE_COOKIE_NAME, lang_code)
return response return response

View File

@ -410,6 +410,9 @@ these changes.
* The ``Model._meta.get_(add|change|delete)_permission`` methods will * The ``Model._meta.get_(add|change|delete)_permission`` methods will
be removed. be removed.
* The session key ``django_language`` will no longer be read for backwards
compatibility.
1.9 1.9
--- ---

View File

@ -302,6 +302,13 @@ Internationalization
* The :attr:`django.middleware.locale.LocaleMiddleware.response_redirect_class` * The :attr:`django.middleware.locale.LocaleMiddleware.response_redirect_class`
attribute allows you to customize the redirects issued by the middleware. attribute allows you to customize the redirects issued by the middleware.
* The :class:`~django.middleware.locale.LocaleMiddleware` now stores the user's
selected language with the session key ``_language``. Previously it was
stored with the key ``django_language``, but keys reserved for Django should
start with an underscore. For backwards compatibility ``django_language`` is
still read from in 1.7. Sessions will be migrated to the new ``_language``
key as they are written.
Management Commands Management Commands
^^^^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^

View File

@ -1594,8 +1594,13 @@ following this algorithm:
root URLconf. See :ref:`url-internationalization` for more information root URLconf. See :ref:`url-internationalization` for more information
about the language prefix and how to internationalize URL patterns. about the language prefix and how to internationalize URL patterns.
* Failing that, it looks for a ``django_language`` key in the current * Failing that, it looks for a ``_language`` key in the current user's session.
user's session.
.. versionchanged:: 1.7
In previous versions, the key was named ``django_language`` but it was
renamed to start with an underscore to denote a Django reserved session
key.
* Failing that, it looks for a cookie. * Failing that, it looks for a cookie.

View File

@ -1190,17 +1190,39 @@ class LocaleMiddlewareTests(TransRealMixin, TestCase):
# Clear the session data before request # Clear the session data before request
session.save() session.save()
self.client.get('/en/simple/') response = self.client.get('/en/simple/')
self.assertEqual(self.client.session['django_language'], 'en') self.assertEqual(self.client.session['_language'], 'en')
# Clear the session data before request # Clear the session data before request
session.save() session.save()
self.client.get('/fr/simple/') response = self.client.get('/fr/simple/')
self.assertEqual(self.client.session['django_language'], 'fr') self.assertEqual(self.client.session['_language'], 'fr')
# Check that language is not changed in session # Check that language is not changed in session
self.client.get('/en/simple/') response = self.client.get('/en/simple/')
self.assertEqual(self.client.session['django_language'], 'fr') self.assertEqual(self.client.session['_language'], 'fr')
@override_settings(
MIDDLEWARE_CLASSES=(
'django.contrib.sessions.middleware.SessionMiddleware',
'django.middleware.locale.LocaleMiddleware',
'django.middleware.common.CommonMiddleware',
),
)
def test_backwards_session_language(self):
"""
Check that language is stored in session if missing.
"""
# Create session with old language key name
engine = import_module(settings.SESSION_ENGINE)
session = engine.SessionStore()
session['django_language'] = 'en'
session.save()
self.client.cookies[settings.SESSION_COOKIE_NAME] = session.session_key
# request other language; should default to old language key value
response = self.client.get('/fr/simple/')
self.assertEqual(self.client.session['_language'], 'en')
@override_settings( @override_settings(

View File

@ -34,7 +34,7 @@ class I18NTests(TestCase):
post_data = dict(language=lang_code, next='/views/') post_data = dict(language=lang_code, next='/views/')
response = self.client.post('/views/i18n/setlang/', data=post_data) response = self.client.post('/views/i18n/setlang/', data=post_data)
self.assertRedirects(response, 'http://testserver/views/') self.assertRedirects(response, 'http://testserver/views/')
self.assertEqual(self.client.session['django_language'], lang_code) self.assertEqual(self.client.session['_language'], lang_code)
def test_setlang_unsafe_next(self): def test_setlang_unsafe_next(self):
""" """
@ -45,7 +45,7 @@ class I18NTests(TestCase):
post_data = dict(language=lang_code, next='//unsafe/redirection/') post_data = dict(language=lang_code, next='//unsafe/redirection/')
response = self.client.post('/views/i18n/setlang/', data=post_data) response = self.client.post('/views/i18n/setlang/', data=post_data)
self.assertEqual(response.url, 'http://testserver/') self.assertEqual(response.url, 'http://testserver/')
self.assertEqual(self.client.session['django_language'], lang_code) self.assertEqual(self.client.session['_language'], lang_code)
def test_setlang_reversal(self): def test_setlang_reversal(self):
self.assertEqual(reverse('set_language'), '/views/i18n/setlang/') self.assertEqual(reverse('set_language'), '/views/i18n/setlang/')