working commit

This commit is contained in:
2026-03-05 21:38:26 +02:00
parent b9f1f384fb
commit 48a2d95f14
3 changed files with 62 additions and 51 deletions
+18 -5
View File
@@ -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
View File
@@ -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
+2 -4
View File
@@ -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))