2018-12-17 11:51:46 -05:00
|
|
|
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
2019-11-29 06:59:40 -05:00
|
|
|
// See LICENSE.txt for license information.
|
2018-12-17 11:51:46 -05:00
|
|
|
|
|
|
|
|
package fileutils
|
|
|
|
|
|
|
|
|
|
import (
|
2026-03-03 15:31:05 -05:00
|
|
|
"fmt"
|
2018-12-17 11:51:46 -05:00
|
|
|
"os"
|
|
|
|
|
"path/filepath"
|
2026-03-03 15:31:05 -05:00
|
|
|
"strings"
|
2024-05-15 11:05:13 -04:00
|
|
|
|
|
|
|
|
"github.com/mattermost/mattermost/server/v8"
|
2018-12-17 11:51:46 -05:00
|
|
|
)
|
|
|
|
|
|
2022-05-05 12:12:31 -04:00
|
|
|
func CommonBaseSearchPaths() []string {
|
|
|
|
|
paths := []string{
|
2018-12-17 11:51:46 -05:00
|
|
|
".",
|
|
|
|
|
"..",
|
|
|
|
|
"../..",
|
|
|
|
|
"../../..",
|
2022-05-05 12:12:31 -04:00
|
|
|
"../../../..",
|
2018-12-17 11:51:46 -05:00
|
|
|
}
|
2022-05-05 12:12:31 -04:00
|
|
|
|
2024-05-15 11:05:13 -04:00
|
|
|
// this enables the server to be used in tests from a different repository
|
|
|
|
|
paths = append(paths, server.GetPackagePath())
|
2022-05-05 12:12:31 -04:00
|
|
|
return paths
|
|
|
|
|
}
|
2018-12-17 11:51:46 -05:00
|
|
|
|
2020-05-14 20:11:22 -04:00
|
|
|
func findPath(path string, baseSearchPaths []string, workingDirFirst bool, filter func(os.FileInfo) bool) string {
|
2018-12-17 11:51:46 -05:00
|
|
|
if filepath.IsAbs(path) {
|
|
|
|
|
if _, err := os.Stat(path); err == nil {
|
|
|
|
|
return path
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
searchPaths := []string{}
|
2020-05-14 20:11:22 -04:00
|
|
|
if workingDirFirst {
|
|
|
|
|
searchPaths = append(searchPaths, baseSearchPaths...)
|
|
|
|
|
}
|
2018-12-17 11:51:46 -05:00
|
|
|
|
2020-05-14 20:11:22 -04:00
|
|
|
// Attempt to search relative to the location of the running binary either before
|
|
|
|
|
// or after searching relative to the working directory, depending on `workingDirFirst`.
|
2018-12-17 11:51:46 -05:00
|
|
|
var binaryDir string
|
|
|
|
|
if exe, err := os.Executable(); err == nil {
|
|
|
|
|
if exe, err = filepath.EvalSymlinks(exe); err == nil {
|
|
|
|
|
if exe, err = filepath.Abs(exe); err == nil {
|
|
|
|
|
binaryDir = filepath.Dir(exe)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if binaryDir != "" {
|
|
|
|
|
for _, baseSearchPath := range baseSearchPaths {
|
|
|
|
|
searchPaths = append(
|
|
|
|
|
searchPaths,
|
|
|
|
|
filepath.Join(binaryDir, baseSearchPath),
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-14 20:11:22 -04:00
|
|
|
if !workingDirFirst {
|
|
|
|
|
searchPaths = append(searchPaths, baseSearchPaths...)
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-17 11:51:46 -05:00
|
|
|
for _, parent := range searchPaths {
|
|
|
|
|
found, err := filepath.Abs(filepath.Join(parent, path))
|
|
|
|
|
if err != nil {
|
|
|
|
|
continue
|
|
|
|
|
} else if fileInfo, err := os.Stat(found); err == nil {
|
|
|
|
|
if filter != nil {
|
|
|
|
|
if filter(fileInfo) {
|
|
|
|
|
return found
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return found
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-14 20:11:22 -04:00
|
|
|
func FindPath(path string, baseSearchPaths []string, filter func(os.FileInfo) bool) string {
|
|
|
|
|
return findPath(path, baseSearchPaths, true, filter)
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-17 11:51:46 -05:00
|
|
|
// FindFile looks for the given file in nearby ancestors relative to the current working
|
|
|
|
|
// directory as well as the directory of the executable.
|
|
|
|
|
func FindFile(path string) string {
|
2022-05-05 12:12:31 -04:00
|
|
|
return FindPath(path, CommonBaseSearchPaths(), func(fileInfo os.FileInfo) bool {
|
2018-12-17 11:51:46 -05:00
|
|
|
return !fileInfo.IsDir()
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// fileutils.FindDir looks for the given directory in nearby ancestors relative to the current working
|
|
|
|
|
// directory as well as the directory of the executable, falling back to `./` if not found.
|
|
|
|
|
func FindDir(dir string) (string, bool) {
|
2022-05-05 12:12:31 -04:00
|
|
|
found := FindPath(dir, CommonBaseSearchPaths(), func(fileInfo os.FileInfo) bool {
|
2018-12-17 11:51:46 -05:00
|
|
|
return fileInfo.IsDir()
|
|
|
|
|
})
|
|
|
|
|
if found == "" {
|
|
|
|
|
return "./", false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return found, true
|
|
|
|
|
}
|
2020-05-14 20:11:22 -04:00
|
|
|
|
|
|
|
|
// FindDirRelBinary looks for the given directory in nearby ancestors relative to the
|
|
|
|
|
// directory of the executable, then relative to the working directory, falling back to `./` if not found.
|
|
|
|
|
func FindDirRelBinary(dir string) (string, bool) {
|
2022-05-05 12:12:31 -04:00
|
|
|
found := findPath(dir, CommonBaseSearchPaths(), false, func(fileInfo os.FileInfo) bool {
|
2020-05-14 20:11:22 -04:00
|
|
|
return fileInfo.IsDir()
|
|
|
|
|
})
|
|
|
|
|
if found == "" {
|
|
|
|
|
return "./", false
|
|
|
|
|
}
|
|
|
|
|
return found, true
|
|
|
|
|
}
|
2026-03-03 15:31:05 -05:00
|
|
|
|
|
|
|
|
// CheckDirectoryConflict checks if either directory is a subdirectory of the other.
|
|
|
|
|
// Returns true if there is a conflict (one is a subdirectory of the other or they are the same).
|
|
|
|
|
// Returns an error if the directory paths cannot be resolved.
|
|
|
|
|
// If a directory does not exist, the absolute path is used without symlink resolution.
|
|
|
|
|
func CheckDirectoryConflict(dir1, dir2 string) (bool, error) {
|
|
|
|
|
absDir1, err := filepath.Abs(dir1)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return false, fmt.Errorf("failed to resolve absolute path for %q: %w", dir1, err)
|
|
|
|
|
}
|
|
|
|
|
absDir2, err := filepath.Abs(dir2)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return false, fmt.Errorf("failed to resolve absolute path for %q: %w", dir2, err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if resolved, err := filepath.EvalSymlinks(absDir1); err == nil {
|
|
|
|
|
absDir1 = resolved
|
|
|
|
|
} else if !os.IsNotExist(err) {
|
|
|
|
|
return false, fmt.Errorf("failed to evaluate symlinks for %q: %w", dir1, err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if resolved, err := filepath.EvalSymlinks(absDir2); err == nil {
|
|
|
|
|
absDir2 = resolved
|
|
|
|
|
} else if !os.IsNotExist(err) {
|
|
|
|
|
return false, fmt.Errorf("failed to evaluate symlinks for %q: %w", dir2, err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !strings.HasSuffix(absDir1, string(filepath.Separator)) {
|
|
|
|
|
absDir1 += string(filepath.Separator)
|
|
|
|
|
}
|
|
|
|
|
if !strings.HasSuffix(absDir2, string(filepath.Separator)) {
|
|
|
|
|
absDir2 += string(filepath.Separator)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return strings.HasPrefix(absDir1, absDir2) || strings.HasPrefix(absDir2, absDir1), nil
|
|
|
|
|
}
|