mirror of
https://github.com/containers/podman.git
synced 2025-05-17 23:26:08 +08:00
252 lines
6.2 KiB
Go
252 lines
6.2 KiB
Go
package e2e_test
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/containers/podman/v4/pkg/machine"
|
|
"github.com/containers/podman/v4/pkg/machine/define"
|
|
"github.com/containers/storage/pkg/stringid"
|
|
. "github.com/onsi/ginkgo/v2"
|
|
. "github.com/onsi/gomega"
|
|
"github.com/onsi/gomega/format"
|
|
. "github.com/onsi/gomega/gexec"
|
|
"github.com/onsi/gomega/types"
|
|
"golang.org/x/exp/slices"
|
|
)
|
|
|
|
var originalHomeDir = os.Getenv("HOME")
|
|
|
|
const (
|
|
defaultTimeout = 240 * time.Second
|
|
)
|
|
|
|
type machineCommand interface {
|
|
buildCmd(m *machineTestBuilder) []string
|
|
}
|
|
|
|
type MachineTestBuilder interface {
|
|
setName(name string) *MachineTestBuilder
|
|
setCmd(mc machineCommand) *MachineTestBuilder
|
|
setTimeout(duration time.Duration) *MachineTestBuilder
|
|
run() (*machineSession, error)
|
|
}
|
|
type machineSession struct {
|
|
*Session
|
|
}
|
|
|
|
type machineTestBuilder struct {
|
|
cmd []string
|
|
imagePath string
|
|
name string
|
|
names []string
|
|
podmanBinary string
|
|
timeout time.Duration
|
|
}
|
|
|
|
// waitWithTimeout waits for a command to complete for a given
|
|
// number of seconds
|
|
func (ms *machineSession) waitWithTimeout(timeout time.Duration) {
|
|
Eventually(ms, timeout).Should(Exit())
|
|
}
|
|
|
|
func (ms *machineSession) Bytes() []byte {
|
|
return []byte(ms.outputToString())
|
|
}
|
|
|
|
func (ms *machineSession) outputToStringSlice() []string {
|
|
var results []string
|
|
output := string(ms.Out.Contents())
|
|
for _, line := range strings.Split(output, "\n") {
|
|
if line != "" {
|
|
results = append(results, line)
|
|
}
|
|
}
|
|
return results
|
|
}
|
|
|
|
// outputToString returns the output from a session in string form
|
|
func (ms *machineSession) outputToString() string {
|
|
if ms == nil || ms.Out == nil || ms.Out.Contents() == nil {
|
|
return ""
|
|
}
|
|
|
|
fields := strings.Fields(string(ms.Out.Contents()))
|
|
return strings.Join(fields, " ")
|
|
}
|
|
|
|
// errorToString returns the error output from a session in string form
|
|
func (ms *machineSession) errorToString() string {
|
|
if ms == nil || ms.Err == nil || ms.Err.Contents() == nil {
|
|
return ""
|
|
}
|
|
return string(ms.Err.Contents())
|
|
}
|
|
|
|
// newMB constructor for machine test builders
|
|
func newMB() (*machineTestBuilder, error) {
|
|
mb := machineTestBuilder{
|
|
timeout: defaultTimeout,
|
|
}
|
|
cwd, err := os.Getwd()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
mb.podmanBinary = filepath.Join(cwd, podmanBinary)
|
|
if os.Getenv("PODMAN_BINARY") != "" {
|
|
mb.podmanBinary = os.Getenv("PODMAN_BINARY")
|
|
}
|
|
if os.Getenv("MACHINE_TEST_TIMEOUT") != "" {
|
|
seconds, err := strconv.Atoi(os.Getenv("MACHINE_TEST_TIMEOUT"))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
mb.timeout = time.Duration(seconds) * time.Second
|
|
}
|
|
return &mb, nil
|
|
}
|
|
|
|
// setName sets the name of the virtuaql machine for the command
|
|
func (m *machineTestBuilder) setName(name string) *machineTestBuilder {
|
|
m.name = name
|
|
return m
|
|
}
|
|
|
|
// setCmd takes a machineCommand struct and assembles a cmd line
|
|
// representation of the podman machine command
|
|
func (m *machineTestBuilder) setCmd(mc machineCommand) *machineTestBuilder {
|
|
// If no name for the machine exists, we set a random name.
|
|
if !slices.Contains(m.names, m.name) {
|
|
if len(m.name) < 1 {
|
|
m.name = randomString()
|
|
}
|
|
m.names = append(m.names, m.name)
|
|
}
|
|
m.cmd = mc.buildCmd(m)
|
|
return m
|
|
}
|
|
|
|
func (m *machineTestBuilder) setTimeout(timeout time.Duration) *machineTestBuilder {
|
|
m.timeout = timeout
|
|
return m
|
|
}
|
|
|
|
// toQemuInspectInfo is only for inspecting qemu machines. Other providers will need
|
|
// to make their own.
|
|
func (m *machineTestBuilder) toQemuInspectInfo() ([]machine.InspectInfo, int, error) {
|
|
args := []string{"machine", "inspect"}
|
|
args = append(args, m.names...)
|
|
session, err := runWrapper(m.podmanBinary, args, defaultTimeout, true)
|
|
if err != nil {
|
|
return nil, -1, err
|
|
}
|
|
mii := []machine.InspectInfo{}
|
|
err = json.Unmarshal(session.Bytes(), &mii)
|
|
return mii, session.ExitCode(), err
|
|
}
|
|
|
|
func (m *machineTestBuilder) runWithoutWait() (*machineSession, error) {
|
|
return runWrapper(m.podmanBinary, m.cmd, m.timeout, false)
|
|
}
|
|
|
|
func (m *machineTestBuilder) run() (*machineSession, error) {
|
|
return runWrapper(m.podmanBinary, m.cmd, m.timeout, true)
|
|
}
|
|
|
|
func runWrapper(podmanBinary string, cmdArgs []string, timeout time.Duration, wait bool) (*machineSession, error) {
|
|
if len(os.Getenv("DEBUG")) > 0 {
|
|
cmdArgs = append([]string{"--log-level=debug"}, cmdArgs...)
|
|
}
|
|
GinkgoWriter.Println(podmanBinary + " " + strings.Join(cmdArgs, " "))
|
|
c := exec.Command(podmanBinary, cmdArgs...)
|
|
session, err := Start(c, GinkgoWriter, GinkgoWriter)
|
|
if err != nil {
|
|
Fail(fmt.Sprintf("Unable to start session: %q", err))
|
|
return nil, err
|
|
}
|
|
ms := machineSession{session}
|
|
if wait {
|
|
ms.waitWithTimeout(timeout)
|
|
}
|
|
return &ms, nil
|
|
}
|
|
|
|
// randomString returns a string of given length composed of random characters
|
|
func randomString() string {
|
|
return stringid.GenerateRandomID()[0:12]
|
|
}
|
|
|
|
type ValidJSONMatcher struct {
|
|
types.GomegaMatcher
|
|
}
|
|
|
|
func BeValidJSON() *ValidJSONMatcher {
|
|
return &ValidJSONMatcher{}
|
|
}
|
|
|
|
func (matcher *ValidJSONMatcher) Match(actual interface{}) (success bool, err error) {
|
|
s, ok := actual.(string)
|
|
if !ok {
|
|
return false, fmt.Errorf("ValidJSONMatcher expects a string, not %q", actual)
|
|
}
|
|
|
|
var i interface{}
|
|
if err := json.Unmarshal([]byte(s), &i); err != nil {
|
|
return false, err
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
func (matcher *ValidJSONMatcher) FailureMessage(actual interface{}) (message string) {
|
|
return format.Message(actual, "to be valid JSON")
|
|
}
|
|
|
|
func (matcher *ValidJSONMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
|
return format.Message(actual, "to _not_ be valid JSON")
|
|
}
|
|
|
|
func skipIfVmtype(vmType define.VMType, message string) {
|
|
if isVmtype(vmType) {
|
|
Skip(message)
|
|
}
|
|
}
|
|
|
|
func skipIfNotVmtype(vmType define.VMType, message string) {
|
|
if !isVmtype(vmType) {
|
|
Skip(message)
|
|
}
|
|
}
|
|
|
|
func skipIfWSL(message string) {
|
|
skipIfVmtype(define.WSLVirt, message)
|
|
}
|
|
|
|
func isVmtype(vmType define.VMType) bool {
|
|
return testProvider.VMType() == vmType
|
|
}
|
|
|
|
// isWSL is a simple wrapper to determine if the testprovider is WSL
|
|
func isWSL() bool {
|
|
return isVmtype(define.WSLVirt)
|
|
}
|
|
|
|
func getFCOSDownloadLocation(p machine.VirtProvider) string {
|
|
dd, err := p.NewDownload("")
|
|
if err != nil {
|
|
Fail("unable to create new download")
|
|
}
|
|
|
|
fcd, err := dd.GetFCOSDownload(defaultStream)
|
|
if err != nil {
|
|
Fail("unable to get virtual machine image")
|
|
}
|
|
|
|
return fcd.Location
|
|
}
|