package backend_golang import ( "archive/zip" "bufio" "embed" "errors" "io" "io/fs" "os" "os/exec" "path/filepath" "runtime" "strings" ) func Cmd(args ...string) (string, error) { switch platform := runtime.GOOS; platform { case "windows": if err := os.WriteFile("./cmd-helper.bat", []byte("start %*"), 0644); err != nil { return "", err } cmdHelper, err := filepath.Abs("./cmd-helper") if err != nil { return "", err } if strings.Contains(cmdHelper, " ") { for _, arg := range args { if strings.Contains(arg, " ") { return "", errors.New("path contains space") // golang bug https://github.com/golang/go/issues/17149#issuecomment-473976818 } } } cmd := exec.Command(cmdHelper, args...) out, err := cmd.CombinedOutput() if err != nil { return "", err } return string(out), nil case "darwin": ex, err := os.Executable() if err != nil { return "", err } exDir := filepath.Dir(ex) + "/../../../" cmd := exec.Command("osascript", "-e", `tell application "Terminal" to do script "`+"cd "+exDir+" && "+strings.Join(args, " ")+`"`) err = cmd.Start() if err != nil { return "", err } cmd.Wait() return "", nil case "linux": cmd := exec.Command(args[0], args[1:]...) err := cmd.Start() if err != nil { return "", err } cmd.Wait() return "", nil } return "", errors.New("unsupported OS") } func CopyEmbed(efs embed.FS) error { prefix := "" if runtime.GOOS == "darwin" { ex, err := os.Executable() if err != nil { return err } prefix = filepath.Dir(ex) + "/../../../" } err := fs.WalkDir(efs, ".", func(path string, d fs.DirEntry, err error) error { if d.IsDir() { return nil } if err != nil { return err } content, err := efs.ReadFile(path) if err != nil { return err } path = prefix + path err = os.MkdirAll(path[:strings.LastIndex(path, "/")], 0755) if err != nil { return err } err = os.WriteFile(path, content, 0644) if err != nil { return err } return nil }) return err } func GetPython() (string, error) { switch platform := runtime.GOOS; platform { case "windows": _, err := os.Stat("py310/python.exe") if err != nil { _, err := os.Stat("python-3.10.11-embed-amd64.zip") if err != nil { return "", errors.New("python zip not found") } else { err := Unzip("python-3.10.11-embed-amd64.zip", "py310") if err != nil { return "", errors.New("failed to unzip python") } else { python, err := filepath.Abs("py310/python.exe") if err != nil { return "", err } return python, nil } } } else { python, err := filepath.Abs("py310/python.exe") if err != nil { return "", err } return python, nil } case "darwin": return "python3", nil case "linux": return "python3", nil } return "", errors.New("unsupported OS") } func ChangeFileLine(filePath string, lineNumber int, newText string) error { file, err := os.OpenFile(filePath, os.O_RDWR, 0644) if err != nil { return err } defer file.Close() scanner := bufio.NewScanner(file) content := make([]string, 0) for scanner.Scan() { content = append(content, scanner.Text()) } content[lineNumber-1] = newText newContent := strings.Join(content, "\n") err = file.Truncate(0) if err != nil { return err } _, err = file.Seek(0, io.SeekStart) if err != nil { return err } _, err = file.WriteString(newContent) if err != nil { return err } return nil } // https://gist.github.com/paulerickson/6d8650947ee4e3f3dbcc28fde10eaae7 func Unzip(source, destination string) error { archive, err := zip.OpenReader(source) if err != nil { return err } defer archive.Close() for _, file := range archive.Reader.File { reader, err := file.Open() if err != nil { return err } defer reader.Close() path := filepath.Join(destination, file.Name) // Remove file if it already exists; no problem if it doesn't; other cases can error out below _ = os.Remove(path) // Create a directory at path, including parents err = os.MkdirAll(path, os.ModePerm) if err != nil { return err } // If file is _supposed_ to be a directory, we're done if file.FileInfo().IsDir() { continue } // otherwise, remove that directory (_not_ including parents) err = os.Remove(path) if err != nil { return err } // and create the actual file. This ensures that the parent directories exist! // An archive may have a single file with a nested path, rather than a file for each parent dir writer, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, file.Mode()) if err != nil { return err } defer writer.Close() _, err = io.Copy(writer, reader) if err != nil { return err } } return nil }