Django and Elastic Beanstalk URL health checks

前端 未结 2 1707
闹比i
闹比i 2020-12-30 10:50

I have a Django webapp. It runs inside Docker on Elastic Beanstalk.

I\'d like to specify a health check URL for slightly more advanced health checking than \"can the

相关标签:
2条回答
  • 2020-12-30 11:21

    If the ELB health check is sending its request with a host header containing the elastic beanstalk domain (*.elasticbeanstalk.com, or an EC2 domain *.amazonaws.com) then the standard ALLOWED_HOSTS can still be used with a wildcard entry of '.amazonaws.com' or '.elasticbeanstalk.com'.

    In my case I received standard ipv4 addresses as the health check hosts, so a different solution was needed. If you can't predict the host at all, and it might be safer to assume you can't, you would need to take a route such as one of the following.

    You can use Apache to handle approved hosts instead of propagating ambiguous requests to Django. Since the host header is intended to be the hostname of the server receiving the request, this solution changes the header of valid requests to use the expected site hostname. With elastic beanstalk you'll need to configure Apache using .ebextensions as described here. Under the .ebextensions directory in your project root, add the following to a .config file.

    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>
    

    Replacing /status/ with your health check URL and example.com with your site's appropriate domain. This tells Apache to check all incoming requests and change the host headers on requests with the appropriate health check user agent that are requesting the appropriate health check URL.

    If you would really prefer not to configure Apache, you could write a custom middleware to authenticate health checks. The middleware would have to override Django's CommonMiddleware which calls HttpRequest's get_host() method that validates the request's host. You could do something like this

    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)
    

    Which just allows any health check requests to skip the host validation. You'd then replace django.middleware.common.CommonMiddleware with path.CommonOverrideMiddleware in your settings.py.

    I would recommend using the Apache configuration approach to avoid any details in the middleware, and to completely isolate Django from host issues.

    0 讨论(0)
  • 2020-12-30 11:26

    This is what I use, and it works well:

    import socket
    local_ip = str(socket.gethostbyname(socket.gethostname()))
    ALLOWED_HOSTS=[local_ip, '.mydomain.com', 'mydomain.elasticbeanstalk.com' ]
    

    where you replace mydomain and mydomain.elasticbeanstalk.com with your own.

    0 讨论(0)
提交回复
热议问题