Files
moby/integration/system/event_test.go
2025-10-20 12:55:52 -05:00

136 lines
4.0 KiB
Go

package system
import (
"context"
"errors"
"io"
"net/http"
"testing"
"time"
"github.com/moby/moby/api/types/events"
"github.com/moby/moby/api/types/mount"
"github.com/moby/moby/client"
"github.com/moby/moby/v2/integration/internal/container"
"github.com/moby/moby/v2/internal/testutil/request"
"gotest.tools/v3/assert"
is "gotest.tools/v3/assert/cmp"
"gotest.tools/v3/skip"
)
func TestEventsExecDie(t *testing.T) {
skip.If(t, testEnv.DaemonInfo.OSType == "windows", "FIXME. Suspect may need to wait until container is running before exec")
ctx := setupTest(t)
apiClient := testEnv.APIClient()
cID := container.Run(ctx, t, apiClient)
id, err := apiClient.ContainerExecCreate(ctx, cID, client.ExecCreateOptions{
Cmd: []string{"echo", "hello"},
})
assert.NilError(t, err)
msg, errs := apiClient.Events(ctx, client.EventsListOptions{
Filters: make(client.Filters).Add("container", cID).Add("event", string(events.ActionExecDie)),
})
err = apiClient.ContainerExecStart(ctx, id.ID, client.ExecStartOptions{
Detach: true,
Tty: false,
})
assert.NilError(t, err)
select {
case m := <-msg:
assert.Equal(t, m.Type, events.ContainerEventType)
assert.Equal(t, m.Actor.ID, cID)
assert.Equal(t, m.Action, events.ActionExecDie)
assert.Equal(t, m.Actor.Attributes["execID"], id.ID)
assert.Equal(t, m.Actor.Attributes["exitCode"], "0")
case err = <-errs:
assert.NilError(t, err)
case <-time.After(time.Second * 3):
t.Fatal("timeout hit")
}
}
// TestEventsNonBlocking verifies that the API responds immediately (not blocking),
// if there are no events.
func TestEventsNonBlocking(t *testing.T) {
ctx := setupTest(t)
// makes sure the API responds immediately (we use "less than 3 sec" to
// have some grace-period).
expectedTime := time.Now().Add(3 * time.Second)
emptyResp, emptyBody, err := request.Get(ctx, "/events")
assert.NilError(t, err)
defer emptyBody.Close()
assert.Check(t, is.DeepEqual(http.StatusOK, emptyResp.StatusCode))
assert.Check(t, time.Now().Before(expectedTime), "timeout waiting for events api to respond, should have responded immediately")
}
// TestEventsVolumeCreate verifies that volume create events are only fired
// once: when creating the volume, and not when attaching to a container.
func TestEventsVolumeCreate(t *testing.T) {
skip.If(t, testEnv.DaemonInfo.OSType == "windows", "FIXME: Windows doesn't trigger the events? Could be a race")
ctx := setupTest(t)
ctx, cancel := context.WithCancel(ctx)
defer cancel()
apiClient := testEnv.APIClient()
since := request.DaemonUnixTime(ctx, t, apiClient, testEnv)
volName := t.Name()
getEvents := func(messages <-chan events.Message, errs <-chan error) ([]events.Message, error) {
var evts []events.Message
for {
select {
case m := <-messages:
evts = append(evts, m)
case err := <-errs:
if errors.Is(err, io.EOF) {
return evts, nil
}
return nil, err
case <-time.After(time.Second * 3):
return nil, errors.New("timeout hit")
}
}
}
_, err := apiClient.VolumeCreate(ctx, client.VolumeCreateOptions{Name: volName})
assert.NilError(t, err)
filter := make(client.Filters).
Add("type", "volume").
Add("event", "create").
Add("volume", volName)
messages, errs := apiClient.Events(ctx, client.EventsListOptions{
Since: since,
Until: request.DaemonUnixTime(ctx, t, apiClient, testEnv),
Filters: filter,
})
volEvents, err := getEvents(messages, errs)
assert.NilError(t, err)
assert.Equal(t, len(volEvents), 1, "expected volume create event when creating a volume")
container.Create(ctx, t, apiClient, container.WithMount(mount.Mount{
Type: mount.TypeVolume,
Source: volName,
Target: "/tmp/foo",
}))
messages, errs = apiClient.Events(ctx, client.EventsListOptions{
Since: since,
Until: request.DaemonUnixTime(ctx, t, apiClient, testEnv),
Filters: filter,
})
volEvents, err = getEvents(messages, errs)
assert.NilError(t, err)
assert.Equal(t, len(volEvents), 1, "expected volume create event to be fired only once")
}