main.go 4.0 KB

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