mirror of
https://github.com/ceph/ceph-csi.git
synced 2025-01-03 18:49:30 +00:00
129 lines
3.8 KiB
Go
129 lines
3.8 KiB
Go
|
/*
|
||
|
* path.go - Utility functions for dealing with filesystem paths
|
||
|
*
|
||
|
* Copyright 2017 Google Inc.
|
||
|
* Author: Joe Richey (joerichey@google.com)
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
||
|
* use this file except in compliance with the License. You may obtain a copy of
|
||
|
* the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||
|
* License for the specific language governing permissions and limitations under
|
||
|
* the License.
|
||
|
*/
|
||
|
|
||
|
package filesystem
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"log"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
|
||
|
"golang.org/x/sys/unix"
|
||
|
|
||
|
"github.com/pkg/errors"
|
||
|
)
|
||
|
|
||
|
// OpenFileOverridingUmask calls os.OpenFile but with the umask overridden so
|
||
|
// that no permission bits are masked out if the file is created.
|
||
|
func OpenFileOverridingUmask(name string, flag int, perm os.FileMode) (*os.File, error) {
|
||
|
oldMask := unix.Umask(0)
|
||
|
defer unix.Umask(oldMask)
|
||
|
return os.OpenFile(name, flag, perm)
|
||
|
}
|
||
|
|
||
|
// canonicalizePath turns path into an absolute path without symlinks.
|
||
|
func canonicalizePath(path string) (string, error) {
|
||
|
path, err := filepath.Abs(path)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
path, err = filepath.EvalSymlinks(path)
|
||
|
|
||
|
// Get a better error if we have an invalid path
|
||
|
if pathErr, ok := err.(*os.PathError); ok {
|
||
|
err = errors.Wrap(pathErr.Err, pathErr.Path)
|
||
|
}
|
||
|
|
||
|
return path, err
|
||
|
}
|
||
|
|
||
|
// loggedStat runs os.Stat, but it logs the error if stat returns any error
|
||
|
// other than nil or IsNotExist.
|
||
|
func loggedStat(name string) (os.FileInfo, error) {
|
||
|
info, err := os.Stat(name)
|
||
|
if err != nil && !os.IsNotExist(err) {
|
||
|
log.Print(err)
|
||
|
}
|
||
|
return info, err
|
||
|
}
|
||
|
|
||
|
// loggedLstat runs os.Lstat (doesn't dereference trailing symlink), but it logs
|
||
|
// the error if lstat returns any error other than nil or IsNotExist.
|
||
|
func loggedLstat(name string) (os.FileInfo, error) {
|
||
|
info, err := os.Lstat(name)
|
||
|
if err != nil && !os.IsNotExist(err) {
|
||
|
log.Print(err)
|
||
|
}
|
||
|
return info, err
|
||
|
}
|
||
|
|
||
|
// isDir returns true if the path exists and is that of a directory.
|
||
|
func isDir(path string) bool {
|
||
|
info, err := loggedStat(path)
|
||
|
return err == nil && info.IsDir()
|
||
|
}
|
||
|
|
||
|
// isRegularFile returns true if the path exists and is that of a regular file.
|
||
|
func isRegularFile(path string) bool {
|
||
|
info, err := loggedStat(path)
|
||
|
return err == nil && info.Mode().IsRegular()
|
||
|
}
|
||
|
|
||
|
// HaveReadAccessTo returns true if the process has read access to a file or
|
||
|
// directory, without actually opening it.
|
||
|
func HaveReadAccessTo(path string) bool {
|
||
|
return unix.Access(path, unix.R_OK) == nil
|
||
|
}
|
||
|
|
||
|
// DeviceNumber represents a combined major:minor device number.
|
||
|
type DeviceNumber uint64
|
||
|
|
||
|
func (num DeviceNumber) String() string {
|
||
|
return fmt.Sprintf("%d:%d", unix.Major(uint64(num)), unix.Minor(uint64(num)))
|
||
|
}
|
||
|
|
||
|
func newDeviceNumberFromString(str string) (DeviceNumber, error) {
|
||
|
var major, minor uint32
|
||
|
if count, _ := fmt.Sscanf(str, "%d:%d", &major, &minor); count != 2 {
|
||
|
return 0, errors.Errorf("invalid device number string %q", str)
|
||
|
}
|
||
|
return DeviceNumber(unix.Mkdev(major, minor)), nil
|
||
|
}
|
||
|
|
||
|
// getDeviceNumber returns the device number of the device node at the given
|
||
|
// path. If there is a symlink at the path, it is dereferenced.
|
||
|
func getDeviceNumber(path string) (DeviceNumber, error) {
|
||
|
var stat unix.Stat_t
|
||
|
if err := unix.Stat(path, &stat); err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
return DeviceNumber(stat.Rdev), nil
|
||
|
}
|
||
|
|
||
|
// getNumberOfContainingDevice returns the device number of the filesystem which
|
||
|
// contains the given file. If the file is a symlink, it is not dereferenced.
|
||
|
func getNumberOfContainingDevice(path string) (DeviceNumber, error) {
|
||
|
var stat unix.Stat_t
|
||
|
if err := unix.Lstat(path, &stat); err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
return DeviceNumber(stat.Dev), nil
|
||
|
}
|