2021-05-24 00:11:58 +03:00
|
|
|
package rootfstest
|
|
|
|
|
|
|
|
import (
|
|
|
|
"archive/tar"
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"io"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
Tarrer interface {
|
2021-05-24 00:11:58 +03:00
|
|
|
Tar(*tar.Writer) error
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Tarball []Tarrer
|
|
|
|
|
|
|
|
// Extractable is an empty interface for comparing extracted outputs in tests.
|
|
|
|
// Using that just to avoid the ugly `interface{}`.
|
|
|
|
Extractable interface{}
|
|
|
|
|
|
|
|
Dir struct {
|
|
|
|
Name string
|
|
|
|
Uid int
|
|
|
|
}
|
|
|
|
|
|
|
|
File struct {
|
|
|
|
Name string
|
|
|
|
Uid int
|
2021-05-24 00:11:58 +03:00
|
|
|
Contents *bytes.Buffer
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Manifest []string
|
|
|
|
|
|
|
|
Hardlink struct {
|
|
|
|
Name string
|
|
|
|
Uid int
|
|
|
|
}
|
|
|
|
|
|
|
|
dockerManifestJSON []struct {
|
|
|
|
Layers []string `json:"Layers"`
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2021-05-24 00:11:58 +03:00
|
|
|
func (tb Tarball) Buffer() *bytes.Buffer {
|
|
|
|
var buf bytes.Buffer
|
2021-05-24 00:11:58 +03:00
|
|
|
tw := tar.NewWriter(&buf)
|
|
|
|
for _, member := range tb {
|
|
|
|
member.Tar(tw)
|
|
|
|
}
|
|
|
|
tw.Close()
|
2021-05-24 00:11:58 +03:00
|
|
|
return &buf
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
|
|
|
|
2021-05-24 00:11:58 +03:00
|
|
|
func (d Dir) Tar(tw *tar.Writer) error {
|
2021-05-24 00:11:58 +03:00
|
|
|
hdr := &tar.Header{
|
|
|
|
Typeflag: tar.TypeDir,
|
|
|
|
Name: d.Name,
|
|
|
|
Mode: 0644,
|
|
|
|
Uid: d.Uid,
|
|
|
|
}
|
2021-05-24 00:11:58 +03:00
|
|
|
return tw.WriteHeader(hdr)
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
|
|
|
|
2021-05-24 00:11:58 +03:00
|
|
|
func (f File) Tar(tw *tar.Writer) error {
|
|
|
|
var contents []byte
|
2021-05-24 00:11:58 +03:00
|
|
|
if f.Contents != nil {
|
2021-05-24 00:11:58 +03:00
|
|
|
contents = f.Contents.Bytes()
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
|
|
|
hdr := &tar.Header{
|
|
|
|
Typeflag: tar.TypeReg,
|
|
|
|
Name: f.Name,
|
|
|
|
Mode: 0644,
|
|
|
|
Uid: f.Uid,
|
2021-05-24 00:11:58 +03:00
|
|
|
Size: int64(len(contents)),
|
|
|
|
}
|
|
|
|
if err := tw.WriteHeader(hdr); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if _, err := tw.Write(contents); err != nil {
|
|
|
|
return err
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
2021-05-24 00:11:58 +03:00
|
|
|
return nil
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
|
|
|
|
2021-05-24 00:11:58 +03:00
|
|
|
func (m Manifest) Tar(tw *tar.Writer) error {
|
2021-05-24 00:11:58 +03:00
|
|
|
b, err := json.Marshal(dockerManifestJSON{{Layers: m}})
|
|
|
|
if err != nil {
|
2021-05-24 00:11:58 +03:00
|
|
|
return err
|
2021-05-24 00:11:58 +03:00
|
|
|
}
|
2021-05-24 00:11:58 +03:00
|
|
|
return File{
|
2021-05-24 00:11:58 +03:00
|
|
|
Name: "manifest.json",
|
|
|
|
Contents: bytes.NewBuffer(b),
|
|
|
|
}.Tar(tw)
|
|
|
|
}
|
|
|
|
|
2021-05-24 00:11:58 +03:00
|
|
|
func (h Hardlink) Tar(tw *tar.Writer) error {
|
|
|
|
return tw.WriteHeader(&tar.Header{
|
2021-05-24 00:11:58 +03:00
|
|
|
Typeflag: tar.TypeLink,
|
|
|
|
Name: h.Name,
|
|
|
|
Mode: 0644,
|
|
|
|
Uid: h.Uid,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func Extract(t *testing.T, r io.Reader) []Extractable {
|
|
|
|
t.Helper()
|
|
|
|
ret := []Extractable{}
|
|
|
|
tr := tar.NewReader(r)
|
|
|
|
for {
|
|
|
|
hdr, err := tr.Next()
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var elem Extractable
|
|
|
|
switch hdr.Typeflag {
|
|
|
|
case tar.TypeDir:
|
|
|
|
elem = Dir{Name: hdr.Name, Uid: hdr.Uid}
|
|
|
|
case tar.TypeLink:
|
|
|
|
elem = Hardlink{Name: hdr.Name}
|
|
|
|
case tar.TypeReg:
|
|
|
|
f := File{Name: hdr.Name, Uid: hdr.Uid}
|
|
|
|
if hdr.Size > 0 {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
io.Copy(&buf, tr)
|
|
|
|
f.Contents = &buf
|
|
|
|
}
|
|
|
|
elem = f
|
|
|
|
}
|
|
|
|
ret = append(ret, elem)
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|