diff --git a/config/config.go b/config/config.go index 118b13d48..520a7ad9d 100644 --- a/config/config.go +++ b/config/config.go @@ -33,21 +33,24 @@ import ( ) var ( - patRulePath = regexp.MustCompile(`^[^*]*(\*[^/]*)?$`) - unchangeableHeaders = map[string]struct{}{ + patRulePath = regexp.MustCompile(`^[^*]*(\*[^/]*)?$`) + reservedHeaders = map[string]struct{}{ // NOTE: authorization is checked specially, // see RemoteWriteConfig.UnmarshalYAML. // "authorization": {}, "host": {}, "content-encoding": {}, + "content-length": {}, "content-type": {}, - "x-prometheus-remote-write-version": {}, "user-agent": {}, "connection": {}, "keep-alive": {}, "proxy-authenticate": {}, "proxy-authorization": {}, "www-authenticate": {}, + "accept-encoding": {}, + "x-prometheus-remote-write-version": {}, + "x-prometheus-remote-read-version": {}, } ) @@ -616,13 +619,8 @@ func (c *RemoteWriteConfig) UnmarshalYAML(unmarshal func(interface{}) error) err return errors.New("empty or null relabeling rule in remote write config") } } - for header := range c.Headers { - if strings.ToLower(header) == "authorization" { - return errors.New("authorization header must be changed via the basic_auth or authorization parameter") - } - if _, ok := unchangeableHeaders[strings.ToLower(header)]; ok { - return errors.Errorf("%s is an unchangeable header", header) - } + if err := validateHeaders(c.Headers); err != nil { + return err } // The UnmarshalYAML method of HTTPClientConfig is not being called because it's not a pointer. @@ -631,6 +629,18 @@ func (c *RemoteWriteConfig) UnmarshalYAML(unmarshal func(interface{}) error) err return c.HTTPClientConfig.Validate() } +func validateHeaders(headers map[string]string) error { + for header := range headers { + if strings.ToLower(header) == "authorization" { + return errors.New("authorization header must be changed via the basic_auth or authorization parameter") + } + if _, ok := reservedHeaders[strings.ToLower(header)]; ok { + return errors.Errorf("%s is a reserved header. It must not be changed", header) + } + } + return nil +} + // QueueConfig is the configuration for the queue used to write to remote // storage. type QueueConfig struct { @@ -667,10 +677,11 @@ type MetadataConfig struct { // RemoteReadConfig is the configuration for reading from remote storage. type RemoteReadConfig struct { - URL *config.URL `yaml:"url"` - RemoteTimeout model.Duration `yaml:"remote_timeout,omitempty"` - ReadRecent bool `yaml:"read_recent,omitempty"` - Name string `yaml:"name,omitempty"` + URL *config.URL `yaml:"url"` + RemoteTimeout model.Duration `yaml:"remote_timeout,omitempty"` + Headers map[string]string `yaml:"headers,omitempty"` + ReadRecent bool `yaml:"read_recent,omitempty"` + Name string `yaml:"name,omitempty"` // We cannot do proper Go type embedding below as the parser will then parse // values arbitrarily into the overflow maps of further-down types. @@ -696,6 +707,9 @@ func (c *RemoteReadConfig) UnmarshalYAML(unmarshal func(interface{}) error) erro if c.URL == nil { return errors.New("url for remote_read is empty") } + if err := validateHeaders(c.Headers); err != nil { + return err + } // The UnmarshalYAML method of HTTPClientConfig is not being called because it's not a pointer. // We cannot make it a pointer as the parser panics for inlined pointer structs. // Thus we just do its validation here. diff --git a/config/config_test.go b/config/config_test.go index bccae5cea..e7409c156 100644 --- a/config/config_test.go +++ b/config/config_test.go @@ -972,7 +972,10 @@ var expectedErrors = []struct { errMsg: `url for remote_read is empty`, }, { filename: "remote_write_header.bad.yml", - errMsg: `x-prometheus-remote-write-version is an unchangeable header`, + errMsg: `x-prometheus-remote-write-version is a reserved header. It must not be changed`, + }, { + filename: "remote_read_header.bad.yml", + errMsg: `x-prometheus-remote-write-version is a reserved header. It must not be changed`, }, { filename: "remote_write_authorization_header.bad.yml", errMsg: `authorization header must be changed via the basic_auth or authorization parameter`, diff --git a/config/testdata/remote_read_header.bad.yml b/config/testdata/remote_read_header.bad.yml new file mode 100644 index 000000000..116b63ce1 --- /dev/null +++ b/config/testdata/remote_read_header.bad.yml @@ -0,0 +1,5 @@ +remote_read: + - url: localhost:9090 + name: queue1 + headers: + "x-prometheus-remote-write-version": "somehack" \ No newline at end of file diff --git a/docs/configuration/configuration.md b/docs/configuration/configuration.md index cd657035b..0d1b3b228 100644 --- a/docs/configuration/configuration.md +++ b/docs/configuration/configuration.md @@ -1850,6 +1850,11 @@ required_matchers: # Timeout for requests to the remote read endpoint. [ remote_timeout: | default = 1m ] +# Custom HTTP headers to be sent along with each remote read request. +# Be aware that headers that are set by Prometheus itself can't be overwritten. +headers: + [ : ... ] + # Whether reads should be made for queries for time ranges that # the local storage should have complete data for. [ read_recent: | default = false ] diff --git a/storage/remote/client.go b/storage/remote/client.go index dae873d86..40b22a1b7 100644 --- a/storage/remote/client.go +++ b/storage/remote/client.go @@ -83,7 +83,6 @@ type Client struct { url *config_util.URL Client *http.Client timeout time.Duration - headers map[string]string retryOnRateLimit bool @@ -115,6 +114,9 @@ func NewReadClient(name string, conf *ClientConfig) (ReadClient, error) { } t := httpClient.Transport + if len(conf.Headers) > 0 { + t = newInjectHeadersRoundTripper(conf.Headers, t) + } httpClient.Transport = &nethttp.Transport{ RoundTripper: t, } @@ -138,6 +140,9 @@ func NewWriteClient(name string, conf *ClientConfig) (WriteClient, error) { } t := httpClient.Transport + if len(conf.Headers) > 0 { + t = newInjectHeadersRoundTripper(conf.Headers, t) + } httpClient.Transport = &nethttp.Transport{ RoundTripper: t, } @@ -148,10 +153,25 @@ func NewWriteClient(name string, conf *ClientConfig) (WriteClient, error) { Client: httpClient, retryOnRateLimit: conf.RetryOnRateLimit, timeout: time.Duration(conf.Timeout), - headers: conf.Headers, }, nil } +func newInjectHeadersRoundTripper(h map[string]string, underlyingRT http.RoundTripper) *injectHeadersRoundTripper { + return &injectHeadersRoundTripper{headers: h, RoundTripper: underlyingRT} +} + +type injectHeadersRoundTripper struct { + headers map[string]string + http.RoundTripper +} + +func (t *injectHeadersRoundTripper) RoundTrip(req *http.Request) (*http.Response, error) { + for key, value := range t.headers { + req.Header.Set(key, value) + } + return t.RoundTripper.RoundTrip(req) +} + const defaultBackoff = 0 type RecoverableError struct { @@ -168,9 +188,7 @@ func (c *Client) Store(ctx context.Context, req []byte) error { // recoverable. return err } - for k, v := range c.headers { - httpReq.Header.Set(k, v) - } + httpReq.Header.Add("Content-Encoding", "snappy") httpReq.Header.Set("Content-Type", "application/x-protobuf") httpReq.Header.Set("User-Agent", UserAgent) diff --git a/storage/remote/storage.go b/storage/remote/storage.go index 2ca540ed3..131ab73b7 100644 --- a/storage/remote/storage.go +++ b/storage/remote/storage.go @@ -111,6 +111,7 @@ func (s *Storage) ApplyConfig(conf *config.Config) error { URL: rrConf.URL, Timeout: rrConf.RemoteTimeout, HTTPClientConfig: rrConf.HTTPClientConfig, + Headers: rrConf.Headers, }) if err != nil { return err