forked from oras-project/oras
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
oras pull with cache (oras-project#203)
- Loading branch information
Showing
4 changed files
with
154 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,114 @@ | ||
package main | ||
|
||
import ( | ||
"context" | ||
|
||
orascontent "github.com/deislabs/oras/pkg/content" | ||
|
||
"github.com/containerd/containerd/content" | ||
"github.com/containerd/containerd/content/local" | ||
"github.com/containerd/containerd/errdefs" | ||
digest "github.com/opencontainers/go-digest" | ||
ocispec "github.com/opencontainers/image-spec/specs-go/v1" | ||
"github.com/pkg/errors" | ||
) | ||
|
||
type cachedStore struct { | ||
base content.Ingester | ||
cache content.Store | ||
} | ||
|
||
// newStoreWithCache creates a store with a consistent cache layer. | ||
func newStoreWithCache(base content.Ingester, cacheRoot string) (orascontent.ProvideIngester, error) { | ||
cache, err := local.NewStore(cacheRoot) | ||
if err != nil { | ||
return nil, err | ||
} | ||
return &cachedStore{ | ||
base: base, | ||
cache: cache, | ||
}, nil | ||
} | ||
|
||
// ReaderAt reads the cache if available, and then check the base store. | ||
func (s *cachedStore) ReaderAt(ctx context.Context, desc ocispec.Descriptor) (content.ReaderAt, error) { | ||
return s.cache.ReaderAt(ctx, desc) | ||
} | ||
|
||
// Writer writes to the cache if not exists, and then copy the cache to the base store. | ||
func (s *cachedStore) Writer(ctx context.Context, opts ...content.WriterOpt) (content.Writer, error) { | ||
var wOpts content.WriterOpts | ||
for _, opt := range opts { | ||
if err := opt(&wOpts); err != nil { | ||
return nil, err | ||
} | ||
} | ||
switch wOpts.Desc.MediaType { | ||
case ocispec.MediaTypeImageManifest, ocispec.MediaTypeImageIndex: | ||
return s.cache.Writer(ctx, opts...) | ||
} | ||
|
||
cacheWriter, err := s.cache.Writer(ctx, opts...) | ||
if err != nil { | ||
if !errdefs.IsAlreadyExists(err) { | ||
return nil, err | ||
} | ||
if err := s.applyCache(ctx, wOpts.Desc, opts...); err != nil { | ||
return nil, err | ||
} | ||
return nil, errdefs.ErrAlreadyExists | ||
} | ||
|
||
return &callbackWriter{ | ||
Writer: cacheWriter, | ||
onCommit: func(ctx context.Context) error { | ||
return s.applyCache(ctx, wOpts.Desc, opts...) | ||
}, | ||
}, nil | ||
} | ||
|
||
// applyCache copies the content from cache to the base store | ||
func (s *cachedStore) applyCache(ctx context.Context, desc ocispec.Descriptor, opts ...content.WriterOpt) error { | ||
cw, err := s.base.Writer(ctx, opts...) | ||
if err != nil { | ||
if errdefs.IsAlreadyExists(err) { | ||
return nil | ||
} | ||
return err | ||
} | ||
|
||
ws, err := cw.Status() | ||
if err != nil { | ||
return err | ||
} | ||
|
||
if ws.Offset != desc.Size { | ||
ra, err := s.cache.ReaderAt(ctx, desc) | ||
if err != nil { | ||
return err | ||
} | ||
defer ra.Close() | ||
|
||
if err := content.CopyReaderAt(cw, ra, desc.Size); err != nil { | ||
return err | ||
} | ||
} | ||
|
||
if err := cw.Commit(ctx, desc.Size, desc.Digest); err != nil && !errdefs.IsAlreadyExists(err) { | ||
return errors.Wrapf(err, "failed commit on ref %q", ws.Ref) | ||
} | ||
|
||
return nil | ||
} | ||
|
||
type callbackWriter struct { | ||
content.Writer | ||
onCommit func(ctx context.Context) error | ||
} | ||
|
||
func (cw *callbackWriter) Commit(ctx context.Context, size int64, expected digest.Digest, opts ...content.Opt) error { | ||
if err := cw.Writer.Commit(ctx, size, expected, opts...); err != nil { | ||
return err | ||
} | ||
return cw.onCommit(ctx) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters