mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-10-26 13:43:48 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			179 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			6.1 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 driver
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"os"
 | |
| )
 | |
| 
 | |
| var ErrNotSupported = fmt.Errorf("not supported")
 | |
| 
 | |
| // Driver provides all of the system-level functions in a common interface.
 | |
| // The context should call these with full paths and should never use the `os`
 | |
| // package or any other package to access resources on the filesystem. This
 | |
| // mechanism let's us carefully control access to the context and maintain
 | |
| // path and resource integrity. It also gives us an interface to reason about
 | |
| // direct resource access.
 | |
| //
 | |
| // Implementations don't need to do much other than meet the interface. For
 | |
| // example, it is not required to wrap os.FileInfo to return correct paths for
 | |
| // the call to Name().
 | |
| type Driver interface {
 | |
| 	// Note that Open() returns a File interface instead of *os.File. This
 | |
| 	// is because os.File is a struct, so if Open was to return *os.File,
 | |
| 	// the only way to fulfill the interface would be to call os.Open()
 | |
| 	Open(path string) (File, error)
 | |
| 	OpenFile(path string, flag int, perm os.FileMode) (File, error)
 | |
| 
 | |
| 	Stat(path string) (os.FileInfo, error)
 | |
| 	Lstat(path string) (os.FileInfo, error)
 | |
| 	Readlink(p string) (string, error)
 | |
| 	Mkdir(path string, mode os.FileMode) error
 | |
| 	Remove(path string) error
 | |
| 
 | |
| 	Link(oldname, newname string) error
 | |
| 	Lchmod(path string, mode os.FileMode) error
 | |
| 	Lchown(path string, uid, gid int64) error
 | |
| 	Symlink(oldname, newname string) error
 | |
| 
 | |
| 	MkdirAll(path string, perm os.FileMode) error
 | |
| 	RemoveAll(path string) error
 | |
| 
 | |
| 	// TODO(aaronl): These methods might move outside the main Driver
 | |
| 	// interface in the future as more platforms are added.
 | |
| 	Mknod(path string, mode os.FileMode, major int, minor int) error
 | |
| 	Mkfifo(path string, mode os.FileMode) error
 | |
| }
 | |
| 
 | |
| // File is the interface for interacting with files returned by continuity's Open
 | |
| // This is needed since os.File is a struct, instead of an interface, so it can't
 | |
| // be used.
 | |
| type File interface {
 | |
| 	io.ReadWriteCloser
 | |
| 	io.Seeker
 | |
| 	Readdir(n int) ([]os.FileInfo, error)
 | |
| }
 | |
| 
 | |
| func NewSystemDriver() (Driver, error) {
 | |
| 	// TODO(stevvooe): Consider having this take a "hint" path argument, which
 | |
| 	// would be the context root. The hint could be used to resolve required
 | |
| 	// filesystem support when assembling the driver to use.
 | |
| 	return &driver{}, nil
 | |
| }
 | |
| 
 | |
| // XAttrDriver should be implemented on operation systems and filesystems that
 | |
| // have xattr support for regular files and directories.
 | |
| type XAttrDriver interface {
 | |
| 	// Getxattr returns all of the extended attributes for the file at path.
 | |
| 	// Typically, this takes a syscall call to Listxattr and Getxattr.
 | |
| 	Getxattr(path string) (map[string][]byte, error)
 | |
| 
 | |
| 	// Setxattr sets all of the extended attributes on file at path, following
 | |
| 	// any symbolic links, if necessary. All attributes on the target are
 | |
| 	// replaced by the values from attr. If the operation fails to set any
 | |
| 	// attribute, those already applied will not be rolled back.
 | |
| 	Setxattr(path string, attr map[string][]byte) error
 | |
| }
 | |
| 
 | |
| // LXAttrDriver should be implemented by drivers on operating systems and
 | |
| // filesystems that support setting and getting extended attributes on
 | |
| // symbolic links. If this is not implemented, extended attributes will be
 | |
| // ignored on symbolic links.
 | |
| type LXAttrDriver interface {
 | |
| 	// LGetxattr returns all of the extended attributes for the file at path
 | |
| 	// and does not follow symlinks. Typically, this takes a syscall call to
 | |
| 	// Llistxattr and Lgetxattr.
 | |
| 	LGetxattr(path string) (map[string][]byte, error)
 | |
| 
 | |
| 	// LSetxattr sets all of the extended attributes on file at path, without
 | |
| 	// following symbolic links. All attributes on the target are replaced by
 | |
| 	// the values from attr. If the operation fails to set any attribute,
 | |
| 	// those already applied will not be rolled back.
 | |
| 	LSetxattr(path string, attr map[string][]byte) error
 | |
| }
 | |
| 
 | |
| type DeviceInfoDriver interface {
 | |
| 	DeviceInfo(fi os.FileInfo) (maj uint64, min uint64, err error)
 | |
| }
 | |
| 
 | |
| // driver is a simple default implementation that sends calls out to the "os"
 | |
| // package. Extend the "driver" type in system-specific files to add support,
 | |
| // such as xattrs, which can add support at compile time.
 | |
| type driver struct{}
 | |
| 
 | |
| var _ File = &os.File{}
 | |
| 
 | |
| // LocalDriver is the exported Driver struct for convenience.
 | |
| var LocalDriver Driver = &driver{}
 | |
| 
 | |
| func (d *driver) Open(p string) (File, error) {
 | |
| 	return os.Open(p)
 | |
| }
 | |
| 
 | |
| func (d *driver) OpenFile(path string, flag int, perm os.FileMode) (File, error) {
 | |
| 	return os.OpenFile(path, flag, perm)
 | |
| }
 | |
| 
 | |
| func (d *driver) Stat(p string) (os.FileInfo, error) {
 | |
| 	return os.Stat(p)
 | |
| }
 | |
| 
 | |
| func (d *driver) Lstat(p string) (os.FileInfo, error) {
 | |
| 	return os.Lstat(p)
 | |
| }
 | |
| 
 | |
| func (d *driver) Readlink(p string) (string, error) {
 | |
| 	return os.Readlink(p)
 | |
| }
 | |
| 
 | |
| func (d *driver) Mkdir(p string, mode os.FileMode) error {
 | |
| 	return os.Mkdir(p, mode)
 | |
| }
 | |
| 
 | |
| // Remove is used to unlink files and remove directories.
 | |
| // This is following the golang os package api which
 | |
| // combines the operations into a higher level Remove
 | |
| // function. If explicit unlinking or directory removal
 | |
| // to mirror system call is required, they should be
 | |
| // split up at that time.
 | |
| func (d *driver) Remove(path string) error {
 | |
| 	return os.Remove(path)
 | |
| }
 | |
| 
 | |
| func (d *driver) Link(oldname, newname string) error {
 | |
| 	return os.Link(oldname, newname)
 | |
| }
 | |
| 
 | |
| func (d *driver) Lchown(name string, uid, gid int64) error {
 | |
| 	// TODO: error out if uid excesses int bit width?
 | |
| 	return os.Lchown(name, int(uid), int(gid))
 | |
| }
 | |
| 
 | |
| func (d *driver) Symlink(oldname, newname string) error {
 | |
| 	return os.Symlink(oldname, newname)
 | |
| }
 | |
| 
 | |
| func (d *driver) MkdirAll(path string, perm os.FileMode) error {
 | |
| 	return os.MkdirAll(path, perm)
 | |
| }
 | |
| 
 | |
| func (d *driver) RemoveAll(path string) error {
 | |
| 	return os.RemoveAll(path)
 | |
| }
 | 
