mirror of
https://github.com/containers/skopeo.git
synced 2025-04-28 03:10:18 +00:00
Add support to mark images for deletion from repository Requires: * V2 API and schema * registry configured to allow deletes * run registry garbage collection to free up disk space Signed-off-by: Jhon Honce <jhonce@redhat.com>
118 lines
3.2 KiB
Go
118 lines
3.2 KiB
Go
package directory
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/projectatomic/skopeo/types"
|
|
)
|
|
|
|
// manifestPath returns a path for the manifest within a directory using our conventions.
|
|
func manifestPath(dir string) string {
|
|
return filepath.Join(dir, "manifest.json")
|
|
}
|
|
|
|
// manifestPath returns a path for a layer tarball within a directory using our conventions.
|
|
func layerPath(dir string, digest string) string {
|
|
// FIXME: Should we keep the digest identification?
|
|
return filepath.Join(dir, strings.TrimPrefix(digest, "sha256:")+".tar")
|
|
}
|
|
|
|
// manifestPath returns a path for a signature within a directory using our conventions.
|
|
func signaturePath(dir string, index int) string {
|
|
return filepath.Join(dir, fmt.Sprintf("signature-%d", index+1))
|
|
}
|
|
|
|
type dirImageDestination struct {
|
|
dir string
|
|
}
|
|
|
|
// NewDirImageDestination returns an ImageDestination for writing to an existing directory.
|
|
func NewDirImageDestination(dir string) types.ImageDestination {
|
|
return &dirImageDestination{dir}
|
|
}
|
|
|
|
func (d *dirImageDestination) CanonicalDockerReference() (string, error) {
|
|
return "", fmt.Errorf("Can not determine canonical Docker reference for a local directory")
|
|
}
|
|
|
|
func (d *dirImageDestination) PutManifest(manifest []byte) error {
|
|
return ioutil.WriteFile(manifestPath(d.dir), manifest, 0644)
|
|
}
|
|
|
|
func (d *dirImageDestination) PutLayer(digest string, stream io.Reader) error {
|
|
layerFile, err := os.Create(layerPath(d.dir, digest))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer layerFile.Close()
|
|
if _, err := io.Copy(layerFile, stream); err != nil {
|
|
return err
|
|
}
|
|
if err := layerFile.Sync(); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d *dirImageDestination) PutSignatures(signatures [][]byte) error {
|
|
for i, sig := range signatures {
|
|
if err := ioutil.WriteFile(signaturePath(d.dir, i), sig, 0644); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type dirImageSource struct {
|
|
dir string
|
|
}
|
|
|
|
// NewDirImageSource returns an ImageSource reading from an existing directory.
|
|
func NewDirImageSource(dir string) types.ImageSource {
|
|
return &dirImageSource{dir}
|
|
}
|
|
|
|
// IntendedDockerReference returns the full, unambiguous, Docker reference for this image, _as specified by the user_
|
|
// (not as the image itself, or its underlying storage, claims). This can be used e.g. to determine which public keys are trusted for this image.
|
|
// May be "" if unknown.
|
|
func (s *dirImageSource) IntendedDockerReference() string {
|
|
return ""
|
|
}
|
|
|
|
// it's up to the caller to determine the MIME type of the returned manifest's bytes
|
|
func (s *dirImageSource) GetManifest(_ []string) ([]byte, string, error) {
|
|
m, err := ioutil.ReadFile(manifestPath(s.dir))
|
|
if err != nil {
|
|
return nil, "", err
|
|
}
|
|
return m, "", err
|
|
}
|
|
|
|
func (s *dirImageSource) GetLayer(digest string) (io.ReadCloser, error) {
|
|
return os.Open(layerPath(s.dir, digest))
|
|
}
|
|
|
|
func (s *dirImageSource) GetSignatures() ([][]byte, error) {
|
|
signatures := [][]byte{}
|
|
for i := 0; ; i++ {
|
|
signature, err := ioutil.ReadFile(signaturePath(s.dir, i))
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
break
|
|
}
|
|
return nil, err
|
|
}
|
|
signatures = append(signatures, signature)
|
|
}
|
|
return signatures, nil
|
|
}
|
|
|
|
func (s *dirImageSource) Delete() error {
|
|
return fmt.Errorf("directory#dirImageSource.Delete() not implmented")
|
|
}
|