使用Djangos按回复数订购配方查询集

2024-05-14 09:04:42 发布

您现在位置:Python中文网/ 问答频道 /正文

我有一个recipe模型:

class Recipe(models.Model):
    title = models.CharField(_("Recipe Title"), max_length=250)
    slug = models.SlugField(_('slug'), unique=True)
    author = models.ForeignKey(User, verbose_name=_('user'))
...

我需要为大多数被评论的菜谱创建一个视图,我正在使用django-disqus来处理菜谱的评论,但是我不知道如何根据菜谱的评论编号来排序查询集。你知道吗

class PopRecipeListView(GlobalQueryMixin, ListView):
    model = Recipe
    context_object_name = "recipe_list"
    template_name = 'recipe/recipe_top_list.html'

    def get_queryset(self):
        qs = super(PopRecipeListView, self).get_queryset()

        if qs:
            qs = qs.extra(
                select={
                    'comments': # get comment numbers
                }
            ).filter(shared=True).order_by('-rate')[:20]

        return qs

我在文档中发现:https://disqus.com/api/docs/forums/listThreads/ 我可以得到一个带有站点和论坛id的json文件,并对它们进行迭代以获得正确的密钥,但我认为这并不容易。你知道吗

再深入一点,我可以使用以下方法获取json文件: var url = "https://disqus.com/api/3.0/forums/listThreads.json?api_key=" + api_key + "&forum=" + forum; //+ "&limit=100";,但我不确定。你知道吗

好的,我有下一个函数来获取参数:

def get_comments_by_site():
    api_key = settings.DISQUS_API_KEY
    forum = settings.DISQUS_WEBSITE_SHORTNAME

    url = 'https://disqus.com/api/3.0/forums/listThreads.json?api_key={0}&forum={1}&limit=100'.format(api_key, forum)

    raw = requests.get(url)
    dict = json.loads(raw)

    values = {}

    for thread in dict['response']:
        item = {thread['slug']: thread['posts']}
        values.update(item)

    return values


def get_comments_number(slug):
    return values[slug]

我创建了两个不同的函数,因为我希望View一次获取所有值,然后在query中调用secong函数从获取的字典中获取值。你知道吗

我创建了一个mixin来集成这两个函数:

class DisqusCommentsNumber(object):
    values = {}

    def get_comments_by_site(self):
        api_key = settings.DISQUS_API_KEY
        forum = settings.DISQUS_WEBSITE_SHORTNAME

        url = 'https://disqus.com/api/3.0/forums/listThreads.json?api_key={0}&forum={1}&limit=100'.format(api_key,
                                                                                                          forum)

        raw = requests.get(url)
        dict = json.loads(raw)

        for thread in dict['response']:
            item = {thread['slug']: thread['posts']}
            self.values.update(item)

        return self.values

    def get_comments_number(self, slug):
        return self.values[slug]

我在这里使用annotate

def get_queryset(self):
    qs = super(PopRecipeListView, self).get_queryset()

    if qs:
        qs = qs.annotate(num_comments=get_comments_number('slug')).filter(shared=True).order_by('-num_comments')

    return qs

我用extra替换了annotate,并且:

定义获取查询集(self): qs=super(PopRecipeListView,self).get\u queryset()

if qs:
    qs = qs.extra(
        select={
            'num_comments': '{0}'.format(
                self.get_comments_number(Recipe.slug)
            )
        }
    ).filter(shared=True).order_by('-num_comments')

return qs

现在我得到一个错误: KeyError at /recipe/popular/ <django.db.models.query_utils.DeferredAttribute object at 0x7f90048f7470>

我在想mixin有问题。你知道吗

我更新了dispatch函数中获取所有值的视图,但出现以下错误:

KeyError at /recipe/popular/
<django.db.models.query_utils.DeferredAttribute object at 0x7f6ffebfa4e0>

代码here。你知道吗

我正在测试另一种方法: 我为get this value创建了一个函数,因为模型是一个属性:

@property
def get_comments_number(self):
    api_key = settings.DISQUS_API_KEY
    forum = settings.DISQUS_WEBSITE_SHORTNAME
    url = 'https://disqus.com/api/3.0/forums/listThreads.json?api_key={0}&forum={1}&thread:ident={2}'.format(
        api_key,
        forum,
        self.slug
    )
    raw = requests.get(url)
    thread = json.loads(raw.content)

    return thread['response'][0]['posts']

这样和一个新的字段排序会更容易,但是,如何使用这个函数获取值并保存在数据库中呢? 这项任务的最佳方法是什么?你知道吗

谢谢你。你知道吗


Tags: key函数selfapijsonurlgetreturn
2条回答

尝试.annotate(),然后按这个新注释的属性排序。你知道吗

请看这里:https://docs.djangoproject.com/en/2.0/ref/models/querysets/#annotate

最后我改变了我的方法:

我在recipe模型中创建了一个新的函数属性:

    @property
    def get_comments_number(self):
        api_key = settings.DISQUS_API_KEY
        forum = settings.DISQUS_WEBSITE_SHORTNAME
        url = 'https://disqus.com/api/3.0/forums/listThreads.json?api_key={0}&forum={1}&thread:ident={2}'.format(
            api_key,
            forum,
            self.slug
        )
        raw = requests.get(url)
        thread = json.loads(raw.content)

        return thread['response'][0]['posts']

之后,我在模型中添加了一个新字段来保存comment's number值。你知道吗

最后,我在DetailView中调用该属性以更新注释的编号:

def调度(self、request、*args、**kwargs):

# TODO: maybe it is necessary create a task for update this in other cases
r = self.get_object()
r.comments = r.get_comments_number
r.save()

# check if the recipe is a private recipe if so through a 404 error
if r.shared == False and self.object.author != request.user:
    output = _("Recipe {0} is marked Private").format(self.get_object().slug)
    raise Http404(output)
else:
    return super(RecipeDetailView, self).dispatch(request, *args, **kwargs)

现在我可以在另一个视图中按注释编号进行过滤:

qs = qs.filter(shared=True).order_by('-comments')

就这样。你知道吗

相关问题 更多 >

    热门问题