mirror of https://github.com/containers/podman.git
183 lines
6.0 KiB
Go
183 lines
6.0 KiB
Go
//go:build linux
|
|
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"os/user"
|
|
"path"
|
|
"path/filepath"
|
|
"strconv"
|
|
"syscall"
|
|
"testing"
|
|
|
|
"github.com/containers/podman/v5/pkg/systemd/quadlet"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestIsUnambiguousName(t *testing.T) {
|
|
tests := []struct {
|
|
input string
|
|
res bool
|
|
}{
|
|
// Ambiguous names
|
|
{"fedora", false},
|
|
{"fedora:latest", false},
|
|
{"library/fedora", false},
|
|
{"library/fedora:latest", false},
|
|
{"busybox@sha256:d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05a", false},
|
|
{"busybox:latest@sha256:d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05a", false},
|
|
{"d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05", false},
|
|
{"d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05aa", false},
|
|
|
|
// Unambiguous names
|
|
{"quay.io/fedora", true},
|
|
{"docker.io/fedora", true},
|
|
{"docker.io/library/fedora:latest", true},
|
|
{"localhost/fedora", true},
|
|
{"localhost:5000/fedora:latest", true},
|
|
{"example.foo.this.may.be.garbage.but.maybe.not:1234/fedora:latest", true},
|
|
{"docker.io/library/busybox@sha256:d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05a", true},
|
|
{"docker.io/library/busybox:latest@sha256:d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05a", true},
|
|
{"docker.io/fedora@sha256:d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05a", true},
|
|
{"sha256:d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05a", true},
|
|
{"d366a4665ab44f0648d7a00ae3fae139d55e32f9712c67accd604bb55df9d05a", true},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
res := isUnambiguousName(test.input)
|
|
assert.Equal(t, res, test.res, "%q", test.input)
|
|
}
|
|
}
|
|
|
|
func TestUnitDirs(t *testing.T) {
|
|
u, err := user.Current()
|
|
assert.Nil(t, err)
|
|
uidInt, err := strconv.Atoi(u.Uid)
|
|
assert.Nil(t, err)
|
|
|
|
if os.Getenv("_UNSHARED") != "true" {
|
|
unitDirs := getUnitDirs(false)
|
|
rootDirs := []string{}
|
|
rootDirs = appendSubPaths(rootDirs, quadlet.UnitDirTemp, false, userLevelFilter)
|
|
rootDirs = appendSubPaths(rootDirs, quadlet.UnitDirAdmin, false, userLevelFilter)
|
|
rootDirs = appendSubPaths(rootDirs, quadlet.UnitDirDistro, false, userLevelFilter)
|
|
assert.Equal(t, unitDirs, rootDirs, "rootful unit dirs should match")
|
|
|
|
configDir, err := os.UserConfigDir()
|
|
assert.Nil(t, err)
|
|
|
|
rootlessDirs := []string{}
|
|
|
|
runtimeDir, found := os.LookupEnv("XDG_RUNTIME_DIR")
|
|
if found {
|
|
rootlessDirs = appendSubPaths(rootlessDirs, path.Join(runtimeDir, "containers/systemd"), false, nil)
|
|
}
|
|
rootlessDirs = appendSubPaths(rootlessDirs, path.Join(configDir, "containers/systemd"), false, nil)
|
|
rootlessDirs = appendSubPaths(rootlessDirs, filepath.Join(quadlet.UnitDirAdmin, "users"), true, nonNumericFilter)
|
|
rootlessDirs = appendSubPaths(rootlessDirs, filepath.Join(quadlet.UnitDirAdmin, "users", u.Uid), true, userLevelFilter)
|
|
rootlessDirs = append(rootlessDirs, filepath.Join(quadlet.UnitDirAdmin, "users"))
|
|
|
|
unitDirs = getUnitDirs(true)
|
|
assert.Equal(t, unitDirs, rootlessDirs, "rootless unit dirs should match")
|
|
|
|
name, err := os.MkdirTemp("", "dir")
|
|
assert.Nil(t, err)
|
|
// remove the temporary directory at the end of the program
|
|
defer os.RemoveAll(name)
|
|
|
|
t.Setenv("QUADLET_UNIT_DIRS", name)
|
|
unitDirs = getUnitDirs(false)
|
|
assert.Equal(t, unitDirs, []string{name}, "rootful should use environment variable")
|
|
|
|
unitDirs = getUnitDirs(true)
|
|
assert.Equal(t, unitDirs, []string{name}, "rootless should use environment variable")
|
|
|
|
symLinkTestBaseDir, err := os.MkdirTemp("", "podman-symlinktest")
|
|
assert.Nil(t, err)
|
|
// remove the temporary directory at the end of the program
|
|
defer os.RemoveAll(symLinkTestBaseDir)
|
|
|
|
actualDir := filepath.Join(symLinkTestBaseDir, "actual")
|
|
err = os.Mkdir(actualDir, 0755)
|
|
assert.Nil(t, err)
|
|
innerDir := filepath.Join(actualDir, "inner")
|
|
err = os.Mkdir(innerDir, 0755)
|
|
assert.Nil(t, err)
|
|
symlink := filepath.Join(symLinkTestBaseDir, "symlink")
|
|
err = os.Symlink(actualDir, symlink)
|
|
assert.Nil(t, err)
|
|
t.Setenv("QUADLET_UNIT_DIRS", symlink)
|
|
unitDirs = getUnitDirs(true)
|
|
assert.Equal(t, unitDirs, []string{actualDir, innerDir}, "directory resolution should follow symlink")
|
|
|
|
// because chroot is only available for root,
|
|
// unshare the namespace and map user to root
|
|
c := exec.Command("/proc/self/exe", os.Args[1:]...)
|
|
c.Stdin = os.Stdin
|
|
c.Stdout = os.Stdout
|
|
c.Stderr = os.Stderr
|
|
c.SysProcAttr = &syscall.SysProcAttr{
|
|
Cloneflags: syscall.CLONE_NEWUSER,
|
|
UidMappings: []syscall.SysProcIDMap{
|
|
{
|
|
ContainerID: 0,
|
|
HostID: uidInt,
|
|
Size: 1,
|
|
},
|
|
},
|
|
}
|
|
c.Env = append(os.Environ(), "_UNSHARED=true")
|
|
err = c.Run()
|
|
assert.Nil(t, err)
|
|
} else {
|
|
fmt.Println(os.Args)
|
|
|
|
symLinkTestBaseDir, err := os.MkdirTemp("", "podman-symlinktest2")
|
|
assert.Nil(t, err)
|
|
defer os.RemoveAll(symLinkTestBaseDir)
|
|
rootF, err := os.Open("/")
|
|
assert.Nil(t, err)
|
|
defer rootF.Close()
|
|
defer func() {
|
|
err := rootF.Chdir()
|
|
assert.Nil(t, err)
|
|
err = syscall.Chroot(".")
|
|
assert.Nil(t, err)
|
|
}()
|
|
err = syscall.Chroot(symLinkTestBaseDir)
|
|
assert.Nil(t, err)
|
|
|
|
err = os.MkdirAll(quadlet.UnitDirAdmin, 0755)
|
|
assert.Nil(t, err)
|
|
err = os.RemoveAll(quadlet.UnitDirAdmin)
|
|
assert.Nil(t, err)
|
|
|
|
systemdDir := filepath.Join("/", "systemd")
|
|
userDir := filepath.Join("/", "users")
|
|
err = os.Mkdir(systemdDir, 0755)
|
|
assert.Nil(t, err)
|
|
err = os.Mkdir(userDir, 0755)
|
|
assert.Nil(t, err)
|
|
err = os.Symlink(userDir, filepath.Join(systemdDir, "users"))
|
|
assert.Nil(t, err)
|
|
err = os.Symlink(systemdDir, quadlet.UnitDirAdmin)
|
|
assert.Nil(t, err)
|
|
|
|
uidDir := filepath.Join(userDir, u.Uid)
|
|
err = os.Mkdir(uidDir, 0755)
|
|
assert.Nil(t, err)
|
|
uidDir2 := filepath.Join(userDir, strconv.Itoa(uidInt+1))
|
|
err = os.Mkdir(uidDir2, 0755)
|
|
assert.Nil(t, err)
|
|
|
|
t.Setenv("QUADLET_UNIT_DIRS", "")
|
|
unitDirs := getUnitDirs(false)
|
|
assert.NotContains(t, unitDirs, userDir, "rootful should not contain rootless")
|
|
unitDirs = getUnitDirs(true)
|
|
assert.NotContains(t, unitDirs, uidDir2, "rootless should not contain other users'")
|
|
}
|
|
}
|