micro/internal/buffer/save.go

342 lines
7.7 KiB
Go
Raw Normal View History

2018-08-27 23:53:08 +00:00
package buffer
import (
"bufio"
2018-08-27 23:53:08 +00:00
"bytes"
2019-01-19 20:37:59 +00:00
"errors"
2018-08-27 23:53:08 +00:00
"io"
"io/fs"
2018-08-27 23:53:08 +00:00
"os"
"os/exec"
"os/signal"
"path/filepath"
2020-01-02 06:25:00 +00:00
"runtime"
2019-01-24 23:25:59 +00:00
"unicode"
2018-08-27 23:53:08 +00:00
2020-05-04 14:16:15 +00:00
"github.com/zyedidia/micro/v2/internal/config"
"github.com/zyedidia/micro/v2/internal/screen"
"github.com/zyedidia/micro/v2/internal/util"
"golang.org/x/text/encoding/htmlindex"
"golang.org/x/text/transform"
2018-08-27 23:53:08 +00:00
)
2019-01-14 02:06:58 +00:00
// LargeFileThreshold is the number of bytes when fastdirty is forced
// because hashing is too slow
const LargeFileThreshold = 50000
type wrappedFile struct {
writeCloser io.WriteCloser
withSudo bool
screenb bool
cmd *exec.Cmd
sigChan chan os.Signal
}
func openFile(name string, withSudo bool) (wrappedFile, error) {
var err error
var writeCloser io.WriteCloser
var screenb bool
var cmd *exec.Cmd
var sigChan chan os.Signal
if withSudo {
cmd = exec.Command(config.GlobalSettings["sucmd"].(string), "dd", "bs=4k", "of="+name)
writeCloser, err = cmd.StdinPipe()
if err != nil {
return wrappedFile{}, err
}
sigChan = make(chan os.Signal, 1)
signal.Reset(os.Interrupt)
signal.Notify(sigChan, os.Interrupt)
screenb = screen.TempFini()
// need to start the process now, otherwise when we flush the file
// contents to its stdin it might hang because the kernel's pipe size
// is too small to handle the full file contents all at once
err = cmd.Start()
if err != nil {
screen.TempStart(screenb)
signal.Notify(util.Sigterm, os.Interrupt)
signal.Stop(sigChan)
return wrappedFile{}, err
}
} else {
writeCloser, err = os.OpenFile(name, os.O_WRONLY|os.O_CREATE, util.FileMode)
if err != nil {
return wrappedFile{}, err
}
}
return wrappedFile{writeCloser, withSudo, screenb, cmd, sigChan}, nil
}
func (wf wrappedFile) Write(b *Buffer) (int, error) {
enc, err := htmlindex.Get(b.Settings["encoding"].(string))
if err != nil {
return 0, err
}
file := bufio.NewWriter(transform.NewWriter(wf.writeCloser, enc.NewEncoder()))
b.Lock()
defer b.Unlock()
if len(b.lines) == 0 {
return 0, nil
}
// end of line
var eol []byte
if b.Endings == FFDos {
eol = []byte{'\r', '\n'}
} else {
eol = []byte{'\n'}
}
if !wf.withSudo {
f := wf.writeCloser.(*os.File)
err = f.Truncate(0)
if err != nil {
return 0, err
}
}
// write lines
size, err := file.Write(b.lines[0].data)
if err != nil {
return 0, err
}
for _, l := range b.lines[1:] {
if _, err = file.Write(eol); err != nil {
return 0, err
}
if _, err = file.Write(l.data); err != nil {
return 0, err
}
size += len(eol) + len(l.data)
}
err = file.Flush()
if err == nil && !wf.withSudo {
// Call Sync() on the file to make sure the content is safely on disk.
f := wf.writeCloser.(*os.File)
err = f.Sync()
}
return size, err
}
func (wf wrappedFile) Close() error {
err := wf.writeCloser.Close()
if wf.withSudo {
// wait for dd to finish and restart the screen if we used sudo
err := wf.cmd.Wait()
screen.TempStart(wf.screenb)
2020-02-09 05:40:50 +00:00
signal.Notify(util.Sigterm, os.Interrupt)
signal.Stop(wf.sigChan)
2020-02-09 05:40:50 +00:00
if err != nil {
return err
2020-02-09 05:40:50 +00:00
}
}
return err
}
2020-02-09 05:40:50 +00:00
func (b *Buffer) overwriteFile(name string) (int, error) {
file, err := openFile(name, false)
if err != nil {
return 0, err
2020-02-09 05:40:50 +00:00
}
size, err := file.Write(b)
2020-02-09 05:40:50 +00:00
err2 := file.Close()
if err2 != nil && err == nil {
err = err2
}
2024-05-29 20:33:33 +00:00
return size, err
}
2018-08-27 23:53:08 +00:00
// Save saves the buffer to its default path
func (b *Buffer) Save() error {
return b.SaveAs(b.Path)
}
// AutoSave saves the buffer to its default path
func (b *Buffer) AutoSave() error {
// Doing full b.Modified() check every time would be costly, due to the hash
// calculation. So use just isModified even if fastdirty is not set.
if !b.isModified {
return nil
}
return b.saveToFile(b.Path, false, true)
}
2018-08-27 23:53:08 +00:00
// SaveAs saves the buffer to a specified path (filename), creating the file if it does not exist
2019-08-04 04:01:57 +00:00
func (b *Buffer) SaveAs(filename string) error {
return b.saveToFile(filename, false, false)
2019-12-22 22:24:00 +00:00
}
func (b *Buffer) SaveWithSudo() error {
return b.SaveAsWithSudo(b.Path)
}
func (b *Buffer) SaveAsWithSudo(filename string) error {
return b.saveToFile(filename, true, false)
2019-12-22 22:24:00 +00:00
}
func (b *Buffer) saveToFile(filename string, withSudo bool, autoSave bool) error {
2019-08-04 04:01:57 +00:00
var err error
2019-08-04 21:22:24 +00:00
if b.Type.Readonly {
return errors.New("Cannot save readonly buffer")
}
2019-01-19 20:37:59 +00:00
if b.Type.Scratch {
return errors.New("Cannot save scratch buffer")
}
if withSudo && runtime.GOOS == "windows" {
2020-02-09 05:40:50 +00:00
return errors.New("Save with sudo not supported on Windows")
}
2019-01-19 20:37:59 +00:00
if !autoSave && b.Settings["rmtrailingws"].(bool) {
2019-01-24 23:25:59 +00:00
for i, l := range b.lines {
2020-05-20 20:47:08 +00:00
leftover := util.CharacterCount(bytes.TrimRightFunc(l.data, unicode.IsSpace))
2019-01-24 23:25:59 +00:00
2020-05-20 20:47:08 +00:00
linelen := util.CharacterCount(l.data)
2019-01-24 23:25:59 +00:00
b.Remove(Loc{leftover, i}, Loc{linelen, i})
}
b.RelocateCursors()
}
2018-08-27 23:53:08 +00:00
if b.Settings["eofnewline"].(bool) {
end := b.End()
if b.RuneAt(Loc{end.X - 1, end.Y}) != '\n' {
b.insert(end, []byte{'\n'})
2018-08-27 23:53:08 +00:00
}
}
// Update the last time this file was updated after saving
defer func() {
b.ModTime, _ = util.GetModTime(filename)
2019-06-15 19:50:37 +00:00
err = b.Serialize()
2018-08-27 23:53:08 +00:00
}()
2024-05-29 20:33:33 +00:00
filename, err = util.ReplaceHome(filename)
if err != nil {
return err
}
2024-05-29 20:33:33 +00:00
newFile := false
fileInfo, err := os.Stat(filename)
if err != nil {
if !errors.Is(err, fs.ErrNotExist) {
return err
}
newFile = true
}
if err == nil && fileInfo.IsDir() {
2024-05-29 20:33:33 +00:00
return errors.New("Error: " + filename + " is a directory and cannot be saved")
}
if err == nil && !fileInfo.Mode().IsRegular() {
2024-05-29 20:33:33 +00:00
return errors.New("Error: " + filename + " is not a regular file and cannot be saved")
}
absFilename, err := filepath.Abs(filename)
if err != nil {
return err
}
2019-08-04 04:01:57 +00:00
// Get the leading path to the file | "." is returned if there's no leading path provided
if dirname := filepath.Dir(absFilename); dirname != "." {
// Check if the parent dirs don't exist
if _, statErr := os.Stat(dirname); errors.Is(statErr, fs.ErrNotExist) {
2019-08-04 04:01:57 +00:00
// Prompt to make sure they want to create the dirs that are missing
if b.Settings["mkparents"].(bool) {
// Create all leading dir(s) since they don't exist
if mkdirallErr := os.MkdirAll(dirname, os.ModePerm); mkdirallErr != nil {
// If there was an error creating the dirs
return mkdirallErr
}
} else {
return errors.New("Parent dirs don't exist, enable 'mkparents' for auto creation")
}
}
}
2018-08-27 23:53:08 +00:00
2024-05-29 20:33:33 +00:00
size, err := b.safeWrite(absFilename, withSudo, newFile)
if err != nil {
return err
}
2018-08-27 23:53:08 +00:00
if !b.Settings["fastdirty"].(bool) {
2024-05-29 20:33:33 +00:00
if size > LargeFileThreshold {
2018-08-27 23:53:08 +00:00
// For large files 'fastdirty' needs to be on
b.Settings["fastdirty"] = true
} else {
calcHash(b, &b.origHash)
}
}
b.Path = filename
2024-05-29 20:33:33 +00:00
b.AbsPath = absFilename
2019-01-14 21:52:25 +00:00
b.isModified = false
b.ReloadSettings(true)
2019-08-04 04:01:57 +00:00
return err
2018-08-27 23:53:08 +00:00
}
2024-05-29 20:33:33 +00:00
// safeWrite writes the buffer to a file in a "safe" way, preventing loss of the
// contents of the file if it fails to write the new contents.
// This means that the file is not overwritten directly but by writing to the
// backup file first.
func (b *Buffer) safeWrite(path string, withSudo bool, newFile bool) (int, error) {
file, err := openFile(path, withSudo)
if err != nil {
return 0, err
}
defer func() {
if newFile && err != nil {
os.Remove(path)
}
}()
2024-05-29 20:33:33 +00:00
backupDir := b.backupDir()
if _, err := os.Stat(backupDir); err != nil {
if !errors.Is(err, fs.ErrNotExist) {
return 0, err
}
if err = os.Mkdir(backupDir, os.ModePerm); err != nil {
return 0, err
}
}
backupName := util.DetermineEscapePath(backupDir, path)
_, err = b.overwriteFile(backupName)
2024-05-29 20:33:33 +00:00
if err != nil {
os.Remove(backupName)
return 0, err
}
b.forceKeepBackup = true
size, err := file.Write(b)
2024-05-29 20:33:33 +00:00
if err != nil {
return size, err
}
b.forceKeepBackup = false
if !b.keepBackup() {
os.Remove(backupName)
}
err2 := file.Close()
if err2 != nil && err == nil {
err = err2
}
2024-05-29 20:33:33 +00:00
return size, err
}