2019-06-14 13:26:55 +03:00
|
|
|
// Copyright 2019 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.
|
|
|
|
|
process_collector: fill in most statistics on macOS (#1600)
* process_collector: fill in most statistics on macOS
Unfortunately, the virtual memory, resident memory, and network stats will
require access to undocumented C functions. I was warned off of cgo in IRC
because it would then have to be enabled in a bunch of different projects that
use this module, but I already was against it because that would break the
ability to cross-compile. There is no interface to `dlopen` built into golang.
The `github.com/ebitengine/purego` module looks promising (I can cross-compile
and call these methods), but I'm currently getting unexpected results. I'll
follow up with that separately if I can get it working, but hopefully this stuff
is pretty uncontroversial.
Tested on macOS 10.14.6 (amd64), macOS 14.6.1 (amd64), and macOS 15.0 (arm64)
by spawning `/usr/bin/ulimit -a -S` and `/usr/sbin/lsof -c $my_process` from
the test exporter process, and `ps -o lstart,vsize,rss,utime,stime,command` from
the shell, and comparing results with the exported metrics.
I can't find documentation for `RLIMIT_AS` on macOS (specifically if it's in
bytes or pages). It's currently being reported back as `RLIM_INFINITY`, which
seems reasonable, because I've come across reports that the value is ignored
anyway[1]. The bash 3.2 code for the built-in `ulimit` divides the value
reported by `getrusage(2)` by 1024 when printing, as it does for `RLIMIT_DATA`,
which is documented as being bytes in `getrusage(2)`. The help for `ulimit`
indicates it prints both in kbytes, so it's reasonable to assume this is already
in bytes.
[1] https://issues.chromium.org/issues/40581251#comment3
Signed-off-by: Matt Harbison <mharbison72@gmail.com>
* Update prometheus/process_collector_darwin.go
Co-authored-by: Ben Kochie <superq@gmail.com>
Signed-off-by: Matt Harbison <57785103+mharbison72@users.noreply.github.com>
---------
Signed-off-by: Matt Harbison <mharbison72@gmail.com>
Signed-off-by: Matt Harbison <57785103+mharbison72@users.noreply.github.com>
Co-authored-by: Ben Kochie <superq@gmail.com>
Co-authored-by: Bartlomiej Plotka <bwplotka@gmail.com>
2024-09-04 17:56:01 +03:00
|
|
|
//go:build !windows && !js && !wasip1 && !darwin
|
|
|
|
// +build !windows,!js,!wasip1,!darwin
|
2019-06-14 13:26:55 +03:00
|
|
|
|
|
|
|
package prometheus
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/prometheus/procfs"
|
|
|
|
)
|
|
|
|
|
|
|
|
func canCollectProcess() bool {
|
|
|
|
_, err := procfs.NewDefaultFS()
|
|
|
|
return err == nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *processCollector) processCollect(ch chan<- Metric) {
|
|
|
|
pid, err := c.pidFn()
|
|
|
|
if err != nil {
|
|
|
|
c.reportError(ch, nil, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := procfs.NewProc(pid)
|
|
|
|
if err != nil {
|
|
|
|
c.reportError(ch, nil, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if stat, err := p.Stat(); err == nil {
|
|
|
|
ch <- MustNewConstMetric(c.cpuTotal, CounterValue, stat.CPUTime())
|
|
|
|
ch <- MustNewConstMetric(c.vsize, GaugeValue, float64(stat.VirtualMemory()))
|
|
|
|
ch <- MustNewConstMetric(c.rss, GaugeValue, float64(stat.ResidentMemory()))
|
|
|
|
if startTime, err := stat.StartTime(); err == nil {
|
|
|
|
ch <- MustNewConstMetric(c.startTime, GaugeValue, startTime)
|
|
|
|
} else {
|
|
|
|
c.reportError(ch, c.startTime, err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
c.reportError(ch, nil, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if fds, err := p.FileDescriptorsLen(); err == nil {
|
|
|
|
ch <- MustNewConstMetric(c.openFDs, GaugeValue, float64(fds))
|
|
|
|
} else {
|
|
|
|
c.reportError(ch, c.openFDs, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if limits, err := p.Limits(); err == nil {
|
|
|
|
ch <- MustNewConstMetric(c.maxFDs, GaugeValue, float64(limits.OpenFiles))
|
|
|
|
ch <- MustNewConstMetric(c.maxVsize, GaugeValue, float64(limits.AddressSpace))
|
|
|
|
} else {
|
|
|
|
c.reportError(ch, nil, err)
|
|
|
|
}
|
2024-07-17 13:19:40 +03:00
|
|
|
|
|
|
|
if netstat, err := p.Netstat(); err == nil {
|
|
|
|
var inOctets, outOctets float64
|
|
|
|
if netstat.IpExt.InOctets != nil {
|
|
|
|
inOctets = *netstat.IpExt.InOctets
|
|
|
|
}
|
|
|
|
if netstat.IpExt.OutOctets != nil {
|
|
|
|
outOctets = *netstat.IpExt.OutOctets
|
|
|
|
}
|
|
|
|
ch <- MustNewConstMetric(c.inBytes, CounterValue, inOctets)
|
|
|
|
ch <- MustNewConstMetric(c.outBytes, CounterValue, outOctets)
|
|
|
|
} else {
|
|
|
|
c.reportError(ch, nil, err)
|
|
|
|
}
|
2019-06-14 13:26:55 +03:00
|
|
|
}
|
2024-10-07 13:08:32 +03:00
|
|
|
|
|
|
|
// describe returns all descriptions of the collector for others than windows, js, wasip1 and darwin.
|
|
|
|
// Ensure that this list of descriptors is kept in sync with the metrics collected
|
|
|
|
// in the processCollect method. Any changes to the metrics in processCollect
|
|
|
|
// (such as adding or removing metrics) should be reflected in this list of descriptors.
|
|
|
|
func (c *processCollector) describe(ch chan<- *Desc) {
|
|
|
|
ch <- c.cpuTotal
|
|
|
|
ch <- c.openFDs
|
|
|
|
ch <- c.maxFDs
|
|
|
|
ch <- c.vsize
|
|
|
|
ch <- c.maxVsize
|
|
|
|
ch <- c.rss
|
|
|
|
ch <- c.startTime
|
|
|
|
ch <- c.inBytes
|
|
|
|
ch <- c.outBytes
|
|
|
|
}
|