2019-04-29 15:08:09 -07:00
|
|
|
package agent
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/hashicorp/consul/agent/structs"
|
|
|
|
)
|
|
|
|
|
2020-06-09 12:11:23 -07:00
|
|
|
const ConfigEntryNotFoundErr string = "Config entry not found"
|
|
|
|
|
2019-04-29 15:08:09 -07:00
|
|
|
// Config switches on the different CRUD operations for config entries.
|
|
|
|
func (s *HTTPServer) Config(resp http.ResponseWriter, req *http.Request) (interface{}, error) {
|
|
|
|
switch req.Method {
|
|
|
|
case "GET":
|
|
|
|
return s.configGet(resp, req)
|
|
|
|
|
|
|
|
case "DELETE":
|
|
|
|
return s.configDelete(resp, req)
|
|
|
|
|
|
|
|
default:
|
|
|
|
return nil, MethodNotAllowedError{req.Method, []string{"GET", "DELETE"}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// configGet gets either a specific config entry, or lists all config entries
|
|
|
|
// of a kind if no name is provided.
|
|
|
|
func (s *HTTPServer) configGet(resp http.ResponseWriter, req *http.Request) (interface{}, error) {
|
|
|
|
var args structs.ConfigEntryQuery
|
|
|
|
if done := s.parse(resp, req, &args.Datacenter, &args.QueryOptions); done {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
pathArgs := strings.SplitN(strings.TrimPrefix(req.URL.Path, "/v1/config/"), "/", 2)
|
|
|
|
|
2020-01-29 19:12:48 -05:00
|
|
|
switch len(pathArgs) {
|
|
|
|
case 2:
|
2020-01-24 10:04:58 -05:00
|
|
|
if err := s.parseEntMetaNoWildcard(req, &args.EnterpriseMeta); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-04-29 15:08:09 -07:00
|
|
|
// Both kind/name provided.
|
|
|
|
args.Kind = pathArgs[0]
|
|
|
|
args.Name = pathArgs[1]
|
|
|
|
|
|
|
|
var reply structs.ConfigEntryResponse
|
|
|
|
if err := s.agent.RPC("ConfigEntry.Get", &args, &reply); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-09-26 10:42:17 -05:00
|
|
|
setMeta(resp, &reply.QueryMeta)
|
2019-04-29 15:08:09 -07:00
|
|
|
|
|
|
|
if reply.Entry == nil {
|
2020-06-09 12:11:23 -07:00
|
|
|
return nil, NotFoundError{Reason: fmt.Sprintf("%s for %q / %q", ConfigEntryNotFoundErr, pathArgs[0], pathArgs[1])}
|
2019-04-29 15:08:09 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return reply.Entry, nil
|
|
|
|
case 1:
|
2020-01-29 19:12:48 -05:00
|
|
|
if err := s.parseEntMeta(req, &args.EnterpriseMeta); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-04-29 15:08:09 -07:00
|
|
|
// Only kind provided, list entries.
|
|
|
|
args.Kind = pathArgs[0]
|
|
|
|
|
|
|
|
var reply structs.IndexedConfigEntries
|
|
|
|
if err := s.agent.RPC("ConfigEntry.List", &args, &reply); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-09-26 10:42:17 -05:00
|
|
|
setMeta(resp, &reply.QueryMeta)
|
2019-04-29 15:08:09 -07:00
|
|
|
|
|
|
|
return reply.Entries, nil
|
|
|
|
default:
|
2019-04-30 18:19:19 -04:00
|
|
|
return nil, NotFoundError{Reason: "Must provide either a kind or both kind and name"}
|
2019-04-29 15:08:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// configDelete deletes the given config entry.
|
|
|
|
func (s *HTTPServer) configDelete(resp http.ResponseWriter, req *http.Request) (interface{}, error) {
|
|
|
|
var args structs.ConfigEntryRequest
|
|
|
|
s.parseDC(req, &args.Datacenter)
|
|
|
|
s.parseToken(req, &args.Token)
|
|
|
|
pathArgs := strings.SplitN(strings.TrimPrefix(req.URL.Path, "/v1/config/"), "/", 2)
|
|
|
|
|
|
|
|
if len(pathArgs) != 2 {
|
|
|
|
resp.WriteHeader(http.StatusNotFound)
|
|
|
|
fmt.Fprintf(resp, "Must provide both a kind and name to delete")
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
entry, err := structs.MakeConfigEntry(pathArgs[0], pathArgs[1])
|
|
|
|
if err != nil {
|
|
|
|
resp.WriteHeader(http.StatusBadRequest)
|
|
|
|
fmt.Fprintf(resp, "%v", err)
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
args.Entry = entry
|
2020-01-24 10:04:58 -05:00
|
|
|
// Parse enterprise meta.
|
|
|
|
meta := args.Entry.GetEnterpriseMeta()
|
|
|
|
if err := s.parseEntMetaNoWildcard(req, meta); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-04-29 15:08:09 -07:00
|
|
|
|
|
|
|
var reply struct{}
|
|
|
|
if err := s.agent.RPC("ConfigEntry.Delete", &args, &reply); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return reply, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConfigCreate applies the given config entry update.
|
|
|
|
func (s *HTTPServer) ConfigApply(resp http.ResponseWriter, req *http.Request) (interface{}, error) {
|
|
|
|
args := structs.ConfigEntryRequest{
|
|
|
|
Op: structs.ConfigEntryUpsert,
|
|
|
|
}
|
|
|
|
s.parseDC(req, &args.Datacenter)
|
|
|
|
s.parseToken(req, &args.Token)
|
|
|
|
|
|
|
|
var raw map[string]interface{}
|
2019-10-29 11:13:36 -07:00
|
|
|
if err := decodeBodyDeprecated(req, &raw, nil); err != nil {
|
2019-04-29 15:08:09 -07:00
|
|
|
return nil, BadRequestError{Reason: fmt.Sprintf("Request decoding failed: %v", err)}
|
|
|
|
}
|
|
|
|
|
|
|
|
if entry, err := structs.DecodeConfigEntry(raw); err == nil {
|
|
|
|
args.Entry = entry
|
|
|
|
} else {
|
|
|
|
return nil, BadRequestError{Reason: fmt.Sprintf("Request decoding failed: %v", err)}
|
|
|
|
}
|
|
|
|
|
2020-01-24 10:04:58 -05:00
|
|
|
// Parse enterprise meta.
|
|
|
|
var meta structs.EnterpriseMeta
|
|
|
|
if err := s.parseEntMetaNoWildcard(req, &meta); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
args.Entry.GetEnterpriseMeta().Merge(&meta)
|
|
|
|
|
2019-04-29 15:08:09 -07:00
|
|
|
// Check for cas value
|
|
|
|
if casStr := req.URL.Query().Get("cas"); casStr != "" {
|
|
|
|
casVal, err := strconv.ParseUint(casStr, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
args.Op = structs.ConfigEntryUpsertCAS
|
|
|
|
args.Entry.GetRaftIndex().ModifyIndex = casVal
|
|
|
|
}
|
|
|
|
|
2019-04-30 19:27:16 -04:00
|
|
|
var reply bool
|
|
|
|
if err := s.agent.RPC("ConfigEntry.Apply", &args, &reply); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return reply, nil
|
2019-04-29 15:08:09 -07:00
|
|
|
}
|