Handle hard links in remote builds

Fixes: https://github.com/containers/podman/issues/9893

Signed-off-by: Daniel J Walsh <dwalsh@redhat.com>
This commit is contained in:
Daniel J Walsh
2021-05-20 16:09:40 -04:00
parent 490915c68e
commit f22791aec7
5 changed files with 80 additions and 12 deletions

View File

@ -382,6 +382,9 @@ bin/podman.cross.%: .gopathok
.PHONY: local-cross .PHONY: local-cross
local-cross: $(CROSS_BUILD_TARGETS) ## Cross compile podman binary for multiple architectures local-cross: $(CROSS_BUILD_TARGETS) ## Cross compile podman binary for multiple architectures
.PHONY: cross
cross: local-cross
# Update nix/nixpkgs.json its latest stable commit # Update nix/nixpkgs.json its latest stable commit
.PHONY: nixpkgs .PHONY: nixpkgs
nixpkgs: nixpkgs:

View File

@ -28,6 +28,11 @@ import (
"github.com/sirupsen/logrus" "github.com/sirupsen/logrus"
) )
type devino struct {
Dev uint64
Ino uint64
}
var ( var (
iidRegex = regexp.MustCompile(`^[0-9a-f]{12}`) iidRegex = regexp.MustCompile(`^[0-9a-f]{12}`)
) )
@ -402,7 +407,7 @@ func nTar(excludes []string, sources ...string) (io.ReadCloser, error) {
defer pw.Close() defer pw.Close()
defer gw.Close() defer gw.Close()
defer tw.Close() defer tw.Close()
seen := make(map[devino]string)
for _, src := range sources { for _, src := range sources {
s, err := filepath.Abs(src) s, err := filepath.Abs(src)
if err != nil { if err != nil {
@ -431,25 +436,40 @@ func nTar(excludes []string, sources ...string) (io.ReadCloser, error) {
} }
if info.Mode().IsRegular() { // add file item if info.Mode().IsRegular() { // add file item
f, lerr := os.Open(path) di, isHardLink := checkHardLink(info)
if lerr != nil { if err != nil {
return lerr return err
} }
hdr, lerr := tar.FileInfoHeader(info, name) hdr, err := tar.FileInfoHeader(info, "")
if lerr != nil { if err != nil {
f.Close() return err
return lerr
} }
orig, ok := seen[di]
if ok {
hdr.Typeflag = tar.TypeLink
hdr.Linkname = orig
hdr.Size = 0
return tw.WriteHeader(hdr)
}
f, err := os.Open(path)
if err != nil {
return err
}
hdr.Name = name hdr.Name = name
if lerr := tw.WriteHeader(hdr); lerr != nil { if err := tw.WriteHeader(hdr); err != nil {
f.Close() f.Close()
return lerr return err
} }
_, cerr := io.Copy(tw, f) _, err = io.Copy(tw, f)
f.Close() f.Close()
return cerr if err == nil && isHardLink {
seen[di] = name
}
return err
} else if info.Mode().IsDir() { // add folders } else if info.Mode().IsDir() { // add folders
hdr, lerr := tar.FileInfoHeader(info, name) hdr, lerr := tar.FileInfoHeader(info, name)
if lerr != nil { if lerr != nil {

View File

@ -0,0 +1,16 @@
// +build !windows
package images
import (
"os"
"syscall"
)
func checkHardLink(fi os.FileInfo) (devino, bool) {
st := fi.Sys().(*syscall.Stat_t)
return devino{
Dev: uint64(st.Dev),
Ino: uint64(st.Ino),
}, st.Nlink > 1
}

View File

@ -0,0 +1,9 @@
package images
import (
"os"
)
func checkHardLink(fi os.FileInfo) (devino, bool) {
return devino{}, false
}

View File

@ -766,6 +766,26 @@ EOF
is "$output" ".*/tmp/bogus: no such file or directory" is "$output" ".*/tmp/bogus: no such file or directory"
} }
@test "podman build COPY hardlinks " {
tmpdir=$PODMAN_TMPDIR/build-test
mkdir -p $tmpdir
dockerfile=$tmpdir/Dockerfile
cat >$dockerfile <<EOF
FROM $IMAGE
COPY . /test
EOF
ln $dockerfile $tmpdir/hardlink
run_podman build -t build_test $tmpdir
run_podman run --rm build_test stat -c '%i' /test/Dockerfile
dinode=$output
run_podman run --rm build_test stat -c '%i' /test/hardlink
is "$output" "$dinode" "COPY hardlinks work"
run_podman rmi -f build_test
}
function teardown() { function teardown() {
# A timeout or other error in 'build' can leave behind stale images # A timeout or other error in 'build' can leave behind stale images
# that podman can't even see and which will cascade into subsequent # that podman can't even see and which will cascade into subsequent