2019-11-07 09:17:42 +02:00
|
|
|
package piperutils
|
|
|
|
|
|
|
|
import (
|
2020-03-23 11:38:31 +02:00
|
|
|
"archive/zip"
|
2019-12-05 15:22:38 +02:00
|
|
|
"errors"
|
2020-03-23 11:38:31 +02:00
|
|
|
"fmt"
|
2020-06-15 09:47:33 +02:00
|
|
|
"github.com/bmatcuk/doublestar"
|
2019-12-05 15:22:38 +02:00
|
|
|
"io"
|
2020-02-25 15:17:00 +02:00
|
|
|
"io/ioutil"
|
2019-11-07 09:17:42 +02:00
|
|
|
"os"
|
2020-03-23 11:38:31 +02:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2019-11-07 09:17:42 +02:00
|
|
|
)
|
|
|
|
|
2020-02-25 15:17:00 +02:00
|
|
|
// FileUtils ...
|
|
|
|
type FileUtils interface {
|
2020-07-28 17:06:17 +02:00
|
|
|
Abs(path string) (string, error)
|
2020-02-25 15:17:00 +02:00
|
|
|
FileExists(filename string) (bool, error)
|
|
|
|
Copy(src, dest string) (int64, error)
|
|
|
|
FileRead(path string) ([]byte, error)
|
|
|
|
FileWrite(path string, content []byte, perm os.FileMode) error
|
|
|
|
MkdirAll(path string, perm os.FileMode) error
|
2020-07-16 14:25:01 +02:00
|
|
|
Chmod(path string, mode os.FileMode) error
|
2020-07-30 10:35:46 +02:00
|
|
|
Glob(pattern string) (matches []string, err error)
|
2020-02-25 15:17:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Files ...
|
|
|
|
type Files struct {
|
|
|
|
}
|
|
|
|
|
2020-04-03 21:32:38 +02:00
|
|
|
// FileExists returns true if the file system entry for the given path exists and is not a directory.
|
2020-02-25 15:17:00 +02:00
|
|
|
func (f Files) FileExists(filename string) (bool, error) {
|
2019-11-07 09:17:42 +02:00
|
|
|
info, err := os.Stat(filename)
|
2019-12-11 11:13:23 +02:00
|
|
|
|
2019-11-07 09:17:42 +02:00
|
|
|
if os.IsNotExist(err) {
|
2019-12-11 11:13:23 +02:00
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
2019-11-07 09:17:42 +02:00
|
|
|
}
|
2019-12-11 11:13:23 +02:00
|
|
|
|
|
|
|
return !info.IsDir(), nil
|
2019-11-07 09:17:42 +02:00
|
|
|
}
|
2019-12-05 15:22:38 +02:00
|
|
|
|
2020-04-03 21:32:38 +02:00
|
|
|
// FileExists returns true if the file system entry for the given path exists and is not a directory.
|
2020-02-25 15:17:00 +02:00
|
|
|
func FileExists(filename string) (bool, error) {
|
|
|
|
return Files{}.FileExists(filename)
|
|
|
|
}
|
|
|
|
|
2020-06-15 09:47:33 +02:00
|
|
|
// DirExists returns true if the file system entry for the given path exists and is a directory.
|
|
|
|
func (f Files) DirExists(path string) (bool, error) {
|
|
|
|
info, err := os.Stat(path)
|
|
|
|
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return info.IsDir(), nil
|
|
|
|
}
|
|
|
|
|
2019-12-05 15:22:38 +02:00
|
|
|
// Copy ...
|
2020-02-25 15:17:00 +02:00
|
|
|
func (f Files) Copy(src, dst string) (int64, error) {
|
2019-12-05 15:22:38 +02:00
|
|
|
|
2020-02-25 15:17:00 +02:00
|
|
|
exists, err := f.FileExists(src)
|
2019-12-11 11:13:23 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !exists {
|
2019-12-11 09:17:16 +02:00
|
|
|
return 0, errors.New("Source file '" + src + "' does not exist")
|
2019-12-05 15:22:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
source, err := os.Open(src)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-09-18 11:54:45 +02:00
|
|
|
defer func() { _ = source.Close() }()
|
2019-12-05 15:22:38 +02:00
|
|
|
|
|
|
|
destination, err := os.Create(dst)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-09-18 11:54:45 +02:00
|
|
|
defer func() { _ = destination.Close() }()
|
2019-12-05 15:22:38 +02:00
|
|
|
nBytes, err := io.Copy(destination, source)
|
|
|
|
return nBytes, err
|
|
|
|
}
|
2020-02-25 15:17:00 +02:00
|
|
|
|
2020-09-18 11:54:45 +02:00
|
|
|
//Chmod is a wrapper for os.Chmod().
|
2020-07-16 14:25:01 +02:00
|
|
|
func (f Files) Chmod(path string, mode os.FileMode) error {
|
|
|
|
return os.Chmod(path, mode)
|
|
|
|
}
|
|
|
|
|
2020-03-23 11:38:31 +02:00
|
|
|
// Unzip will decompress a zip archive, moving all files and folders
|
|
|
|
// within the zip file (parameter 1) to an output directory (parameter 2).
|
|
|
|
// from https://golangcode.com/unzip-files-in-go/ with the following license:
|
|
|
|
// MIT License
|
|
|
|
//
|
|
|
|
// Copyright (c) 2017 Edd Turtle
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
// SOFTWARE.
|
|
|
|
func Unzip(src, dest string) ([]string, error) {
|
|
|
|
|
|
|
|
var filenames []string
|
|
|
|
|
|
|
|
r, err := zip.OpenReader(src)
|
|
|
|
if err != nil {
|
|
|
|
return filenames, err
|
|
|
|
}
|
2020-09-18 11:54:45 +02:00
|
|
|
defer func() { _ = r.Close() }()
|
2020-03-23 11:38:31 +02:00
|
|
|
|
|
|
|
for _, f := range r.File {
|
|
|
|
|
|
|
|
// Store filename/path for returning and using later on
|
|
|
|
fpath := filepath.Join(dest, f.Name)
|
|
|
|
|
|
|
|
// Check for ZipSlip. More Info: http://bit.ly/2MsjAWE
|
|
|
|
if !strings.HasPrefix(fpath, filepath.Clean(dest)+string(os.PathSeparator)) {
|
|
|
|
return filenames, fmt.Errorf("%s: illegal file path", fpath)
|
|
|
|
}
|
|
|
|
|
|
|
|
filenames = append(filenames, fpath)
|
|
|
|
|
|
|
|
if f.FileInfo().IsDir() {
|
|
|
|
// Make Folder
|
2020-09-18 11:54:45 +02:00
|
|
|
err := os.MkdirAll(fpath, os.ModePerm)
|
|
|
|
if err != nil {
|
|
|
|
return filenames, fmt.Errorf("failed to create directory: %w", err)
|
|
|
|
}
|
2020-03-23 11:38:31 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make File
|
|
|
|
if err = os.MkdirAll(filepath.Dir(fpath), os.ModePerm); err != nil {
|
|
|
|
return filenames, err
|
|
|
|
}
|
|
|
|
|
|
|
|
outFile, err := os.OpenFile(fpath, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, f.Mode())
|
|
|
|
if err != nil {
|
|
|
|
return filenames, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rc, err := f.Open()
|
|
|
|
if err != nil {
|
|
|
|
return filenames, err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = io.Copy(outFile, rc)
|
|
|
|
|
|
|
|
// Close the file without defer to close before next iteration of loop
|
2020-09-18 11:54:45 +02:00
|
|
|
_ = outFile.Close()
|
|
|
|
_ = rc.Close()
|
2020-03-23 11:38:31 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return filenames, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return filenames, nil
|
|
|
|
}
|
|
|
|
|
2020-02-25 15:17:00 +02:00
|
|
|
// Copy ...
|
|
|
|
func Copy(src, dst string) (int64, error) {
|
|
|
|
return Files{}.Copy(src, dst)
|
|
|
|
}
|
|
|
|
|
2020-06-15 09:47:33 +02:00
|
|
|
// FileRead is a wrapper for ioutil.ReadFile().
|
2020-02-25 15:17:00 +02:00
|
|
|
func (f Files) FileRead(path string) ([]byte, error) {
|
|
|
|
return ioutil.ReadFile(path)
|
|
|
|
}
|
|
|
|
|
2020-06-15 09:47:33 +02:00
|
|
|
// FileWrite is a wrapper for ioutil.WriteFile().
|
2020-02-25 15:17:00 +02:00
|
|
|
func (f Files) FileWrite(path string, content []byte, perm os.FileMode) error {
|
|
|
|
return ioutil.WriteFile(path, content, perm)
|
|
|
|
}
|
|
|
|
|
2020-09-18 11:54:45 +02:00
|
|
|
// FileRemove is a wrapper for os.Remove().
|
2020-06-15 09:47:33 +02:00
|
|
|
func (f Files) FileRemove(path string) error {
|
|
|
|
return os.Remove(path)
|
|
|
|
}
|
|
|
|
|
2020-09-18 11:54:45 +02:00
|
|
|
// FileRename is a wrapper for os.Rename().
|
|
|
|
func (f Files) FileRename(oldPath, newPath string) error {
|
|
|
|
return os.Rename(oldPath, newPath)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FileOpen is a wrapper for os.OpenFile().
|
|
|
|
func (f *Files) FileOpen(name string, flag int, perm os.FileMode) (*os.File, error) {
|
|
|
|
return os.OpenFile(name, flag, perm)
|
|
|
|
}
|
|
|
|
|
2020-06-15 09:47:33 +02:00
|
|
|
// MkdirAll is a wrapper for os.MkdirAll().
|
2020-02-25 15:17:00 +02:00
|
|
|
func (f Files) MkdirAll(path string, perm os.FileMode) error {
|
|
|
|
return os.MkdirAll(path, perm)
|
|
|
|
}
|
2020-06-15 09:47:33 +02:00
|
|
|
|
2020-09-18 11:54:45 +02:00
|
|
|
// RemoveAll is a wrapper for os.RemoveAll().
|
|
|
|
func (f Files) RemoveAll(path string) error {
|
|
|
|
return os.RemoveAll(path)
|
|
|
|
}
|
|
|
|
|
2020-06-15 09:47:33 +02:00
|
|
|
// Glob is a wrapper for doublestar.Glob().
|
|
|
|
func (f Files) Glob(pattern string) (matches []string, err error) {
|
|
|
|
return doublestar.Glob(pattern)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Getwd is a wrapper for os.Getwd().
|
|
|
|
func (f Files) Getwd() (string, error) {
|
|
|
|
return os.Getwd()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Chdir is a wrapper for os.Chdir().
|
|
|
|
func (f Files) Chdir(path string) error {
|
|
|
|
return os.Chdir(path)
|
|
|
|
}
|
2020-07-15 10:31:36 +02:00
|
|
|
|
|
|
|
// Stat is a wrapper for os.Stat()
|
|
|
|
func (f Files) Stat(path string) (os.FileInfo, error) {
|
|
|
|
return os.Stat(path)
|
|
|
|
}
|
2020-07-28 17:06:17 +02:00
|
|
|
|
|
|
|
// Abs is a wrapper for filepath.Abs()
|
|
|
|
func (f Files) Abs(path string) (string, error) {
|
|
|
|
return filepath.Abs(path)
|
|
|
|
}
|