2018-11-05 08:50:00 +05:30
|
|
|
// Copyright 2018 The Prometheus 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 procfs
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2019-07-06 20:46:43 +05:30
|
|
|
|
|
|
|
"github.com/prometheus/procfs/internal/fs"
|
2018-11-05 08:50:00 +05:30
|
|
|
)
|
|
|
|
|
|
|
|
// Originally, this USER_HZ value was dynamically retrieved via a sysconf call
|
|
|
|
// which required cgo. However, that caused a lot of problems regarding
|
|
|
|
// cross-compilation. Alternatives such as running a binary to determine the
|
|
|
|
// value, or trying to derive it in some other way were all problematic. After
|
|
|
|
// much research it was determined that USER_HZ is actually hardcoded to 100 on
|
|
|
|
// all Go-supported platforms as of the time of this writing. This is why we
|
|
|
|
// decided to hardcode it here as well. It is not impossible that there could
|
|
|
|
// be systems with exceptions, but they should be very exotic edge cases, and
|
|
|
|
// in that case, the worst outcome will be two misreported metrics.
|
|
|
|
//
|
|
|
|
// See also the following discussions:
|
|
|
|
//
|
|
|
|
// - https://github.com/prometheus/node_exporter/issues/52
|
|
|
|
// - https://github.com/prometheus/procfs/pull/2
|
|
|
|
// - http://stackoverflow.com/questions/17410841/how-does-user-hz-solve-the-jiffy-scaling-issue
|
|
|
|
const userHZ = 100
|
|
|
|
|
|
|
|
// ProcStat provides status information about the process,
|
|
|
|
// read from /proc/[pid]/stat.
|
|
|
|
type ProcStat struct {
|
|
|
|
// The process ID.
|
|
|
|
PID int
|
|
|
|
// The filename of the executable.
|
|
|
|
Comm string
|
|
|
|
// The process state.
|
|
|
|
State string
|
|
|
|
// The PID of the parent of this process.
|
|
|
|
PPID int
|
|
|
|
// The process group ID of the process.
|
|
|
|
PGRP int
|
|
|
|
// The session ID of the process.
|
|
|
|
Session int
|
|
|
|
// The controlling terminal of the process.
|
|
|
|
TTY int
|
|
|
|
// The ID of the foreground process group of the controlling terminal of
|
|
|
|
// the process.
|
|
|
|
TPGID int
|
|
|
|
// The kernel flags word of the process.
|
|
|
|
Flags uint
|
|
|
|
// The number of minor faults the process has made which have not required
|
|
|
|
// loading a memory page from disk.
|
|
|
|
MinFlt uint
|
|
|
|
// The number of minor faults that the process's waited-for children have
|
|
|
|
// made.
|
|
|
|
CMinFlt uint
|
|
|
|
// The number of major faults the process has made which have required
|
|
|
|
// loading a memory page from disk.
|
|
|
|
MajFlt uint
|
|
|
|
// The number of major faults that the process's waited-for children have
|
|
|
|
// made.
|
|
|
|
CMajFlt uint
|
|
|
|
// Amount of time that this process has been scheduled in user mode,
|
|
|
|
// measured in clock ticks.
|
|
|
|
UTime uint
|
|
|
|
// Amount of time that this process has been scheduled in kernel mode,
|
|
|
|
// measured in clock ticks.
|
|
|
|
STime uint
|
|
|
|
// Amount of time that this process's waited-for children have been
|
|
|
|
// scheduled in user mode, measured in clock ticks.
|
|
|
|
CUTime uint
|
|
|
|
// Amount of time that this process's waited-for children have been
|
|
|
|
// scheduled in kernel mode, measured in clock ticks.
|
|
|
|
CSTime uint
|
|
|
|
// For processes running a real-time scheduling policy, this is the negated
|
|
|
|
// scheduling priority, minus one.
|
|
|
|
Priority int
|
|
|
|
// The nice value, a value in the range 19 (low priority) to -20 (high
|
|
|
|
// priority).
|
|
|
|
Nice int
|
|
|
|
// Number of threads in this process.
|
|
|
|
NumThreads int
|
|
|
|
// The time the process started after system boot, the value is expressed
|
|
|
|
// in clock ticks.
|
|
|
|
Starttime uint64
|
|
|
|
// Virtual memory size in bytes.
|
2019-07-06 20:46:43 +05:30
|
|
|
VSize uint
|
2018-11-05 08:50:00 +05:30
|
|
|
// Resident set size in pages.
|
|
|
|
RSS int
|
|
|
|
|
2019-07-06 20:46:43 +05:30
|
|
|
proc fs.FS
|
2018-11-05 08:50:00 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
// NewStat returns the current status information of the process.
|
2019-08-28 12:25:22 +05:30
|
|
|
//
|
|
|
|
// Deprecated: use p.Stat() instead
|
2018-11-05 08:50:00 +05:30
|
|
|
func (p Proc) NewStat() (ProcStat, error) {
|
2019-08-28 12:25:22 +05:30
|
|
|
return p.Stat()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stat returns the current status information of the process.
|
|
|
|
func (p Proc) Stat() (ProcStat, error) {
|
2018-11-05 08:50:00 +05:30
|
|
|
f, err := os.Open(p.path("stat"))
|
|
|
|
if err != nil {
|
|
|
|
return ProcStat{}, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
data, err := ioutil.ReadAll(f)
|
|
|
|
if err != nil {
|
|
|
|
return ProcStat{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
ignore int
|
|
|
|
|
2019-07-06 20:46:43 +05:30
|
|
|
s = ProcStat{PID: p.PID, proc: p.fs}
|
2018-11-05 08:50:00 +05:30
|
|
|
l = bytes.Index(data, []byte("("))
|
|
|
|
r = bytes.LastIndex(data, []byte(")"))
|
|
|
|
)
|
|
|
|
|
|
|
|
if l < 0 || r < 0 {
|
|
|
|
return ProcStat{}, fmt.Errorf(
|
|
|
|
"unexpected format, couldn't extract comm: %s",
|
|
|
|
data,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Comm = string(data[l+1 : r])
|
|
|
|
_, err = fmt.Fscan(
|
|
|
|
bytes.NewBuffer(data[r+2:]),
|
|
|
|
&s.State,
|
|
|
|
&s.PPID,
|
|
|
|
&s.PGRP,
|
|
|
|
&s.Session,
|
|
|
|
&s.TTY,
|
|
|
|
&s.TPGID,
|
|
|
|
&s.Flags,
|
|
|
|
&s.MinFlt,
|
|
|
|
&s.CMinFlt,
|
|
|
|
&s.MajFlt,
|
|
|
|
&s.CMajFlt,
|
|
|
|
&s.UTime,
|
|
|
|
&s.STime,
|
|
|
|
&s.CUTime,
|
|
|
|
&s.CSTime,
|
|
|
|
&s.Priority,
|
|
|
|
&s.Nice,
|
|
|
|
&s.NumThreads,
|
|
|
|
&ignore,
|
|
|
|
&s.Starttime,
|
|
|
|
&s.VSize,
|
|
|
|
&s.RSS,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return ProcStat{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// VirtualMemory returns the virtual memory size in bytes.
|
2019-07-06 20:46:43 +05:30
|
|
|
func (s ProcStat) VirtualMemory() uint {
|
2018-11-05 08:50:00 +05:30
|
|
|
return s.VSize
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResidentMemory returns the resident memory size in bytes.
|
|
|
|
func (s ProcStat) ResidentMemory() int {
|
|
|
|
return s.RSS * os.Getpagesize()
|
|
|
|
}
|
|
|
|
|
|
|
|
// StartTime returns the unix timestamp of the process in seconds.
|
|
|
|
func (s ProcStat) StartTime() (float64, error) {
|
2019-07-06 20:46:43 +05:30
|
|
|
fs := FS{proc: s.proc}
|
2019-08-28 12:25:22 +05:30
|
|
|
stat, err := fs.Stat()
|
2018-11-05 08:50:00 +05:30
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return float64(stat.BootTime) + (float64(s.Starttime) / userHZ), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CPUTime returns the total CPU user and system time in seconds.
|
|
|
|
func (s ProcStat) CPUTime() float64 {
|
|
|
|
return float64(s.UTime+s.STime) / userHZ
|
|
|
|
}
|