Django和Elastic Beanstalk URL运行状况检查

前端之家收集整理的这篇文章主要介绍了Django和Elastic Beanstalk URL运行状况检查前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我有一个Django webapp.它在Elastic Beanstalk上的Docker中运行.

我想指定一个health check URL用于稍微更高级的健康检查,而不是“ELB可以建立TCP连接”.

完全合理的是,ELB通过使用实例的主机名(例如ec2-127-0-0-1.compute-1.amazonaws.com)作为主机头连接到HTTP实例来完成此操作.

Django有ALLOWED_HOSTS验证传入请求的主机头.我通过环境变量将它设置为我的应用程序的外部域.

不出所料,完全合理的是,由于缺少匹配的主机,Django因此拒绝ELB URL健康检查.

我们不想禁用ALLOWED_HOSTS,因为我们希望能够信任get_host().

到目前为止,解决方案似乎是:

>以某种方式说服Django不关心某些特定路径的ALLOWED_HOSTS(即健康检查URL)
>做一些时髦的事情,比如在启动时调用EC2 info API来获取主机的FQDN并将其附加到ALLOWED_HOSTS

这些都不是特别令人愉快.有人可以推荐更好/现有的解决方案吗?

(为避免疑问,我认为这个问题与“禁用ALLOWED_HOSTS,面向主机上过滤的HTTPD前端”的情况相同 – 我希望健康检查能够点击Django,而不是前面的HTTPD)

解决方法

如果ELB运行状况检查使用包含弹性beanstalk域(* .elasticbeanstalk.com或EC2域* .amazonaws.com)的主机头发送其请求,则标准ALLOWED_HOSTS仍可用于’的通配符条目. amazonaws.com’或’.elasticbeanstalk.com’.

在我的情况下,我收到标准的ipv4地址作为健康检查主机,因此需要一个不同的解决方案.如果您根本无法预测主机,并且假设您不能预测主机可能更安全,则需要采取以下路线之一.

您可以使用Apache来处理已批准的主机,而不是将不明确的请求传播到Django.由于主机头旨在成为接收请求的服务器的主机名,因此此解决方案会更改有效请求的标头以使用预期的站点主机名.使用弹性beanstalk,您需要使用.ebextensions配置Apache,如here所述.在项目根目录下的.ebextensions目录下,将以下内容添加到.config文件中.

files:
  "/etc/httpd/conf.d/eb_healthcheck.conf":
    mode: "000644"
    owner: root
    group: root
    content: |
        <If "req('User-Agent') == 'ELB-HealthChecker/1.0' && %{REQUEST_URI} == '/status/'">
            RequestHeader set Host "example.com"
        </If>

使用您站点的相应域替换/状态/使用您的运行状况检查URL和example.com.这告诉Apache检查所有传入请求并使用正在请求相应运行状况检查URL的相应​​运行状况检查用户代理更改请求上的主机头.

如果您真的不想配置Apache,可以编写自定义中间件来验证运行状况检查.中间件必须覆盖Django的CommonMiddleware,它调用HttpRequest的get_host()方法来验证请求的主机.你可以这样做

from django.middleware.common import CommonMiddleware


class CommonOverrideMiddleware(CommonMiddleware):
    def process_request(self,request):
        if not('HTTP_USER_AGENT' in request.Meta and request.Meta['HTTP_USER_AGENT'] == 'ELB-HealthChecker/1.0' and request.get_full_path() == '/status/'):
            return super().process_request(request)

这只允许任何健康检查请求跳过主机验证.然后在settings.py中用path.CommonOverrideMiddleware替换django.middleware.common.CommonMiddleware.

我建议使用Apache配置方法来避免中间件中的任何细节,并完全将Django与主机问题隔离开来.

猜你在找的Python相关文章