mirror of
https://github.com/mattermost/mattermost.git
synced 2026-02-03 20:40:00 -05:00
137 lines
3.5 KiB
Go
137 lines
3.5 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package utils
|
|
|
|
import (
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
)
|
|
|
|
func CommonBaseSearchPaths() []string {
|
|
paths := []string{
|
|
".",
|
|
"..",
|
|
"../..",
|
|
"../../..",
|
|
"../../../..",
|
|
}
|
|
|
|
return paths
|
|
}
|
|
|
|
func findPath(path string, baseSearchPaths []string, workingDirFirst bool, filter func(os.FileInfo) bool) string {
|
|
if filepath.IsAbs(path) {
|
|
if _, err := os.Stat(path); err == nil {
|
|
return path
|
|
}
|
|
|
|
return ""
|
|
}
|
|
|
|
searchPaths := []string{}
|
|
if workingDirFirst {
|
|
searchPaths = append(searchPaths, baseSearchPaths...)
|
|
}
|
|
|
|
// Attempt to search relative to the location of the running binary either before
|
|
// or after searching relative to the working directory, depending on `workingDirFirst`.
|
|
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),
|
|
)
|
|
}
|
|
}
|
|
|
|
if !workingDirFirst {
|
|
searchPaths = append(searchPaths, baseSearchPaths...)
|
|
}
|
|
|
|
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 ""
|
|
}
|
|
|
|
func FindPath(path string, baseSearchPaths []string, filter func(os.FileInfo) bool) string {
|
|
return findPath(path, baseSearchPaths, true, filter)
|
|
}
|
|
|
|
// 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 {
|
|
return FindPath(path, CommonBaseSearchPaths(), func(fileInfo os.FileInfo) bool {
|
|
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) {
|
|
found := FindPath(dir, CommonBaseSearchPaths(), func(fileInfo os.FileInfo) bool {
|
|
return fileInfo.IsDir()
|
|
})
|
|
if found == "" {
|
|
return "./", false
|
|
}
|
|
|
|
return found, true
|
|
}
|
|
|
|
// 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) {
|
|
found := findPath(dir, CommonBaseSearchPaths(), false, func(fileInfo os.FileInfo) bool {
|
|
return fileInfo.IsDir()
|
|
})
|
|
if found == "" {
|
|
return "./", false
|
|
}
|
|
return found, true
|
|
}
|
|
|
|
// Valid characters are: alphanumeric, dash, underscore
|
|
var safeFileNameRegex = regexp.MustCompile(`[^\w\-\_]`)
|
|
|
|
// SanitizeFileName takes a string and returns a safe file name without an extension.
|
|
func SanitizeFileName(input string) string {
|
|
// Trim leading or trailing dots or spaces
|
|
safeName := strings.Trim(input, ". ")
|
|
// Replace dots with nothing
|
|
safeName = strings.ReplaceAll(safeName, ".", "")
|
|
|
|
// Replace all invalid characters with an underscore
|
|
safeName = safeFileNameRegex.ReplaceAllString(safeName, "_")
|
|
|
|
// Limit length
|
|
const maxLength = 100
|
|
if len(safeName) > maxLength {
|
|
safeName = safeName[:maxLength]
|
|
}
|
|
|
|
return safeName
|
|
}
|