kubemetrics.go 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. package metrics
  2. import (
  3. "fmt"
  4. "strings"
  5. "sync"
  6. "github.com/opencost/opencost/pkg/clustercache"
  7. "github.com/opencost/opencost/pkg/prom"
  8. "github.com/prometheus/client_golang/prometheus"
  9. batchv1 "k8s.io/api/batch/v1"
  10. v1 "k8s.io/api/core/v1"
  11. "k8s.io/apimachinery/pkg/api/resource"
  12. "k8s.io/apimachinery/pkg/util/validation"
  13. )
  14. //--------------------------------------------------------------------------
  15. // Kube Metric Registration
  16. //--------------------------------------------------------------------------
  17. // initializer
  18. var kubeMetricInit sync.Once
  19. // KubeMetricsOpts represents our Kubernetes metrics emission options.
  20. type KubeMetricsOpts struct {
  21. EmitKubecostControllerMetrics bool
  22. EmitNamespaceAnnotations bool
  23. EmitPodAnnotations bool
  24. EmitKubeStateMetrics bool
  25. EmitKubeStateMetricsV1Only bool
  26. }
  27. // DefaultKubeMetricsOpts returns KubeMetricsOpts with default values set
  28. func DefaultKubeMetricsOpts() *KubeMetricsOpts {
  29. return &KubeMetricsOpts{
  30. EmitKubecostControllerMetrics: true,
  31. EmitNamespaceAnnotations: false,
  32. EmitPodAnnotations: false,
  33. EmitKubeStateMetrics: true,
  34. EmitKubeStateMetricsV1Only: false,
  35. }
  36. }
  37. // InitKubeMetrics initializes kubernetes metric emission using the provided options.
  38. func InitKubeMetrics(clusterCache clustercache.ClusterCache, metricsConfig *MetricsConfig, opts *KubeMetricsOpts) {
  39. if opts == nil {
  40. opts = DefaultKubeMetricsOpts()
  41. }
  42. kubeMetricInit.Do(func() {
  43. if opts.EmitKubecostControllerMetrics {
  44. prometheus.MustRegister(KubecostServiceCollector{
  45. KubeClusterCache: clusterCache,
  46. metricsConfig: *metricsConfig,
  47. })
  48. prometheus.MustRegister(KubecostDeploymentCollector{
  49. KubeClusterCache: clusterCache,
  50. metricsConfig: *metricsConfig,
  51. })
  52. prometheus.MustRegister(KubecostStatefulsetCollector{
  53. KubeClusterCache: clusterCache,
  54. metricsConfig: *metricsConfig,
  55. })
  56. }
  57. if opts.EmitPodAnnotations {
  58. prometheus.MustRegister(KubecostPodCollector{
  59. KubeClusterCache: clusterCache,
  60. metricsConfig: *metricsConfig,
  61. })
  62. }
  63. if opts.EmitNamespaceAnnotations {
  64. prometheus.MustRegister(KubecostNamespaceCollector{
  65. KubeClusterCache: clusterCache,
  66. metricsConfig: *metricsConfig,
  67. })
  68. }
  69. if opts.EmitKubeStateMetrics {
  70. prometheus.MustRegister(KubeNodeCollector{
  71. KubeClusterCache: clusterCache,
  72. metricsConfig: *metricsConfig,
  73. })
  74. prometheus.MustRegister(KubeNamespaceCollector{
  75. KubeClusterCache: clusterCache,
  76. metricsConfig: *metricsConfig,
  77. })
  78. prometheus.MustRegister(KubeDeploymentCollector{
  79. KubeClusterCache: clusterCache,
  80. metricsConfig: *metricsConfig,
  81. })
  82. prometheus.MustRegister(KubePodCollector{
  83. KubeClusterCache: clusterCache,
  84. metricsConfig: *metricsConfig,
  85. })
  86. prometheus.MustRegister(KubePVCollector{
  87. KubeClusterCache: clusterCache,
  88. metricsConfig: *metricsConfig,
  89. })
  90. prometheus.MustRegister(KubePVCCollector{
  91. KubeClusterCache: clusterCache,
  92. metricsConfig: *metricsConfig,
  93. })
  94. prometheus.MustRegister(KubeJobCollector{
  95. KubeClusterCache: clusterCache,
  96. metricsConfig: *metricsConfig,
  97. })
  98. } else if opts.EmitKubeStateMetricsV1Only {
  99. prometheus.MustRegister(KubeNodeCollector{
  100. KubeClusterCache: clusterCache,
  101. metricsConfig: *metricsConfig,
  102. })
  103. prometheus.MustRegister(KubeNamespaceCollector{
  104. KubeClusterCache: clusterCache,
  105. metricsConfig: *metricsConfig,
  106. })
  107. prometheus.MustRegister(KubePodLabelsCollector{
  108. KubeClusterCache: clusterCache,
  109. metricsConfig: *metricsConfig,
  110. })
  111. }
  112. })
  113. }
  114. //--------------------------------------------------------------------------
  115. // Kube Metric Helpers
  116. //--------------------------------------------------------------------------
  117. // getPersistentVolumeClaimClass returns StorageClassName. If no storage class was
  118. // requested, it returns "".
  119. func getPersistentVolumeClaimClass(claim *v1.PersistentVolumeClaim) string {
  120. // Use beta annotation first
  121. if class, found := claim.Annotations[v1.BetaStorageClassAnnotation]; found {
  122. return class
  123. }
  124. if claim.Spec.StorageClassName != nil {
  125. return *claim.Spec.StorageClassName
  126. }
  127. // Special non-empty string to indicate absence of storage class.
  128. return "<none>"
  129. }
  130. // toResourceUnitValue accepts a resource name and quantity and returns the sanitized resource, the unit, and the value in the units.
  131. // Returns an empty string for resource and unit if there was a failure.
  132. func toResourceUnitValue(resourceName v1.ResourceName, quantity resource.Quantity) (resource string, unit string, value float64) {
  133. resource = prom.SanitizeLabelName(string(resourceName))
  134. switch resourceName {
  135. case v1.ResourceCPU:
  136. unit = "core"
  137. value = float64(quantity.MilliValue()) / 1000
  138. return
  139. case v1.ResourceStorage:
  140. fallthrough
  141. case v1.ResourceEphemeralStorage:
  142. fallthrough
  143. case v1.ResourceMemory:
  144. unit = "byte"
  145. value = float64(quantity.Value())
  146. return
  147. case v1.ResourcePods:
  148. unit = "integer"
  149. value = float64(quantity.Value())
  150. return
  151. default:
  152. if isHugePageResourceName(resourceName) || isAttachableVolumeResourceName(resourceName) {
  153. unit = "byte"
  154. value = float64(quantity.Value())
  155. return
  156. }
  157. if isExtendedResourceName(resourceName) {
  158. unit = "integer"
  159. value = float64(quantity.Value())
  160. return
  161. }
  162. }
  163. resource = ""
  164. unit = ""
  165. value = 0.0
  166. return
  167. }
  168. // isHugePageResourceName checks for a huge page container resource name
  169. func isHugePageResourceName(name v1.ResourceName) bool {
  170. return strings.HasPrefix(string(name), v1.ResourceHugePagesPrefix)
  171. }
  172. // isAttachableVolumeResourceName checks for attached volume container resource name
  173. func isAttachableVolumeResourceName(name v1.ResourceName) bool {
  174. return strings.HasPrefix(string(name), v1.ResourceAttachableVolumesPrefix)
  175. }
  176. // isExtendedResourceName checks for extended container resource name
  177. func isExtendedResourceName(name v1.ResourceName) bool {
  178. if isNativeResource(name) || strings.HasPrefix(string(name), v1.DefaultResourceRequestsPrefix) {
  179. return false
  180. }
  181. // Ensure it satisfies the rules in IsQualifiedName() after converted into quota resource name
  182. nameForQuota := fmt.Sprintf("%s%s", v1.DefaultResourceRequestsPrefix, string(name))
  183. if errs := validation.IsQualifiedName(nameForQuota); len(errs) != 0 {
  184. return false
  185. }
  186. return true
  187. }
  188. // isNativeResource checks for a kubernetes.io/ prefixed resource name
  189. func isNativeResource(name v1.ResourceName) bool {
  190. return !strings.Contains(string(name), "/") || isPrefixedNativeResource(name)
  191. }
  192. func isPrefixedNativeResource(name v1.ResourceName) bool {
  193. return strings.Contains(string(name), v1.ResourceDefaultNamespacePrefix)
  194. }
  195. func failureReason(jc *batchv1.JobCondition, reason string) bool {
  196. if jc == nil {
  197. return false
  198. }
  199. return jc.Reason == reason
  200. }
  201. // boolFloat64 converts a boolean input into a 1 or 0
  202. func boolFloat64(b bool) float64 {
  203. if b {
  204. return 1
  205. }
  206. return 0
  207. }
  208. // toStringPtr is used to create a new string pointer from iteration vars
  209. func toStringPtr(s string) *string { return &s }