3

我正在使用 Django 设置,我可以在其中接收包含查询字符串的 url 作为 GET 的一部分。我希望能够处理查询字符串中提供的数据并返回针对该数据进行调整但不包含 URL 中的查询字符串的页面。

通常我只会使用reverse(),但我不确定在这种情况下如何应用它。以下是情况的详细信息:

示例网址: .../test/123/?list_options=1&list_options=2&list_options=3

网址.py

urlpatterns = patterns('', 
    url(r'test/(P<testrun_id>\d+)/'), views.testrun, name='testrun')
    )

视图.py

def testrun(request, testrun_id):
    if 'list_options' in request.GET.keys():
        lopt = request.GET.getlist('list_options')
        :
        :
        [process lopt list] 
        :
        :

    :
    :
    [other processing]
    :
    :

    context = { ...stuff... }
    return render(request, 'test_tracker/testview.html', context)

处理示例 URL 后,Django 将返回我想要的页面,但 URL 最后仍包含查询字符串。剥离不需要的查询字符串的标准方法是返回 testrun 函数return HttpResponseRedirect(reverse('testrun', args=(testrun_id,)))。但是,如果我在这里这样做,那么我将通过testrun函数获得一个无限循环。此外,我不确定原始请求中的 list_options 数据在重定向后是否仍然可用,因为它已从 URL 中删除。

我应该如何解决这个问题?我可以看到,将 list_options 变量的解析移到一个单独的函数中以避免无限递归可能是有意义的,但如果我这样做,我担心它会丢失请求中的 list_options 数据。有没有一种巧妙的方法可以同时将查询字符串从 URL 的末尾删除,并在一个地方返回我想要的页面,这样我就可以避免将事情分成多个函数?

编辑:一点额外的背景,因为有几个“你为什么要这样做?” 查询。

我正在设计的网站是报告我正在开发的软件的各种测试结果。这个特定页面用于报告单个测试的结果,我通常会从更大的测试列表中链接到它。

list_options数组是一种在我刚刚来自的列表中指定其他测试的方法。这使我可以使用其他相关测试填充下拉菜单,以便我可以轻松地在它们之间切换。

因此,我很容易最终会传入 15-20 个不同的值并创建巨大的 URL,这是我想避免的。如果我不建议 URL 中的任何其他内容,则该页面旨在具有一组默认的其他测试来填写相关菜单,因此如果我删除list_options. 如果用户希望直接返回页面,他不会关心列表中的其他测试,因此如果该信息不可用,这不是问题。

4

1 回答 1

3

首先要注意一点。由于各种原因,这可能不是一个好主意:

  • 书签。想象一下,这.../link?q=bar&order=foo将过滤一些搜索结果并按特定顺序对结果进行排序。如果您将自动删除查询字符串,那么您将有效地禁止用户为特定的搜索查询添加书签。
  • 测试。任何时候你添加任何自动化,事情都可能而且很可能会以你从未想象过的方式出错。坚持简单而有效的方法总是更好,因为它们被广泛使用,因此不易出错。我将在下面举一个例子。
  • 维护。这不是一个标准的行为模型,因此这将使未来的开发人员更难维护,因为首先他们必须首先了解正在发生的事情。

如果您仍然想实现这一点,最简单的方法之一是使用会话。这个想法是,当有查询字符串时,将其内容保存到会话中,然后在没有查询字符串时检索它。例如:

def testrun(request, testrun_id):
    # save the get data
    if request.META['QUERY_STRING']:
        request.session['testrun_get'] = request.GET
        # the following will not have querystring hence no infinite loop
        return HttpResponseRedirect(reverse('testrun', args=(testrun_id,)))

    # there is no querystring so retreive it from session
    # however someone could visit the url without the querystring
    # without visiting the querystring version first hence
    # you have to test for it
    get_data = request.session.get('testrun_get', None)
    if get_data:
        if 'list_options' in get_data.keys():
            ...
    else:
        # do some default option
        ...

    context = { ...stuff... }
    return render(request, 'test_tracker/testview.html', context)

这应该可以工作,但是它很容易损坏,并且没有办法轻松修复它。这应该说明上面的第二个项目符号。例如,假设用户想要并排比较两个搜索查询。所以他会尝试在同一个浏览.../link?q=bar&order=foo`.../link?q=cat&order=dog的不同标签中访问。到目前为止一切顺利,因为每个页面都会打开正确的结果,但是一旦用户尝试刷新第一个打开的选项卡,他就会从第二个选项卡获得结果,因为这是当前存储在会话中的内容,并且因为浏览器将有一个两个选项卡的单个会话令牌。

即使您会找到其他方法来实现您想要的而不使用会话,我想您也会遇到类似的问题,因为 HTTP 是无状态的,因此您必须在服务器上存储状态。

实际上有一种方法可以在不破坏大部分功能的情况下做到这一点——将状态存储在客户端而不是服务器端。因此,您将拥有一个没有查询字符串的 url,然后让 javascript 查询一些 API 以获取您需要在该页面上显示的任何内容。但是,这将迫使您制作某种 API 并使用一些不完全属于您的问题范围的 javascript。因此,可以干净利落地做,但这不仅仅涉及使用 Django。

于 2013-08-08T18:27:37.197 回答