Skip to content

Commit

Permalink
feat(client): kube build support
Browse files Browse the repository at this point in the history
  • Loading branch information
axel7083 committed Oct 31, 2024
1 parent de99041 commit 66fa014
Show file tree
Hide file tree
Showing 13 changed files with 1,018 additions and 214 deletions.
10 changes: 3 additions & 7 deletions pkg/bindings/images/build.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import (
"encoding/json"
"errors"
"fmt"
util2 "github.com/containers/podman/v5/pkg/bindings/internal/util"
"io"
"io/fs"
"net/http"
Expand Down Expand Up @@ -35,11 +36,6 @@ import (
"github.com/sirupsen/logrus"
)

type devino struct {
Dev uint64
Ino uint64
}

var iidRegex = regexp.Delayed(`^[0-9a-f]{12}`)

type BuildResponse struct {
Expand Down Expand Up @@ -723,7 +719,7 @@ func nTar(excludes []string, sources ...string) (io.ReadCloser, error) {
defer pw.Close()
defer gw.Close()
defer tw.Close()
seen := make(map[devino]string)
seen := make(map[util2.Devino]string)
for i, src := range sources {
source, err := filepath.Abs(src)
if err != nil {
Expand Down Expand Up @@ -786,7 +782,7 @@ func nTar(excludes []string, sources ...string) (io.ReadCloser, error) {
if err != nil {
return err
}
di, isHardLink := checkHardLink(info)
di, isHardLink := util2.CheckHardLink(info)
if err != nil {
return err
}
Expand Down
9 changes: 0 additions & 9 deletions pkg/bindings/images/build_windows.go

This file was deleted.

Original file line number Diff line number Diff line change
@@ -1,15 +1,15 @@
//go:build !windows

package images
package util

import (
"os"
"syscall"
)

func checkHardLink(fi os.FileInfo) (devino, bool) {
func CheckHardLink(fi os.FileInfo) (Devino, bool) {
st := fi.Sys().(*syscall.Stat_t)
return devino{
return Devino{
Dev: uint64(st.Dev), //nolint: unconvert
Ino: st.Ino,
}, st.Nlink > 1
Expand Down
9 changes: 9 additions & 0 deletions pkg/bindings/internal/util/build_windows.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package util

import (
"os"
)

func CheckHardLink(fi os.FileInfo) (Devino, bool) {
return Devino{}, false
}
187 changes: 187 additions & 0 deletions pkg/bindings/internal/util/tar_builder.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,187 @@
package util

import (
"archive/tar"
"fmt"
"github.com/containers/storage/pkg/fileutils"
"github.com/containers/storage/pkg/ioutils"
"github.com/hashicorp/go-multierror"
gzip "github.com/klauspost/pgzip"
"io"
"io/fs"
"os"
"path/filepath"
)

type Devino struct {
Dev uint64
Ino uint64
}

type TarBuilder struct {
sources []sourceMapping
excludes []string
}

type sourceMapping struct {
source string // Absolute path of the source directory/file
target string // Custom path inside the tar archive
}

// NewTarBuilder returns a new TarBuilder
func NewTarBuilder() *TarBuilder {
return &TarBuilder{
sources: []sourceMapping{},
excludes: []string{},
}
}

// Add adds a new source directory or file and the corresponding target inside the tar.
func (tb *TarBuilder) Add(source string, target string) error {
absSource, err := filepath.Abs(source)
if err != nil {
return fmt.Errorf("failed to get absolute path for source: %v", err)
}
tb.sources = append(tb.sources, sourceMapping{source: absSource, target: target})
return nil
}

// Exclude adds patterns to be excluded during tar creation.
func (tb *TarBuilder) Exclude(patterns ...string) {
tb.excludes = append(tb.excludes, patterns...)
}

// Build generates the tarball and returns a ReadCloser for the tar stream.
func (tb *TarBuilder) Build() (io.ReadCloser, error) {
if len(tb.sources) == 0 {
return nil, fmt.Errorf("no source(s) added for tar creation")
}

pm, err := fileutils.NewPatternMatcher(tb.excludes)
if err != nil {
return nil, fmt.Errorf("processing excludes list %v: %w", tb.excludes, err)
}

pr, pw := io.Pipe()
gw := gzip.NewWriter(pw)
tw := tar.NewWriter(gw)

var merr *multierror.Error
go func() {
defer pw.Close()
defer gw.Close()
defer tw.Close()

seen := make(map[Devino]string)

for _, src := range tb.sources {
err = filepath.WalkDir(src.source, func(path string, dentry fs.DirEntry, err error) error {
if err != nil {
return err
}

// Build the relative path under the custom target path
relPath, err := filepath.Rel(src.source, path)
if err != nil {
return err
}
targetPath := filepath.ToSlash(filepath.Join(src.target, relPath))

// Check exclusion patterns
if !filepath.IsAbs(targetPath) {
excluded, err := pm.Matches(targetPath)
if err != nil {
return fmt.Errorf("checking if %q is excluded: %w", targetPath, err)
}
if excluded {
return nil
}
}

switch {
case dentry.Type().IsRegular(): // Handle files
info, err := dentry.Info()
if err != nil {
return err
}
di, isHardLink := CheckHardLink(info)
if err != nil {
return err
}

hdr, err := tar.FileInfoHeader(info, "")
if err != nil {
return err
}
hdr.Name = targetPath
hdr.Uid, hdr.Gid = 0, 0
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
}
defer f.Close()

if err := tw.WriteHeader(hdr); err != nil {
return err
}
_, err = io.Copy(tw, f)
if err == nil && isHardLink {
seen[di] = targetPath
}
return err
case dentry.IsDir(): // Handle directories
info, err := dentry.Info()
if err != nil {
return err
}
hdr, lerr := tar.FileInfoHeader(info, targetPath)
if lerr != nil {
return lerr
}
hdr.Name = targetPath
hdr.Uid, hdr.Gid = 0, 0
return tw.WriteHeader(hdr)
case dentry.Type()&os.ModeSymlink != 0: // Handle symlinks
link, err := os.Readlink(path)
if err != nil {
return err
}
info, err := dentry.Info()
if err != nil {
return err
}
hdr, lerr := tar.FileInfoHeader(info, link)
if lerr != nil {
return lerr
}
hdr.Name = targetPath
hdr.Uid, hdr.Gid = 0, 0
return tw.WriteHeader(hdr)
}
return nil
})

if err != nil {
merr = multierror.Append(merr, err)
}
}
}()

rc := ioutils.NewReadCloserWrapper(pr, func() error {
if merr != nil {
merr = multierror.Append(merr, pr.Close())
return merr.ErrorOrNil()
}
return pr.Close()
})

return rc, nil
}
Loading

0 comments on commit 66fa014

Please sign in to comment.