working commit
This commit is contained in:
+18
-5
@@ -29,14 +29,16 @@ type Algorithm int
|
|||||||
const (
|
const (
|
||||||
Undefined Algorithm = iota
|
Undefined Algorithm = iota
|
||||||
SHA256
|
SHA256
|
||||||
|
SHA384
|
||||||
SHA512
|
SHA512
|
||||||
)
|
)
|
||||||
|
|
||||||
func DigestType(digest string) Algorithm {
|
func DigestType(digest string) Algorithm {
|
||||||
var err error
|
var err error
|
||||||
var typ Algorithm
|
var algorithm Algorithm
|
||||||
digest = strings.ToLower(digest)
|
digest = strings.ToLower(digest)
|
||||||
digest = strings.TrimPrefix(digest, "sha256:")
|
digest = strings.TrimPrefix(digest, "sha256:")
|
||||||
|
digest = strings.TrimPrefix(digest, "sha384:")
|
||||||
digest = strings.TrimPrefix(digest, "sha512:")
|
digest = strings.TrimPrefix(digest, "sha512:")
|
||||||
decoded, err := hex.DecodeString(digest)
|
decoded, err := hex.DecodeString(digest)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -44,13 +46,15 @@ func DigestType(digest string) Algorithm {
|
|||||||
}
|
}
|
||||||
switch len(decoded) {
|
switch len(decoded) {
|
||||||
case 32:
|
case 32:
|
||||||
typ = SHA256
|
algorithm = SHA256
|
||||||
|
case 48:
|
||||||
|
algorithm = SHA384
|
||||||
case 64:
|
case 64:
|
||||||
typ = SHA512
|
algorithm = SHA512
|
||||||
default:
|
default:
|
||||||
typ = Undefined
|
algorithm = Undefined
|
||||||
}
|
}
|
||||||
return typ
|
return algorithm
|
||||||
}
|
}
|
||||||
|
|
||||||
type Digest struct {
|
type Digest struct {
|
||||||
@@ -64,6 +68,8 @@ func NewDigestFromBytes(algorithm Algorithm, payload []byte) *Digest {
|
|||||||
switch algorithm {
|
switch algorithm {
|
||||||
case SHA512:
|
case SHA512:
|
||||||
hasher = sha512.New()
|
hasher = sha512.New()
|
||||||
|
case SHA384:
|
||||||
|
hasher = sha512.New384()
|
||||||
default:
|
default:
|
||||||
hasher = sha256.New()
|
hasher = sha256.New()
|
||||||
}
|
}
|
||||||
@@ -82,6 +88,7 @@ func ParseDigest(str string) (*Digest, error) {
|
|||||||
|
|
||||||
str = strings.ToLower(str)
|
str = strings.ToLower(str)
|
||||||
str = strings.TrimPrefix(str, "sha256:")
|
str = strings.TrimPrefix(str, "sha256:")
|
||||||
|
str = strings.TrimPrefix(str, "sha384:")
|
||||||
str = strings.TrimPrefix(str, "sha512:")
|
str = strings.TrimPrefix(str, "sha512:")
|
||||||
|
|
||||||
decoded, err := hex.DecodeString(str)
|
decoded, err := hex.DecodeString(str)
|
||||||
@@ -93,6 +100,8 @@ func ParseDigest(str string) (*Digest, error) {
|
|||||||
switch len(decoded) {
|
switch len(decoded) {
|
||||||
case 32:
|
case 32:
|
||||||
digest.algorithm = SHA256
|
digest.algorithm = SHA256
|
||||||
|
case 48:
|
||||||
|
digest.algorithm = SHA384
|
||||||
case 64:
|
case 64:
|
||||||
digest.algorithm = SHA512
|
digest.algorithm = SHA512
|
||||||
default:
|
default:
|
||||||
@@ -115,6 +124,8 @@ func (dig *Digest) Prefix() string {
|
|||||||
switch dig.algorithm {
|
switch dig.algorithm {
|
||||||
case SHA256:
|
case SHA256:
|
||||||
prefix = "sha256"
|
prefix = "sha256"
|
||||||
|
case SHA384:
|
||||||
|
prefix = "sha384"
|
||||||
case SHA512:
|
case SHA512:
|
||||||
prefix = "sha512"
|
prefix = "sha512"
|
||||||
}
|
}
|
||||||
@@ -126,6 +137,8 @@ func (dig *Digest) Encoded() string {
|
|||||||
switch dig.algorithm {
|
switch dig.algorithm {
|
||||||
case SHA256:
|
case SHA256:
|
||||||
prefix = "sha256"
|
prefix = "sha256"
|
||||||
|
case SHA384:
|
||||||
|
prefix = "sha384"
|
||||||
case SHA512:
|
case SHA512:
|
||||||
prefix = "sha512"
|
prefix = "sha512"
|
||||||
}
|
}
|
||||||
|
|||||||
+42
-42
@@ -3,6 +3,7 @@ package repocli
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
@@ -12,19 +13,12 @@ import (
|
|||||||
ocispec "github.com/opencontainers/image-spec/specs-go/v1"
|
ocispec "github.com/opencontainers/image-spec/specs-go/v1"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Imageer interface {
|
|
||||||
WriteManifest(mime string, manifest []byte) error
|
|
||||||
ReadManifest(digest string) (bool, string, []byte, error)
|
|
||||||
WriteBlob(digest string, reader io.Reader) error
|
|
||||||
ReadBlob(digest string, writer io.Writer) error
|
|
||||||
}
|
|
||||||
|
|
||||||
type Imager struct {
|
type Imager struct {
|
||||||
index ocispec.Index
|
index ocispec.Index
|
||||||
place string
|
place string
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewImager(place string) *Imager {
|
func NewEmptyImager(place string) *Imager {
|
||||||
imager := &Imager{
|
imager := &Imager{
|
||||||
index: ocispec.Index{
|
index: ocispec.Index{
|
||||||
MediaType: MediaTypeOIIv1,
|
MediaType: MediaTypeOIIv1,
|
||||||
@@ -32,7 +26,7 @@ func NewImager(place string) *Imager {
|
|||||||
},
|
},
|
||||||
place: place,
|
place: place,
|
||||||
}
|
}
|
||||||
//imager.SchemaVersion = 2
|
imager.index.SchemaVersion = 2
|
||||||
return imager
|
return imager
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -42,32 +36,30 @@ func (ima *Imager) WriteManifest(ctx context.Context, digest, mime string, paylo
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
var subdir string
|
subdir := string(digestobj.Algorithm())
|
||||||
switch digestobj.Algorithm() {
|
|
||||||
case ocidigest.SHA256:
|
|
||||||
subdir = "sha256"
|
|
||||||
case ocidigest.SHA384:
|
|
||||||
subdir = "sha384"
|
|
||||||
case ocidigest.SHA512:
|
|
||||||
subdir = "sha512"
|
|
||||||
default:
|
|
||||||
err := errors.New("Unknown digest type")
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
dir := filepath.Join(ima.place, subdir)
|
dir := filepath.Join(ima.place, subdir)
|
||||||
err = os.MkdirAll(dir, 0750)
|
err = os.MkdirAll(dir, 0750)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
fpath := filepath.Join(dir, digestobj.Encoded())
|
mpath := filepath.Join(dir, digestobj.Encoded())
|
||||||
file, err := os.OpenFile(fpath, os.O_CREATE|os.O_WRONLY, 0640)
|
mfile, err := os.OpenFile(mpath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0640)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
defer mfile.Close()
|
||||||
reader := bytes.NewReader(payload)
|
reader := bytes.NewReader(payload)
|
||||||
size, err := io.Copy(file, reader)
|
|
||||||
if size != int64(len(payload)) {
|
verifier := digestobj.Verifier()
|
||||||
err = errors.New("Mismatch sizes")
|
mwriter := io.MultiWriter(verifier, mfile)
|
||||||
|
recsize, err := io.Copy(mwriter, reader)
|
||||||
|
size := int64(len(payload))
|
||||||
|
if size != recsize {
|
||||||
|
err = errors.New("Mismatch manigest sizes")
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if !verifier.Verified() {
|
||||||
|
err = errors.New("Mismatch manifest digests")
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
descr := ocispec.Descriptor{
|
descr := ocispec.Descriptor{
|
||||||
@@ -76,7 +68,26 @@ func (ima *Imager) WriteManifest(ctx context.Context, digest, mime string, paylo
|
|||||||
Size: size,
|
Size: size,
|
||||||
}
|
}
|
||||||
ima.index.Manifests = append(ima.index.Manifests, descr)
|
ima.index.Manifests = append(ima.index.Manifests, descr)
|
||||||
// TODO: flush index
|
indexdat, err := json.Marshal(ima.index)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// Flush index
|
||||||
|
ipath := filepath.Join(dir, "index.json")
|
||||||
|
ifile, err := os.OpenFile(ipath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0640)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer ifile.Close()
|
||||||
|
reader = bytes.NewReader(indexdat)
|
||||||
|
recsize, err = io.Copy(ifile, reader)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if recsize != int64(len(indexdat)) {
|
||||||
|
err = errors.New("Mismatch index sizes")
|
||||||
|
return err
|
||||||
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -86,31 +97,20 @@ func (ima *Imager) WriteLayer(ctx context.Context, digest string, size int64, re
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
var subdir string
|
subdir := string(digestobj.Algorithm())
|
||||||
switch digestobj.Algorithm() {
|
|
||||||
case ocidigest.SHA256:
|
|
||||||
subdir = "sha256"
|
|
||||||
case ocidigest.SHA384:
|
|
||||||
subdir = "sha384"
|
|
||||||
case ocidigest.SHA512:
|
|
||||||
subdir = "sha512"
|
|
||||||
default:
|
|
||||||
err := errors.New("Unknown digest type")
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
dir := filepath.Join(ima.place, subdir)
|
dir := filepath.Join(ima.place, subdir)
|
||||||
err = os.MkdirAll(dir, 0750)
|
err = os.MkdirAll(dir, 0750)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
fpath := filepath.Join(dir, digestobj.Encoded())
|
fpath := filepath.Join(dir, digestobj.Encoded())
|
||||||
file, err := os.OpenFile(fpath, os.O_CREATE|os.O_WRONLY, 0640)
|
file, err := os.OpenFile(fpath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0640)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
verifier := digestobj.Verifier()
|
verifier := digestobj.Verifier()
|
||||||
mw := io.MultiWriter(verifier, file)
|
mwriter := io.MultiWriter(verifier, file)
|
||||||
recsize, err := io.Copy(mw, reader)
|
recsize, err := io.Copy(mwriter, reader)
|
||||||
if size != recsize {
|
if size != recsize {
|
||||||
err = errors.New("Mismatch sizes")
|
err = errors.New("Mismatch sizes")
|
||||||
return err
|
return err
|
||||||
|
|||||||
@@ -1,16 +1,14 @@
|
|||||||
package repocli
|
package repocli
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
|
|
||||||
ocidigest "github.com/opencontainers/go-digest"
|
ocidigest "github.com/opencontainers/go-digest"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
//"io"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestDigest(t *testing.T) {
|
func TestDigest(t *testing.T) {
|
||||||
@@ -23,7 +21,7 @@ func TestDigest(t *testing.T) {
|
|||||||
|
|
||||||
ctx, _ := context.WithTimeout(context.Background(), 10*time.Second)
|
ctx, _ := context.WithTimeout(context.Background(), 10*time.Second)
|
||||||
tmpdir := t.TempDir()
|
tmpdir := t.TempDir()
|
||||||
imager := NewImager(tmpdir)
|
imager := NewEmptyImager(tmpdir)
|
||||||
require.NotNil(t, imager)
|
require.NotNil(t, imager)
|
||||||
digest := fmt.Sprintf("%s:%s", digestobj.Algorithm(), digestobj.Encoded())
|
digest := fmt.Sprintf("%s:%s", digestobj.Algorithm(), digestobj.Encoded())
|
||||||
size := int64(len(payload))
|
size := int64(len(payload))
|
||||||
|
|||||||
Reference in New Issue
Block a user