|
| 1 | +package controller |
| 2 | + |
| 3 | +import ( |
| 4 | + "fmt" |
| 5 | + "sync" |
| 6 | + |
| 7 | + apierrs "k8s.io/apimachinery/pkg/api/errors" |
| 8 | + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" |
| 9 | + utilruntime "k8s.io/apimachinery/pkg/util/runtime" |
| 10 | + |
| 11 | + imageapi "github.com/openshift/origin/pkg/image/apis/image" |
| 12 | + metrics "github.com/openshift/origin/pkg/image/metrics/prometheus" |
| 13 | +) |
| 14 | + |
| 15 | +const reasonUnknown = "Unknown" |
| 16 | +const reasonInvalidImageReference = "InvalidImageReference" |
| 17 | + |
| 18 | +// ImportMetricCounter counts numbers of successful and failed imports for the purpose of metrics collection. |
| 19 | +type ImportMetricCounter struct { |
| 20 | + counterMutex sync.Mutex |
| 21 | + importSuccessCounts metrics.ImportSuccessCounts |
| 22 | + importErrorCounts metrics.ImportErrorCounts |
| 23 | +} |
| 24 | + |
| 25 | +// NewImportMetricCounter returns a new ImportMetricCounter |
| 26 | +func NewImportMetricCounter() *ImportMetricCounter { |
| 27 | + return &ImportMetricCounter{ |
| 28 | + importSuccessCounts: make(metrics.ImportSuccessCounts), |
| 29 | + importErrorCounts: make(metrics.ImportErrorCounts), |
| 30 | + } |
| 31 | +} |
| 32 | + |
| 33 | +// Increment processes the given image stream import object as a result of successful or failed import and |
| 34 | +// increments the counters. The given error will be used to construct reason of the error_count metric unless |
| 35 | +// any reason is found in the image stream import object. It's safe to call this method with any of the |
| 36 | +// parameters nil. |
| 37 | +func (c *ImportMetricCounter) Increment(isi *imageapi.ImageStreamImport, err error) { |
| 38 | + if isi == nil { |
| 39 | + if err == nil { |
| 40 | + return |
| 41 | + } |
| 42 | + |
| 43 | + c.counterMutex.Lock() |
| 44 | + defer c.counterMutex.Unlock() |
| 45 | + info := defaultErrorInfoReason(&metrics.ImportErrorInfo{}, err) |
| 46 | + c.importErrorCounts[*info]++ |
| 47 | + return |
| 48 | + } |
| 49 | + |
| 50 | + c.countRepositoryImport(isi, err) |
| 51 | + |
| 52 | + if len(isi.Status.Images) == 0 { |
| 53 | + return |
| 54 | + } |
| 55 | + |
| 56 | + c.counterMutex.Lock() |
| 57 | + defer c.counterMutex.Unlock() |
| 58 | + |
| 59 | + enumerateIsImportStatuses(isi, func(info *metrics.ImportErrorInfo) { |
| 60 | + if len(info.Reason) == 0 { |
| 61 | + c.importSuccessCounts[info.Registry]++ |
| 62 | + } else { |
| 63 | + c.importErrorCounts[*defaultErrorInfoReason(info, err)]++ |
| 64 | + } |
| 65 | + }) |
| 66 | +} |
| 67 | + |
| 68 | +// countRepositoryImport increments either success or error counter if the isimport contains repository |
| 69 | +// request. |
| 70 | +func (c *ImportMetricCounter) countRepositoryImport(isi *imageapi.ImageStreamImport, err error) { |
| 71 | + errInfo := getIsImportRepositoryInfo(isi) |
| 72 | + if errInfo == nil { |
| 73 | + return |
| 74 | + } |
| 75 | + |
| 76 | + c.counterMutex.Lock() |
| 77 | + defer c.counterMutex.Unlock() |
| 78 | + |
| 79 | + if len(errInfo.Reason) == 0 { |
| 80 | + c.importSuccessCounts[errInfo.Registry]++ |
| 81 | + } else { |
| 82 | + c.importErrorCounts[*defaultErrorInfoReason(errInfo, err)]++ |
| 83 | + } |
| 84 | +} |
| 85 | + |
| 86 | +// Collect is supposed to be called by the metrics collector. It returns the actual state of counters. |
| 87 | +func (c *ImportMetricCounter) Collect() (metrics.ImportSuccessCounts, metrics.ImportErrorCounts, error) { |
| 88 | + c.counterMutex.Lock() |
| 89 | + defer c.counterMutex.Unlock() |
| 90 | + |
| 91 | + success := metrics.ImportSuccessCounts{} |
| 92 | + for registry, count := range c.importSuccessCounts { |
| 93 | + success[registry] = count |
| 94 | + } |
| 95 | + |
| 96 | + failures := metrics.ImportErrorCounts{} |
| 97 | + for info, count := range c.importErrorCounts { |
| 98 | + failures[info] = count |
| 99 | + } |
| 100 | + |
| 101 | + return success, failures, nil |
| 102 | +} |
| 103 | + |
| 104 | +// getIsImportRepositoryInfo returns an import error info if the given isi contains repository request. |
| 105 | +// If the request succeeded, its Reason will be empty. |
| 106 | +func getIsImportRepositoryInfo(isi *imageapi.ImageStreamImport) *metrics.ImportErrorInfo { |
| 107 | + if isi.Status.Repository == nil || isi.Spec.Repository == nil { |
| 108 | + return nil |
| 109 | + } |
| 110 | + ref := isi.Spec.Repository.From |
| 111 | + if ref.Kind != "DockerImage" { |
| 112 | + return nil |
| 113 | + } |
| 114 | + imgRef, err := imageapi.ParseDockerImageReference(ref.Name) |
| 115 | + if err != nil { |
| 116 | + utilruntime.HandleError(fmt.Errorf( |
| 117 | + "failed to parse isi.spec.repository.from.name %q: %v", |
| 118 | + ref.Name, err)) |
| 119 | + return nil |
| 120 | + } |
| 121 | + |
| 122 | + info := mkImportInfo(imgRef.DockerClientDefaults().Registry, &isi.Status.Repository.Status) |
| 123 | + return &info |
| 124 | +} |
| 125 | + |
| 126 | +// enumerateIsImportStatuses iterates over images of the given image stream import. For any valid recorded |
| 127 | +// import the cb callback will be colled with the obtains information. |
| 128 | +// If the image import is successful, the object passed to the cb will contain empty Reason. |
| 129 | +func enumerateIsImportStatuses(isi *imageapi.ImageStreamImport, cb func(*metrics.ImportErrorInfo)) { |
| 130 | + if len(isi.Status.Images) == 0 { |
| 131 | + return |
| 132 | + } |
| 133 | + |
| 134 | + for i, status := range isi.Status.Images { |
| 135 | + var registry string |
| 136 | + |
| 137 | + imgRef, err := getImageDockerReferenceForImage(isi, i) |
| 138 | + if err != nil { |
| 139 | + utilruntime.HandleError(err) |
| 140 | + } else { |
| 141 | + if imgRef == nil { |
| 142 | + continue |
| 143 | + } |
| 144 | + registry = imgRef.DockerClientDefaults().Registry |
| 145 | + } |
| 146 | + |
| 147 | + info := mkImportInfo(registry, &status.Status) |
| 148 | + if err != nil { |
| 149 | + info.Reason = reasonInvalidImageReference |
| 150 | + } |
| 151 | + cb(&info) |
| 152 | + } |
| 153 | +} |
| 154 | + |
| 155 | +func getImageDockerReferenceForImage( |
| 156 | + isi *imageapi.ImageStreamImport, |
| 157 | + index int, |
| 158 | +) (*imageapi.DockerImageReference, error) { |
| 159 | + var ( |
| 160 | + imgRef imageapi.DockerImageReference |
| 161 | + err error |
| 162 | + ) |
| 163 | + |
| 164 | + // prefer the specification as the source of truth because the reference in status may belong to an |
| 165 | + // older image imported from somewhere else |
| 166 | + if index >= 0 && index < len(isi.Spec.Images) { |
| 167 | + imgSpec := &isi.Spec.Images[index] |
| 168 | + if imgSpec.From.Kind == "DockerImage" { |
| 169 | + imgRef, err = imageapi.ParseDockerImageReference(imgSpec.From.Name) |
| 170 | + if err == nil { |
| 171 | + return &imgRef, nil |
| 172 | + } |
| 173 | + err = fmt.Errorf("failed to parse isi.spec.images[%d].from.name %q: %v", |
| 174 | + index, imgSpec.From.Name, err) |
| 175 | + } |
| 176 | + } |
| 177 | + |
| 178 | + // fall-back to the image in status |
| 179 | + if index < 0 || index >= len(isi.Status.Images) { |
| 180 | + return nil, err |
| 181 | + } |
| 182 | + |
| 183 | + img := isi.Status.Images[index].Image |
| 184 | + if img == nil { |
| 185 | + return nil, err |
| 186 | + } |
| 187 | + |
| 188 | + imgRef, err = imageapi.ParseDockerImageReference(img.DockerImageReference) |
| 189 | + if err != nil { |
| 190 | + return nil, fmt.Errorf( |
| 191 | + "failed to parse isi.status.images[%d].image.dockerImageReference %q: %v", |
| 192 | + index, img.DockerImageReference, err) |
| 193 | + } |
| 194 | + |
| 195 | + return &imgRef, nil |
| 196 | +} |
| 197 | + |
| 198 | +// mkImportInfo returns an import error info for the given status. If the import succeeded, the Reason field |
| 199 | +// will be empty. |
| 200 | +func mkImportInfo(registry string, status *metav1.Status) metrics.ImportErrorInfo { |
| 201 | + var reason string |
| 202 | + if status.Status != metav1.StatusSuccess { |
| 203 | + reason = string(status.Reason) |
| 204 | + if len(reason) == 0 { |
| 205 | + reason = reasonUnknown |
| 206 | + } |
| 207 | + } |
| 208 | + return metrics.ImportErrorInfo{ |
| 209 | + Registry: registry, |
| 210 | + Reason: reason, |
| 211 | + } |
| 212 | +} |
| 213 | + |
| 214 | +// defaultErrorInfoReason fills the Reason field of the import error info from the given error unless already |
| 215 | +// set. |
| 216 | +func defaultErrorInfoReason(info *metrics.ImportErrorInfo, err error) *metrics.ImportErrorInfo { |
| 217 | + if len(info.Reason) == 0 && err != nil { |
| 218 | + info.Reason = string(apierrs.ReasonForError(err)) |
| 219 | + if len(info.Reason) == 0 { |
| 220 | + info.Reason = reasonUnknown |
| 221 | + } |
| 222 | + } |
| 223 | + return info |
| 224 | +} |
0 commit comments