main.go 3.5 KB

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