问题:如何在Django中获取所有请求标头?
我需要获取所有Django请求标头。根据我的阅读,Django只是将所有内容request.META
与大量其他数据一起转储到变量中。获取客户端发送到我的Django应用程序的所有标头的最佳方法是什么?
我将使用这些来构建httplib
请求。
回答 0
根据文档,这 request.META
是“包含所有可用HTTP标头的标准Python词典”。如果要获取所有标头,则可以简单地遍历字典。
代码的哪一部分执行此操作取决于您的确切要求。有权访问的任何地方都request
应该这样做。
更新资料
我需要在Middleware类中访问它,但是当我对其进行迭代时,除了HTTP标头之外,我还获得了很多其他值。
从文档中:
除了
CONTENT_LENGTH
和之外CONTENT_TYPE
,如上所述,通过将所有字符HTTP
都转换META
为大写字母,用下划线替换所有连字符并将名称添加HTTP_
前缀,将请求中的所有标头都转换为键。
(添加了强调)
要HTTP
单独获取标题,只需按前缀为的键进行过滤HTTP_
。
更新2
您能否告诉我如何通过从request.META变量中滤除所有以HTTP_开头并除去开头的HTTP_部分的键来构建标头字典。
当然。这是一种方法。
import re
regex = re.compile('^HTTP_')
dict((regex.sub('', header), value) for (header, value)
in request.META.items() if header.startswith('HTTP_'))
回答 1
从Django 2.2开始,您可以使用request.headers
访问HTTP标头。从HttpRequest.headers的文档中:
不区分大小写,类似于dict的对象,该对象提供对请求中所有HTTP前缀的标头(加上Content-Length和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
这是另一种实现方法,与上面的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_TYPE
和CONTENT_LENGTH
请求标头以及HTTP_
那些标头。request_headers['some_key]
== request.META['some_key']
。
如果需要包括/省略某些标题,请进行相应的修改。Django在这里列出了一堆,但不是全部:https : //docs.djangoproject.com/en/dev/ref/request-response/#django.http.HttpRequest.META
Django请求标头的算法:
-
用下划线替换连字符_
- 转换为大写。
- 前置
HTTP_
到原请求的所有头,除了CONTENT_TYPE
和CONTENT_LENGTH
。
每个标头的值都应保持不变。
回答 3
request.META.get(’HTTP_AUTHORIZATION’)
/python3.6/site-packages/rest_framework/authentication.py
你可以从这个文件中得到…
回答 4
我认为没有简单的方法可以仅获取HTTP标头。您必须遍历request.META字典以获得所需的所有内容。
django-debug-toolbar采用相同的方法显示标题信息。看一下负责检索头信息的文件。
回答 5
如果要从请求标头获取客户端密钥,则可以尝试以下操作:
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)
回答 6
对于它的价值,看来您的意图是使用传入的HTTP请求来形成另一个HTTP请求。有点像网关。有一个出色的模块django-revproxy可以完成此任务。
资料来源很好地参考了如何完成您想做的事情。
回答 7
<b>request.META</b><br>
{% for k_meta, v_meta in request.META.items %}
<code>{{ k_meta }}</code> : {{ v_meta }} <br>
{% endfor %}
回答 8
只需从Django 2.2开始使用HttpRequest.headers。以下示例直接取自Django官方文档的 “ 请求和响应对象”部分。
>>> 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)