mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-10-31 16:13:45 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			161 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			161 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
|    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"
 | |
| 	"log"
 | |
| 	"os"
 | |
| 	"sort"
 | |
| 
 | |
| 	pb "github.com/containerd/continuity/proto"
 | |
| 	"github.com/golang/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))
 | |
| 	}
 | |
| 
 | |
| 	return proto.MarshalText(w, &bm)
 | |
| }
 | |
| 
 | |
| // 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: %v", 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
 | |
| 			}
 | |
| 			log.Printf("error getting resource %q: %v", p, err)
 | |
| 			return 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: %v", 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
 | |
| }
 | 
