73 """Returns real IP of the request. Since not all proxies set all the HTTP
74 headers and incoming headers can be faked it may happen that the IP cannot
75 be determined correctly.
77 .. sidebar:: :py:obj:`flask.Request.remote_addr`
79 SearXNG uses Werkzeug's ProxyFix_ (with it default ``x_for=1``).
81 This function tries to get the remote IP in the order listed below,
82 additional some tests are done and if inconsistencies or errors are
83 detected, they are logged.
85 The remote IP of the request is taken from (first match):
87 - X-Forwarded-For_ header
88 - `X-real-IP header <https://github.com/searxng/searxng/issues/1237#issuecomment-1147564516>`__
89 - :py:obj:`flask.Request.remote_addr`
92 https://werkzeug.palletsprojects.com/middleware/proxy_fix/
95 https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/X-Forwarded-For
99 forwarded_for = request.headers.get(
"X-Forwarded-For")
100 real_ip = request.headers.get(
'X-Real-IP')
101 remote_addr = request.remote_addr
106 if not forwarded_for:
111 forwarded_for = [x.strip()
for x
in forwarded_for.split(
',')]
112 x_for: int = cfg[
'real_ip.x_for']
113 forwarded_for = forwarded_for[-min(len(forwarded_for), x_for)]
118 if forwarded_for
and real_ip
and forwarded_for != real_ip:
119 logger.warning(
"IP from X-Real-IP (%s) is not equal to IP from X-Forwarded-For (%s)", real_ip, forwarded_for)
121 if forwarded_for
and remote_addr
and forwarded_for != remote_addr:
123 "IP from WSGI environment (%s) is not equal to IP from X-Forwarded-For (%s)", remote_addr, forwarded_for
126 if real_ip
and remote_addr
and real_ip != remote_addr:
127 logger.warning(
"IP from WSGI environment (%s) is not equal to IP from X-Real-IP (%s)", remote_addr, real_ip)
129 request_ip = ip_address(forwarded_for
or real_ip
or remote_addr
or '0.0.0.0')
130 if request_ip.version == 6
and request_ip.ipv4_mapped:
131 request_ip = request_ip.ipv4_mapped
134 return str(request_ip)