1 Star 0 Fork 1

mysnapcore / mysnapd

forked from tupelo-shen / mysnapd 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
cp.go 5.46 KB
一键复制 编辑 原始数据 按行查看 历史
tupelo-shen 提交于 2022-11-06 23:46 . fix: osutil 1 commit
// -*- Mode: Go; indent-tabs-mode: t -*-
/*
* Copyright (C) 2014-2015 Canonical Ltd
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 3 as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
package osutil
import (
"bytes"
"fmt"
"os"
"os/exec"
"path/filepath"
)
// CopyFlag is used to tweak the behaviour of CopyFile
type CopyFlag uint8
const (
// CopyFlagDefault is the default behaviour
CopyFlagDefault CopyFlag = 0
// CopyFlagSync does a sync after copying the files
CopyFlagSync CopyFlag = 1 << iota
// CopyFlagOverwrite overwrites the target if it exists
CopyFlagOverwrite
// CopyFlagPreserveAll preserves mode,owner,time attributes
CopyFlagPreserveAll
)
var (
openfile = doOpenFile
copyfile = doCopyFile
)
type fileish interface {
Close() error
Sync() error
Fd() uintptr
Stat() (os.FileInfo, error)
Read([]byte) (int, error)
Write([]byte) (int, error)
}
func doOpenFile(name string, flag int, perm os.FileMode) (fileish, error) {
return os.OpenFile(name, flag, perm)
}
// CopyFile copies src to dst
func CopyFile(src, dst string, flags CopyFlag) (err error) {
if flags&CopyFlagPreserveAll != 0 {
// Our native copy code does not preserve all attributes
// (yet). If the user needs this functionality we just
// fallback to use the system's "cp" binary to do the copy.
if err := runCpPreserveAll(src, dst, "copy all"); err != nil {
return err
}
if flags&CopyFlagSync != 0 {
return runSync()
}
return nil
}
fin, err := openfile(src, os.O_RDONLY, 0)
if err != nil {
return fmt.Errorf("unable to open %s: %v", src, err)
}
defer func() {
if cerr := fin.Close(); cerr != nil && err == nil {
err = fmt.Errorf("when closing %s: %v", src, cerr)
}
}()
fi, err := fin.Stat()
if err != nil {
return fmt.Errorf("unable to stat %s: %v", src, err)
}
outflags := os.O_WRONLY | os.O_CREATE | os.O_TRUNC
if flags&CopyFlagOverwrite == 0 {
outflags |= os.O_EXCL
}
fout, err := openfile(dst, outflags, fi.Mode())
if err != nil {
return fmt.Errorf("unable to create %s: %v", dst, err)
}
defer func() {
if cerr := fout.Close(); cerr != nil && err == nil {
err = fmt.Errorf("when closing %s: %v", dst, cerr)
}
}()
if err := copyfile(fin, fout, fi); err != nil {
return fmt.Errorf("unable to copy %s to %s: %v", src, dst, err)
}
if flags&CopyFlagSync != 0 {
if err = fout.Sync(); err != nil {
return fmt.Errorf("unable to sync %s: %v", dst, err)
}
}
return nil
}
// AtomicWriteFileCopy writes to dst a copy of src using AtomicFile
// internally to create the destination.
// The destination path is always overwritten. The destination and
// the owning directory are synced after copy completes. Pass additional flags
// for AtomicFile wrapping the destination.
func AtomicWriteFileCopy(dst, src string, flags AtomicWriteFlags) (err error) {
fin, err := openfile(src, os.O_RDONLY, 0)
if err != nil {
return fmt.Errorf("unable to open source file %s: %v", src, err)
}
defer func() {
if cerr := fin.Close(); cerr != nil && err == nil {
err = fmt.Errorf("when closing %s: %v", src, cerr)
}
}()
fi, err := fin.Stat()
if err != nil {
return fmt.Errorf("unable to stat %s: %v", src, err)
}
fout, err := NewAtomicFile(dst, fi.Mode(), flags, NoChown, NoChown)
if err != nil {
return fmt.Errorf("cannot create atomic file: %v", err)
}
fout.SetModTime(fi.ModTime())
defer func() {
if cerr := fout.Cancel(); cerr != ErrCannotCancel && err == nil {
err = fmt.Errorf("cannot cancel temporary file copy %s: %v", fout.Name(), cerr)
}
}()
if err := copyfile(fin, fout, fi); err != nil {
return fmt.Errorf("unable to copy %s to %s: %v", src, fout.Name(), err)
}
if err := fout.Commit(); err != nil {
return fmt.Errorf("cannot commit atomic file copy: %v", err)
}
return nil
}
func runCmd(cmd *exec.Cmd, errdesc string) error {
if output, err := cmd.CombinedOutput(); err != nil {
output = bytes.TrimSpace(output)
if exitCode, err := ExitCode(err); err == nil {
return &CopySpecialFileError{
desc: errdesc,
exitCode: exitCode,
output: output,
}
}
return &CopySpecialFileError{
desc: errdesc,
err: err,
output: output,
}
}
return nil
}
func runSync(args ...string) error {
return runCmd(exec.Command("sync", args...), "sync")
}
func runCpPreserveAll(path, dest, errdesc string) error {
return runCmd(exec.Command("cp", "-av", path, dest), errdesc)
}
// CopySpecialFile is used to copy all the things that are not files
// (like device nodes, named pipes etc)
func CopySpecialFile(path, dest string) error {
if err := runCpPreserveAll(path, dest, "copy device node"); err != nil {
return err
}
return runSync(filepath.Dir(dest))
}
// CopySpecialFileError is returned if a special file copy fails
type CopySpecialFileError struct {
desc string
exitCode int
output []byte
err error
}
func (e CopySpecialFileError) Error() string {
if e.err == nil {
return fmt.Sprintf("failed to %s: %q (%v)", e.desc, e.output, e.exitCode)
}
return fmt.Sprintf("failed to %s: %q (%v)", e.desc, e.output, e.err)
}
Go
1
https://gitee.com/mysnapcore/mysnapd.git
git@gitee.com:mysnapcore/mysnapd.git
mysnapcore
mysnapd
mysnapd
v0.1.0

搜索帮助