mirror of
https://github.com/containers/podman.git
synced 2025-05-17 15:18:43 +08:00
libpod: fix lookup for subpath in volumes
a subdirectory that is below a mount destination is detected as a subpath. Closes: https://github.com/containers/podman/issues/15789 Signed-off-by: Giuseppe Scrivano <gscrivan@redhat.com>
This commit is contained in:
@ -119,15 +119,29 @@ func findVolume(c *Container, containerPath string) (*Volume, error) {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
// isSubDir checks whether path is a subdirectory of root.
|
||||
func isSubDir(path, root string) bool {
|
||||
// check if the specified container path is below a bind mount.
|
||||
rel, err := filepath.Rel(root, path)
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
return rel != ".." && !strings.HasPrefix(rel, "../")
|
||||
}
|
||||
|
||||
// isPathOnVolume returns true if the specified containerPath is a subdir of any
|
||||
// Volume's destination.
|
||||
func isPathOnVolume(c *Container, containerPath string) bool {
|
||||
cleanedContainerPath := filepath.Clean(containerPath)
|
||||
for _, vol := range c.config.NamedVolumes {
|
||||
if cleanedContainerPath == filepath.Clean(vol.Dest) {
|
||||
cleanedDestination := filepath.Clean(vol.Dest)
|
||||
if cleanedContainerPath == cleanedDestination {
|
||||
return true
|
||||
}
|
||||
for dest := vol.Dest; dest != "/" && dest != "."; dest = filepath.Dir(dest) {
|
||||
if isSubDir(cleanedContainerPath, cleanedDestination) {
|
||||
return true
|
||||
}
|
||||
for dest := cleanedDestination; dest != "/" && dest != "."; dest = filepath.Dir(dest) {
|
||||
if cleanedContainerPath == dest {
|
||||
return true
|
||||
}
|
||||
@ -157,10 +171,14 @@ func findBindMount(c *Container, containerPath string) *specs.Mount {
|
||||
func isPathOnMount(c *Container, containerPath string) bool {
|
||||
cleanedContainerPath := filepath.Clean(containerPath)
|
||||
for _, m := range c.config.Spec.Mounts {
|
||||
if cleanedContainerPath == filepath.Clean(m.Destination) {
|
||||
cleanedDestination := filepath.Clean(m.Destination)
|
||||
if cleanedContainerPath == cleanedDestination {
|
||||
return true
|
||||
}
|
||||
for dest := m.Destination; dest != "/" && dest != "."; dest = filepath.Dir(dest) {
|
||||
if isSubDir(cleanedContainerPath, cleanedDestination) {
|
||||
return true
|
||||
}
|
||||
for dest := cleanedDestination; dest != "/" && dest != "."; dest = filepath.Dir(dest) {
|
||||
if cleanedContainerPath == dest {
|
||||
return true
|
||||
}
|
||||
|
28
libpod/container_path_resolution_test.go
Normal file
28
libpod/container_path_resolution_test.go
Normal file
@ -0,0 +1,28 @@
|
||||
package libpod
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
)
|
||||
|
||||
func TestIsSubDir(t *testing.T) {
|
||||
assert.True(t, isSubDir("/foo", "/foo"))
|
||||
assert.True(t, isSubDir("/foo/bar", "/foo"))
|
||||
assert.True(t, isSubDir("/foo/bar", "/foo/"))
|
||||
assert.True(t, isSubDir("/foo/bar", "/foo//"))
|
||||
assert.True(t, isSubDir("/foo/bar/", "/foo"))
|
||||
assert.True(t, isSubDir("/foo/bar/baz/", "/foo"))
|
||||
assert.True(t, isSubDir("/foo/bar/baz/", "/foo/bar"))
|
||||
assert.True(t, isSubDir("/foo/bar/baz/", "/foo/bar/"))
|
||||
assert.False(t, isSubDir("/foo/bar/baz/", "/foobar/"))
|
||||
assert.False(t, isSubDir("/foo/bar/baz/../../", "/foobar/"))
|
||||
assert.False(t, isSubDir("/foo/bar/baz/", "../foo/bar"))
|
||||
assert.False(t, isSubDir("/foo/bar/baz/", "../foo/"))
|
||||
assert.False(t, isSubDir("/foo/bar/baz/", "../foo"))
|
||||
assert.False(t, isSubDir("/", ".."))
|
||||
assert.False(t, isSubDir("//", ".."))
|
||||
assert.False(t, isSubDir("//", "../"))
|
||||
assert.False(t, isSubDir("//", "..//"))
|
||||
assert.True(t, isSubDir("/foo/bar/baz/../../", "/foo/"))
|
||||
}
|
@ -46,6 +46,15 @@ var _ = Describe("Podman run", func() {
|
||||
Expect(session).Should(Exit(126))
|
||||
})
|
||||
|
||||
It("podman run a container using a --workdir under a bind mount", func() {
|
||||
volume, err := CreateTempDirInTempDir()
|
||||
Expect(err).To(BeNil())
|
||||
|
||||
session := podmanTest.Podman([]string{"run", "--volume", fmt.Sprintf("%s:/var_ovl/:O", volume), "--workdir", "/var_ovl/log", ALPINE, "true"})
|
||||
session.WaitWithDefaultTimeout()
|
||||
Expect(session).Should(Exit(0))
|
||||
})
|
||||
|
||||
It("podman run a container on an image with a workdir", func() {
|
||||
dockerfile := fmt.Sprintf(`FROM %s
|
||||
RUN mkdir -p /home/foobar /etc/foobar; chown bin:bin /etc/foobar
|
||||
|
Reference in New Issue
Block a user