210 lines
5.5 KiB
Go

// Copyright (c) 2021 Gitpod GmbH. All rights reserved.
// Licensed under the GNU Affero General Public License (AGPL).
// See License-AGPL.txt in the project root for license information.
package imagebuilder
import (
"context"
"errors"
"io"
"testing"
"time"
"golang.org/x/sync/errgroup"
"golang.org/x/xerrors"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"
"sigs.k8s.io/e2e-framework/pkg/envconf"
"sigs.k8s.io/e2e-framework/pkg/features"
csapi "github.com/gitpod-io/gitpod/content-service/api"
imgapi "github.com/gitpod-io/gitpod/image-builder/api"
"github.com/gitpod-io/gitpod/test/pkg/integration"
)
func TestBaseImageBuild(t *testing.T) {
f := features.New("database").
WithLabel("component", "image-builder").
Assess("it should build a base image", func(_ context.Context, t *testing.T, cfg *envconf.Config) context.Context {
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute)
defer cancel()
api := integration.NewComponentAPI(ctx, cfg.Namespace(), kubeconfig, cfg.Client())
t.Cleanup(func() {
api.Done(t)
})
client, err := api.ImageBuilder()
if err != nil {
t.Fatal("cannot start build", err)
}
bld, err := client.Build(ctx, &imgapi.BuildRequest{
ForceRebuild: true,
TriggeredBy: "integration-test",
Source: &imgapi.BuildSource{
From: &imgapi.BuildSource_File{
File: &imgapi.BuildSourceDockerfile{
DockerfileVersion: "some-version",
DockerfilePath: "test/tests/components/image-builder/test.Dockerfile",
ContextPath: ".",
Source: &csapi.WorkspaceInitializer{
Spec: &csapi.WorkspaceInitializer_Git{
Git: &csapi.GitInitializer{
RemoteUri: "https://github.com/gitpod-io/gitpod.git",
TargetMode: csapi.CloneTargetMode_REMOTE_BRANCH,
CloneTaget: "main",
Config: &csapi.GitConfig{
Authentication: csapi.GitAuthMethod_NO_AUTH,
},
},
},
},
},
},
},
})
if err != nil {
t.Fatal("cannot start build", err)
}
var ref string
for {
msg, err := bld.Recv()
if errors.Is(err, io.EOF) {
break
}
if err != nil {
if st, ok := status.FromError(err); ok && st.Code() == codes.Unavailable {
continue
}
t.Fatal(err)
}
ref = msg.Ref
if msg.Status == imgapi.BuildStatus_done_success {
break
} else if msg.Status == imgapi.BuildStatus_done_failure {
t.Fatalf("image build failed: %s", msg.Message)
} else {
t.Logf("build output: %s, %s", msg.Message, msg.Info)
}
}
if ref == "" {
t.Fatal("ref was empty")
}
return ctx
}).
Feature()
testEnv.Test(t, f)
}
func TestParallelBaseImageBuild(t *testing.T) {
f := features.New("image-builder").
WithLabel("component", "image-builder").
Assess("it should allow parallel build of images", func(_ context.Context, t *testing.T, cfg *envconf.Config) context.Context {
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute)
defer cancel()
api := integration.NewComponentAPI(ctx, cfg.Namespace(), kubeconfig, cfg.Client())
t.Cleanup(func() {
api.Done(t)
})
client, err := api.ImageBuilder()
if err != nil {
t.Fatalf("cannot start build: %q", err)
}
req := &imgapi.BuildRequest{
ForceRebuild: true,
TriggeredBy: "integration-test",
Source: &imgapi.BuildSource{
From: &imgapi.BuildSource_File{
File: &imgapi.BuildSourceDockerfile{
DockerfileVersion: "some-version",
DockerfilePath: "test/tests/components/image-builder/test.Dockerfile",
ContextPath: ".",
Source: &csapi.WorkspaceInitializer{
Spec: &csapi.WorkspaceInitializer_Git{
Git: &csapi.GitInitializer{
RemoteUri: "https://github.com/gitpod-io/gitpod.git",
TargetMode: csapi.CloneTargetMode_REMOTE_BRANCH,
CloneTaget: "main",
Config: &csapi.GitConfig{
Authentication: csapi.GitAuthMethod_NO_AUTH,
},
},
},
},
},
},
},
}
bld0, err := client.Build(ctx, req)
if err != nil {
t.Fatalf("cannot start build: %v", err)
}
bld1, err := client.Build(ctx, req)
if err != nil {
t.Fatalf("cannot start build: %v", err)
}
watchBuild := func(cl imgapi.ImageBuilder_BuildClient) error {
for {
msg, err := cl.Recv()
if errors.Is(err, io.EOF) {
break
}
if err != nil {
return xerrors.Errorf("image builder error: %v", err)
}
if err := ctx.Err(); err != nil {
return xerrors.Errorf("context error: %v", err)
}
if msg.Status == imgapi.BuildStatus_done_success {
break
} else if msg.Status == imgapi.BuildStatus_done_failure {
return xerrors.Errorf("image build failed: %s", msg.Message)
} else {
t.Logf("build output: %s", msg.Message)
}
}
return nil
}
var eg errgroup.Group
eg.Go(func() error { return watchBuild(bld0) })
eg.Go(func() error { return watchBuild(bld1) })
/*
blds, err := client.ListBuilds(ctx, &imgapi.ListBuildsRequest{})
if err != nil {
t.Fatalf("cannot list builds: %v", err)
}
// TODO(cw): make this assertion resiliant against other on-going builds
if l := len(blds.Builds); l != 1 {
t.Errorf("image builder is running not just one build, but %d", l)
}
*/
err = eg.Wait()
if err != nil {
t.Fatal(err)
}
return ctx
}).
Feature()
testEnv.Test(t, f)
}