【问题标题】:django admin, extending admin with custom viewsdjango admin,使用自定义视图扩展 admin
【发布时间】:2016-03-08 18:44:24
【问题描述】:

我想就此事请求一些帮助。

我已按照this 指南向我的管理员添加视图。

我正在使用与网站相同的代码,但出现错误:

Reverse for 'app_list' with arguments '()' and keyword arguments '{'app_label': 'reports'}' not found.

admin/my_view 工作正常,但如果我尝试访问管理员中的其他页面,例如索引页面,则会出现错误

代码如下:

#someapp/admin.py

from django.conf.urls import patterns
from django.contrib import admin
from django.http import HttpResponse

def my_view(request):
    return HttpResponse("Hello!")

def get_admin_urls(urls):
    def get_urls():
        my_urls = patterns('',
            (r'^my_view/$', admin.site.admin_view(my_view))
         )
        return my_urls + urls
    return get_urls

admin_urls = get_admin_urls(admin.site.get_urls())
admin.site.get_urls = admin_urls

我正在使用 django 1.8.2、python 2.7

另外,在管理员的索引页面中添加一些视图或上下文的最佳方法是什么?

更新

我已经尝试过 Antoine Pinsard 的回答并尝试了这个:

#admin.py
from django.contrib.admin import AdminSite
class MyAdminSite(AdminSite):

    def get_urls(self):
        from django.conf.urls import url
        urls = super(MyAdminSite, self).get_urls()
        urls += [
            url(r'^my_view/$', self.admin_view(self.my_view))
        ]
        return urls

    def my_view(self, request):
        return HttpResponse("Hello!")

admin_site = MyAdminSite()

并将 urls.py 的 url(r'^admin/', include(admin.site.urls)), 替换为 url(r'^admin/', include(admin_site.urls)),

得到:

Reverse for 'app_list' with arguments '()' and keyword arguments

回溯:

Request Method: GET
Request URL: http://localhost:8000/@dmin/

Django Version: 1.8.2
Python Version: 2.7.10
Installed Applications:
('grappelli',
 'django.contrib.admin',
 'django.contrib.auth',
 'django.contrib.contenttypes',
 'django.contrib.sessions',
 'django.contrib.messages',
 'django.contrib.staticfiles',
 'django.contrib.sessions.models',
 'frontend',
 'file_maintenance',
 'reports',
 'transactions',
 'admin_reorder',
 'admin_notifications',
 'django_twilio',
 'daterange_filter')
Installed Middleware:
('django.contrib.sessions.middleware.SessionMiddleware',
 'django.contrib.auth.models.User',
 'django.middleware.common.CommonMiddleware',
 'django.middleware.csrf.CsrfViewMiddleware',
 'django.contrib.auth.middleware.AuthenticationMiddleware',
 'django.contrib.auth.middleware.SessionAuthenticationMiddleware',
 'django.contrib.messages.middleware.MessageMiddleware',
 'django.middleware.clickjacking.XFrameOptionsMiddleware',
 'django.middleware.security.SecurityMiddleware',
 'admin_reorder.middleware.ModelAdminReorder')


Traceback:
File "C:\Python27\lib\site-packages\django\core\handlers\base.py" in get_response
  157.                     response = middleware_method(request, response)
File "C:\Python27\lib\site-packages\admin_reorder\middleware.py" in process_template_response
  134.         self.init_config(request, app_list)
File "C:\Python27\lib\site-packages\admin_reorder\middleware.py" in init_config
  21.         admin_index = admin.site.index(request)
File "C:\Python27\lib\site-packages\django\views\decorators\cache.py" in _wrapped_view_func
  57.         response = view_func(request, *args, **kwargs)
File "C:\Python27\lib\site-packages\django\contrib\admin\sites.py" in index
  438.                                 current_app=self.name,
File "C:\Python27\lib\site-packages\django\core\urlresolvers.py" in reverse
  579.     return force_text(iri_to_uri(resolver._reverse_with_prefix(view, prefix, *args, **kwargs)))
File "C:\Python27\lib\site-packages\django\core\urlresolvers.py" in _reverse_with_prefix
  496.                              (lookup_view_s, args, kwargs, len(patterns), patterns))

Exception Type: NoReverseMatch at /@dmin/
Exception Value: Reverse for 'app_list' with arguments '()' and keyword arguments '{'app_label': 'django_twilio'}' not found. 0 pattern(s) tried: []

【问题讨论】:

标签: python django


【解决方案1】:

本指南看起来很旧。我宁愿建议你关注django docs

someapp/admin.py

from django.contrib.admin import AdminSite
from django.http import HttpResponse

