mirror of https://github.com/gogits/gogs.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
188 lines
5.1 KiB
188 lines
5.1 KiB
// 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" |
|
) |
|
|
|
// 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. |
|
VSize int |
|
// Resident set size in pages. |
|
RSS int |
|
|
|
fs FS |
|
} |
|
|
|
// NewStat returns the current status information of the process. |
|
func (p Proc) NewStat() (ProcStat, error) { |
|
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 |
|
|
|
s = ProcStat{PID: p.PID, fs: p.fs} |
|
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. |
|
func (s ProcStat) VirtualMemory() int { |
|
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) { |
|
stat, err := s.fs.NewStat() |
|
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 |
|
}
|
|
|