influxdb/http/config.go

117 lines
2.9 KiB
Go

package http
import (
"encoding/json"
"fmt"
"net/http"
"time"
"github.com/go-chi/chi"
"github.com/go-chi/chi/middleware"
"github.com/influxdata/influxdb/v2"
"github.com/influxdata/influxdb/v2/authorizer"
"github.com/influxdata/influxdb/v2/kit/cli"
"github.com/influxdata/influxdb/v2/kit/platform"
"github.com/influxdata/influxdb/v2/kit/platform/errors"
kithttp "github.com/influxdata/influxdb/v2/kit/transport/http"
"github.com/spf13/pflag"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
)
const prefixConfig = "/api/v2/config"
func errInvalidType(dest interface{}, flag string) error {
return &errors.Error{
Code: errors.EInternal,
Err: fmt.Errorf("unknown destination type %T for %q", dest, flag),
}
}
type parsedOpt map[string]optValue
type optValue []byte
func (o optValue) MarshalJSON() ([]byte, error) { return o, nil }
type ConfigHandler struct {
chi.Router
log *zap.Logger
api *kithttp.API
config parsedOpt
}
// NewConfigHandler creates a handler that will return a JSON object with key/value pairs for the configuration values
// used during the launcher startup. The opts slice provides a list of options names along with a pointer to their
// value.
func NewConfigHandler(log *zap.Logger, opts []cli.Opt) (*ConfigHandler, error) {
h := &ConfigHandler{
log: log,
api: kithttp.NewAPI(kithttp.WithLog(log)),
}
if err := h.parseOptions(opts); err != nil {
return nil, err
}
r := chi.NewRouter()
r.Use(
middleware.Recoverer,
middleware.RequestID,
middleware.RealIP,
h.mwAuthorize,
)
r.Get("/", h.handleGetConfig)
h.Router = r
return h, nil
}
func (h *ConfigHandler) Prefix() string {
return prefixConfig
}
func (h *ConfigHandler) parseOptions(opts []cli.Opt) error {
h.config = make(parsedOpt)
for _, o := range opts {
var b []byte
switch o.DestP.(type) {
// Known types for configuration values. Currently, these can all be encoded directly with json.Marshal.
case *string, *int, *int32, *int64, *bool, *time.Duration, *[]string, *map[string]string, pflag.Value, *platform.ID, *zapcore.Level:
var err error
b, err = json.Marshal(o.DestP)
if err != nil {
return err
}
default:
// Return an error if we don't know how to marshal this type.
return errInvalidType(o.DestP, o.Flag)
}
h.config[o.Flag] = b
}
return nil
}
func (h *ConfigHandler) handleGetConfig(w http.ResponseWriter, r *http.Request) {
h.api.Respond(w, r, http.StatusOK, map[string]parsedOpt{"config": h.config})
}
func (h *ConfigHandler) mwAuthorize(next http.Handler) http.Handler {
fn := func(w http.ResponseWriter, r *http.Request) {
if err := authorizer.IsAllowedAll(r.Context(), influxdb.OperPermissions()); err != nil {
h.api.Err(w, r, &errors.Error{
Code: errors.EUnauthorized,
Msg: fmt.Sprintf("access to %s requires operator permissions", h.Prefix()),
})
return
}
next.ServeHTTP(w, r)
}
return http.HandlerFunc(fn)
}