我如何在 Django 中获得所有请求头?

我需要所有姜戈请求的标题。据我所知,Django 只是将所有内容和其他大量数据一起转储到 request.META变量中。什么是最好的方法来获得 所有的头,客户端发送到我的 Django 应用程序?

我要用这些来构建一个 httplib请求。

162207 次浏览

根据 文件 request.META是一个“包含所有可用 HTTP 头的标准 Python 字典”。如果你想得到 所有的标题,你可以简单地通过字典迭代。

代码的哪一部分完成这项工作取决于您的确切需求。任何可以访问 request的地方都应该这样做。

更新

我需要在一个中间件类中访问它,但是当我在它上面迭代时,除了 HTTP 头之外,我得到了很多值。

根据文件:

如上所述,除了 CONTENT_LENGTHCONTENT_TYPE之外,请求中的任何 HTTP标头都会通过将所有字符转换为大写字母,用下划线和 在名称 中添加一个 HTTP_前缀替换任何连字符,从而转换为 META键。

(加重语气)

要获得单独的 HTTP标头,只需按照以 HTTP_为前缀的键进行过滤。

更新2

你能告诉我如何通过过滤以 HTTP _ 开头的 request.META 变量中的所有键并去掉前面的 HTTP _ 部分来构建一个头字典吗。

当然。这里有一个方法可以做到这一点。

import re
regex = re.compile('^HTTP_')
dict((regex.sub('', header), value) for (header, value)
in request.META.items() if header.startswith('HTTP_'))

我不认为有任何简单的方法可以只获得 HTTP 头。您必须通过请求迭代。

Django-debug-toolbar 采用相同的方法来显示头信息。

这是另一种做法,非常类似于 Manoj Govindan上面的答案:

import re
regex_http_          = re.compile(r'^HTTP_.+$')
regex_content_type   = re.compile(r'^CONTENT_TYPE$')
regex_content_length = re.compile(r'^CONTENT_LENGTH$')


request_headers = {}
for header in request.META:
if regex_http_.match(header) or regex_content_type.match(header) or regex_content_length.match(header):
request_headers[header] = request.META[header]

这也将获取 CONTENT_TYPECONTENT_LENGTH请求标头,以及 HTTP_请求标头。

如果需要包含/省略某些头文件,可以相应地进行修改。 Django 在这里列出了一些,但不是全部: https://docs.djangoproject.com/en/dev/ref/request-response/#django.http.HttpRequest.META

Django 的请求头算法:

  1. 用下划线 _替换连字符 -
  2. 转换为大写。
  3. 除了 CONTENT_TYPECONTENT_LENGTH之外,在原始请求中将 HTTP_添加到所有标头。

每个标头的值应该保持不变。

无论如何,您的目的似乎是使用传入的 HTTP 请求来形成另一个 HTTP 请求。有点像通道。有一个优秀的模块 姜戈-雷夫代理正是完成了这一点。

这个源代码对于如何完成你正在尝试做的事情是一个很好的参考。

<b>request.META</b><br>
{% for k_meta, v_meta in request.META.items %}
<code>\{\{ k_meta }}</code> : \{\{ v_meta }} <br>
{% endfor %}

Get (‘ HTTP _ AUTHORIZATION’) /python3.6/site-packages/rest_framework/authentication.py

你可以从这个文件里找到。

如果您想从请求头中获取客户端密钥,可以尝试以下操作:

from rest_framework.authentication import BaseAuthentication
from rest_framework import exceptions
from apps.authentication.models import CerebroAuth


class CerebroAuthentication(BaseAuthentication):
def authenticate(self, request):
client_id = request.META.get('HTTP_AUTHORIZATION')
if not client_id:
raise exceptions.AuthenticationFailed('Client key not provided')
client_id = client_id.split()
if len(client_id) == 1 or len(client_id) > 2:
msg = ('Invalid secrer key header. No credentials provided.')
raise exceptions.AuthenticationFailed(msg)
try:
client = CerebroAuth.objects.get(client_id=client_id[1])
except CerebroAuth.DoesNotExist:
raise exceptions.AuthenticationFailed('No such client')
return (client, None)

从 Django 2.2开始,您可以使用 request.headers访问 HTTP 头:

一个不区分大小写、类似 dict 的对象,提供从请求访问所有 HTTP 前缀头(加上 Content-Llength 和 Content-Type)的权限。

每个标题的名称在显示时都使用了大小写(例如 User-Agent):

>>> request.headers
{'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6', ...}


>>> 'User-Agent' in request.headers
True
>>> 'user-agent' in request.headers
True


>>> request.headers['User-Agent']
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)
>>> request.headers['user-agent']
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)


>>> request.headers.get('User-Agent')
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)
>>> request.headers.get('user-agent')
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)

要获取所有的头,可以使用 request.headers.keys()request.headers.items()

简单地说,你可以从 姜戈2.2开始使用 HttpRequest.headers。下面的例子直接取自 请求和响应对象章节下的官方 姜戈文档

>>> request.headers
{'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6', ...}


>>> 'User-Agent' in request.headers
True
>>> 'user-agent' in request.headers
True


>>> request.headers['User-Agent']
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)
>>> request.headers['user-agent']
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)


>>> request.headers.get('User-Agent')
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)
>>> request.headers.get('user-agent')
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6)

只要注意在标题键中使用“-”而不是“ _”即可

因为如果在标题中使用“ Current _ User”,则无法通过

request.headers.get('Current_User')   <-- None

用户“ xxx-xxx”作为密钥