package fs import ( "fmt" "io" "os" "path/filepath" "regexp" "strings" "sync" "sync/atomic" "time" "github.com/VictoriaMetrics/VictoriaMetrics/lib/envutil" "github.com/VictoriaMetrics/VictoriaMetrics/lib/fasttime" "github.com/VictoriaMetrics/VictoriaMetrics/lib/filestream" "github.com/VictoriaMetrics/VictoriaMetrics/lib/logger" ) var disableFSyncForTesting = envutil.GetenvBool("DISABLE_FSYNC_FOR_TESTING") var tmpFileNum atomic.Uint64 // MustSyncPath syncs contents of the given path. func MustSyncPath(path string) { mustSyncPath(path) } // MustWriteSync writes data to the file at path and then calls fsync on the created file. // // The fsync guarantees that the written data survives hardware reset after successful call. // // This function may leave the file at the path in inconsistent state on app crash // in the middle of the write. // Use MustWriteAtomic if the file at the path must be either written in full // or not written at all on app crash in the middle of the write. func MustWriteSync(path string, data []byte) { f := filestream.MustCreate(path, false) if _, err := f.Write(data); err != nil { f.MustClose() // Do not call MustRemoveAll(path), so the user could inspect // the file contents during investigation of the issue. logger.Panicf("FATAL: cannot write %d bytes to %q: %s", len(data), path, err) } // Sync and close the file. f.MustClose() } // MustWriteAtomic atomically writes data to the given file path. // // This function returns only after the file is fully written and synced // to the underlying storage. // // This function guarantees that the file at path either fully written or not written at all on app crash // in the middle of the write. // // If the file at path already exists, then the file is overwritten atomically if canOverwrite is true. // Otherwise, error is returned. func MustWriteAtomic(path string, data []byte, canOverwrite bool) { // Check for the existing file. It is expected that // the MustWriteAtomic function cannot be called concurrently // with the same `path`. if IsPathExist(path) && !canOverwrite { logger.Panicf("FATAL: cannot create file %q, since it already exists", path) } // Write data to a temporary file. n := tmpFileNum.Add(1) tmpPath := fmt.Sprintf("%s.tmp.%d", path, n) MustWriteSync(tmpPath, data) // Atomically move the temporary file from tmpPath to path. if err := os.Rename(tmpPath, path); err != nil { // do not call MustRemoveAll(tmpPath) here, so the user could inspect // the file contents during investigation of the issue. logger.Panicf("FATAL: cannot move temporary file %q to %q: %s", tmpPath, path, err) } // Sync the containing directory, so the file is guaranteed to appear in the directory. // See https://www.quora.com/When-should-you-fsync-the-containing-directory-in-addition-to-the-file-itself absPath, err := filepath.Abs(path) if err != nil { logger.Panicf("FATAL: cannot obtain absolute path to %q: %s", path, err) } parentDirPath := filepath.Dir(absPath) MustSyncPath(parentDirPath) } // IsTemporaryFileName returns true if fn matches temporary file name pattern // from MustWriteAtomic. func IsTemporaryFileName(fn string) bool { return tmpFileNameRe.MatchString(fn) } // tmpFileNameRe is regexp for temporary file name - see MustWriteAtomic for details. var tmpFileNameRe = regexp.MustCompile(`\.tmp\.\d+$`) // MustMkdirIfNotExist creates the given path dir if it isn't exist. func MustMkdirIfNotExist(path string) { if IsPathExist(path) { return } mustMkdirSync(path) } // MustMkdirFailIfExist creates the given path dir if it isn't exist. // // If the directory at the given path already exists, then the function logs the error and exits. func MustMkdirFailIfExist(path string) { if IsPathExist(path) { logger.Panicf("FATAL: the %q already exists", path) } mustMkdirSync(path) } func mustMkdirSync(path string) { if err := os.MkdirAll(path, 0755); err != nil { logger.Panicf("FATAL: cannot create directory: %s", err) } // Sync the parent directory, so the created directory becomes visible // in the fs after power loss. parentDirPath := filepath.Dir(path) MustSyncPath(parentDirPath) } // RemoveDirContents removes all the contents of the given dir if it exists. // // It doesn't remove the dir itself, so the dir may be mounted // to a separate partition. func RemoveDirContents(dir string) { if !IsPathExist(dir) { // The path doesn't exist, so nothing to remove. return } d, err := os.Open(dir) if err != nil { logger.Panicf("FATAL: cannot open dir: %s", err) } defer MustClose(d) names, err := d.Readdirnames(-1) if err != nil { logger.Panicf("FATAL: cannot read contents of the dir %q: %s", dir, err) } for _, name := range names { if name == "." || name == ".." || name == "lost+found" { // Skip special dirs. continue } fullPath := filepath.Join(dir, name) MustRemoveAll(fullPath) } MustSyncPath(dir) } // MustClose must close the given file f. func MustClose(f *os.File) { fname := f.Name() if err := f.Close(); err != nil { logger.Panicf("FATAL: cannot close %q: %s", fname, err) } } // MustFileSize returns file size for the given path. func MustFileSize(path string) uint64 { fi, err := os.Stat(path) if err != nil { logger.Panicf("FATAL: cannot stat %q: %s", path, err) } if fi.IsDir() { logger.Panicf("FATAL: %q must be a file, not a directory", path) } return uint64(fi.Size()) } // IsPathExist returns whether the given path exists. func IsPathExist(path string) bool { if _, err := os.Stat(path); err != nil { if os.IsNotExist(err) { return false } logger.Panicf("FATAL: cannot stat %q: %s", path, err) } return true } func mustSyncParentDirIfExists(path string) { parentDirPath := filepath.Dir(path) if !IsPathExist(parentDirPath) { return } MustSyncPath(parentDirPath) } // MustRemoveDirAtomic removes the given dir atomically. // // It uses the following algorithm: // // 1. Atomically rename the "