add some tests
This commit is contained in:
parent
b49779600c
commit
dff0a38206
@ -8,4 +8,8 @@ go_library(
|
|||||||
],
|
],
|
||||||
importpath = "github.com/motiejus/code/undocker/rootfs",
|
importpath = "github.com/motiejus/code/undocker/rootfs",
|
||||||
visibility = ["//visibility:public"],
|
visibility = ["//visibility:public"],
|
||||||
|
deps = [
|
||||||
|
"@com_github_stretchr_testify//assert",
|
||||||
|
"@com_github_stretchr_testify//require",
|
||||||
|
],
|
||||||
)
|
)
|
||||||
|
@ -3,7 +3,6 @@ package rootfs
|
|||||||
import (
|
import (
|
||||||
"archive/tar"
|
"archive/tar"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
@ -50,12 +49,12 @@ func RootFS(in io.ReadSeeker, out io.Writer) error {
|
|||||||
case hdr.Name == _manifestJSON:
|
case hdr.Name == _manifestJSON:
|
||||||
dec := json.NewDecoder(tr)
|
dec := json.NewDecoder(tr)
|
||||||
if err := dec.Decode(&manifest); err != nil {
|
if err := dec.Decode(&manifest); err != nil {
|
||||||
return fmt.Errorf("decode manifest.json: %w", err)
|
return err
|
||||||
}
|
}
|
||||||
case strings.HasSuffix(hdr.Name, "/layer.tar"):
|
case strings.HasSuffix(hdr.Name, "/layer.tar"):
|
||||||
here, err := in.Seek(0, io.SeekCurrent)
|
here, err := in.Seek(0, io.SeekCurrent)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("seek: %w", err)
|
return err
|
||||||
}
|
}
|
||||||
layerOffsets[hdr.Name] = here
|
layerOffsets[hdr.Name] = here
|
||||||
}
|
}
|
||||||
@ -74,7 +73,7 @@ func RootFS(in io.ReadSeeker, out io.Writer) error {
|
|||||||
// for all kinds of files.
|
// for all kinds of files.
|
||||||
for i, offset := range layers {
|
for i, offset := range layers {
|
||||||
if _, err := in.Seek(offset, io.SeekStart); err != nil {
|
if _, err := in.Seek(offset, io.SeekStart); err != nil {
|
||||||
fmt.Errorf("seek: %w", err)
|
return err
|
||||||
}
|
}
|
||||||
tr = tar.NewReader(in)
|
tr = tar.NewReader(in)
|
||||||
|
|
||||||
@ -83,6 +82,9 @@ func RootFS(in io.ReadSeeker, out io.Writer) error {
|
|||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
file2layer[hdr.Name] = i
|
file2layer[hdr.Name] = i
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -90,7 +92,7 @@ func RootFS(in io.ReadSeeker, out io.Writer) error {
|
|||||||
// phase 3: iterate through all layers and write files.
|
// phase 3: iterate through all layers and write files.
|
||||||
for i, offset := range layers {
|
for i, offset := range layers {
|
||||||
if _, err := in.Seek(offset, io.SeekStart); err != nil {
|
if _, err := in.Seek(offset, io.SeekStart); err != nil {
|
||||||
fmt.Errorf("seek: %w", err)
|
return err
|
||||||
}
|
}
|
||||||
tr = tar.NewReader(in)
|
tr = tar.NewReader(in)
|
||||||
|
|
||||||
@ -99,6 +101,9 @@ func RootFS(in io.ReadSeeker, out io.Writer) error {
|
|||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
if file2layer[hdr.Name] != i {
|
if file2layer[hdr.Name] != i {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,11 @@ import (
|
|||||||
"archive/tar"
|
"archive/tar"
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"io"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
type layers []string
|
type layers []string
|
||||||
@ -24,8 +28,8 @@ type tarrable interface {
|
|||||||
|
|
||||||
type file struct {
|
type file struct {
|
||||||
name string
|
name string
|
||||||
contents []byte
|
|
||||||
uid int
|
uid int
|
||||||
|
contents []byte
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f file) tar(tw *tar.Writer) {
|
func (f file) tar(tw *tar.Writer) {
|
||||||
@ -56,6 +60,27 @@ func (t tarball) bytes() []byte {
|
|||||||
return buf.Bytes()
|
return buf.Bytes()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func extract(t *testing.T, tarball io.Reader) []file {
|
||||||
|
t.Helper()
|
||||||
|
var ret []file
|
||||||
|
tr := tar.NewReader(tarball)
|
||||||
|
for {
|
||||||
|
hdr, err := tr.Next()
|
||||||
|
if err == io.EOF {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
elem := file{name: hdr.Name, uid: hdr.Uid}
|
||||||
|
if hdr.Typeflag == tar.TypeReg {
|
||||||
|
buf := bytes.Buffer{}
|
||||||
|
io.Copy(&buf, tr)
|
||||||
|
elem.contents = buf.Bytes()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
_layer0 = tarball{
|
_layer0 = tarball{
|
||||||
dir{name: "/", uid: 0},
|
dir{name: "/", uid: 0},
|
||||||
@ -77,6 +102,35 @@ var (
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
func RootFSTest(t *testing.T) {
|
func TestRootFS(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
image tarball
|
||||||
|
want []file
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "empty",
|
||||||
|
image: tarball{},
|
||||||
|
want: []file{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "basic overwrite, 2 layers",
|
||||||
|
image: _image,
|
||||||
|
want: []file{
|
||||||
|
{name: "/", uid: 1},
|
||||||
|
{name: "file", uid: 0, contents: []byte("from 1")},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
in := bytes.NewReader(tt.image.bytes())
|
||||||
|
out := bytes.Buffer{}
|
||||||
|
|
||||||
|
require.NoError(t, RootFS(in, &out))
|
||||||
|
got := extract(t, &out)
|
||||||
|
assert.Equal(t, tt.want, got)
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user