Skip to content

Commit

Permalink
feat: dump kube pod logs on test failure
Browse files Browse the repository at this point in the history
  • Loading branch information
stuartwdouglas committed Oct 1, 2024
1 parent 24e2d87 commit 387ad5f
Show file tree
Hide file tree
Showing 3 changed files with 40 additions and 8 deletions.
4 changes: 2 additions & 2 deletions backend/controller/scaling/kube_scaling_integration_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ func TestKubeScaling(t *testing.T) {
in.Call("echo", "echo", "Bob", func(t testing.TB, response string) {
assert.Equal(t, "Hello, Bob!!!", response)
}),
in.VerifyKubeState(func(ctx context.Context, t testing.TB, namespace string, client *kubernetes.Clientset) {
in.VerifyKubeState(func(ctx context.Context, t testing.TB, namespace string, client kubernetes.Clientset) {
deps, err := client.AppsV1().Deployments(namespace).List(ctx, v1.ListOptions{})
assert.NoError(t, err)
for _, dep := range deps.Items {
Expand Down Expand Up @@ -76,7 +76,7 @@ func TestKubeScaling(t *testing.T) {
err := failure.Load()
assert.NoError(t, err)
},
in.VerifyKubeState(func(ctx context.Context, t testing.TB, namespace string, client *kubernetes.Clientset) {
in.VerifyKubeState(func(ctx context.Context, t testing.TB, namespace string, client kubernetes.Clientset) {
deps, err := client.AppsV1().Deployments(namespace).List(ctx, v1.ListOptions{})
assert.NoError(t, err)
depCount := 0
Expand Down
7 changes: 4 additions & 3 deletions internal/integration/actions.go
Original file line number Diff line number Diff line change
Expand Up @@ -227,7 +227,7 @@ func Deploy(module string) Action {
if ic.Provisioner != nil {
args = append(args, "--use-provisioner", "--provisioner-endpoint=http://localhost:8893")
}
if ic.kubeClient != nil {
if ic.kubeClient.Ok() {
args = append(args, "--build-env", "GOOS=linux", "--build-env", "GOARCH=amd64", "--build-env", "CGO_ENABLED=0")
}
args = append(args, module)
Expand Down Expand Up @@ -395,9 +395,10 @@ func Call[Req any, Resp any](module, verb string, request Req, check func(t test
}

// VerifyKubeState lets you test the current kube state
func VerifyKubeState(check func(ctx context.Context, t testing.TB, namespace string, client *kubernetes.Clientset)) Action {
func VerifyKubeState(check func(ctx context.Context, t testing.TB, namespace string, client kubernetes.Clientset)) Action {
return func(t testing.TB, ic TestContext) {
check(ic.Context, t, ic.kubeNamespace, ic.kubeClient)
clientset := ic.kubeClient.MustGet()
check(ic.Context, t, ic.kubeNamespace, clientset)
}
}

Expand Down
37 changes: 34 additions & 3 deletions internal/integration/harness.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import (
"context"
"errors"
"fmt"
"io"
"os"
"path/filepath"
"runtime"
Expand All @@ -20,6 +21,8 @@ import (
"github.com/alecthomas/assert/v2"
"github.com/alecthomas/types/optional"
"github.com/otiai10/copy"
kubecore "k8s.io/api/core/v1"
kubemeta "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/kubernetes"

"github.com/TBD54566975/ftl/backend/controller/scaling/k8sscaling"
Expand All @@ -39,7 +42,7 @@ func (i TestContext) integrationTestTimeout() time.Duration {
if err != nil {
panic(err)
}
if i.kubeClient != nil {
if i.kubeClient.Ok() {
// kube can be slow, give it some time
return d * 5
}
Expand Down Expand Up @@ -271,7 +274,7 @@ func run(t *testing.T, actionsOrOptions ...ActionOrOption) {
Verbs: verbs,
realT: t,
language: language,
kubeClient: kubeClient,
kubeClient: optional.Ptr(kubeClient),
kubeNamespace: kubeNamespace,
}

Expand Down Expand Up @@ -348,7 +351,7 @@ type TestContext struct {
// The Language under test
language string
// Set if the test is running on kubernetes
kubeClient *kubernetes.Clientset
kubeClient optional.Option[kubernetes.Clientset]
kubeNamespace string

Controller ftlv1connect.ControllerServiceClient
Expand Down Expand Up @@ -378,6 +381,34 @@ func (i TestContext) AssertWithRetry(t testing.TB, assertion Action) {
}
select {
case <-waitCtx.Done():
if client, ok := i.kubeClient.Get(); ok {
Infof("Kube logs:")
list, err := client.CoreV1().Pods(i.kubeNamespace).List(i, kubemeta.ListOptions{})
if err == nil {
for _, pod := range list.Items {
Infof("\n\n\n========== Pod %s ==========", pod.Name)
for _, container := range pod.Spec.Containers {
Infof("\n\n\n----------- Pod %s Container %s --------", pod.Name, container.Name)
req := client.CoreV1().Pods(i.kubeNamespace).GetLogs(pod.Name, &kubecore.PodLogOptions{Container: container.Name})
podLogs, err := req.Stream(context.Background())
if err != nil {
Infof("Error getting logs for pod %s: %v", pod.Name, err)
continue
}
buf := new(bytes.Buffer)
_, err = io.Copy(buf, podLogs)
if err != nil {
Infof("Error copying logs for pod %s: %v", pod.Name, err)
continue
}
str := buf.String()
Infof("%s", str)
_ = podLogs.Close()
}
}
}

}
t.Fatalf("Timed out waiting for assertion to pass: %s", err)

case <-time.After(time.Millisecond * 200):
Expand Down

0 comments on commit 387ad5f

Please sign in to comment.