从Django中的Generic View DetailView重定向

时间:2011-06-23 15:26:26

标签: django generics seo views

我正在使用Django基于类的DetailView通用视图来查找要显示的对象。在某些情况下,我希望退出并发出HTTP重定向,而不是显示对象。我看不出我是怎么做的。它适用于用户点击我的应用中的对象但不使用规范网址的情况。因此,例如,在StackOverflow URL上采用以下形式:

http://stackoverflow.com/<content_type>/<pk>/<seo_friendly_slug>

例如:

http://stackoverflow.com/questions/5661806/django-debug-toolbar-with-django-cms-and-django-1-3

您实际上可以输入任何内容作为seo_friendly_slug部分,它会将您重定向到通过PK查找的对象的正确规范URL。

我希望在我的DetailView中也这样做。检索对象,检查它是否为规范URL,如果没有,则重定向到项目的get_absolute_url URL。

我无法在get_object中返回HttpResponseRedirect,因为它期望查找对象。我似乎无法从get_context_data返回它,因为它只是期待上下文数据。

也许我只需要编写一个手动视图,但我想知道是否有人知道这是否可能?

谢谢!

2 个答案:

答案 0 :(得分:15)

这不适合DetailView。为此,您需要覆盖BaseDetailView的get方法,如下所示:

class BaseDetailView(SingleObjectMixin, View):
    def get(self, request, **kwargs):
        self.object = self.get_object()
        context = self.get_context_data(object=self.object)
        return self.render_to_response(context)

因此,在您的班级中,您需要提供一个新的get方法,该方法在获取对象和设置上下文之间进行URL检查。类似的东西:

def get(self, request, **kwargs):
    self.object = self.get_object()
    if self.request.path != self.object.get_absolute_url():
        return HttpResponseRedirect(self.object.get_absolute_url())
    else:
        context = self.get_context_data(object=self.object)
        return self.render_to_response(context)

当你最终覆盖了这么多的功能时,它是否值得为此实际使用通用视图值得怀疑,但是你知道。

答案 1 :(得分:10)

根据Rolo的回答和评论,我提出了以下通用视图来实现此目的:

from django import http
from django.views import generic


class CanonicalDetailView(generic.DetailView):
    """
        A DetailView which redirects to the absolute_url, if necessary.
    """
    def get_object(self, *args, **kwargs):
        # Return any previously-cached object
        if getattr(self, 'object', None):
            return self.object
        return super(CanonicalDetailView, self).get_object(*args, **kwargs)

    def get(self, *args, **kwargs):
        # Make sure to use the canonical URL
        self.object = self.get_object()
        obj_url = self.object.get_absolute_url()
        if self.request.path != obj_url:
            return http.HttpResponsePermanentRedirect(obj_url)
        return super(CanonicalDetailView, self).get(*args, **kwargs);

它的使用方法与普通的DetailView相同,适用于任何正确实现get_absolute_url的模型。