123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118 |
- // +build linux darwin
- /*
- Copyright 2014 The Kubernetes Authors.
- 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 fs
- import (
- "bytes"
- "fmt"
- "os/exec"
- "strings"
- "golang.org/x/sys/unix"
- "k8s.io/apimachinery/pkg/api/resource"
- "k8s.io/kubernetes/pkg/volume/util/fsquota"
- )
- // FSInfo linux returns (available bytes, byte capacity, byte usage, total inodes, inodes free, inode usage, error)
- // for the filesystem that path resides upon.
- func FsInfo(path string) (int64, int64, int64, int64, int64, int64, error) {
- statfs := &unix.Statfs_t{}
- err := unix.Statfs(path, statfs)
- if err != nil {
- return 0, 0, 0, 0, 0, 0, err
- }
- // Available is blocks available * fragment size
- available := int64(statfs.Bavail) * int64(statfs.Bsize)
- // Capacity is total block count * fragment size
- capacity := int64(statfs.Blocks) * int64(statfs.Bsize)
- // Usage is block being used * fragment size (aka block size).
- usage := (int64(statfs.Blocks) - int64(statfs.Bfree)) * int64(statfs.Bsize)
- inodes := int64(statfs.Files)
- inodesFree := int64(statfs.Ffree)
- inodesUsed := inodes - inodesFree
- return available, capacity, usage, inodes, inodesFree, inodesUsed, nil
- }
- // DiskUsage gets disk usage of specified path.
- func DiskUsage(path string) (*resource.Quantity, error) {
- // First check whether the quota system knows about this directory
- // A nil quantity with no error means that the path does not support quotas
- // and we should use other mechanisms.
- data, err := fsquota.GetConsumption(path)
- if data != nil {
- return data, nil
- } else if err != nil {
- return nil, fmt.Errorf("unable to retrieve disk consumption via quota for %s: %v", path, err)
- }
- // Uses the same niceness level as cadvisor.fs does when running du
- // Uses -B 1 to always scale to a blocksize of 1 byte
- out, err := exec.Command("nice", "-n", "19", "du", "-x", "-s", "-B", "1", path).CombinedOutput()
- if err != nil {
- return nil, fmt.Errorf("failed command 'du' ($ nice -n 19 du -x -s -B 1) on path %s with error %v", path, err)
- }
- used, err := resource.ParseQuantity(strings.Fields(string(out))[0])
- if err != nil {
- return nil, fmt.Errorf("failed to parse 'du' output %s due to error %v", out, err)
- }
- used.Format = resource.BinarySI
- return &used, nil
- }
- // Find uses the equivalent of the command `find <path> -dev -printf '.' | wc -c` to count files and directories.
- // While this is not an exact measure of inodes used, it is a very good approximation.
- func Find(path string) (int64, error) {
- if path == "" {
- return 0, fmt.Errorf("invalid directory")
- }
- // First check whether the quota system knows about this directory
- // A nil quantity with no error means that the path does not support quotas
- // and we should use other mechanisms.
- inodes, err := fsquota.GetInodes(path)
- if inodes != nil {
- return inodes.Value(), nil
- } else if err != nil {
- return 0, fmt.Errorf("unable to retrieve inode consumption via quota for %s: %v", path, err)
- }
- var counter byteCounter
- var stderr bytes.Buffer
- findCmd := exec.Command("find", path, "-xdev", "-printf", ".")
- findCmd.Stdout, findCmd.Stderr = &counter, &stderr
- if err := findCmd.Start(); err != nil {
- return 0, fmt.Errorf("failed to exec cmd %v - %v; stderr: %v", findCmd.Args, err, stderr.String())
- }
- if err := findCmd.Wait(); err != nil {
- return 0, fmt.Errorf("cmd %v failed. stderr: %s; err: %v", findCmd.Args, stderr.String(), err)
- }
- return counter.bytesWritten, nil
- }
- // Simple io.Writer implementation that counts how many bytes were written.
- type byteCounter struct{ bytesWritten int64 }
- func (b *byteCounter) Write(p []byte) (int, error) {
- b.bytesWritten += int64(len(p))
- return len(p), nil
- }
|