cb689a8382
* Get concrete types out of dynamic payload wip pull out value setting to func * Add TestEventSTream_SetPayloadValue Add more assertions use alias type in unmarshalJSON to handle payload rawmessage shorten unmarshal and remove anonymous wrap struct * use map structure and helper functions to return concrete types * ensure times are properly handled * update test name * put all decode logic in a single function Co-authored-by: Kris Hicks <khicks@hashicorp.com>
251 lines
5.9 KiB
Go
251 lines
5.9 KiB
Go
package api
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/hashicorp/nomad/api/internal/testutil"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestEvent_Stream(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
c, s := makeClient(t, nil, nil)
|
|
defer s.Stop()
|
|
|
|
// register job to generate events
|
|
jobs := c.Jobs()
|
|
job := testJob()
|
|
resp2, _, err := jobs.Register(job, nil)
|
|
require.Nil(t, err)
|
|
require.NotNil(t, resp2)
|
|
|
|
// build event stream request
|
|
events := c.EventStream()
|
|
q := &QueryOptions{}
|
|
topics := map[Topic][]string{
|
|
"Eval": {"*"},
|
|
}
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
defer cancel()
|
|
|
|
streamCh, err := events.Stream(ctx, topics, 0, q)
|
|
require.NoError(t, err)
|
|
|
|
select {
|
|
case event := <-streamCh:
|
|
if event.Err != nil {
|
|
require.Fail(t, err.Error())
|
|
}
|
|
require.Equal(t, len(event.Events), 1)
|
|
require.Equal(t, "Eval", string(event.Events[0].Topic))
|
|
case <-time.After(5 * time.Second):
|
|
require.Fail(t, "failed waiting for event stream event")
|
|
}
|
|
}
|
|
|
|
func TestEvent_Stream_Err_InvalidQueryParam(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
c, s := makeClient(t, nil, nil)
|
|
defer s.Stop()
|
|
|
|
// register job to generate events
|
|
jobs := c.Jobs()
|
|
job := testJob()
|
|
resp2, _, err := jobs.Register(job, nil)
|
|
require.Nil(t, err)
|
|
require.NotNil(t, resp2)
|
|
|
|
// build event stream request
|
|
events := c.EventStream()
|
|
q := &QueryOptions{}
|
|
topics := map[Topic][]string{
|
|
"Eval": {"::*"},
|
|
}
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
defer cancel()
|
|
|
|
_, err = events.Stream(ctx, topics, 0, q)
|
|
require.Error(t, err)
|
|
require.Contains(t, err.Error(), "400")
|
|
require.Contains(t, err.Error(), "Invalid key value pair")
|
|
}
|
|
|
|
func TestEvent_Stream_CloseCtx(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
c, s := makeClient(t, nil, nil)
|
|
defer s.Stop()
|
|
|
|
// register job to generate events
|
|
jobs := c.Jobs()
|
|
job := testJob()
|
|
resp2, _, err := jobs.Register(job, nil)
|
|
require.Nil(t, err)
|
|
require.NotNil(t, resp2)
|
|
|
|
// build event stream request
|
|
events := c.EventStream()
|
|
q := &QueryOptions{}
|
|
topics := map[Topic][]string{
|
|
"Eval": {"*"},
|
|
}
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
streamCh, err := events.Stream(ctx, topics, 0, q)
|
|
require.NoError(t, err)
|
|
|
|
// cancel the request
|
|
cancel()
|
|
|
|
select {
|
|
case event, ok := <-streamCh:
|
|
require.False(t, ok)
|
|
require.Nil(t, event)
|
|
case <-time.After(5 * time.Second):
|
|
require.Fail(t, "failed waiting for event stream event")
|
|
}
|
|
}
|
|
|
|
func TestEventStream_PayloadValue(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
c, s := makeClient(t, nil, func(c *testutil.TestServerConfig) {
|
|
c.DevMode = true
|
|
})
|
|
defer s.Stop()
|
|
|
|
// register job to generate events
|
|
jobs := c.Jobs()
|
|
job := testJob()
|
|
resp2, _, err := jobs.Register(job, nil)
|
|
require.Nil(t, err)
|
|
require.NotNil(t, resp2)
|
|
|
|
// build event stream request
|
|
events := c.EventStream()
|
|
q := &QueryOptions{}
|
|
topics := map[Topic][]string{
|
|
"Node": {"*"},
|
|
}
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
defer cancel()
|
|
|
|
streamCh, err := events.Stream(ctx, topics, 0, q)
|
|
require.NoError(t, err)
|
|
|
|
select {
|
|
case event := <-streamCh:
|
|
if event.Err != nil {
|
|
require.NoError(t, err)
|
|
}
|
|
for _, e := range event.Events {
|
|
n, err := e.Node()
|
|
require.NoError(t, err)
|
|
require.NotEqual(t, "", n.ID)
|
|
}
|
|
case <-time.After(5 * time.Second):
|
|
require.Fail(t, "failed waiting for event stream event")
|
|
}
|
|
}
|
|
|
|
func TestEventStream_PayloadValueHelpers(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
testCases := []struct {
|
|
topic Topic
|
|
event Event
|
|
input []byte
|
|
err string
|
|
expectFn func(t *testing.T, event Event)
|
|
}{
|
|
{
|
|
input: []byte(`{"Topic": "Deployment", "Payload": {"Deployment":{"ID":"some-id","JobID":"some-job-id", "TaskGroups": {"tg1": {"RequireProgressBy": "2020-11-05T11:52:54.370774000-05:00"}}}}}`),
|
|
expectFn: func(t *testing.T, event Event) {
|
|
eventTime, err := time.Parse(time.RFC3339, "2020-11-05T11:52:54.370774000-05:00")
|
|
require.NoError(t, err)
|
|
require.Equal(t, TopicDeployment, event.Topic)
|
|
|
|
d, err := event.Deployment()
|
|
require.NoError(t, err)
|
|
require.NoError(t, err)
|
|
require.Equal(t, &Deployment{
|
|
ID: "some-id",
|
|
JobID: "some-job-id",
|
|
TaskGroups: map[string]*DeploymentState{
|
|
"tg1": {
|
|
RequireProgressBy: eventTime,
|
|
},
|
|
},
|
|
}, d)
|
|
},
|
|
},
|
|
{
|
|
input: []byte(`{"Topic": "Eval", "Payload": {"Eval":{"ID":"some-id","Namespace":"some-namespace-id"}}}`),
|
|
expectFn: func(t *testing.T, event Event) {
|
|
require.Equal(t, TopicEval, event.Topic)
|
|
eval, err := event.Evaluation()
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, &Evaluation{
|
|
ID: "some-id",
|
|
Namespace: "some-namespace-id",
|
|
}, eval)
|
|
},
|
|
},
|
|
{
|
|
input: []byte(`{"Topic": "Alloc", "Payload": {"Alloc":{"ID":"some-id","Namespace":"some-namespace-id"}}}`),
|
|
expectFn: func(t *testing.T, event Event) {
|
|
require.Equal(t, TopicAlloc, event.Topic)
|
|
a, err := event.Allocation()
|
|
require.NoError(t, err)
|
|
require.Equal(t, &Allocation{
|
|
ID: "some-id",
|
|
Namespace: "some-namespace-id",
|
|
}, a)
|
|
},
|
|
},
|
|
{
|
|
input: []byte(`{"Topic": "Job", "Payload": {"Job":{"ID":"some-id","Namespace":"some-namespace-id"}}}`),
|
|
expectFn: func(t *testing.T, event Event) {
|
|
require.Equal(t, TopicJob, event.Topic)
|
|
j, err := event.Job()
|
|
require.NoError(t, err)
|
|
require.Equal(t, &Job{
|
|
ID: stringToPtr("some-id"),
|
|
Namespace: stringToPtr("some-namespace-id"),
|
|
}, j)
|
|
},
|
|
},
|
|
{
|
|
input: []byte(`{"Topic": "Node", "Payload": {"Node":{"ID":"some-id","Datacenter":"some-dc-id"}}}`),
|
|
expectFn: func(t *testing.T, event Event) {
|
|
require.Equal(t, TopicNode, event.Topic)
|
|
n, err := event.Node()
|
|
require.NoError(t, err)
|
|
require.Equal(t, &Node{
|
|
ID: "some-id",
|
|
Datacenter: "some-dc-id",
|
|
}, n)
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(string(tc.topic), func(t *testing.T) {
|
|
var out Event
|
|
err := json.Unmarshal(tc.input, &out)
|
|
require.NoError(t, err)
|
|
tc.expectFn(t, out)
|
|
})
|
|
}
|
|
}
|