1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
12 "git.arvados.org/arvados.git/sdk/go/httpserver"
16 Name string // to use in Via header
19 type HTTPError struct {
24 func (h HTTPError) Error() string {
28 var dropHeaders = map[string]bool{
29 // Headers that shouldn't be forwarded when proxying. See
30 // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers
33 "Proxy-Authenticate": true,
34 "Proxy-Authorization": true,
35 // (comment/space here makes gofmt1.10 agree with gofmt1.11)
40 // Headers that would interfere with Go's automatic
41 // compression/decompression if we forwarded them.
42 "Accept-Encoding": true,
43 "Content-Encoding": true,
44 "Transfer-Encoding": true,
46 // Content-Length depends on encoding.
47 "Content-Length": true,
49 // Defend against Rails vulnerability CVE-2023-22795 -
50 // we don't use this functionality anyway, so it costs us nothing.
51 // <https://discuss.rubyonrails.org/t/cve-2023-22795-possible-redos-based-dos-vulnerability-in-action-dispatch/82118>
52 "If-None-Match": true,
55 type ResponseFilter func(*http.Response, error) (*http.Response, error)
57 // Forward a request to upstream service, and return response or error.
61 client *http.Client) (*http.Response, error) {
63 // Copy headers from incoming request, then add/replace proxy
64 // headers like Via and X-Forwarded-For.
65 hdrOut := http.Header{}
66 for k, v := range reqIn.Header {
72 for _, xffIn := range reqIn.Header["X-Forwarded-For"] {
77 xff += reqIn.RemoteAddr
78 hdrOut.Set("X-Forwarded-For", xff)
79 if hdrOut.Get("X-Forwarded-Proto") == "" {
80 hdrOut.Set("X-Forwarded-Proto", reqIn.URL.Scheme)
82 hdrOut.Add("Via", reqIn.Proto+" arvados-controller")
84 reqOut := (&http.Request{
90 }).WithContext(reqIn.Context())
91 return client.Do(reqOut)
94 // Copy a response (or error) to the downstream client
95 func (p *proxy) ForwardResponse(w http.ResponseWriter, resp *http.Response, err error) (int64, error) {
97 if he, ok := err.(HTTPError); ok {
98 httpserver.Error(w, he.Message, he.Code)
100 httpserver.Error(w, err.Error(), http.StatusBadGateway)
105 defer resp.Body.Close()
106 for k, v := range resp.Header {
107 for _, v := range v {
111 w.WriteHeader(resp.StatusCode)
112 return io.Copy(w, resp.Body)