main.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. package main
  2. import (
  3. "bytes"
  4. "github.com/coroot/coroot-node-agent/common"
  5. "github.com/coroot/coroot-node-agent/containers"
  6. "github.com/coroot/coroot-node-agent/flags"
  7. "github.com/coroot/coroot-node-agent/node"
  8. "github.com/coroot/coroot-node-agent/tracing"
  9. "github.com/prometheus/client_golang/prometheus"
  10. "github.com/prometheus/client_golang/prometheus/promhttp"
  11. "golang.org/x/mod/semver"
  12. "golang.org/x/sys/unix"
  13. "k8s.io/klog/v2"
  14. "net/http"
  15. _ "net/http/pprof"
  16. "os"
  17. "path"
  18. "runtime"
  19. "strings"
  20. )
  21. var (
  22. version = "unknown"
  23. )
  24. const minSupportedKernelVersion = "4.16"
  25. func uname() (string, string, error) {
  26. runtime.LockOSThread()
  27. defer runtime.UnlockOSThread()
  28. f, err := os.Open("/proc/1/ns/uts")
  29. if err != nil {
  30. return "", "", err
  31. }
  32. defer f.Close()
  33. self, err := os.Open("/proc/self/ns/uts")
  34. if err != nil {
  35. return "", "", err
  36. }
  37. defer self.Close()
  38. defer func() {
  39. unix.Setns(int(self.Fd()), unix.CLONE_NEWUTS)
  40. }()
  41. err = unix.Setns(int(f.Fd()), unix.CLONE_NEWUTS)
  42. if err != nil {
  43. return "", "", err
  44. }
  45. var utsname unix.Utsname
  46. if err := unix.Uname(&utsname); err != nil {
  47. return "", "", err
  48. }
  49. hostname := string(bytes.Split(utsname.Nodename[:], []byte{0})[0])
  50. kernelVersion := string(bytes.Split(utsname.Release[:], []byte{0})[0])
  51. return hostname, kernelVersion, nil
  52. }
  53. func machineID() string {
  54. for _, p := range []string{"sys/devices/virtual/dmi/id/product_uuid", "etc/machine-id", "var/lib/dbus/machine-id"} {
  55. payload, err := os.ReadFile(path.Join("/proc/1/root", p))
  56. if err != nil {
  57. klog.Warningln("failed to read machine-id:", err)
  58. continue
  59. }
  60. id := strings.TrimSpace(strings.Replace(string(payload), "-", "", -1))
  61. klog.Infoln("machine-id: ", id)
  62. return id
  63. }
  64. return ""
  65. }
  66. func main() {
  67. klog.Infoln("agent version:", version)
  68. hostname, kv, err := uname()
  69. if err != nil {
  70. klog.Exitln("failed to get uname:", err)
  71. }
  72. klog.Infoln("hostname:", hostname)
  73. klog.Infoln("kernel version:", kv)
  74. ver := common.KernelMajorMinor(kv)
  75. if ver == "" {
  76. klog.Exitln("invalid kernel version:", kv)
  77. }
  78. if semver.Compare("v"+ver, "v"+minSupportedKernelVersion) == -1 {
  79. klog.Exitf("the minimum Linux kernel version required is %s or later", minSupportedKernelVersion)
  80. }
  81. machineId := machineID()
  82. tracing.Init(machineId, hostname, version)
  83. registry := prometheus.NewRegistry()
  84. registerer := prometheus.WrapRegistererWith(prometheus.Labels{"machine_id": machineId}, registry)
  85. registerer.MustRegister(info("node_agent_info", version))
  86. if err := registerer.Register(node.NewCollector(hostname, kv)); err != nil {
  87. klog.Exitln(err)
  88. }
  89. cs, err := containers.NewRegistry(registerer, kv)
  90. if err != nil {
  91. klog.Exitln(err)
  92. }
  93. defer cs.Close()
  94. http.Handle("/metrics", promhttp.HandlerFor(registry, promhttp.HandlerOpts{ErrorLog: logger{}, Registry: registerer}))
  95. klog.Infoln("listening on:", *flags.ListenAddress)
  96. klog.Errorln(http.ListenAndServe(*flags.ListenAddress, nil))
  97. }
  98. func info(name, version string) prometheus.Collector {
  99. g := prometheus.NewGauge(prometheus.GaugeOpts{
  100. Name: name,
  101. ConstLabels: prometheus.Labels{"version": version},
  102. })
  103. g.Set(1)
  104. return g
  105. }
  106. type logger struct{}
  107. func (l logger) Println(v ...interface{}) {
  108. klog.Errorln(v...)
  109. }