mirror of
https://github.com/django/django.git
synced 2025-10-24 06:06:09 +00:00
Fixed #21386 -- Removed admindocs dependence on sites framework
* Removed ADMIN_FOR setting and warn warning * Group view functions by namespace instead of site * Added a test verifying namespaces are listed Thanks to Claude Paroz for reviewing and ideas for improvement.
This commit is contained in:
committed by
Claude Paroz
parent
f1b3ab9c21
commit
a39d672ec7
@@ -15,15 +15,12 @@
|
|||||||
{% block content %}
|
{% block content %}
|
||||||
<h1>{% blocktrans %}Template: "{{ name }}"{% endblocktrans %}</h1>
|
<h1>{% blocktrans %}Template: "{{ name }}"{% endblocktrans %}</h1>
|
||||||
|
|
||||||
{% regroup templates|dictsort:"site_id" by site as templates_by_site %}
|
<h2>{% blocktrans %}Search path for template "{{ name }}":{% endblocktrans %}</h2>
|
||||||
{% for group in templates_by_site %}
|
<ol>
|
||||||
<h2>{% blocktrans with group.grouper as grouper %}Search path for template "{{ name }}" on {{ grouper }}:{% endblocktrans %}</h2>
|
{% for template in templates|dictsort:"order" %}
|
||||||
<ol>
|
<li><code>{{ template.file }}</code>{% if not template.exists %} <em>{% trans '(does not exist)' %}</em>{% endif %}</li>
|
||||||
{% for template in group.list|dictsort:"order" %}
|
|
||||||
<li><code>{{ template.file }}</code>{% if not template.exists %} <em>{% trans '(does not exist)' %}</em>{% endif %}</li>
|
|
||||||
{% endfor %}
|
|
||||||
</ol>
|
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
</ol>
|
||||||
|
|
||||||
<p class="small"><a href="{% url 'django-admindocs-docroot' %}">‹ {% trans 'Back to Documentation' %}</a></p>
|
<p class="small"><a href="{% url 'django-admindocs-docroot' %}">‹ {% trans 'Back to Documentation' %}</a></p>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@@ -15,29 +15,40 @@
|
|||||||
|
|
||||||
<h1>{% trans 'View documentation' %}</h1>
|
<h1>{% trans 'View documentation' %}</h1>
|
||||||
|
|
||||||
{% regroup views|dictsort:"site_id" by site as views_by_site %}
|
{% regroup views|dictsort:'namespace' by namespace as views_by_ns %}
|
||||||
|
|
||||||
<div id="content-related" class="sidebar">
|
<div id="content-related" class="sidebar">
|
||||||
<div class="module">
|
<div class="module">
|
||||||
<h2>{% trans 'Jump to site' %}</h2>
|
<h2>{% trans 'Jump to namespace' %}</h2>
|
||||||
<ul>
|
<ul>
|
||||||
{% for site_views in views_by_site %}
|
{% for ns_views in views_by_ns %}
|
||||||
<li><a href="#site{{ site_views.grouper.id }}">{{ site_views.grouper.name }}</a></li>
|
<li><a href="#ns|{{ ns_views.grouper }}">
|
||||||
{% endfor %}
|
{% if ns_views.grouper %}{{ ns_views.grouper }}
|
||||||
|
{% else %}{% trans "Empty namespace" %}{% endif %}
|
||||||
|
</a></li>
|
||||||
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="content-main">
|
<div id="content-main">
|
||||||
|
|
||||||
{% for site_views in views_by_site %}
|
{% for ns_views in views_by_ns %}
|
||||||
<div class="module">
|
<div class="module">
|
||||||
<h2 id="site{{ site_views.grouper.id }}">{% blocktrans with site_views.grouper.name as name %}Views by URL on {{ name }}{% endblocktrans %}</h2>
|
<h2 id="ns|{{ ns_views.grouper }}">
|
||||||
|
{% if ns_views.grouper %}
|
||||||
|
{% blocktrans with ns_views.grouper as name %}Views by namespace {{ name }}{% endblocktrans %}
|
||||||
|
{% else %}
|
||||||
|
{% blocktrans %}Views by empty namespace{% endblocktrans %}
|
||||||
|
{% endif %}
|
||||||
|
</h2>
|
||||||
|
|
||||||
{% for view in site_views.list|dictsort:"url" %}
|
{% for view in ns_views.list|dictsort:"url" %}
|
||||||
{% ifchanged %}
|
{% ifchanged %}
|
||||||
<h3><a href="{% url 'django-admindocs-views-detail' view=view.full_name %}">{{ view.url }}</a></h3>
|
<h3><a href="{% url 'django-admindocs-views-detail' view=view.full_name %}">{{ view.url }}</a></h3>
|
||||||
<p class="small quiet">{% blocktrans with view.full_name as name %}View function: {{ name }}{% endblocktrans %}</p>
|
<p class="small quiet">{% blocktrans with view.full_name as full_name and view.url_name as url_name %}
|
||||||
|
View function: <code>{{ full_name }}</code>. Name: <code>{{ url_name }}</code>.
|
||||||
|
{% endblocktrans %}</p>
|
||||||
<p>{{ view.title }}</p>
|
<p>{{ view.title }}</p>
|
||||||
<hr />
|
<hr />
|
||||||
{% endifchanged %}
|
{% endifchanged %}
|
||||||
|
@@ -2,6 +2,7 @@ from importlib import import_module
|
|||||||
import inspect
|
import inspect
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
import warnings
|
||||||
|
|
||||||
from django import template
|
from django import template
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
@@ -13,7 +14,6 @@ from django.core.exceptions import ViewDoesNotExist
|
|||||||
from django.http import Http404
|
from django.http import Http404
|
||||||
from django.core import urlresolvers
|
from django.core import urlresolvers
|
||||||
from django.contrib.admindocs import utils
|
from django.contrib.admindocs import utils
|
||||||
from django.contrib.sites.models import Site
|
|
||||||
from django.utils.decorators import method_decorator
|
from django.utils.decorators import method_decorator
|
||||||
from django.utils._os import upath
|
from django.utils._os import upath
|
||||||
from django.utils import six
|
from django.utils import six
|
||||||
@@ -23,10 +23,10 @@ from django.views.generic import TemplateView
|
|||||||
# Exclude methods starting with these strings from documentation
|
# Exclude methods starting with these strings from documentation
|
||||||
MODEL_METHODS_EXCLUDE = ('_', 'add_', 'delete', 'save', 'set_')
|
MODEL_METHODS_EXCLUDE = ('_', 'add_', 'delete', 'save', 'set_')
|
||||||
|
|
||||||
|
if getattr(settings, 'ADMIN_FOR', None):
|
||||||
class GenericSite(object):
|
warnings.warn('The ADMIN_FOR setting has been removed, you can remove '
|
||||||
domain = 'example.com'
|
'this setting from your configuration.', DeprecationWarning,
|
||||||
name = 'my site'
|
stacklevel=2)
|
||||||
|
|
||||||
|
|
||||||
class BaseAdminDocsView(TemplateView):
|
class BaseAdminDocsView(TemplateView):
|
||||||
@@ -129,26 +129,17 @@ class ViewIndexView(BaseAdminDocsView):
|
|||||||
template_name = 'admin_doc/view_index.html'
|
template_name = 'admin_doc/view_index.html'
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
if settings.ADMIN_FOR:
|
|
||||||
settings_modules = [import_module(m) for m in settings.ADMIN_FOR]
|
|
||||||
else:
|
|
||||||
settings_modules = [settings]
|
|
||||||
|
|
||||||
views = []
|
views = []
|
||||||
for settings_mod in settings_modules:
|
urlconf = import_module(settings.ROOT_URLCONF)
|
||||||
urlconf = import_module(settings_mod.ROOT_URLCONF)
|
view_functions = extract_views_from_urlpatterns(urlconf.urlpatterns)
|
||||||
view_functions = extract_views_from_urlpatterns(urlconf.urlpatterns)
|
for (func, regex, namespace, name) in view_functions:
|
||||||
if Site._meta.installed:
|
views.append({
|
||||||
site_obj = Site.objects.get(pk=settings_mod.SITE_ID)
|
'full_name': '%s.%s' % (func.__module__, getattr(func, '__name__', func.__class__.__name__)),
|
||||||
else:
|
'url': simplify_regex(regex),
|
||||||
site_obj = GenericSite()
|
'url_name': ':'.join((namespace or []) + (name and [name] or [])),
|
||||||
for (func, regex) in view_functions:
|
'namespace': ':'.join((namespace or [])),
|
||||||
views.append({
|
'name': name,
|
||||||
'full_name': '%s.%s' % (func.__module__, getattr(func, '__name__', func.__class__.__name__)),
|
})
|
||||||
'site_id': settings_mod.SITE_ID,
|
|
||||||
'site': site_obj,
|
|
||||||
'url': simplify_regex(regex),
|
|
||||||
})
|
|
||||||
kwargs.update({'views': views})
|
kwargs.update({'views': views})
|
||||||
return super(ViewIndexView, self).get_context_data(**kwargs)
|
return super(ViewIndexView, self).get_context_data(**kwargs)
|
||||||
|
|
||||||
@@ -292,22 +283,14 @@ class TemplateDetailView(BaseAdminDocsView):
|
|||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
template = self.kwargs['template']
|
template = self.kwargs['template']
|
||||||
templates = []
|
templates = []
|
||||||
for site_settings_module in settings.ADMIN_FOR:
|
for dir in settings.TEMPLATE_DIRS:
|
||||||
settings_mod = import_module(site_settings_module)
|
template_file = os.path.join(dir, template)
|
||||||
if Site._meta.installed:
|
templates.append({
|
||||||
site_obj = Site.objects.get(pk=settings_mod.SITE_ID)
|
'file': template_file,
|
||||||
else:
|
'exists': os.path.exists(template_file),
|
||||||
site_obj = GenericSite()
|
'contents': lambda: open(template_file).read() if os.path.exists(template_file) else '',
|
||||||
for dir in settings_mod.TEMPLATE_DIRS:
|
'order': list(settings.TEMPLATE_DIRS).index(dir),
|
||||||
template_file = os.path.join(dir, template)
|
})
|
||||||
templates.append({
|
|
||||||
'file': template_file,
|
|
||||||
'exists': os.path.exists(template_file),
|
|
||||||
'contents': lambda: open(template_file).read() if os.path.exists(template_file) else '',
|
|
||||||
'site_id': settings_mod.SITE_ID,
|
|
||||||
'site': site_obj,
|
|
||||||
'order': list(settings_mod.TEMPLATE_DIRS).index(dir),
|
|
||||||
})
|
|
||||||
kwargs.update({
|
kwargs.update({
|
||||||
'name': template,
|
'name': template,
|
||||||
'templates': templates,
|
'templates': templates,
|
||||||
@@ -356,7 +339,7 @@ def get_readable_field_data_type(field):
|
|||||||
return field.description % field.__dict__
|
return field.description % field.__dict__
|
||||||
|
|
||||||
|
|
||||||
def extract_views_from_urlpatterns(urlpatterns, base=''):
|
def extract_views_from_urlpatterns(urlpatterns, base='', namespace=None):
|
||||||
"""
|
"""
|
||||||
Return a list of views from a list of urlpatterns.
|
Return a list of views from a list of urlpatterns.
|
||||||
|
|
||||||
@@ -369,10 +352,15 @@ def extract_views_from_urlpatterns(urlpatterns, base=''):
|
|||||||
patterns = p.url_patterns
|
patterns = p.url_patterns
|
||||||
except ImportError:
|
except ImportError:
|
||||||
continue
|
continue
|
||||||
views.extend(extract_views_from_urlpatterns(patterns, base + p.regex.pattern))
|
views.extend(extract_views_from_urlpatterns(
|
||||||
|
patterns,
|
||||||
|
base + p.regex.pattern,
|
||||||
|
(namespace or []) + (p.namespace and [p.namespace] or [])
|
||||||
|
))
|
||||||
elif hasattr(p, 'callback'):
|
elif hasattr(p, 'callback'):
|
||||||
try:
|
try:
|
||||||
views.append((p.callback, base + p.regex.pattern))
|
views.append((p.callback, base + p.regex.pattern,
|
||||||
|
namespace, p.name))
|
||||||
except ViewDoesNotExist:
|
except ViewDoesNotExist:
|
||||||
continue
|
continue
|
||||||
else:
|
else:
|
||||||
|
@@ -28,8 +28,6 @@ the following:
|
|||||||
``r'^admin/'`` entry, so that requests to ``/admin/doc/`` don't get
|
``r'^admin/'`` entry, so that requests to ``/admin/doc/`` don't get
|
||||||
handled by the latter entry.
|
handled by the latter entry.
|
||||||
* Install the docutils Python module (http://docutils.sf.net/).
|
* Install the docutils Python module (http://docutils.sf.net/).
|
||||||
* **Optional:** Linking to templates requires the :setting:`ADMIN_FOR`
|
|
||||||
setting to be configured.
|
|
||||||
* **Optional:** Using the admindocs bookmarklets requires
|
* **Optional:** Using the admindocs bookmarklets requires
|
||||||
``django.contrib.admindocs.middleware.XViewMiddleware`` to be installed.
|
``django.contrib.admindocs.middleware.XViewMiddleware`` to be installed.
|
||||||
|
|
||||||
|
@@ -2097,25 +2097,6 @@ The default value for the X-Frame-Options header used by
|
|||||||
:doc:`clickjacking protection </ref/clickjacking/>` documentation.
|
:doc:`clickjacking protection </ref/clickjacking/>` documentation.
|
||||||
|
|
||||||
|
|
||||||
Admindocs
|
|
||||||
=========
|
|
||||||
|
|
||||||
Settings for :mod:`django.contrib.admindocs`.
|
|
||||||
|
|
||||||
.. setting:: ADMIN_FOR
|
|
||||||
|
|
||||||
ADMIN_FOR
|
|
||||||
---------
|
|
||||||
|
|
||||||
Default: ``()`` (Empty tuple)
|
|
||||||
|
|
||||||
Used for admin-site settings modules, this should be a tuple of settings
|
|
||||||
modules (in the format ``'foo.bar.baz'``) for which this site is an admin.
|
|
||||||
|
|
||||||
The admin site uses this in its automatically-introspected documentation of
|
|
||||||
models, views and template tags.
|
|
||||||
|
|
||||||
|
|
||||||
Auth
|
Auth
|
||||||
====
|
====
|
||||||
|
|
||||||
|
@@ -880,3 +880,9 @@ Callable arguments were evaluated when a queryset was constructed rather than
|
|||||||
when it was evaluated, thus this feature didn't offer any benefit compared to
|
when it was evaluated, thus this feature didn't offer any benefit compared to
|
||||||
evaluating arguments before passing them to queryset and created confusion that
|
evaluating arguments before passing them to queryset and created confusion that
|
||||||
the arguments may have been evaluated at query time.
|
the arguments may have been evaluated at query time.
|
||||||
|
|
||||||
|
``ADMIN_FOR`` setting
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
The ``ADMIN_FOR`` feature, part of the admindocs, has been removed. You can
|
||||||
|
remove the setting from your configuration at your convenience.
|
||||||
|
@@ -1,5 +1,7 @@
|
|||||||
import unittest
|
import unittest
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
from django.contrib.sites.models import Site
|
||||||
from django.contrib.admindocs import utils
|
from django.contrib.admindocs import utils
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from django.core.urlresolvers import reverse
|
from django.core.urlresolvers import reverse
|
||||||
@@ -7,6 +9,33 @@ from django.test import TestCase
|
|||||||
from django.test.utils import override_settings
|
from django.test.utils import override_settings
|
||||||
|
|
||||||
|
|
||||||
|
class MiscTests(TestCase):
|
||||||
|
urls = 'admin_docs.urls'
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self._old_installed = Site._meta.app_config.installed
|
||||||
|
User.objects.create_superuser('super', None, 'secret')
|
||||||
|
self.client.login(username='super', password='secret')
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
Site._meta.app_config.installed = self._old_installed
|
||||||
|
|
||||||
|
@override_settings(
|
||||||
|
SITE_ID=None,
|
||||||
|
INSTALLED_APPS=[app for app in settings.INSTALLED_APPS
|
||||||
|
if app != 'django.contrib.sites'],
|
||||||
|
)
|
||||||
|
def test_no_sites_framework(self):
|
||||||
|
"""
|
||||||
|
Without the sites framework, should not access SITE_ID or Site
|
||||||
|
objects. Deleting settings is fine here as UserSettingsHolder is used.
|
||||||
|
"""
|
||||||
|
Site._meta.app_config.installed = False
|
||||||
|
Site.objects.all().delete()
|
||||||
|
del settings.SITE_ID
|
||||||
|
self.client.get('/admindocs/views/') # should not raise
|
||||||
|
|
||||||
|
|
||||||
@override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
|
@override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
|
||||||
@unittest.skipUnless(utils.docutils_is_available, "no docutils installed.")
|
@unittest.skipUnless(utils.docutils_is_available, "no docutils installed.")
|
||||||
class AdminDocViewTests(TestCase):
|
class AdminDocViewTests(TestCase):
|
||||||
@@ -46,6 +75,8 @@ class AdminDocViewTests(TestCase):
|
|||||||
self.assertContains(response,
|
self.assertContains(response,
|
||||||
'<h3><a href="/admindocs/views/django.contrib.admindocs.views.BaseAdminDocsView/">/admindocs/</a></h3>',
|
'<h3><a href="/admindocs/views/django.contrib.admindocs.views.BaseAdminDocsView/">/admindocs/</a></h3>',
|
||||||
html=True)
|
html=True)
|
||||||
|
self.assertContains(response, 'Views by namespace test')
|
||||||
|
self.assertContains(response, 'Name: <code>test:func</code>.')
|
||||||
|
|
||||||
def test_view_detail(self):
|
def test_view_detail(self):
|
||||||
response = self.client.get(
|
response = self.client.get(
|
||||||
|
@@ -1,11 +1,16 @@
|
|||||||
from django.conf.urls import include, patterns
|
from django.conf.urls import include, patterns, url
|
||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
|
|
||||||
from . import views
|
from . import views
|
||||||
|
|
||||||
|
ns_patterns = patterns('',
|
||||||
|
url(r'^xview/func/$', views.xview_dec(views.xview), name='func'),
|
||||||
|
)
|
||||||
|
|
||||||
urlpatterns = patterns('',
|
urlpatterns = patterns('',
|
||||||
(r'^admin/', include(admin.site.urls)),
|
(r'^admin/', include(admin.site.urls)),
|
||||||
(r'^admindocs/', include('django.contrib.admindocs.urls')),
|
(r'^admindocs/', include('django.contrib.admindocs.urls')),
|
||||||
|
(r'^', include(ns_patterns, namespace='test')),
|
||||||
(r'^xview/func/$', views.xview_dec(views.xview)),
|
(r'^xview/func/$', views.xview_dec(views.xview)),
|
||||||
(r'^xview/class/$', views.xview_dec(views.XViewClass.as_view())),
|
(r'^xview/class/$', views.xview_dec(views.XViewClass.as_view())),
|
||||||
)
|
)
|
||||||
|
Reference in New Issue
Block a user