2019-11-16 00:12:57 +01:00
|
|
|
// Copyright 2019 The Prometheus Authors
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
// Package https allows the implementation of TLS.
|
|
|
|
package https
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
|
2020-05-01 14:26:51 +02:00
|
|
|
"github.com/go-kit/kit/log"
|
|
|
|
"github.com/go-kit/kit/log/level"
|
2019-11-16 00:12:57 +01:00
|
|
|
"github.com/pkg/errors"
|
2020-05-01 14:26:51 +02:00
|
|
|
config_util "github.com/prometheus/common/config"
|
2019-11-16 00:12:57 +01:00
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
2020-05-01 14:26:51 +02:00
|
|
|
var (
|
|
|
|
errNoTLSConfig = errors.New("TLS config is not present")
|
|
|
|
)
|
|
|
|
|
2019-11-16 00:12:57 +01:00
|
|
|
type Config struct {
|
2020-05-01 14:26:51 +02:00
|
|
|
TLSConfig TLSStruct `yaml:"tls_config"`
|
|
|
|
Users map[string]config_util.Secret `yaml:"basic_auth_users"`
|
2019-11-16 00:12:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type TLSStruct struct {
|
2020-04-25 13:42:45 +02:00
|
|
|
TLSCertPath string `yaml:"cert_file"`
|
|
|
|
TLSKeyPath string `yaml:"key_file"`
|
|
|
|
ClientAuth string `yaml:"client_auth_type"`
|
|
|
|
ClientCAs string `yaml:"client_ca_file"`
|
2019-11-16 00:12:57 +01:00
|
|
|
}
|
|
|
|
|
2020-05-01 14:26:51 +02:00
|
|
|
func getConfig(configPath string) (*Config, error) {
|
2019-11-16 00:12:57 +01:00
|
|
|
content, err := ioutil.ReadFile(configPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
c := &Config{}
|
2020-05-01 14:26:51 +02:00
|
|
|
err = yaml.UnmarshalStrict(content, c)
|
|
|
|
return c, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func getTLSConfig(configPath string) (*tls.Config, error) {
|
|
|
|
c, err := getConfig(configPath)
|
2019-11-16 00:12:57 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-04-16 09:58:51 +02:00
|
|
|
return ConfigToTLSConfig(&c.TLSConfig)
|
2019-11-16 00:12:57 +01:00
|
|
|
}
|
|
|
|
|
2020-04-16 09:58:51 +02:00
|
|
|
// ConfigToTLSConfig generates the golang tls.Config from the TLSStruct config.
|
|
|
|
func ConfigToTLSConfig(c *TLSStruct) (*tls.Config, error) {
|
2020-05-01 14:26:51 +02:00
|
|
|
if c.TLSCertPath == "" && c.TLSKeyPath == "" && c.ClientAuth == "" && c.ClientCAs == "" {
|
|
|
|
return nil, errNoTLSConfig
|
2020-04-25 15:51:32 +02:00
|
|
|
}
|
2020-05-01 14:26:51 +02:00
|
|
|
|
|
|
|
if c.TLSCertPath == "" {
|
|
|
|
return nil, errors.New("missing cert_file")
|
2019-11-16 00:12:57 +01:00
|
|
|
}
|
2020-05-01 14:26:51 +02:00
|
|
|
if c.TLSKeyPath == "" {
|
|
|
|
return nil, errors.New("missing key_file")
|
|
|
|
}
|
|
|
|
cfg := &tls.Config{
|
|
|
|
MinVersion: tls.VersionTLS12,
|
2019-11-16 00:12:57 +01:00
|
|
|
}
|
|
|
|
loadCert := func() (*tls.Certificate, error) {
|
|
|
|
cert, err := tls.LoadX509KeyPair(c.TLSCertPath, c.TLSKeyPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to load X509KeyPair")
|
|
|
|
}
|
|
|
|
return &cert, nil
|
|
|
|
}
|
|
|
|
// Confirm that certificate and key paths are valid.
|
|
|
|
if _, err := loadCert(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
cfg.GetCertificate = func(*tls.ClientHelloInfo) (*tls.Certificate, error) {
|
|
|
|
return loadCert()
|
|
|
|
}
|
|
|
|
|
2020-05-01 14:26:51 +02:00
|
|
|
if c.ClientCAs != "" {
|
2019-11-16 00:12:57 +01:00
|
|
|
clientCAPool := x509.NewCertPool()
|
|
|
|
clientCAFile, err := ioutil.ReadFile(c.ClientCAs)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
clientCAPool.AppendCertsFromPEM(clientCAFile)
|
|
|
|
cfg.ClientCAs = clientCAPool
|
|
|
|
}
|
2020-05-01 14:26:51 +02:00
|
|
|
|
|
|
|
switch c.ClientAuth {
|
|
|
|
case "RequestClientCert":
|
|
|
|
cfg.ClientAuth = tls.RequestClientCert
|
|
|
|
case "RequireClientCert":
|
|
|
|
cfg.ClientAuth = tls.RequireAnyClientCert
|
|
|
|
case "VerifyClientCertIfGiven":
|
|
|
|
cfg.ClientAuth = tls.VerifyClientCertIfGiven
|
|
|
|
case "RequireAndVerifyClientCert":
|
|
|
|
cfg.ClientAuth = tls.RequireAndVerifyClientCert
|
|
|
|
case "", "NoClientCert":
|
|
|
|
cfg.ClientAuth = tls.NoClientCert
|
|
|
|
default:
|
|
|
|
return nil, errors.New("Invalid ClientAuth: " + c.ClientAuth)
|
2019-11-16 00:12:57 +01:00
|
|
|
}
|
2020-05-01 14:26:51 +02:00
|
|
|
|
|
|
|
if c.ClientCAs != "" && cfg.ClientAuth == tls.NoClientCert {
|
2019-11-16 00:12:57 +01:00
|
|
|
return nil, errors.New("Client CA's have been configured without a Client Auth Policy")
|
|
|
|
}
|
2020-05-01 14:26:51 +02:00
|
|
|
|
2019-11-16 00:12:57 +01:00
|
|
|
return cfg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Listen starts the server on the given address. If tlsConfigPath isn't empty the server connection will be started using TLS.
|
2020-05-01 14:26:51 +02:00
|
|
|
func Listen(server *http.Server, tlsConfigPath string, logger log.Logger) error {
|
|
|
|
if tlsConfigPath == "" {
|
|
|
|
level.Info(logger).Log("msg", "TLS is disabled and it cannot be enabled on the fly.")
|
2019-11-16 00:12:57 +01:00
|
|
|
return server.ListenAndServe()
|
|
|
|
}
|
2020-05-01 14:26:51 +02:00
|
|
|
|
|
|
|
if err := validateUsers(tlsConfigPath); err != nil {
|
2019-11-16 00:12:57 +01:00
|
|
|
return err
|
|
|
|
}
|
2020-05-01 14:26:51 +02:00
|
|
|
|
|
|
|
// Setup basic authentication.
|
|
|
|
var handler http.Handler = http.DefaultServeMux
|
|
|
|
if server.Handler != nil {
|
|
|
|
handler = server.Handler
|
|
|
|
}
|
|
|
|
server.Handler = &userAuthRoundtrip{
|
|
|
|
tlsConfigPath: tlsConfigPath,
|
|
|
|
logger: logger,
|
|
|
|
handler: handler,
|
|
|
|
}
|
|
|
|
|
|
|
|
config, err := getTLSConfig(tlsConfigPath)
|
|
|
|
switch err {
|
|
|
|
case nil:
|
|
|
|
// Valid TLS config.
|
|
|
|
level.Info(logger).Log("msg", "TLS is enabled and it cannot be disabled on the fly.")
|
|
|
|
case errNoTLSConfig:
|
|
|
|
// No TLS config, back to plain HTTP.
|
|
|
|
level.Info(logger).Log("msg", "TLS is disabled and it cannot be enabled on the fly.")
|
|
|
|
return server.ListenAndServe()
|
|
|
|
default:
|
|
|
|
// Invalid TLS config.
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
server.TLSConfig = config
|
|
|
|
|
2019-11-16 00:12:57 +01:00
|
|
|
// Set the GetConfigForClient method of the HTTPS server so that the config
|
|
|
|
// and certs are reloaded on new connections.
|
|
|
|
server.TLSConfig.GetConfigForClient = func(*tls.ClientHelloInfo) (*tls.Config, error) {
|
|
|
|
return getTLSConfig(tlsConfigPath)
|
|
|
|
}
|
|
|
|
return server.ListenAndServeTLS("", "")
|
|
|
|
}
|