class MyAdminSite(AdminSite):

     def get_urls(self):
         from django.urls import path
         urls = super().get_urls()
         urls += [
             path('my_view/', self.admin_view(self.my_view))
         ]
         return urls

     def my_view(self, request):
         return HttpResponse("Hello!")

admin_site = MyAdminSite()

来源:https://github.com/django/django/blob/2.2/django/contrib/admin/sites.py#L194-L205

您还应该更新您的project/urls.py 并将path('admin/', admin.site.urls) 替换为path('admin/', admin_site.urls)。之前别忘了from someapp.admin import admin_site

【讨论】:

    【解决方案2】:

    您链接的指南很旧,我很惊讶在过去一年左右没有找到直接回答您问题的任何内容。

    1. 在您应用的 admin.py 或方便的地方创建一个新的Admin Site
    2. 在新的 AdminSite 中创建一个函数,使用您的额外 URL 扩充 get_urls() 函数。
    3. 确保您的项目 urls.py 链接到新创建的 AdminSite

    以下适用于 Python 3.5.1 和 Django 1.9.6。


    my_app/admin.py

    from django.contrib import admin
    from django.contrib.admin import AdminSite
    from django.http import HttpResponse
    
    from my_app.models import SomeModel
    
    
    class MyAdminSite(AdminSite):
    
        def custom_view(self, request):
            return HttpResponse("Test")
    
        def get_urls(self):
            from django.conf.urls import url
            urls = super(MyAdminSite, self).get_urls()
            urls += [
                url(r'^custom_view/$', self.admin_view(self.custom_view))
            ]
            return urls
    
    admin_site = MyAdminSite()
    
    
    @admin.register(SomeModel, site=admin_site)
    class SomeModelAdmin(admin.ModelAdmin):
        pass
    

    my_project/urls.py

    from django.conf.urls import url, include
    
    from my_app.admin import admin_site
    
    urlpatterns = [
        url(r'^admin/', admin_site.urls),
        ...
    ]
    

    【讨论】:

    • 赞成一个清晰有效的例子。知道如何使用多个项目而不是一个项目吗?
    • @dKen 我很困惑。你的意思是多个 Django 项目吗?如果是这样,我不确定这是否可能。
    • 非常抱歉,我混淆了我的命名法!我的意思是问我怎样才能让它与多个应用程序一起工作。很抱歉造成混乱!
    • 嗯,这适用于多个应用程序...您只需将实例化的类导入所有应用程序:from common.admin import admin_site
    • @DanielvanFlymen ,第三方应用程序管理员会以这种方式包含作品吗?我的意思是,某些软件包在安装并添加到 INSTALLED_APPS 后,会自动将自己绑定到具有所有功能的索引页面。
    【解决方案3】:

    对于 Django 1.4+,这是解决方案:

    from django.conf.urls import url
    from django.contrib import admin
    from .models import MyModel
    
    
    class MyAdmin(admin.ModelAdmin):
        list_display = (...)
    
        def custom_admin_view(self, request):
            # your logic here
            if request.method == 'POST':
                ...
            else:
                ...
            return HttpResponse(...)
    
        def get_urls(self):
            additional_urls = [
                url(r'^custom/$', self.admin_site.admin_view(self.custom_admin_view), name='custom')
            ]
            # append your custom URL BEFORE default ones
            return additional_urls + super().get_urls()
    
    
    admin.site.register(MyModel, MyAdmin)
    

    在所有其他网址之前附加您的自定义网址很重要。否则,您可能会被重定向到管理员 change_view

    【讨论】:

      【解决方案4】:

      下面的这段代码将override默认的管理站点来编写你的自定义视图。

      适用于 Django >= 2.1:

      ###myproject/admin.py

      from django.contrib import admin
      from django.http import HttpResponse
      
      class MyAdminSite(admin.AdminSite):
      
          def my_view(self, request): # your custom view function
              return HttpResponse("Test")
      
          def get_urls(self):
              from django.urls import path
      
              urlpatterns = super().get_urls()
              urlpatterns += [
                  path('my_view/', self.admin_view(self.my_view))
              ]
              return urlpatterns
      

      ###myproject/apps.py

      from django.contrib.admin.apps import AdminConfig
      
      class MyAdminConfig(AdminConfig):
          default_site = 'myproject.admin.MyAdminSite'
      

      ###myproject/settings.py

      INSTALLED_APPS = [
          ...
          'myproject.apps.MyAdminConfig',  # replaces 'django.contrib.admin'
          ...
      ]
      

      那就去http://127.0.0.1:8000/admin/my_view/

      【讨论】:

        猜你喜欢
        • 2012-02-27
        • 1970-01-01
        • 2018-07-23
        • 2011-10-12
        • 1970-01-01
        • 2015-09-17
        • 2020-12-14
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多