mirror of
https://github.com/ManyakRus/crud_generator.git
synced 2025-01-18 21:09:02 +02:00
205 lines
4.9 KiB
Go
205 lines
4.9 KiB
Go
package utils
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
)
|
|
|
|
const (
|
|
// tmpPermissionForDirectory makes the destination directory writable,
|
|
// so that stuff can be copied recursively even if any original directory is NOT writable.
|
|
// See https://github.com/otiai10/copy/pull/9 for more information.
|
|
tmpPermissionForDirectory = os.FileMode(0755)
|
|
)
|
|
|
|
// Copy copies src to dest, doesn't matter if src is a directory or a file.
|
|
func Copy(src, dest string, opt ...Options) (*Results, error) {
|
|
results := &Results{}
|
|
info, err := os.Lstat(src)
|
|
if err != nil {
|
|
return results, err
|
|
}
|
|
|
|
err = switchboard(src, dest, info, assure(opt...), results)
|
|
return results, err
|
|
}
|
|
|
|
// switchboard switches proper copy functions regarding file type, etc...
|
|
// If there would be anything else here, add a case to this switchboard.
|
|
func switchboard(src, dest string, info os.FileInfo, opt Options, results *Results) error {
|
|
switch {
|
|
case info.Mode()&os.ModeSymlink != 0:
|
|
return onsymlink(src, dest, info, opt, results)
|
|
case info.IsDir():
|
|
return dcopy(src, dest, info, opt, results)
|
|
default:
|
|
return fcopy(src, dest, info, opt, results)
|
|
}
|
|
}
|
|
|
|
// copy decide if this src should be copied or not.
|
|
// Because this "copy" could be called recursively,
|
|
// "info" MUST be given here, NOT nil.
|
|
func copy(src, dest string, info os.FileInfo, opt Options, results *Results) error {
|
|
skip, err := opt.Skip(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if skip {
|
|
return nil
|
|
}
|
|
return switchboard(src, dest, info, opt, results)
|
|
}
|
|
|
|
// fcopy is for just a file,
|
|
// with considering existence of parent directory
|
|
// and file permission.
|
|
func fcopy(src, dest string, info os.FileInfo, opt Options, results *Results) (err error) {
|
|
|
|
if err = os.MkdirAll(filepath.Dir(dest), os.ModePerm); err != nil {
|
|
return
|
|
}
|
|
|
|
handler := DefaultFileCopy
|
|
if opt.FileHandler != nil {
|
|
handler = opt.FileHandler(src, dest, info)
|
|
}
|
|
|
|
return handler(src, dest, info, opt, results)
|
|
}
|
|
|
|
// DefaultFileCopy file copy that can be called from external
|
|
func DefaultFileCopy(src, dest string, info os.FileInfo, opt Options, results *Results) (err error) {
|
|
f, err := os.Create(dest)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer fclose(f, &err)
|
|
|
|
if err = os.Chmod(f.Name(), info.Mode()|opt.AddPermission); err != nil {
|
|
return
|
|
}
|
|
|
|
s, err := os.Open(src)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer fclose(s, &err)
|
|
|
|
if _, err = io.Copy(f, s); err != nil {
|
|
return
|
|
}
|
|
|
|
if opt.Sync {
|
|
err = f.Sync()
|
|
}
|
|
results.FilesCopied++
|
|
|
|
return
|
|
}
|
|
|
|
// dcopy is for a directory,
|
|
// with scanning contents inside the directory
|
|
// and pass everything to "copy" recursively.
|
|
func dcopy(srcdir, destdir string, info os.FileInfo, opt Options, results *Results) (err error) {
|
|
if opt.ShouldCopy != nil && !opt.ShouldCopy(info) {
|
|
results.Info.WriteString(fmt.Sprintf("CopyDir Skipping %s\n", srcdir))
|
|
return nil
|
|
}
|
|
|
|
originalMode := info.Mode()
|
|
// Make dest dir with 0755 so that everything writable.
|
|
if err = os.MkdirAll(destdir, tmpPermissionForDirectory); err != nil {
|
|
return
|
|
}
|
|
results.DirsCopied++
|
|
// Recover dir mode with original one.
|
|
defer chmod(destdir, originalMode|opt.AddPermission, &err)
|
|
|
|
contents, err := ioutil.ReadDir(srcdir)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, content := range contents {
|
|
cs, cd := filepath.Join(srcdir, content.Name()), filepath.Join(destdir, content.Name())
|
|
|
|
if err = copy(cs, cd, content, opt, results); err != nil {
|
|
// If any error, exit immediately
|
|
return
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func onsymlink(src, dest string, info os.FileInfo, opt Options, results *Results) error {
|
|
|
|
switch opt.OnSymlink(src) {
|
|
case Shallow:
|
|
return lcopy(src, dest, results)
|
|
case Deep:
|
|
orig, err := os.Readlink(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
info, err = os.Lstat(orig)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return copy(orig, dest, info, opt, results)
|
|
case Skip:
|
|
fallthrough
|
|
default:
|
|
return nil // do nothing
|
|
}
|
|
}
|
|
|
|
// lcopy is for a symlink,
|
|
// with just creating a new symlink by replicating src symlink.
|
|
func lcopy(src, dest string, results *Results) error {
|
|
src, err := os.Readlink(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
results.SymLinksCreated++
|
|
return os.Symlink(src, dest)
|
|
}
|
|
|
|
// fclose ANYHOW closes file,
|
|
// with asiging error raised during Close,
|
|
// BUT respecting the error already reported.
|
|
func fclose(f *os.File, reported *error) {
|
|
if err := f.Close(); *reported == nil {
|
|
*reported = err
|
|
}
|
|
}
|
|
|
|
// chmod ANYHOW changes file mode,
|
|
// with asiging error raised during Chmod,
|
|
// BUT respecting the error already reported.
|
|
func chmod(dir string, mode os.FileMode, reported *error) {
|
|
if err := os.Chmod(dir, mode); *reported == nil {
|
|
*reported = err
|
|
}
|
|
}
|
|
|
|
// assure Options struct, should be called only once.
|
|
// All optional values MUST NOT BE nil/zero after assured.
|
|
func assure(opts ...Options) Options {
|
|
if len(opts) == 0 {
|
|
return DefaultCopyOptions()
|
|
}
|
|
defopt := DefaultCopyOptions()
|
|
if opts[0].OnSymlink == nil {
|
|
opts[0].OnSymlink = defopt.OnSymlink
|
|
}
|
|
if opts[0].Skip == nil {
|
|
opts[0].Skip = defopt.Skip
|
|
}
|
|
return opts[0]
|
|
}
|