如何在Django中获取所有请求标头?


107

我需要获取所有Django请求标头。根据我的阅读,Django只是将所有内容request.META与大量其他数据一起转储到变量中。获取客户端发送到我的Django应用程序的所有标头的最佳方法是什么?

我将使用这些来构建httplib请求。

Answers:


139

根据文档,这 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_'))

我需要在Middleware类中访问它,但是当我对其进行迭代时,除了HTTP标头之外,我还获得了很多其他值。
Mridang Agarwalla

谢谢Manoj。出于好奇-您能告诉我如何通过从request.META变量中过滤掉所有以a开头的键HTTP_并去除开头HTTP_部分的方式来构建标题字典。通过lambda函数可以做到吗?(我认为它们被称为lambda函数)是因为我可能会做很长的路要走,首先迭代它们,然后检查它是否以a开头,HTTP_然后将其添加到新字典中。再次感谢。
Mridang Agarwalla

再次感谢Manoj。我对其进行了少许修改,以lstrip('HTTP_')代替正则表达式。:)
Mridang Agarwalla 2010年

3
@Mridang Agarwalla:lstrip实际上不会执行您要它执行的操作。 lstrip会删除与您提供的字符串中的任何字符匹配的所有前导字符,因此,如果您有标头"HTTP_TOKEN_ID",它将返回"OKEN_ID",因为"T""TOKEN"匹配的开头匹配传递给lstrip的字符串中的字符。方法是prefix = 'HTTP_'; header = header[len(prefix):]
jcdyer 2014年

2
Django 2.2已支持HttpRequest.headers
Dcalsky

30

从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()


17

这是另一种实现方法,与上面的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_那些标头。request_headers['some_key]== request.META['some_key']

如果需要包括/省略某些标题,请进行相应的修改。Django在这里列出了一堆,但不是全部:https : //docs.djangoproject.com/en/dev/ref/request-response/#django.http.HttpRequest.META

Django请求标头的算法:

  1. -用下划线替换连字符_
  2. 转换为大写。
  3. 前置HTTP_到原请求的所有头,除了CONTENT_TYPECONTENT_LENGTH

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


5
可以将它们全部组合成一个正则表达式,re.compile(r'^(HTTP_.+|CONTENT_TYPE|CONTENT_LENGTH)$')
Rebs

6

request.META.get('HTTP_AUTHORIZATION') /python3.6/site-packages/rest_framework/authentication.py

你可以从这个文件中得到...


3

我认为没有简单的方法可以仅获取HTTP标头。您必须遍历request.META字典以获得所需的所有内容。

django-debug-toolbar采用相同的方法显示标题信息。看一下负责检索头信息的文件


1

如果要从请求标头获取客户端密钥,则可以尝试以下操作:

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)

1

对于它的价值,看来您的意图是使用传入的HTTP请求来形成另一个HTTP请求。有点像网关。有一个出色的模块django-revproxy可以完成此任务。

资料来源很好地参考了如何完成您想做的事情。


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

0

只需从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)
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.