Blame cmd/osbuild-image-tests/main_test.go

Packit Service 4d2de5
// +build integration
Packit Service 4d2de5
Packit Service 4d2de5
package main
Packit Service 4d2de5
Packit Service 4d2de5
import (
Packit Service 4d2de5
	"bytes"
Packit Service 4d2de5
	"context"
Packit Service 4d2de5
	"encoding/json"
Packit Service 4d2de5
	"flag"
Packit Service 4d2de5
	"fmt"
Packit Service 4d2de5
	"io"
Packit Service 4d2de5
	"io/ioutil"
Packit Service 4d2de5
	"log"
Packit Service 4d2de5
	"os"
Packit Service 4d2de5
	"os/exec"
Packit Service 4d2de5
	"path"
Packit Service 4d2de5
	"strings"
Packit Service 4d2de5
	"testing"
Packit Service 4d2de5
	"time"
Packit Service 4d2de5
Packit Service 4d2de5
	"github.com/gophercloud/gophercloud"
Packit Service 4d2de5
	"github.com/gophercloud/gophercloud/openstack"
Packit Service 4d2de5
	"github.com/stretchr/testify/assert"
Packit Service 4d2de5
	"github.com/stretchr/testify/require"
Packit Service 4d2de5
Packit Service 4d2de5
	"github.com/osbuild/osbuild-composer/cmd/osbuild-image-tests/azuretest"
Packit Service 4d2de5
	"github.com/osbuild/osbuild-composer/cmd/osbuild-image-tests/constants"
Packit Service 4d2de5
	"github.com/osbuild/osbuild-composer/cmd/osbuild-image-tests/openstacktest"
Packit Service 4d2de5
	"github.com/osbuild/osbuild-composer/cmd/osbuild-image-tests/vmwaretest"
Packit Service 4d2de5
	"github.com/osbuild/osbuild-composer/internal/common"
Packit Service 4d2de5
)
Packit Service 4d2de5
Packit Service 4d2de5
type testcaseStruct struct {
Packit Service 4d2de5
	ComposeRequest struct {
Packit Service 4d2de5
		Distro   string
Packit Service 4d2de5
		Arch     string
Packit Service 4d2de5
		Filename string
Packit Service 4d2de5
	} `json:"compose-request"`
Packit Service 4d2de5
	Manifest  json.RawMessage
Packit Service 4d2de5
	ImageInfo json.RawMessage `json:"image-info"`
Packit Service 4d2de5
	Boot      *struct {
Packit Service 4d2de5
		Type string
Packit Service 4d2de5
	}
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
var disableLocalBoot = flag.Bool("disable-local-boot", false, "when this flag is given, no images are booted locally using qemu (this does not affect testing in clouds)")
Packit Service 4d2de5
Packit Service 4d2de5
// runOsbuild runs osbuild with the specified manifest and output-directory.
Packit Service 4d2de5
func runOsbuild(manifest []byte, store, outputDirectory string) error {
Packit Service 4d2de5
	cmd := constants.GetOsbuildCommand(store, outputDirectory)
Packit Service 4d2de5
Packit Service 4d2de5
	cmd.Stdin = bytes.NewReader(manifest)
Packit Service 4d2de5
	var outBuffer bytes.Buffer
Packit Service 4d2de5
	cmd.Stdout = &outBuffer
Packit Service 4d2de5
	cmd.Stderr = &outBuffer
Packit Service 4d2de5
Packit Service 4d2de5
	err := cmd.Run()
Packit Service 4d2de5
	if err != nil {
Packit Service 4d2de5
		// Pretty print the osbuild error output.
Packit Service 4d2de5
		buf := new(bytes.Buffer)
Packit Service 4d2de5
		_ = json.Indent(buf, outBuffer.Bytes(), "", "    ")
Packit Service 4d2de5
		fmt.Println(buf)
Packit Service 4d2de5
Packit Service 4d2de5
		return fmt.Errorf("running osbuild failed: %v", err)
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	return nil
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
// testImageInfo runs image-info on image specified by imageImage and
Packit Service 4d2de5
// compares the result with expected image info
Packit Service 4d2de5
func testImageInfo(t *testing.T, imagePath string, rawImageInfoExpected []byte) {
Packit Service 4d2de5
	var imageInfoExpected interface{}
Packit Service 4d2de5
	err := json.Unmarshal(rawImageInfoExpected, &imageInfoExpected)
Packit Service 4d2de5
	require.NoErrorf(t, err, "cannot decode expected image info: %#v", err)
Packit Service 4d2de5
Packit Service 4d2de5
	cmd := constants.GetImageInfoCommand(imagePath)
Packit Service 4d2de5
	cmd.Stderr = os.Stderr
Packit Service 4d2de5
	reader, writer := io.Pipe()
Packit Service 4d2de5
	cmd.Stdout = writer
Packit Service 4d2de5
Packit Service 4d2de5
	err = cmd.Start()
Packit Service 4d2de5
	require.NoErrorf(t, err, "image-info cannot start: %#v", err)
Packit Service 4d2de5
Packit Service 4d2de5
	var imageInfoGot interface{}
Packit Service 4d2de5
	err = json.NewDecoder(reader).Decode(&imageInfoGot)
Packit Service 4d2de5
	require.NoErrorf(t, err, "decoding image-info output failed: %#v", err)
Packit Service 4d2de5
Packit Service 4d2de5
	err = cmd.Wait()
Packit Service 4d2de5
	require.NoErrorf(t, err, "running image-info failed: %#v", err)
Packit Service 4d2de5
Packit Service 4d2de5
	assert.Equal(t, imageInfoExpected, imageInfoGot)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
type timeoutError struct{}
Packit Service 4d2de5
Packit Service 4d2de5
func (*timeoutError) Error() string { return "" }
Packit Service 4d2de5
Packit Service 4d2de5
// trySSHOnce tries to test the running image using ssh once
Packit Service 4d2de5
// It returns timeoutError if ssh command returns 255, if it runs for more
Packit Service 4d2de5
// that 10 seconds or if systemd-is-running returns starting.
Packit Service 4d2de5
// It returns nil if systemd-is-running returns running or degraded.
Packit Service 4d2de5
// It can also return other errors in other error cases.
Packit Service 4d2de5
func trySSHOnce(address string, privateKey string, ns *netNS) error {
Packit Service 4d2de5
	ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
Packit Service 4d2de5
	defer cancel()
Packit Service 4d2de5
Packit Service 4d2de5
	cmdName := "ssh"
Packit Service 4d2de5
	cmdArgs := []string{
Packit Service 4d2de5
		"-p", "22",
Packit Service 4d2de5
		"-i", privateKey,
Packit Service 4d2de5
		"-o", "StrictHostKeyChecking=no",
Packit Service 4d2de5
		"-o", "UserKnownHostsFile=/dev/null",
Packit Service 4d2de5
		"redhat@" + address,
Packit Service 4d2de5
		"systemctl --wait is-system-running",
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	var cmd *exec.Cmd
Packit Service 4d2de5
Packit Service 4d2de5
	if ns != nil {
Packit Service 4d2de5
		cmd = ns.NamespacedCommandContext(ctx, cmdName, cmdArgs...)
Packit Service 4d2de5
	} else {
Packit Service 4d2de5
		cmd = exec.CommandContext(ctx, cmdName, cmdArgs...)
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	output, err := cmd.Output()
Packit Service 4d2de5
Packit Service 4d2de5
	if ctx.Err() == context.DeadlineExceeded {
Packit Service 4d2de5
		return &timeoutError{}
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	if err != nil {
Packit Service 4d2de5
		if exitError, ok := err.(*exec.ExitError); ok {
Packit Service 4d2de5
			if exitError.ExitCode() == 255 {
Packit Service 4d2de5
				return &timeoutError{}
Packit Service 4d2de5
			}
Packit Service 4d2de5
		} else {
Packit Service 4d2de5
			return fmt.Errorf("ssh command failed from unknown reason: %#v", err)
Packit Service 4d2de5
		}
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	outputString := strings.TrimSpace(string(output))
Packit Service 4d2de5
	switch outputString {
Packit Service 4d2de5
	case "running":
Packit Service 4d2de5
		return nil
Packit Service 4d2de5
	case "degraded":
Packit Service 4d2de5
		log.Print("ssh test passed, but the system is degraded")
Packit Service 4d2de5
		return nil
Packit Service 4d2de5
	case "starting":
Packit Service 4d2de5
		return &timeoutError{}
Packit Service 4d2de5
	default:
Packit Service 4d2de5
		return fmt.Errorf("ssh test failed, system status is: %s", outputString)
Packit Service 4d2de5
	}
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
// testSSH tests the running image using ssh.
Packit Service 4d2de5
// It tries 20 attempts before giving up. If a major error occurs, it might
Packit Service 4d2de5
// return earlier.
Packit Service 4d2de5
func testSSH(t *testing.T, address string, privateKey string, ns *netNS) {
Packit Service 4d2de5
	const attempts = 20
Packit Service 4d2de5
	for i := 0; i < attempts; i++ {
Packit Service 4d2de5
		err := trySSHOnce(address, privateKey, ns)
Packit Service 4d2de5
		if err == nil {
Packit Service 4d2de5
			// pass the test
Packit Service 4d2de5
			return
Packit Service 4d2de5
		}
Packit Service 4d2de5
Packit Service 4d2de5
		// if any other error than the timeout one happened, fail the test immediately
Packit Service 4d2de5
		if _, ok := err.(*timeoutError); !ok {
Packit Service 4d2de5
			t.Fatal(err)
Packit Service 4d2de5
		}
Packit Service 4d2de5
Packit Service 4d2de5
		time.Sleep(10 * time.Second)
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	t.Errorf("ssh test failure, %d attempts were made", attempts)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func testBootUsingQemu(t *testing.T, imagePath string) {
Packit Service 4d2de5
	if *disableLocalBoot {
Packit Service 4d2de5
		t.Skip("local booting was disabled by -disable-local-boot, skipping")
Packit Service 4d2de5
	}
Packit Service 4d2de5
	err := withNetworkNamespace(func(ns netNS) error {
Packit Service 4d2de5
		return withBootedQemuImage(imagePath, ns, func() error {
Packit Service 4d2de5
			testSSH(t, "localhost", constants.TestPaths.PrivateKey, &ns)
Packit Service 4d2de5
			return nil
Packit Service 4d2de5
		})
Packit Service 4d2de5
	})
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func testBootUsingNspawnImage(t *testing.T, imagePath string) {
Packit Service 4d2de5
	err := withNetworkNamespace(func(ns netNS) error {
Packit Service 4d2de5
		return withBootedNspawnImage(imagePath, ns, func() error {
Packit Service 4d2de5
			testSSH(t, "localhost", constants.TestPaths.PrivateKey, &ns)
Packit Service 4d2de5
			return nil
Packit Service 4d2de5
		})
Packit Service 4d2de5
	})
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func testBootUsingNspawnDirectory(t *testing.T, imagePath string) {
Packit Service 4d2de5
	err := withNetworkNamespace(func(ns netNS) error {
Packit Service 4d2de5
		return withExtractedTarArchive(imagePath, func(dir string) error {
Packit Service 4d2de5
			return withBootedNspawnDirectory(dir, ns, func() error {
Packit Service 4d2de5
				testSSH(t, "localhost", constants.TestPaths.PrivateKey, &ns)
Packit Service 4d2de5
				return nil
Packit Service 4d2de5
			})
Packit Service 4d2de5
		})
Packit Service 4d2de5
	})
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func testBootUsingAWS(t *testing.T, imagePath string) {
Packit Service 4d2de5
	creds, err := getAWSCredentialsFromEnv()
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// if no credentials are given, fall back to qemu
Packit Service 4d2de5
	if creds == nil {
Packit Service 4d2de5
		log.Print("no AWS credentials given, falling back to booting using qemu")
Packit Service 4d2de5
		testBootUsingQemu(t, imagePath)
Packit Service 4d2de5
		return
Packit Service 4d2de5
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	imageName, err := generateRandomString("osbuild-image-tests-image-")
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	e, err := newEC2(creds)
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// the following line should be done by osbuild-composer at some point
Packit Service 4d2de5
	err = uploadImageToAWS(creds, imagePath, imageName)
Packit Service 4d2de5
	require.NoErrorf(t, err, "upload to amazon failed, resources could have been leaked")
Packit Service 4d2de5
Packit Service 4d2de5
	imageDesc, err := describeEC2Image(e, imageName)
Packit Service 4d2de5
	require.NoErrorf(t, err, "cannot describe the ec2 image")
Packit Service 4d2de5
Packit Service 4d2de5
	// delete the image after the test is over
Packit Service 4d2de5
	defer func() {
Packit Service 4d2de5
		err = deleteEC2Image(e, imageDesc)
Packit Service 4d2de5
		require.NoErrorf(t, err, "cannot delete the ec2 image, resources could have been leaked")
Packit Service 4d2de5
	}()
Packit Service 4d2de5
Packit Service 4d2de5
	// boot the uploaded image and try to connect to it
Packit Service 4d2de5
	err = withSSHKeyPair(func(privateKey, publicKey string) error {
Packit Service 4d2de5
		return withBootedImageInEC2(e, imageDesc, publicKey, func(address string) error {
Packit Service 4d2de5
			testSSH(t, address, privateKey, nil)
Packit Service 4d2de5
			return nil
Packit Service 4d2de5
		})
Packit Service 4d2de5
	})
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func testBootUsingAzure(t *testing.T, imagePath string) {
Packit Service 4d2de5
	creds, err := azuretest.GetAzureCredentialsFromEnv()
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// if no credentials are given, fall back to qemu
Packit Service 4d2de5
	if creds == nil {
Packit Service 4d2de5
		log.Print("no Azure credentials given, falling back to booting using qemu")
Packit Service 4d2de5
		testBootUsingQemu(t, imagePath)
Packit Service 4d2de5
		return
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	// create a random test id to name all the resources used in this test
Packit Service 4d2de5
	testId, err := generateRandomString("")
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	imageName := "image-" + testId + ".vhd"
Packit Service 4d2de5
Packit Service 4d2de5
	// the following line should be done by osbuild-composer at some point
Packit Service 4d2de5
	err = azuretest.UploadImageToAzure(creds, imagePath, imageName)
Packit Service 4d2de5
	require.NoErrorf(t, err, "upload to azure failed, resources could have been leaked")
Packit Service 4d2de5
Packit Service 4d2de5
	// delete the image after the test is over
Packit Service 4d2de5
	defer func() {
Packit Service 4d2de5
		err = azuretest.DeleteImageFromAzure(creds, imageName)
Packit Service 4d2de5
		require.NoErrorf(t, err, "cannot delete the azure image, resources could have been leaked")
Packit Service 4d2de5
	}()
Packit Service 4d2de5
Packit Service 4d2de5
	// boot the uploaded image and try to connect to it
Packit Service 4d2de5
	err = withSSHKeyPair(func(privateKey, publicKey string) error {
Packit Service 4d2de5
		return azuretest.WithBootedImageInAzure(creds, imageName, testId, publicKey, func(address string) error {
Packit Service 4d2de5
			testSSH(t, address, privateKey, nil)
Packit Service 4d2de5
			return nil
Packit Service 4d2de5
		})
Packit Service 4d2de5
	})
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func testBootUsingOpenStack(t *testing.T, imagePath string) {
Packit Service 4d2de5
	creds, err := openstack.AuthOptionsFromEnv()
Packit Service 4d2de5
Packit Service 4d2de5
	// if no credentials are given, fall back to qemu
Packit Service 4d2de5
	if (creds == gophercloud.AuthOptions{}) {
Packit Service 4d2de5
		log.Print("No OpenStack credentials given, falling back to booting using qemu")
Packit Service 4d2de5
		testBootUsingQemu(t, imagePath)
Packit Service 4d2de5
		return
Packit Service 4d2de5
	}
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// provider is the top-level client that all OpenStack services derive from
Packit Service 4d2de5
	provider, err := openstack.AuthenticatedClient(creds)
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// create a random test id to name all the resources used in this test
Packit Service 4d2de5
	imageName, err := generateRandomString("osbuild-image-tests-openstack-image-")
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// the following line should be done by osbuild-composer at some point
Packit Service 4d2de5
	image, err := openstacktest.UploadImageToOpenStack(provider, imagePath, imageName)
Packit Service 4d2de5
	require.NoErrorf(t, err, "Upload to OpenStack failed, resources could have been leaked")
Packit Service 4d2de5
	require.NotNil(t, image)
Packit Service 4d2de5
Packit Service 4d2de5
	// delete the image after the test is over
Packit Service 4d2de5
	defer func() {
Packit Service 4d2de5
		err = openstacktest.DeleteImageFromOpenStack(provider, image.ID)
Packit Service 4d2de5
		require.NoErrorf(t, err, "Cannot delete OpenStack image, resources could have been leaked")
Packit Service 4d2de5
	}()
Packit Service 4d2de5
Packit Service 4d2de5
	// boot the uploaded image and try to connect to it
Packit Service 4d2de5
	err = withSSHKeyPair(func(privateKey, publicKey string) error {
Packit Service 4d2de5
		userData, err := createUserData(publicKey)
Packit Service 4d2de5
		require.NoErrorf(t, err, "Creating user data failed: %v", err)
Packit Service 4d2de5
Packit Service 4d2de5
		return openstacktest.WithBootedImageInOpenStack(provider, image.ID, userData, func(address string) error {
Packit Service 4d2de5
			testSSH(t, address, privateKey, nil)
Packit Service 4d2de5
			return nil
Packit Service 4d2de5
		})
Packit Service 4d2de5
	})
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func testBootUsingVMware(t *testing.T, imagePath string) {
Packit Service 4d2de5
	creds, err := vmwaretest.AuthOptionsFromEnv()
Packit Service 4d2de5
Packit Service 4d2de5
	// if no credentials are given, fall back to qemu
Packit Service 4d2de5
	if (creds == nil) {
Packit Service 4d2de5
		log.Print("No vCenter credentials given, falling back to booting using qemu")
Packit Service 4d2de5
		log.Printf("Error=%v", err)
Packit Service 4d2de5
		testBootUsingQemu(t, imagePath)
Packit Service 4d2de5
		return
Packit Service 4d2de5
	}
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// convert to streamOptimized vmdk
Packit Service 4d2de5
	imagePath, err = vmwaretest.ConvertToStreamOptimizedVmdk(imagePath)
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
	require.NotEqual(t, "", imagePath)
Packit Service 4d2de5
	defer os.Remove(imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	// create a random test id to name all the resources used in this test
Packit Service 4d2de5
	imageName, err := generateRandomString("osbuild-image-tests-vmware-image-")
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	// the following line should be done by osbuild-composer at some point
Packit Service 4d2de5
	err = vmwaretest.ImportImage(creds, imagePath, imageName)
Packit Service 4d2de5
	require.NoErrorf(t, err, "Upload to vCenter failed, resources could have been leaked")
Packit Service 4d2de5
Packit Service 4d2de5
	// delete the image after the test is over
Packit Service 4d2de5
	defer func() {
Packit Service 4d2de5
		err = vmwaretest.DeleteImage(creds, imageName)
Packit Service 4d2de5
		require.NoErrorf(t, err, "Cannot delete image from vCenter, resources could have been leaked")
Packit Service 4d2de5
	}()
Packit Service 4d2de5
Packit Service 4d2de5
	// boot the uploaded image and try to connect to it
Packit Service 4d2de5
	err = vmwaretest.WithSSHKeyPair(func(privateKey, publicKey string) error {
Packit Service 4d2de5
		return vmwaretest.WithBootedImage(creds, imagePath, imageName, publicKey, func(address string) error {
Packit Service 4d2de5
			testSSH(t, address, privateKey, nil)
Packit Service 4d2de5
			return nil
Packit Service 4d2de5
		})
Packit Service 4d2de5
	})
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
// testBoot tests if the image is able to successfully boot
Packit Service 4d2de5
// Before the test it boots the image respecting the specified bootType.
Packit Service 4d2de5
// The test passes if the function is able to connect to the image via ssh
Packit Service 4d2de5
// in defined number of attempts and systemd-is-running returns running
Packit Service 4d2de5
// or degraded status.
Packit Service 4d2de5
func testBoot(t *testing.T, imagePath string, bootType string) {
Packit Service 4d2de5
	switch bootType {
Packit Service 4d2de5
	case "qemu":
Packit Service 4d2de5
		testBootUsingQemu(t, imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	case "nspawn":
Packit Service 4d2de5
		testBootUsingNspawnImage(t, imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	case "nspawn-extract":
Packit Service 4d2de5
		testBootUsingNspawnDirectory(t, imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	case "aws":
Packit Service 4d2de5
		testBootUsingAWS(t, imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	case "azure":
Packit Service 4d2de5
		testBootUsingAzure(t, imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	case "openstack":
Packit Service 4d2de5
		testBootUsingOpenStack(t, imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	case "vmware":
Packit Service 4d2de5
		testBootUsingVMware(t, imagePath)
Packit Service 4d2de5
Packit Service 4d2de5
	default:
Packit Service 4d2de5
		panic("unknown boot type!")
Packit Service 4d2de5
	}
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func kvmAvailable() bool {
Packit Service 4d2de5
	_, err := os.Stat("/dev/kvm")
Packit Service 4d2de5
	// File exists
Packit Service 4d2de5
	if err == nil {
Packit Service 4d2de5
		// KVM is available
Packit Service 4d2de5
		return true
Packit Service 4d2de5
	} else if os.IsNotExist(err) {
Packit Service 4d2de5
		// KVM is not available as /dev/kvm is missing
Packit Service 4d2de5
		return false
Packit Service 4d2de5
	} else {
Packit Service 4d2de5
		// The error was different than non-existing file which is unexpected
Packit Service 4d2de5
		panic(err)
Packit Service 4d2de5
	}
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
// testImage performs a series of tests specified in the testcase
Packit Service 4d2de5
// on an image
Packit Service 4d2de5
func testImage(t *testing.T, testcase testcaseStruct, imagePath string) {
Packit Service 4d2de5
	if testcase.ImageInfo != nil {
Packit Service 4d2de5
		t.Run("image info", func(t *testing.T) {
Packit Service 4d2de5
			testImageInfo(t, imagePath, testcase.ImageInfo)
Packit Service 4d2de5
		})
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	if testcase.Boot != nil {
Packit Service 4d2de5
		if common.CurrentArch() == "aarch64" && !kvmAvailable() {
Packit Service 4d2de5
			t.Log("Running on aarch64 without KVM support, skipping the boot test.")
Packit Service 4d2de5
			return
Packit Service 4d2de5
		}
Packit Service 4d2de5
		t.Run("boot", func(t *testing.T) {
Packit Service 4d2de5
			testBoot(t, imagePath, testcase.Boot.Type)
Packit Service 4d2de5
		})
Packit Service 4d2de5
	}
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
// runTestcase builds the pipeline specified in the testcase and then it
Packit Service 4d2de5
// tests the result
Packit Service 4d2de5
func runTestcase(t *testing.T, testcase testcaseStruct, store string) {
Packit Service 4d2de5
	_ = os.Mkdir("/var/lib/osbuild-composer-tests", 0755)
Packit Service 4d2de5
	outputDirectory, err := ioutil.TempDir("/var/lib/osbuild-composer-tests", "osbuild-image-tests-*")
Packit Service 4d2de5
	require.NoError(t, err, "error creating temporary output directory")
Packit Service 4d2de5
Packit Service 4d2de5
	defer func() {
Packit Service 4d2de5
		err := os.RemoveAll(outputDirectory)
Packit Service 4d2de5
		require.NoError(t, err, "error removing temporary output directory")
Packit Service 4d2de5
	}()
Packit Service 4d2de5
Packit Service 4d2de5
	err = runOsbuild(testcase.Manifest, store, outputDirectory)
Packit Service 4d2de5
	require.NoError(t, err)
Packit Service 4d2de5
Packit Service 4d2de5
	imagePath := fmt.Sprintf("%s/%s", outputDirectory, testcase.ComposeRequest.Filename)
Packit Service 4d2de5
Packit Service 4d2de5
	testImage(t, testcase, imagePath)
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
// getAllCases returns paths to all testcases in the testcase directory
Packit Service 4d2de5
func getAllCases() ([]string, error) {
Packit Service 4d2de5
	cases, err := ioutil.ReadDir(constants.TestPaths.TestCasesDirectory)
Packit Service 4d2de5
	if err != nil {
Packit Service 4d2de5
		return nil, fmt.Errorf("cannot list test cases: %#v", err)
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	casesPaths := []string{}
Packit Service 4d2de5
	for _, c := range cases {
Packit Service 4d2de5
		if c.IsDir() {
Packit Service 4d2de5
			continue
Packit Service 4d2de5
		}
Packit Service 4d2de5
Packit Service 4d2de5
		casePath := fmt.Sprintf("%s/%s", constants.TestPaths.TestCasesDirectory, c.Name())
Packit Service 4d2de5
		casesPaths = append(casesPaths, casePath)
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	return casesPaths, nil
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
// runTests opens, parses and runs all the specified testcases
Packit Service 4d2de5
func runTests(t *testing.T, cases []string) {
Packit Service 4d2de5
	_ = os.Mkdir("/var/lib/osbuild-composer-tests", 0755)
Packit Service 4d2de5
	store, err := ioutil.TempDir("/var/lib/osbuild-composer-tests", "osbuild-image-tests-*")
Packit Service 4d2de5
	require.NoError(t, err, "error creating temporary store")
Packit Service 4d2de5
Packit Service 4d2de5
	defer func() {
Packit Service 4d2de5
		err := os.RemoveAll(store)
Packit Service 4d2de5
		require.NoError(t, err, "error removing temporary store")
Packit Service 4d2de5
	}()
Packit Service 4d2de5
Packit Service 4d2de5
	for _, p := range cases {
Packit Service 4d2de5
		t.Run(path.Base(p), func(t *testing.T) {
Packit Service 4d2de5
			f, err := os.Open(p)
Packit Service 4d2de5
			if err != nil {
Packit Service 4d2de5
				t.Skipf("%s: cannot open test case: %#v", p, err)
Packit Service 4d2de5
			}
Packit Service 4d2de5
Packit Service 4d2de5
			var testcase testcaseStruct
Packit Service 4d2de5
			err = json.NewDecoder(f).Decode(&testcase)
Packit Service 4d2de5
			require.NoErrorf(t, err, "%s: cannot decode test case", p)
Packit Service 4d2de5
Packit Service 4d2de5
			currentArch := common.CurrentArch()
Packit Service 4d2de5
			if testcase.ComposeRequest.Arch != currentArch {
Packit Service 4d2de5
				t.Skipf("the required arch is %s, the current arch is %s", testcase.ComposeRequest.Arch, currentArch)
Packit Service 4d2de5
			}
Packit Service 4d2de5
Packit Service 4d2de5
			runTestcase(t, testcase, store)
Packit Service 4d2de5
		})
Packit Service 4d2de5
Packit Service 4d2de5
	}
Packit Service 4d2de5
}
Packit Service 4d2de5
Packit Service 4d2de5
func TestImages(t *testing.T) {
Packit Service 4d2de5
	cases := flag.Args()
Packit Service 4d2de5
	// if no cases were specified, run the default set
Packit Service 4d2de5
	if len(cases) == 0 {
Packit Service 4d2de5
		var err error
Packit Service 4d2de5
		cases, err = getAllCases()
Packit Service 4d2de5
		require.NoError(t, err)
Packit Service 4d2de5
	}
Packit Service 4d2de5
Packit Service 4d2de5
	runTests(t, cases)
Packit Service 4d2de5
}