mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-04 07:29:56 +08:00
457 lines
9.5 KiB
Go
457 lines
9.5 KiB
Go
package s
|
|
|
|
import (
|
|
"crypto/md5"
|
|
"fmt"
|
|
"github.com/google/shlex"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
)
|
|
|
|
var (
|
|
Output io.Writer = io.Discard
|
|
IndentSize int
|
|
originalOutput io.Writer
|
|
currentIndent int
|
|
dryRun bool
|
|
deferred []func()
|
|
)
|
|
|
|
func checkError(err error) {
|
|
if err != nil {
|
|
println("\nERROR:", err.Error())
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
func mute() {
|
|
originalOutput = Output
|
|
Output = io.Discard
|
|
}
|
|
|
|
func unmute() {
|
|
Output = originalOutput
|
|
}
|
|
|
|
func indent() {
|
|
currentIndent += IndentSize
|
|
}
|
|
|
|
func unindent() {
|
|
currentIndent -= IndentSize
|
|
}
|
|
|
|
func log(message string, args ...interface{}) {
|
|
indent := strings.Repeat(" ", currentIndent)
|
|
_, err := fmt.Fprintf(Output, indent+message+"\n", args...)
|
|
checkError(err)
|
|
}
|
|
|
|
// RENAME a file or directory
|
|
func RENAME(source string, target string) {
|
|
log("RENAME %s -> %s", source, target)
|
|
err := os.Rename(source, target)
|
|
checkError(err)
|
|
}
|
|
|
|
// MUSTDELETE a file.
|
|
func MUSTDELETE(filename string) {
|
|
log("DELETE %s", filename)
|
|
err := os.Remove(filepath.Join(CWD(), filename))
|
|
checkError(err)
|
|
}
|
|
|
|
// DELETE a file.
|
|
func DELETE(filename string) {
|
|
log("DELETE %s", filename)
|
|
_ = os.Remove(filepath.Join(CWD(), filename))
|
|
}
|
|
|
|
func CONTAINS(list string, item string) bool {
|
|
result := strings.Contains(list, item)
|
|
listTrimmed := list
|
|
if len(listTrimmed) > 30 {
|
|
listTrimmed = listTrimmed[:30] + "..."
|
|
}
|
|
log("CONTAINS %s in %s: %t", item, listTrimmed, result)
|
|
return result
|
|
}
|
|
|
|
func SETENV(key string, value string) {
|
|
log("SETENV %s=%s", key, value)
|
|
err := os.Setenv(key, value)
|
|
checkError(err)
|
|
}
|
|
|
|
func CD(dir string) {
|
|
err := os.Chdir(dir)
|
|
checkError(err)
|
|
log("CD %s", dir)
|
|
}
|
|
func MKDIR(path string, mode ...os.FileMode) {
|
|
var perms os.FileMode
|
|
perms = 0755
|
|
if len(mode) == 1 {
|
|
perms = mode[0]
|
|
}
|
|
log("MKDIR %s (perms: %v)", path, perms)
|
|
err := os.MkdirAll(path, perms)
|
|
checkError(err)
|
|
}
|
|
|
|
// ENDIR ensures that the path gets created if it doesn't exist
|
|
func ENDIR(path string, mode ...os.FileMode) {
|
|
var perms os.FileMode
|
|
perms = 0755
|
|
if len(mode) == 1 {
|
|
perms = mode[0]
|
|
}
|
|
_ = os.MkdirAll(path, perms)
|
|
}
|
|
|
|
// COPYDIR recursively copies a directory tree, attempting to preserve permissions.
|
|
// Source directory must exist, destination directory must *not* exist.
|
|
// Symlinks are ignored and skipped.
|
|
// Credit: https://gist.github.com/r0l1/92462b38df26839a3ca324697c8cba04
|
|
func COPYDIR(src string, dst string) {
|
|
log("COPYDIR %s -> %s", src, dst)
|
|
src = filepath.Clean(src)
|
|
dst = filepath.Clean(dst)
|
|
|
|
si, err := os.Stat(src)
|
|
checkError(err)
|
|
if !si.IsDir() {
|
|
checkError(fmt.Errorf("source is not a directory"))
|
|
}
|
|
|
|
_, err = os.Stat(dst)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
checkError(err)
|
|
}
|
|
if err == nil {
|
|
checkError(fmt.Errorf("destination already exists"))
|
|
}
|
|
|
|
indent()
|
|
MKDIR(dst)
|
|
|
|
entries, err := os.ReadDir(src)
|
|
checkError(err)
|
|
|
|
for _, entry := range entries {
|
|
srcPath := filepath.Join(src, entry.Name())
|
|
dstPath := filepath.Join(dst, entry.Name())
|
|
|
|
if entry.IsDir() {
|
|
COPYDIR(srcPath, dstPath)
|
|
} else {
|
|
// Skip symlinks.
|
|
if entry.Type()&os.ModeSymlink != 0 {
|
|
continue
|
|
}
|
|
|
|
COPY(srcPath, dstPath)
|
|
}
|
|
}
|
|
unindent()
|
|
}
|
|
|
|
// COPYDIR2 recursively copies a directory tree, attempting to preserve permissions.
|
|
// Source directory must exist, destination directory can exist.
|
|
// Symlinks are ignored and skipped.
|
|
// Credit: https://gist.github.com/r0l1/92462b38df26839a3ca324697c8cba04
|
|
func COPYDIR2(src string, dst string) {
|
|
log("COPYDIR %s -> %s", src, dst)
|
|
src = filepath.Clean(src)
|
|
dst = filepath.Clean(dst)
|
|
|
|
si, err := os.Stat(src)
|
|
checkError(err)
|
|
if !si.IsDir() {
|
|
checkError(fmt.Errorf("source is not a directory"))
|
|
}
|
|
|
|
indent()
|
|
MKDIR(dst)
|
|
|
|
entries, err := os.ReadDir(src)
|
|
checkError(err)
|
|
|
|
for _, entry := range entries {
|
|
srcPath := filepath.Join(src, entry.Name())
|
|
dstPath := filepath.Join(dst, entry.Name())
|
|
|
|
if entry.IsDir() {
|
|
COPYDIR(srcPath, dstPath)
|
|
} else {
|
|
// Skip symlinks.
|
|
if entry.Type()&os.ModeSymlink != 0 {
|
|
continue
|
|
}
|
|
|
|
COPY(srcPath, dstPath)
|
|
}
|
|
}
|
|
unindent()
|
|
}
|
|
|
|
func SYMLINK(source string, target string) {
|
|
// trim string to first 30 chars
|
|
var trimTarget = target
|
|
if len(trimTarget) > 30 {
|
|
trimTarget = trimTarget[:30] + "..."
|
|
}
|
|
log("SYMLINK %s -> %s", source, trimTarget)
|
|
err := os.Symlink(source, target)
|
|
checkError(err)
|
|
}
|
|
|
|
// COPY file from source to target
|
|
func COPY(source string, target string) {
|
|
log("COPY %s -> %s", source, target)
|
|
src, err := os.Open(source)
|
|
checkError(err)
|
|
defer closefile(src)
|
|
if ISDIR(target) {
|
|
target = filepath.Join(target, filepath.Base(source))
|
|
}
|
|
d, err := os.Create(target)
|
|
checkError(err)
|
|
_, err = io.Copy(d, src)
|
|
checkError(err)
|
|
}
|
|
|
|
// Move file from source to target
|
|
func MOVE(source string, target string) {
|
|
// If target is a directory, append the source filename
|
|
if ISDIR(target) {
|
|
target = filepath.Join(target, filepath.Base(source))
|
|
}
|
|
log("MOVE %s -> %s", source, target)
|
|
err := os.Rename(source, target)
|
|
checkError(err)
|
|
}
|
|
|
|
func CWD() string {
|
|
result, err := os.Getwd()
|
|
checkError(err)
|
|
log("CWD %s", result)
|
|
return result
|
|
}
|
|
|
|
func RMDIR(target string) {
|
|
log("RMDIR %s", target)
|
|
err := os.RemoveAll(target)
|
|
checkError(err)
|
|
}
|
|
|
|
func RM(target string) {
|
|
log("RM %s", target)
|
|
err := os.Remove(target)
|
|
checkError(err)
|
|
}
|
|
|
|
func ECHO(message string) {
|
|
println(message)
|
|
}
|
|
|
|
func TOUCH(filepath string) {
|
|
log("TOUCH %s", filepath)
|
|
f, err := os.Create(filepath)
|
|
checkError(err)
|
|
closefile(f)
|
|
}
|
|
|
|
func EXEC(command string) ([]byte, error) {
|
|
log("EXEC %s", command)
|
|
|
|
// Split input using shlex
|
|
args, err := shlex.Split(command)
|
|
checkError(err)
|
|
// Execute command
|
|
cmd := exec.Command(args[0], args[1:]...)
|
|
cmd.Dir = CWD()
|
|
cmd.Env = os.Environ()
|
|
return cmd.CombinedOutput()
|
|
}
|
|
|
|
func CHMOD(path string, mode os.FileMode) {
|
|
log("CHMOD %s %v", path, mode)
|
|
err := os.Chmod(path, mode)
|
|
checkError(err)
|
|
}
|
|
|
|
// EXISTS - Returns true if the given path exists
|
|
func EXISTS(path string) bool {
|
|
_, err := os.Lstat(path)
|
|
log("EXISTS %s -> %t", path, err == nil)
|
|
return err == nil
|
|
}
|
|
|
|
// ISDIR returns true if the given directory exists
|
|
func ISDIR(path string) bool {
|
|
fi, err := os.Lstat(path)
|
|
if err != nil {
|
|
return false
|
|
}
|
|
|
|
return fi.Mode().IsDir()
|
|
}
|
|
|
|
// ISDIREMPTY returns true if the given directory is empty
|
|
func ISDIREMPTY(dir string) bool {
|
|
|
|
// CREDIT: https://stackoverflow.com/a/30708914/8325411
|
|
f, err := os.Open(dir)
|
|
checkError(err)
|
|
defer closefile(f)
|
|
|
|
_, err = f.Readdirnames(1) // Or f.Readdir(1)
|
|
if err == io.EOF {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// ISFILE returns true if the given file exists
|
|
func ISFILE(path string) bool {
|
|
fi, err := os.Lstat(path)
|
|
if err != nil {
|
|
return false
|
|
}
|
|
|
|
return fi.Mode().IsRegular()
|
|
}
|
|
|
|
// SUBDIRS returns a list of subdirectories for the given directory
|
|
func SUBDIRS(rootDir string) []string {
|
|
var result []string
|
|
|
|
// Iterate root dir
|
|
err := filepath.Walk(rootDir, func(path string, info os.FileInfo, err error) error {
|
|
checkError(err)
|
|
// If we have a directory, save it
|
|
if info.IsDir() {
|
|
result = append(result, path)
|
|
}
|
|
return nil
|
|
})
|
|
checkError(err)
|
|
return result
|
|
}
|
|
|
|
// SAVESTRING will create a file with the given string
|
|
func SAVESTRING(filename string, data string) {
|
|
log("SAVESTRING %s", filename)
|
|
mute()
|
|
SAVEBYTES(filename, []byte(data))
|
|
unmute()
|
|
}
|
|
|
|
// LOADSTRING returns the contents of the given filename as a string
|
|
func LOADSTRING(filename string) string {
|
|
log("LOADSTRING %s", filename)
|
|
mute()
|
|
data := LOADBYTES(filename)
|
|
unmute()
|
|
return string(data)
|
|
}
|
|
|
|
// SAVEBYTES will create a file with the given string
|
|
func SAVEBYTES(filename string, data []byte) {
|
|
log("SAVEBYTES %s", filename)
|
|
err := os.WriteFile(filename, data, 0755)
|
|
checkError(err)
|
|
}
|
|
|
|
// LOADBYTES returns the contents of the given filename as a string
|
|
func LOADBYTES(filename string) []byte {
|
|
log("LOADBYTES %s", filename)
|
|
data, err := os.ReadFile(filename)
|
|
checkError(err)
|
|
return data
|
|
}
|
|
|
|
func closefile(f *os.File) {
|
|
err := f.Close()
|
|
checkError(err)
|
|
}
|
|
|
|
// MD5FILE returns the md5sum of the given file
|
|
func MD5FILE(filename string) string {
|
|
f, err := os.Open(filename)
|
|
checkError(err)
|
|
defer closefile(f)
|
|
|
|
h := md5.New()
|
|
_, err = io.Copy(h, f)
|
|
checkError(err)
|
|
|
|
return fmt.Sprintf("%x", h.Sum(nil))
|
|
}
|
|
|
|
// Sub is the substitution type
|
|
type Sub map[string]string
|
|
|
|
// REPLACEALL replaces all substitution keys with associated values in the given file
|
|
func REPLACEALL(filename string, substitutions Sub) {
|
|
log("REPLACEALL %s (%v)", filename, substitutions)
|
|
data := LOADSTRING(filename)
|
|
for old, newText := range substitutions {
|
|
data = strings.ReplaceAll(data, old, newText)
|
|
}
|
|
SAVESTRING(filename, data)
|
|
}
|
|
|
|
func DOWNLOAD(url string, target string) {
|
|
log("DOWNLOAD %s -> %s", url, target)
|
|
// create HTTP client
|
|
resp, err := http.Get(url)
|
|
checkError(err)
|
|
defer resp.Body.Close()
|
|
|
|
out, err := os.Create(target)
|
|
checkError(err)
|
|
defer out.Close()
|
|
|
|
// Write the body to file
|
|
_, err = io.Copy(out, resp.Body)
|
|
checkError(err)
|
|
}
|
|
|
|
func FINDFILES(root string, filenames ...string) []string {
|
|
var result []string
|
|
// Walk the root directory trying to find all the files
|
|
err := filepath.Walk(root, func(path string, info os.FileInfo, err error) error {
|
|
checkError(err)
|
|
// If we have a file, check if it is in the list
|
|
if info.Mode().IsRegular() {
|
|
for _, filename := range filenames {
|
|
if info.Name() == filename {
|
|
result = append(result, path)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
checkError(err)
|
|
log("FINDFILES in %s -> [%v]", root, strings.Join(result, ", "))
|
|
return result
|
|
}
|
|
|
|
func DEFER(fn func()) {
|
|
log("DEFER")
|
|
deferred = append(deferred, fn)
|
|
}
|
|
|
|
func CALLDEFER() {
|
|
log("CALLDEFER")
|
|
for _, fn := range deferred {
|
|
fn()
|
|
}
|
|
}
|