Merge pull request #1280 from umap-project/ajax-proxy-redirect

Use X-Accel-Redirect for serving ajax-proxy request
This commit is contained in:
Yohan Boniface 2023-08-28 16:37:19 +02:00 committed by GitHub
commit dbba8d2744
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 59 additions and 22 deletions

View file

@ -332,8 +332,30 @@ And then add this new location in your nginx config (before the `/` location):
alias /path/to/umap/var/data/;
}
### Configure ajax proxy cache
uMap allows to use remote URL as data sources, but those URLs are not always
CORS open, so this is why there is this "ajax-proxy" feature, where the URL is
passed to the backend.
Additionally, there is a caching feature, which duration is configurable through
frontend settings. Valid values are: disabled, 5 min, 1 hour, 1 day.
This configuration provides a mix option, where python deals with validating the
URL and parsing the TTL parameter, and then it passes the hand to nginx which
will serve the remote content.
So, roughly:
- the client calls `/ajax-proxy/?url=xxx&ttl=300`
- python will validate the URL (not internal calls…)
- if `UMAP_XSENDFILE_HEADER` is set, then the python returns an empty response
with the path `/proxy/http://url` plus it will set the cache TTL through the
header `X-Accel-Expires`
- this `/proxy/` location is then handled by nginx
In Nginx:
- add the proxy cache
@ -341,28 +363,36 @@ In Nginx:
proxy_cache_path /tmp/nginx_ajax_proxy_cache levels=1:2 keys_zone=ajax_proxy:10m inactive=60m;
proxy_cache_key "$args";
- add this location (before the `/` location):
- add those locations (before the `/` location):
location /ajax-proxy/ {
valid_referers server_names;
if ($invalid_referer) {
return 400;
}
location ~ ^/proxy/(.*) {
internal;
add_header X-Proxy-Cache $upstream_cache_status always;
proxy_cache ajax_proxy;
proxy_cache_valid 1m; # Default. Umap will override using X-Accel-Expires
gzip on;
gzip_proxied any;
gzip_types
application/vnd.google-earth.kml+xml
application/geo+json
application/json
application/javascript
text/xml
application/xml;
uwsgi_pass umap;
include /srv/umap/uwsgi_params;
set $target_url $1;
# URL is encoded, so we need a few hack to clean it back.
if ( $target_url ~ (.+)%3A%2F%2F(.+) ){ # fix :// between scheme and destination
set $target_url $1://$2;
}
if ( $target_url ~ (.+?)%3A(.*) ){ # fix : between destination and port
set $target_url $1:$2;
}
if ( $target_url ~ (.+?)%2F(.*) ){ # fix / after port, the rest will be decoded by proxy_pass
set $target_url $1/$2;
}
add_header X-Proxy-Target $target_url; # For debugging
proxy_read_timeout 10s;
proxy_connect_timeout 5s;
proxy_pass $target_url;
proxy_intercept_errors on;
error_page 301 302 307 = @handle_proxy_redirect;
}
location @handle_proxy_redirect {
set $saved_redirect_location '$upstream_http_location';
proxy_pass $saved_redirect_location;
}

View file

@ -351,6 +351,17 @@ class AjaxProxy(View):
url = validate_url(self.request)
except AssertionError:
return HttpResponseBadRequest()
try:
ttl = int(self.request.GET.get("ttl"))
except (TypeError, ValueError):
ttl = None
if getattr(settings, "UMAP_XSENDFILE_HEADER", None):
response = HttpResponse()
response[settings.UMAP_XSENDFILE_HEADER] = f"/proxy/{url}"
if ttl:
response["X-Accel-Expires"] = ttl
return response
headers = {"User-Agent": "uMapProxy +http://wiki.openstreetmap.org/wiki/UMap"}
request = Request(url, headers=headers)
opener = build_opener()
@ -375,11 +386,7 @@ class AjaxProxy(View):
# Quick hack to prevent Django from adding a Vary: Cookie header
self.request.session.accessed = False
response = HttpResponse(content, status=status_code, content_type=mimetype)
try:
ttl = int(self.request.GET.get("ttl"))
except (TypeError, ValueError):
pass
else:
if ttl:
response["X-Accel-Expires"] = ttl
return response