You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
165 lines
3.7 KiB
Go
165 lines
3.7 KiB
Go
2 years ago
|
/*
|
||
|
Copyright The containerd Authors.
|
||
|
|
||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
you may not use this file except in compliance with the License.
|
||
|
You may obtain a copy of the License at
|
||
|
|
||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||
|
|
||
|
Unless required by applicable law or agreed to in writing, software
|
||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
See the License for the specific language governing permissions and
|
||
|
limitations under the License.
|
||
|
*/
|
||
|
|
||
|
package continuity
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"os"
|
||
|
"sort"
|
||
|
|
||
|
pb "github.com/containerd/continuity/proto"
|
||
|
"google.golang.org/protobuf/encoding/prototext"
|
||
|
"google.golang.org/protobuf/proto"
|
||
|
)
|
||
|
|
||
|
// Manifest provides the contents of a manifest. Users of this struct should
|
||
|
// not typically modify any fields directly.
|
||
|
type Manifest struct {
|
||
|
// Resources specifies all the resources for a manifest in order by path.
|
||
|
Resources []Resource
|
||
|
}
|
||
|
|
||
|
func Unmarshal(p []byte) (*Manifest, error) {
|
||
|
var bm pb.Manifest
|
||
|
|
||
|
if err := proto.Unmarshal(p, &bm); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
var m Manifest
|
||
|
for _, b := range bm.Resource {
|
||
|
r, err := fromProto(b)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
m.Resources = append(m.Resources, r)
|
||
|
}
|
||
|
|
||
|
return &m, nil
|
||
|
}
|
||
|
|
||
|
func Marshal(m *Manifest) ([]byte, error) {
|
||
|
var bm pb.Manifest
|
||
|
for _, resource := range m.Resources {
|
||
|
bm.Resource = append(bm.Resource, toProto(resource))
|
||
|
}
|
||
|
|
||
|
return proto.Marshal(&bm)
|
||
|
}
|
||
|
|
||
|
func MarshalText(w io.Writer, m *Manifest) error {
|
||
|
var bm pb.Manifest
|
||
|
for _, resource := range m.Resources {
|
||
|
bm.Resource = append(bm.Resource, toProto(resource))
|
||
|
}
|
||
|
|
||
|
b, err := prototext.Marshal(&bm)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
_, err = w.Write(b)
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// BuildManifest creates the manifest for the given context
|
||
|
func BuildManifest(ctx Context) (*Manifest, error) {
|
||
|
resourcesByPath := map[string]Resource{}
|
||
|
hardLinks := newHardlinkManager()
|
||
|
|
||
|
if err := ctx.Walk(func(p string, fi os.FileInfo, err error) error {
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("error walking %s: %w", p, err)
|
||
|
}
|
||
|
|
||
|
if p == string(os.PathSeparator) {
|
||
|
// skip root
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
resource, err := ctx.Resource(p, fi)
|
||
|
if err != nil {
|
||
|
if err == ErrNotFound {
|
||
|
return nil
|
||
|
}
|
||
|
return fmt.Errorf("failed to get resource %q: %w", p, err)
|
||
|
}
|
||
|
|
||
|
// add to the hardlink manager
|
||
|
if err := hardLinks.Add(fi, resource); err == nil {
|
||
|
// Resource has been accepted by hardlink manager so we don't add
|
||
|
// it to the resourcesByPath until we merge at the end.
|
||
|
return nil
|
||
|
} else if err != errNotAHardLink {
|
||
|
// handle any other case where we have a proper error.
|
||
|
return fmt.Errorf("adding hardlink %s: %w", p, err)
|
||
|
}
|
||
|
|
||
|
resourcesByPath[p] = resource
|
||
|
|
||
|
return nil
|
||
|
}); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
// merge and post-process the hardlinks.
|
||
|
hardLinked, err := hardLinks.Merge()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
for _, resource := range hardLinked {
|
||
|
resourcesByPath[resource.Path()] = resource
|
||
|
}
|
||
|
|
||
|
var resources []Resource
|
||
|
for _, resource := range resourcesByPath {
|
||
|
resources = append(resources, resource)
|
||
|
}
|
||
|
|
||
|
sort.Stable(ByPath(resources))
|
||
|
|
||
|
return &Manifest{
|
||
|
Resources: resources,
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
// VerifyManifest verifies all the resources in a manifest
|
||
|
// against files from the given context.
|
||
|
func VerifyManifest(ctx Context, manifest *Manifest) error {
|
||
|
for _, resource := range manifest.Resources {
|
||
|
if err := ctx.Verify(resource); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// ApplyManifest applies on the resources in a manifest to
|
||
|
// the given context.
|
||
|
func ApplyManifest(ctx Context, manifest *Manifest) error {
|
||
|
for _, resource := range manifest.Resources {
|
||
|
if err := ctx.Apply(resource); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|