2011-02-21 18:12:23 +08:00
|
|
|
from copy import copy
|
2006-05-02 09:31:56 +08:00
|
|
|
from django.core.exceptions import ImproperlyConfigured
|
2009-03-19 00:55:59 +08:00
|
|
|
from django.utils.importlib import import_module
|
2011-02-21 18:12:23 +08:00
|
|
|
from django.http import HttpRequest
|
2006-05-02 09:31:56 +08:00
|
|
|
|
Fixed #9977 - CsrfMiddleware gets template tag added, session dependency removed, and turned on by default.
This is a large change to CSRF protection for Django. It includes:
* removing the dependency on the session framework.
* deprecating CsrfResponseMiddleware, and replacing with a core template tag.
* turning on CSRF protection by default by adding CsrfViewMiddleware to
the default value of MIDDLEWARE_CLASSES.
* protecting all contrib apps (whatever is in settings.py)
using a decorator.
For existing users of the CSRF functionality, it should be a seamless update,
but please note that it includes DEPRECATION of features in Django 1.1,
and there are upgrade steps which are detailed in the docs.
Many thanks to 'Glenn' and 'bthomas', who did a lot of the thinking and work
on the patch, and to lots of other people including Simon Willison and
Russell Keith-Magee who refined the ideas.
Details of the rationale for these changes is found here:
http://code.djangoproject.com/wiki/CsrfProtection
As of this commit, the CSRF code is mainly in 'contrib'. The code will be
moved to core in a separate commit, to make the changeset as readable as
possible.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@11660 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2009-10-27 07:23:07 +08:00
|
|
|
# Cache of actual callables.
|
2006-05-02 09:31:56 +08:00
|
|
|
_standard_context_processors = None
|
Fixed #9977 - CsrfMiddleware gets template tag added, session dependency removed, and turned on by default.
This is a large change to CSRF protection for Django. It includes:
* removing the dependency on the session framework.
* deprecating CsrfResponseMiddleware, and replacing with a core template tag.
* turning on CSRF protection by default by adding CsrfViewMiddleware to
the default value of MIDDLEWARE_CLASSES.
* protecting all contrib apps (whatever is in settings.py)
using a decorator.
For existing users of the CSRF functionality, it should be a seamless update,
but please note that it includes DEPRECATION of features in Django 1.1,
and there are upgrade steps which are detailed in the docs.
Many thanks to 'Glenn' and 'bthomas', who did a lot of the thinking and work
on the patch, and to lots of other people including Simon Willison and
Russell Keith-Magee who refined the ideas.
Details of the rationale for these changes is found here:
http://code.djangoproject.com/wiki/CsrfProtection
As of this commit, the CSRF code is mainly in 'contrib'. The code will be
moved to core in a separate commit, to make the changeset as readable as
possible.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@11660 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2009-10-27 07:23:07 +08:00
|
|
|
# We need the CSRF processor no matter what the user has in their settings,
|
|
|
|
# because otherwise it is a security vulnerability, and we can't afford to leave
|
|
|
|
# this to human error or failure to read migration instructions.
|
2009-10-27 08:36:34 +08:00
|
|
|
_builtin_context_processors = ('django.core.context_processors.csrf',)
|
2006-05-02 09:31:56 +08:00
|
|
|
|
|
|
|
class ContextPopException(Exception):
|
|
|
|
"pop() has been called more times than push()"
|
|
|
|
pass
|
|
|
|
|
2011-02-27 08:24:35 +08:00
|
|
|
class EmptyClass(object):
|
|
|
|
# No-op class which takes no args to its __init__ method, to help implement
|
|
|
|
# __copy__
|
|
|
|
pass
|
|
|
|
|
2009-12-14 20:08:23 +08:00
|
|
|
class BaseContext(object):
|
|
|
|
def __init__(self, dict_=None):
|
2006-05-02 09:31:56 +08:00
|
|
|
dict_ = dict_ or {}
|
|
|
|
self.dicts = [dict_]
|
|
|
|
|
2011-02-21 18:12:23 +08:00
|
|
|
def __copy__(self):
|
2011-02-27 08:24:35 +08:00
|
|
|
duplicate = EmptyClass()
|
|
|
|
duplicate.__class__ = self.__class__
|
|
|
|
duplicate.__dict__ = self.__dict__.copy()
|
|
|
|
duplicate.dicts = duplicate.dicts[:]
|
2011-02-21 18:12:23 +08:00
|
|
|
return duplicate
|
|
|
|
|
2006-05-02 09:31:56 +08:00
|
|
|
def __repr__(self):
|
|
|
|
return repr(self.dicts)
|
|
|
|
|
|
|
|
def __iter__(self):
|
2009-12-14 20:08:23 +08:00
|
|
|
for d in reversed(self.dicts):
|
2006-05-02 09:31:56 +08:00
|
|
|
yield d
|
|
|
|
|
|
|
|
def push(self):
|
2007-12-03 07:57:22 +08:00
|
|
|
d = {}
|
2009-12-14 20:08:23 +08:00
|
|
|
self.dicts.append(d)
|
2007-12-03 07:57:22 +08:00
|
|
|
return d
|
2006-05-02 09:31:56 +08:00
|
|
|
|
|
|
|
def pop(self):
|
|
|
|
if len(self.dicts) == 1:
|
|
|
|
raise ContextPopException
|
2009-12-14 20:08:23 +08:00
|
|
|
return self.dicts.pop()
|
2006-05-02 09:31:56 +08:00
|
|
|
|
|
|
|
def __setitem__(self, key, value):
|
|
|
|
"Set a variable in the current context"
|
2009-12-14 20:08:23 +08:00
|
|
|
self.dicts[-1][key] = value
|
2006-05-02 09:31:56 +08:00
|
|
|
|
|
|
|
def __getitem__(self, key):
|
|
|
|
"Get a variable's value, starting at the current context and going upward"
|
2009-12-14 20:08:23 +08:00
|
|
|
for d in reversed(self.dicts):
|
2007-04-26 21:30:48 +08:00
|
|
|
if key in d:
|
2006-05-02 09:31:56 +08:00
|
|
|
return d[key]
|
2006-07-04 11:21:44 +08:00
|
|
|
raise KeyError(key)
|
2006-05-02 09:31:56 +08:00
|
|
|
|
|
|
|
def __delitem__(self, key):
|
|
|
|
"Delete a variable from the current context"
|
2009-12-14 20:08:23 +08:00
|
|
|
del self.dicts[-1][key]
|
2006-05-02 09:31:56 +08:00
|
|
|
|
|
|
|
def has_key(self, key):
|
|
|
|
for d in self.dicts:
|
2007-04-26 21:30:48 +08:00
|
|
|
if key in d:
|
2006-05-02 09:31:56 +08:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2009-12-14 20:08:23 +08:00
|
|
|
def __contains__(self, key):
|
|
|
|
return self.has_key(key)
|
2007-02-10 17:02:00 +08:00
|
|
|
|
2006-07-04 11:21:44 +08:00
|
|
|
def get(self, key, otherwise=None):
|
2009-12-14 20:08:23 +08:00
|
|
|
for d in reversed(self.dicts):
|
2007-04-26 21:30:48 +08:00
|
|
|
if key in d:
|
2006-05-02 09:31:56 +08:00
|
|
|
return d[key]
|
|
|
|
return otherwise
|
|
|
|
|
2009-12-14 20:08:23 +08:00
|
|
|
class Context(BaseContext):
|
|
|
|
"A stack container for variable context"
|
2010-10-30 00:48:58 +08:00
|
|
|
def __init__(self, dict_=None, autoescape=True, current_app=None, use_l10n=None):
|
2009-12-14 20:08:23 +08:00
|
|
|
self.autoescape = autoescape
|
2010-10-30 00:48:58 +08:00
|
|
|
self.use_l10n = use_l10n
|
2009-12-14 20:08:23 +08:00
|
|
|
self.current_app = current_app
|
|
|
|
self.render_context = RenderContext()
|
|
|
|
super(Context, self).__init__(dict_)
|
|
|
|
|
2011-02-21 18:12:23 +08:00
|
|
|
def __copy__(self):
|
|
|
|
duplicate = super(Context, self).__copy__()
|
|
|
|
duplicate.render_context = copy(self.render_context)
|
|
|
|
return duplicate
|
|
|
|
|
2006-05-02 09:31:56 +08:00
|
|
|
def update(self, other_dict):
|
2010-11-24 08:36:36 +08:00
|
|
|
"Pushes other_dict to the stack of dictionaries in the Context"
|
2009-03-19 00:55:59 +08:00
|
|
|
if not hasattr(other_dict, '__getitem__'):
|
2008-08-02 05:37:38 +08:00
|
|
|
raise TypeError('other_dict must be a mapping (dictionary-like) object.')
|
2009-12-14 20:08:23 +08:00
|
|
|
self.dicts.append(other_dict)
|
2007-12-03 07:57:22 +08:00
|
|
|
return other_dict
|
2006-05-02 09:31:56 +08:00
|
|
|
|
2009-12-14 20:08:23 +08:00
|
|
|
class RenderContext(BaseContext):
|
|
|
|
"""
|
|
|
|
A stack container for storing Template state.
|
|
|
|
|
|
|
|
RenderContext simplifies the implementation of template Nodes by providing a
|
|
|
|
safe place to store state between invocations of a node's `render` method.
|
|
|
|
|
|
|
|
The RenderContext also provides scoping rules that are more sensible for
|
|
|
|
'template local' variables. The render context stack is pushed before each
|
|
|
|
template is rendered, creating a fresh scope with nothing in it. Name
|
|
|
|
resolution fails if a variable is not found at the top of the RequestContext
|
|
|
|
stack. Thus, variables are local to a specific template and don't affect the
|
|
|
|
rendering of other templates as they would if they were stored in the normal
|
|
|
|
template context.
|
|
|
|
"""
|
|
|
|
def __iter__(self):
|
|
|
|
for d in self.dicts[-1]:
|
|
|
|
yield d
|
|
|
|
|
|
|
|
def has_key(self, key):
|
|
|
|
return key in self.dicts[-1]
|
|
|
|
|
|
|
|
def get(self, key, otherwise=None):
|
|
|
|
d = self.dicts[-1]
|
|
|
|
if key in d:
|
|
|
|
return d[key]
|
|
|
|
return otherwise
|
|
|
|
|
2006-05-02 09:31:56 +08:00
|
|
|
# This is a function rather than module-level procedural code because we only
|
|
|
|
# want it to execute if somebody uses RequestContext.
|
|
|
|
def get_standard_processors():
|
2009-03-31 04:15:01 +08:00
|
|
|
from django.conf import settings
|
2006-05-02 09:31:56 +08:00
|
|
|
global _standard_context_processors
|
|
|
|
if _standard_context_processors is None:
|
|
|
|
processors = []
|
Fixed #9977 - CsrfMiddleware gets template tag added, session dependency removed, and turned on by default.
This is a large change to CSRF protection for Django. It includes:
* removing the dependency on the session framework.
* deprecating CsrfResponseMiddleware, and replacing with a core template tag.
* turning on CSRF protection by default by adding CsrfViewMiddleware to
the default value of MIDDLEWARE_CLASSES.
* protecting all contrib apps (whatever is in settings.py)
using a decorator.
For existing users of the CSRF functionality, it should be a seamless update,
but please note that it includes DEPRECATION of features in Django 1.1,
and there are upgrade steps which are detailed in the docs.
Many thanks to 'Glenn' and 'bthomas', who did a lot of the thinking and work
on the patch, and to lots of other people including Simon Willison and
Russell Keith-Magee who refined the ideas.
Details of the rationale for these changes is found here:
http://code.djangoproject.com/wiki/CsrfProtection
As of this commit, the CSRF code is mainly in 'contrib'. The code will be
moved to core in a separate commit, to make the changeset as readable as
possible.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@11660 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2009-10-27 07:23:07 +08:00
|
|
|
collect = []
|
|
|
|
collect.extend(_builtin_context_processors)
|
|
|
|
collect.extend(settings.TEMPLATE_CONTEXT_PROCESSORS)
|
|
|
|
for path in collect:
|
2006-05-02 09:31:56 +08:00
|
|
|
i = path.rfind('.')
|
|
|
|
module, attr = path[:i], path[i+1:]
|
|
|
|
try:
|
2009-03-19 00:55:59 +08:00
|
|
|
mod = import_module(module)
|
2006-05-02 09:31:56 +08:00
|
|
|
except ImportError, e:
|
2007-12-23 03:32:54 +08:00
|
|
|
raise ImproperlyConfigured('Error importing request processor module %s: "%s"' % (module, e))
|
2006-05-02 09:31:56 +08:00
|
|
|
try:
|
|
|
|
func = getattr(mod, attr)
|
|
|
|
except AttributeError:
|
2007-12-23 03:32:54 +08:00
|
|
|
raise ImproperlyConfigured('Module "%s" does not define a "%s" callable request processor' % (module, attr))
|
2006-05-02 09:31:56 +08:00
|
|
|
processors.append(func)
|
|
|
|
_standard_context_processors = tuple(processors)
|
|
|
|
return _standard_context_processors
|
|
|
|
|
|
|
|
class RequestContext(Context):
|
|
|
|
"""
|
|
|
|
This subclass of template.Context automatically populates itself using
|
|
|
|
the processors defined in TEMPLATE_CONTEXT_PROCESSORS.
|
|
|
|
Additional processors can be specified as a list of callables
|
|
|
|
using the "processors" keyword argument.
|
|
|
|
"""
|
2010-10-30 00:48:58 +08:00
|
|
|
def __init__(self, request, dict=None, processors=None, current_app=None, use_l10n=None):
|
|
|
|
Context.__init__(self, dict, current_app=current_app, use_l10n=use_l10n)
|
2006-05-02 09:31:56 +08:00
|
|
|
if processors is None:
|
|
|
|
processors = ()
|
|
|
|
else:
|
|
|
|
processors = tuple(processors)
|
|
|
|
for processor in get_standard_processors() + processors:
|
|
|
|
self.update(processor(request))
|