From c90e68809d715b23bfc6d9c3facdec2e370dc54a Mon Sep 17 00:00:00 2001 From: Pavel Kutishchev Date: Wed, 23 May 2018 14:30:28 +0200 Subject: [PATCH 1/3] Add processes exporter Signed-off-by: Pavel Kutishchev Signed-off-by: Ben Kochie --- collector/fixtures/e2e-output.txt | 16 +++ collector/fixtures/proc/10/stat | 1 + collector/fixtures/proc/sys/kernel/pid_max | 1 + .../fixtures/proc/sys/kernel/threads-max | 1 + collector/fixtures/proc/sys/pid_max | 1 + collector/fixtures/proc/sys/threads-max | 1 + collector/processes_linux.go | 109 ++++++++++++++++++ collector/processes_linux_test.go | 42 +++++++ end-to-end-test.sh | 1 + 9 files changed, 173 insertions(+) create mode 100644 collector/fixtures/proc/10/stat create mode 100644 collector/fixtures/proc/sys/kernel/pid_max create mode 100644 collector/fixtures/proc/sys/kernel/threads-max create mode 100644 collector/fixtures/proc/sys/pid_max create mode 100644 collector/fixtures/proc/sys/threads-max create mode 100644 collector/processes_linux.go create mode 100644 collector/processes_linux_test.go diff --git a/collector/fixtures/e2e-output.txt b/collector/fixtures/e2e-output.txt index 4eb1dd0e21..218aaf56a1 100644 --- a/collector/fixtures/e2e-output.txt +++ b/collector/fixtures/e2e-output.txt @@ -2094,6 +2094,21 @@ node_nfsd_server_rpcs_total 18628 # HELP node_nfsd_server_threads Total number of NFSd kernel threads that are running. # TYPE node_nfsd_server_threads gauge node_nfsd_server_threads 8 +# HELP node_processes_max_processes Number of max PIDs limit +# TYPE node_processes_max_processes gauge +node_processes_max_processes 123 +# HELP node_processes_max_threads Limit of threads in the system +# TYPE node_processes_max_threads gauge +node_processes_max_threads 7801 +# HELP node_processes_pids Number of PIDs +# TYPE node_processes_pids gauge +node_processes_pids 1 +# HELP node_processes_state Number of processes in each state. +# TYPE node_processes_state gauge +node_processes_state{state="S"} 1 +# HELP node_processes_threads Allocated threads in system +# TYPE node_processes_threads gauge +node_processes_threads 1 # HELP node_procs_blocked Number of processes blocked waiting for I/O to complete. # TYPE node_procs_blocked gauge node_procs_blocked 0 @@ -2149,6 +2164,7 @@ node_scrape_collector_success{collector="netdev"} 1 node_scrape_collector_success{collector="netstat"} 1 node_scrape_collector_success{collector="nfs"} 1 node_scrape_collector_success{collector="nfsd"} 1 +node_scrape_collector_success{collector="processes"} 1 node_scrape_collector_success{collector="qdisc"} 1 node_scrape_collector_success{collector="sockstat"} 1 node_scrape_collector_success{collector="stat"} 1 diff --git a/collector/fixtures/proc/10/stat b/collector/fixtures/proc/10/stat new file mode 100644 index 0000000000..1451c8a44e --- /dev/null +++ b/collector/fixtures/proc/10/stat @@ -0,0 +1 @@ +17 (khungtaskd) S 2 0 0 0 -1 2129984 0 0 0 0 14 0 0 0 20 0 1 0 24 0 0 18446744073709551615 0 0 0 0 0 0 0 2147483647 0 0 0 0 17 0 0 0 0 0 0 0 0 0 0 0 0 0 0 \ No newline at end of file diff --git a/collector/fixtures/proc/sys/kernel/pid_max b/collector/fixtures/proc/sys/kernel/pid_max new file mode 100644 index 0000000000..190a18037c --- /dev/null +++ b/collector/fixtures/proc/sys/kernel/pid_max @@ -0,0 +1 @@ +123 diff --git a/collector/fixtures/proc/sys/kernel/threads-max b/collector/fixtures/proc/sys/kernel/threads-max new file mode 100644 index 0000000000..0ccbf457eb --- /dev/null +++ b/collector/fixtures/proc/sys/kernel/threads-max @@ -0,0 +1 @@ +7801 \ No newline at end of file diff --git a/collector/fixtures/proc/sys/pid_max b/collector/fixtures/proc/sys/pid_max new file mode 100644 index 0000000000..190a18037c --- /dev/null +++ b/collector/fixtures/proc/sys/pid_max @@ -0,0 +1 @@ +123 diff --git a/collector/fixtures/proc/sys/threads-max b/collector/fixtures/proc/sys/threads-max new file mode 100644 index 0000000000..0ccbf457eb --- /dev/null +++ b/collector/fixtures/proc/sys/threads-max @@ -0,0 +1 @@ +7801 \ No newline at end of file diff --git a/collector/processes_linux.go b/collector/processes_linux.go new file mode 100644 index 0000000000..fd6ddb1f86 --- /dev/null +++ b/collector/processes_linux.go @@ -0,0 +1,109 @@ +// 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. + +// +build !noprocesses + +package collector + +import ( + "fmt" + "github.com/prometheus/client_golang/prometheus" + "github.com/prometheus/procfs" +) + +type processCollector struct { + threadAlloc *prometheus.Desc + threadLimit *prometheus.Desc + procsState *prometheus.Desc + pidUsed *prometheus.Desc + pidMax *prometheus.Desc +} + +func init() { + registerCollector("processes", defaultDisabled, NewProcessStatCollector) +} + +func NewProcessStatCollector() (Collector, error) { + subsystem := "processes" + return &processCollector{ + threadAlloc: prometheus.NewDesc( + prometheus.BuildFQName(namespace, subsystem, "threads"), + "Allocated threads in system", + nil, nil, + ), + threadLimit: prometheus.NewDesc( + prometheus.BuildFQName(namespace, subsystem, "max_threads"), + "Limit of threads in the system", + nil, nil, + ), + procsState: prometheus.NewDesc( + prometheus.BuildFQName(namespace, subsystem, "state"), + "Number of processes in each state.", + []string{"state"}, nil, + ), + pidUsed: prometheus.NewDesc(prometheus.BuildFQName(namespace, subsystem, "pids"), + "Number of PIDs", nil, nil, + ), + pidMax: prometheus.NewDesc(prometheus.BuildFQName(namespace, subsystem, "max_processes"), + "Number of max PIDs limit", nil, nil, + ), + }, nil +} +func (t *processCollector) Update(ch chan<- prometheus.Metric) error { + pids, states, threads, err := getAllocatedThreads() + if err != nil { + return fmt.Errorf("Unable to retrieve number of allocated threads %v\n", err) + } + + ch <- prometheus.MustNewConstMetric(t.threadAlloc, prometheus.GaugeValue, float64(threads)) + maxThreads, err := readUintFromFile(procFilePath("sys/kernel/threads-max")) + if err != nil { + return fmt.Errorf("Unable to retrieve limit number of threads %v\n", err) + } + ch <- prometheus.MustNewConstMetric(t.threadLimit, prometheus.GaugeValue, float64(maxThreads)) + + for state := range states { + ch <- prometheus.MustNewConstMetric(t.procsState, prometheus.GaugeValue, float64(states[state]), state) + } + + pidM, err := readUintFromFile(procFilePath("sys/kernel/pid_max")) + if err != nil { + return fmt.Errorf("Unable to retrieve limit number of maximum pids alloved %v\n", err) + } + ch <- prometheus.MustNewConstMetric(t.pidUsed, prometheus.GaugeValue, float64(pids)) + ch <- prometheus.MustNewConstMetric(t.pidMax, prometheus.GaugeValue, float64(pidM)) + + return nil +} + +func getAllocatedThreads() (int, map[string]int32, int, error) { + fs, err := procfs.NewFS(*procPath) + if err != nil { + return 0, nil, 0, err + } + p, err := fs.AllProcs() + if err != nil { + return 0, nil, 0, err + } + thread := 0 + procStates := make(map[string]int32) + for _, pid := range p { + stat, err := pid.NewStat() + if err != nil { + return 0, nil, 0, err + } + procStates[stat.State] += 1 + thread += stat.NumThreads + } + return len(p), procStates, thread, nil +} diff --git a/collector/processes_linux_test.go b/collector/processes_linux_test.go new file mode 100644 index 0000000000..bd1151a834 --- /dev/null +++ b/collector/processes_linux_test.go @@ -0,0 +1,42 @@ +// 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. + +// +build !noprocesses + +package collector + +import ( + "testing" +) + +func TestReadProcessStatus(t *testing.T) { + want := 1 + pids, states, threads, err := getAllocatedThreads() + if err != nil { + t.Fatalf("Cannot retrieve data from procfs getAllocatedThreads function: %v ", err) + } + if threads < want { + t.Fatalf("Current threads: %d Shouldn't be less than wanted %d", threads, want) + } + if states == nil { + + t.Fatalf("Process states cannot be nil %v:", states) + } + maxPid, err := readUintFromFile(procFilePath("sys/kernel/pid_max")) + if err != nil { + t.Fatalf("Unable to retrieve limit number of maximum pids alloved %v\n", err) + } + if uint64(pids) > maxPid || pids == 0 { + t.Fatalf("Total running pids cannot be greater than %d or equals to 0", maxPid) + } +} diff --git a/end-to-end-test.sh b/end-to-end-test.sh index ec99cae5a9..469d2441d7 100755 --- a/end-to-end-test.sh +++ b/end-to-end-test.sh @@ -36,6 +36,7 @@ enabled_collectors=$(cat << COLLECTORS wifi xfs zfs + processes COLLECTORS ) disabled_collectors=$(cat << COLLECTORS From ce8dd9934f6a2306219d65386b410f6925fedf70 Mon Sep 17 00:00:00 2001 From: Ben Kochie Date: Wed, 23 May 2018 17:43:38 +0200 Subject: [PATCH 2/3] Add procPath override to test. Signed-off-by: Ben Kochie --- collector/processes_linux_test.go | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/collector/processes_linux_test.go b/collector/processes_linux_test.go index bd1151a834..1cf9e4f0d5 100644 --- a/collector/processes_linux_test.go +++ b/collector/processes_linux_test.go @@ -17,9 +17,14 @@ package collector import ( "testing" + + "gopkg.in/alecthomas/kingpin.v2" ) func TestReadProcessStatus(t *testing.T) { + if _, err := kingpin.CommandLine.Parse([]string{"--path.procfs", "fixtures/proc"}); err != nil { + t.Fatal(err) + } want := 1 pids, states, threads, err := getAllocatedThreads() if err != nil { From c5d4ab874a8a743bc52213db25bdbc2880dfd249 Mon Sep 17 00:00:00 2001 From: Ben Kochie Date: Wed, 23 May 2018 17:45:42 +0200 Subject: [PATCH 3/3] Update 64k page e2e fixture. Signed-off-by: Ben Kochie --- collector/fixtures/e2e-64k-page-output.txt | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/collector/fixtures/e2e-64k-page-output.txt b/collector/fixtures/e2e-64k-page-output.txt index 3015262370..f2f660f851 100644 --- a/collector/fixtures/e2e-64k-page-output.txt +++ b/collector/fixtures/e2e-64k-page-output.txt @@ -2094,6 +2094,21 @@ node_nfsd_server_rpcs_total 18628 # HELP node_nfsd_server_threads Total number of NFSd kernel threads that are running. # TYPE node_nfsd_server_threads gauge node_nfsd_server_threads 8 +# HELP node_processes_max_processes Number of max PIDs limit +# TYPE node_processes_max_processes gauge +node_processes_max_processes 123 +# HELP node_processes_max_threads Limit of threads in the system +# TYPE node_processes_max_threads gauge +node_processes_max_threads 7801 +# HELP node_processes_pids Number of PIDs +# TYPE node_processes_pids gauge +node_processes_pids 1 +# HELP node_processes_state Number of processes in each state. +# TYPE node_processes_state gauge +node_processes_state{state="S"} 1 +# HELP node_processes_threads Allocated threads in system +# TYPE node_processes_threads gauge +node_processes_threads 1 # HELP node_procs_blocked Number of processes blocked waiting for I/O to complete. # TYPE node_procs_blocked gauge node_procs_blocked 0 @@ -2149,6 +2164,7 @@ node_scrape_collector_success{collector="netdev"} 1 node_scrape_collector_success{collector="netstat"} 1 node_scrape_collector_success{collector="nfs"} 1 node_scrape_collector_success{collector="nfsd"} 1 +node_scrape_collector_success{collector="processes"} 1 node_scrape_collector_success{collector="qdisc"} 1 node_scrape_collector_success{collector="sockstat"} 1 node_scrape_collector_success{collector="stat"} 1