如何更改 nginx 中代理服务器响应的状态代码?

How to change the status code of a proxied server response in nginx?

我很难将 nginx 配置为充当 public S3 端点的代理。我的用例需要更改 S3 响应的状态代码,同时保留响应负载。

S3 可能返回的状态码包括 200 和 403。对于我的用例,我需要将这些状态码映射到 503。

我试过以下方法无效:

location ~* ^/.* {
  [...]
  proxy_intercept_errors on;
  error_page             200 =503 $upstream_http_location
}

Nginx 输出以下错误:

nginx: [emerg] value "200" must be between 300 and 599 in /etc/nginx/nginx.conf:xx

这是一个更完整的片段:

server {
  listen       80;

  location ~* ^/.* {
    proxy_http_version         1.1;
    proxy_method               GET;
    proxy_pass                 http://my-s3-bucket-endpoint;
    proxy_pass_request_body    off;

    proxy_set_header       Content-Length "";
    proxy_set_header       Connection "";
    proxy_set_header       Host my-s3-bucket-endpoint;
    proxy_set_header       Authorization '';

    proxy_hide_header      x-amz-id-2;
    proxy_hide_header      x-amz-request-id;
    proxy_hide_header      Set-Cookie;
    proxy_ignore_headers   "Set-Cookie";

    proxy_cache            S3_CACHE;
    proxy_cache_valid      200 403 503 1h;
    proxy_cache_bypass     $http_cache_purge;
    add_header             X-Cached $upstream_cache_status;

    proxy_intercept_errors on;
    error_page             200 =503 $upstream_http_location;
  }
}

能不能用nginx实现我所需要的?

我找到了一个或多或少合适的解决方案。这有点老套,但它确实有效。

关键是将我的 S3 存储桶的索引文档设置为不存在的文件名。这会导致对 S3 存储桶端点 / 的请求导致 403。

由于 nginx 代理将所有传入请求映射到 S3 存储桶端点/上,结果始终为 403,nginx 代理可以拦截。从那里,error_page 指令告诉它通过请求 S3 存储桶端点中的特定文档(在本例中为 error.json)进行响应,并使用 503 作为响应状态码。

location ~* ^/. {
  proxy_intercept_errors on;
  error_page             403 =503 /error.json;
}

此解决方案涉及将两个请求发送到 S3 存储桶端点(/,/error.json),但至少似乎使用上面更完整的代码段中的配置为两个请求启用了缓存。