2023-03-28 19:39:22 +01:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
|
2014-08-21 14:28:16 -07:00
|
|
|
package agent
|
|
|
|
|
|
|
|
import (
|
2022-11-10 10:26:01 -06:00
|
|
|
"io"
|
2017-10-21 18:39:09 -07:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2014-08-21 14:28:16 -07:00
|
|
|
"os"
|
|
|
|
"testing"
|
2017-10-21 18:39:09 -07:00
|
|
|
"time"
|
2017-10-21 20:08:11 -07:00
|
|
|
|
2019-04-26 12:33:01 -04:00
|
|
|
"github.com/hashicorp/consul/api/watch"
|
2020-01-28 17:50:41 -06:00
|
|
|
"github.com/hashicorp/consul/sdk/testutil"
|
2020-07-10 14:19:12 -04:00
|
|
|
"github.com/hashicorp/go-hclog"
|
|
|
|
"github.com/stretchr/testify/require"
|
2014-08-21 14:28:16 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestMakeWatchHandler(t *testing.T) {
|
|
|
|
defer os.Remove("handler_out")
|
|
|
|
defer os.Remove("handler_index_out")
|
2018-02-22 14:06:06 +00:00
|
|
|
script := "bash -c 'echo $CONSUL_INDEX >> handler_index_out && cat >> handler_out'"
|
2020-01-28 17:50:41 -06:00
|
|
|
handler := makeWatchHandler(testutil.Logger(t), script)
|
2014-08-21 14:28:16 -07:00
|
|
|
handler(100, []string{"foo", "bar", "baz"})
|
2022-11-10 10:26:01 -06:00
|
|
|
raw, err := os.ReadFile("handler_out")
|
2014-08-21 14:28:16 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if string(raw) != "[\"foo\",\"bar\",\"baz\"]\n" {
|
|
|
|
t.Fatalf("bad: %s", raw)
|
|
|
|
}
|
2022-11-10 10:26:01 -06:00
|
|
|
raw, err = os.ReadFile("handler_index_out")
|
2014-08-21 14:28:16 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if string(raw) != "100\n" {
|
|
|
|
t.Fatalf("bad: %s", raw)
|
|
|
|
}
|
|
|
|
}
|
2017-10-21 18:39:09 -07:00
|
|
|
|
|
|
|
func TestMakeHTTPWatchHandler(t *testing.T) {
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
idx := r.Header.Get("X-Consul-Index")
|
|
|
|
if idx != "100" {
|
|
|
|
t.Fatalf("bad: %s", idx)
|
|
|
|
}
|
|
|
|
// Get the first one
|
|
|
|
customHeader := r.Header.Get("X-Custom")
|
|
|
|
if customHeader != "abc" {
|
|
|
|
t.Fatalf("bad: %s", idx)
|
|
|
|
}
|
2022-11-10 10:26:01 -06:00
|
|
|
body, err := io.ReadAll(r.Body)
|
2017-10-21 18:39:09 -07:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if string(body) != "[\"foo\",\"bar\",\"baz\"]\n" {
|
|
|
|
t.Fatalf("bad: %s", body)
|
|
|
|
}
|
|
|
|
w.Write([]byte("Ok, i see"))
|
|
|
|
}))
|
|
|
|
defer server.Close()
|
|
|
|
config := watch.HttpHandlerConfig{
|
|
|
|
Path: server.URL,
|
|
|
|
Header: map[string][]string{"X-Custom": {"abc", "def"}},
|
|
|
|
Timeout: time.Minute,
|
|
|
|
}
|
2020-01-28 17:50:41 -06:00
|
|
|
handler := makeHTTPWatchHandler(testutil.Logger(t), &config)
|
2017-10-21 18:39:09 -07:00
|
|
|
handler(100, []string{"foo", "bar", "baz"})
|
|
|
|
}
|
2020-07-10 14:19:12 -04:00
|
|
|
|
|
|
|
type raw map[string]interface{}
|
|
|
|
|
|
|
|
func TestMakeWatchPlan(t *testing.T) {
|
|
|
|
type testCase struct {
|
|
|
|
name string
|
|
|
|
params map[string]interface{}
|
|
|
|
expected func(t *testing.T, plan *watch.Plan)
|
|
|
|
expectedErr string
|
|
|
|
}
|
|
|
|
fn := func(t *testing.T, tc testCase) {
|
|
|
|
plan, err := makeWatchPlan(hclog.New(nil), tc.params)
|
|
|
|
if tc.expectedErr != "" {
|
|
|
|
require.Error(t, err)
|
|
|
|
require.Contains(t, err.Error(), tc.expectedErr)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
require.NoError(t, err)
|
|
|
|
tc.expected(t, plan)
|
|
|
|
}
|
|
|
|
var testCases = []testCase{
|
|
|
|
{
|
|
|
|
name: "handler_type script, with deprecated handler field",
|
|
|
|
params: raw{
|
|
|
|
"type": "key",
|
|
|
|
"key": "foo",
|
|
|
|
"handler_type": "script",
|
|
|
|
"handler": "./script.sh",
|
|
|
|
},
|
|
|
|
expected: func(t *testing.T, plan *watch.Plan) {
|
|
|
|
require.Equal(t, plan.HandlerType, "script")
|
|
|
|
require.Equal(t, plan.Exempt["handler"], "./script.sh")
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "handler_type script, with single arg",
|
|
|
|
params: raw{
|
|
|
|
"type": "key",
|
|
|
|
"key": "foo",
|
|
|
|
"handler_type": "script",
|
|
|
|
"args": "./script.sh",
|
|
|
|
},
|
|
|
|
expected: func(t *testing.T, plan *watch.Plan) {
|
|
|
|
require.Equal(t, plan.HandlerType, "script")
|
|
|
|
require.Equal(t, plan.Exempt["args"], []string{"./script.sh"})
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "handler_type script, with multiple args from slice of interface",
|
|
|
|
params: raw{
|
|
|
|
"type": "key",
|
|
|
|
"key": "foo",
|
|
|
|
"handler_type": "script",
|
|
|
|
"args": []interface{}{"./script.sh", "arg1"},
|
|
|
|
},
|
|
|
|
expected: func(t *testing.T, plan *watch.Plan) {
|
|
|
|
require.Equal(t, plan.HandlerType, "script")
|
|
|
|
require.Equal(t, plan.Exempt["args"], []string{"./script.sh", "arg1"})
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "handler_type script, with multiple args from slice of strings",
|
|
|
|
params: raw{
|
|
|
|
"type": "key",
|
|
|
|
"key": "foo",
|
|
|
|
"handler_type": "script",
|
|
|
|
"args": []string{"./script.sh", "arg1"},
|
|
|
|
},
|
|
|
|
expected: func(t *testing.T, plan *watch.Plan) {
|
|
|
|
require.Equal(t, plan.HandlerType, "script")
|
|
|
|
require.Equal(t, plan.Exempt["args"], []string{"./script.sh", "arg1"})
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "handler_type script, with not string args",
|
|
|
|
params: raw{
|
|
|
|
"type": "key",
|
|
|
|
"key": "foo",
|
|
|
|
"handler_type": "script",
|
|
|
|
"args": []interface{}{"./script.sh", true},
|
|
|
|
},
|
|
|
|
expectedErr: "Watch args must be a list of strings",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "conflicting handler",
|
|
|
|
params: raw{
|
|
|
|
"type": "key",
|
|
|
|
"key": "foo",
|
|
|
|
"handler_type": "script",
|
|
|
|
"handler": "./script.sh",
|
|
|
|
"args": []interface{}{"arg1"},
|
|
|
|
},
|
|
|
|
expectedErr: "Only one watch handler allowed",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "no handler_type",
|
|
|
|
params: raw{
|
|
|
|
"type": "key",
|
|
|
|
"key": "foo",
|
|
|
|
},
|
|
|
|
expectedErr: "Must define a watch handler",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
fn(t, tc)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|