Browse Source
Currently Node Exporter has a metric called `node_uname_info` which of course exposes uname info. While this is nice, it does not help if you are running different OSes which could have similar uname info. Therefore parse `/etc/os-release` or `/usr/lib/os-release` and expose a `node_os_info` metric which provide information regarding the OS release/version of the node. Also expose the major.minor part of the OS release version as `node_os_version`. Since the os-release files will not change often, cache the parsed content and only refresh the cache if the modification time changes. This `os` collector will read files outside of `/proc` and `/sys`, but the os-release file is widely used and the format is standardized: https://www.freedesktop.org/software/systemd/man/os-release.html Bug: https://github.com/prometheus/node_exporter/issues/1574 Signed-off-by: Benjamin Drung <benjamin.drung@ionos.com>pull/2128/head
Benjamin Drung
3 years ago
committed by
Johannes 'fish' Ziemke
9 changed files with 314 additions and 0 deletions
@ -0,0 +1,12 @@
|
||||
NAME="Ubuntu" |
||||
VERSION="20.04.2 LTS (Focal Fossa)" |
||||
ID=ubuntu |
||||
ID_LIKE=debian |
||||
PRETTY_NAME="Ubuntu 20.04.2 LTS" |
||||
VERSION_ID="20.04" |
||||
HOME_URL="https://www.ubuntu.com/" |
||||
SUPPORT_URL="https://help.ubuntu.com/" |
||||
BUG_REPORT_URL="https://bugs.launchpad.net/ubuntu/" |
||||
PRIVACY_POLICY_URL="https://www.ubuntu.com/legal/terms-and-policies/privacy-policy" |
||||
VERSION_CODENAME=focal |
||||
UBUNTU_CODENAME=focal |
@ -0,0 +1,178 @@
|
||||
// Copyright 2021 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 collector |
||||
|
||||
import ( |
||||
"errors" |
||||
"io" |
||||
"os" |
||||
"regexp" |
||||
"strconv" |
||||
"strings" |
||||
"sync" |
||||
"time" |
||||
|
||||
"github.com/go-kit/log" |
||||
"github.com/go-kit/log/level" |
||||
envparse "github.com/hashicorp/go-envparse" |
||||
"github.com/prometheus/client_golang/prometheus" |
||||
) |
||||
|
||||
const ( |
||||
etcOSRelease = "/etc/os-release" |
||||
usrLibOSRelease = "/usr/lib/os-release" |
||||
) |
||||
|
||||
var ( |
||||
versionRegex = regexp.MustCompile(`^[0-9]+\.?[0-9]*`) |
||||
) |
||||
|
||||
type osRelease struct { |
||||
Name string |
||||
ID string |
||||
IDLike string |
||||
PrettyName string |
||||
Variant string |
||||
VariantID string |
||||
Version string |
||||
VersionID string |
||||
VersionCodename string |
||||
BuildID string |
||||
ImageID string |
||||
ImageVersion string |
||||
} |
||||
|
||||
type osReleaseCollector struct { |
||||
infoDesc *prometheus.Desc |
||||
logger log.Logger |
||||
os *osRelease |
||||
osFilename string // file name of cached release information
|
||||
osMtime time.Time // mtime of cached release file
|
||||
osMutex sync.Mutex |
||||
osReleaseFilenames []string // all os-release file names to check
|
||||
version float64 |
||||
versionDesc *prometheus.Desc |
||||
} |
||||
|
||||
func init() { |
||||
registerCollector("os", defaultEnabled, NewOSCollector) |
||||
} |
||||
|
||||
// NewOSCollector returns a new Collector exposing os-release information.
|
||||
func NewOSCollector(logger log.Logger) (Collector, error) { |
||||
return &osReleaseCollector{ |
||||
logger: logger, |
||||
infoDesc: prometheus.NewDesc( |
||||
prometheus.BuildFQName(namespace, "os", "info"), |
||||
"A metric with a constant '1' value labeled by build_id, id, id_like, image_id, image_version, "+ |
||||
"name, pretty_name, variant, variant_id, version, version_codename, version_id.", |
||||
[]string{"build_id", "id", "id_like", "image_id", "image_version", "name", "pretty_name", |
||||
"variant", "variant_id", "version", "version_codename", "version_id"}, nil, |
||||
), |
||||
osReleaseFilenames: []string{etcOSRelease, usrLibOSRelease}, |
||||
versionDesc: prometheus.NewDesc( |
||||
prometheus.BuildFQName(namespace, "os", "version"), |
||||
"Metric containing the major.minor part of the OS version.", |
||||
[]string{"id", "id_like", "name"}, nil, |
||||
), |
||||
}, nil |
||||
} |
||||
|
||||
func parseOSRelease(r io.Reader) (*osRelease, error) { |
||||
env, err := envparse.Parse(r) |
||||
return &osRelease{ |
||||
Name: env["NAME"], |
||||
ID: env["ID"], |
||||
IDLike: env["ID_LIKE"], |
||||
PrettyName: env["PRETTY_NAME"], |
||||
Variant: env["VARIANT"], |
||||
VariantID: env["VARIANT_ID"], |
||||
Version: env["VERSION"], |
||||
VersionID: env["VERSION_ID"], |
||||
VersionCodename: env["VERSION_CODENAME"], |
||||
BuildID: env["BUILD_ID"], |
||||
ImageID: env["IMAGE_ID"], |
||||
ImageVersion: env["IMAGE_VERSION"], |
||||
}, err |
||||
} |
||||
|
||||
func (c *osReleaseCollector) UpdateStruct(path string) error { |
||||
releaseFile, err := os.Open(path) |
||||
if err != nil { |
||||
return err |
||||
} |
||||
defer releaseFile.Close() |
||||
|
||||
stat, err := releaseFile.Stat() |
||||
if err != nil { |
||||
return err |
||||
} |
||||
|
||||
t := stat.ModTime() |
||||
if path == c.osFilename && t == c.osMtime { |
||||
// osReleaseCollector struct is already up-to-date.
|
||||
return nil |
||||
} |
||||
|
||||
// Acquire a lock to update the osReleaseCollector struct.
|
||||
c.osMutex.Lock() |
||||
defer c.osMutex.Unlock() |
||||
|
||||
level.Debug(c.logger).Log("msg", "file modification time has changed", |
||||
"file", path, "old_value", c.osMtime, "new_value", t) |
||||
c.osFilename = path |
||||
c.osMtime = t |
||||
|
||||
c.os, err = parseOSRelease(releaseFile) |
||||
if err != nil { |
||||
return err |
||||
} |
||||
|
||||
majorMinor := versionRegex.FindString(c.os.VersionID) |
||||
if majorMinor != "" { |
||||
c.version, err = strconv.ParseFloat(majorMinor, 64) |
||||
if err != nil { |
||||
return err |
||||
} |
||||
} else { |
||||
c.version = 0 |
||||
} |
||||
return nil |
||||
} |
||||
|
||||
func (c *osReleaseCollector) Update(ch chan<- prometheus.Metric) error { |
||||
for i, path := range c.osReleaseFilenames { |
||||
err := c.UpdateStruct(*rootfsPath + path) |
||||
if err == nil { |
||||
break |
||||
} |
||||
if errors.Is(err, os.ErrNotExist) { |
||||
if i >= (len(c.osReleaseFilenames) - 1) { |
||||
level.Debug(c.logger).Log("msg", "no os-release file found", "files", strings.Join(c.osReleaseFilenames, ",")) |
||||
return ErrNoData |
||||
} |
||||
continue |
||||
} |
||||
return err |
||||
} |
||||
|
||||
ch <- prometheus.MustNewConstMetric(c.infoDesc, prometheus.GaugeValue, 1.0, |
||||
c.os.BuildID, c.os.ID, c.os.IDLike, c.os.ImageID, c.os.ImageVersion, c.os.Name, c.os.PrettyName, |
||||
c.os.Variant, c.os.VariantID, c.os.Version, c.os.VersionCodename, c.os.VersionID) |
||||
if c.version > 0 { |
||||
ch <- prometheus.MustNewConstMetric(c.versionDesc, prometheus.GaugeValue, c.version, |
||||
c.os.ID, c.os.IDLike, c.os.Name) |
||||
} |
||||
return nil |
||||
} |
@ -0,0 +1,105 @@
|
||||
// Copyright 2021 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 collector |
||||
|
||||
import ( |
||||
"os" |
||||
"reflect" |
||||
"strings" |
||||
"testing" |
||||
|
||||
"github.com/go-kit/log" |
||||
) |
||||
|
||||
const debianBullseye string = `PRETTY_NAME="Debian GNU/Linux 11 (bullseye)" |
||||
NAME="Debian GNU/Linux" |
||||
VERSION_ID="11" |
||||
VERSION="11 (bullseye)" |
||||
VERSION_CODENAME=bullseye |
||||
ID=debian |
||||
HOME_URL="https://www.debian.org/" |
||||
SUPPORT_URL="https://www.debian.org/support" |
||||
BUG_REPORT_URL="https://bugs.debian.org/" |
||||
` |
||||
|
||||
func TestParseOSRelease(t *testing.T) { |
||||
want := &osRelease{ |
||||
Name: "Ubuntu", |
||||
ID: "ubuntu", |
||||
IDLike: "debian", |
||||
PrettyName: "Ubuntu 20.04.2 LTS", |
||||
Version: "20.04.2 LTS (Focal Fossa)", |
||||
VersionID: "20.04", |
||||
VersionCodename: "focal", |
||||
} |
||||
|
||||
osReleaseFile, err := os.Open("fixtures" + usrLibOSRelease) |
||||
if err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
got, err := parseOSRelease(osReleaseFile) |
||||
if err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
if !reflect.DeepEqual(want, got) { |
||||
t.Fatalf("should have %+v osRelease: got %+v", want, got) |
||||
} |
||||
|
||||
want = &osRelease{ |
||||
Name: "Debian GNU/Linux", |
||||
ID: "debian", |
||||
PrettyName: "Debian GNU/Linux 11 (bullseye)", |
||||
Version: "11 (bullseye)", |
||||
VersionID: "11", |
||||
VersionCodename: "bullseye", |
||||
} |
||||
got, err = parseOSRelease(strings.NewReader(debianBullseye)) |
||||
if err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
if !reflect.DeepEqual(want, got) { |
||||
t.Fatalf("should have %+v osRelease: got %+v", want, got) |
||||
} |
||||
} |
||||
|
||||
func TestUpdateStruct(t *testing.T) { |
||||
wantedOS := &osRelease{ |
||||
Name: "Ubuntu", |
||||
ID: "ubuntu", |
||||
IDLike: "debian", |
||||
PrettyName: "Ubuntu 20.04.2 LTS", |
||||
Version: "20.04.2 LTS (Focal Fossa)", |
||||
VersionID: "20.04", |
||||
VersionCodename: "focal", |
||||
} |
||||
wantedVersion := 20.04 |
||||
|
||||
collector, err := NewOSCollector(log.NewNopLogger()) |
||||
if err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
c := collector.(*osReleaseCollector) |
||||
|
||||
err = c.UpdateStruct("fixtures" + usrLibOSRelease) |
||||
if err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
|
||||
if !reflect.DeepEqual(wantedOS, c.os) { |
||||
t.Fatalf("should have %+v osRelease: got %+v", wantedOS, c.os) |
||||
} |
||||
if wantedVersion != c.version { |
||||
t.Errorf("Expected '%v' but got '%v'", wantedVersion, c.version) |
||||
} |
||||
} |
Loading…
Reference in new issue