mirror of
				https://github.com/prometheus/node_exporter.git
				synced 2025-08-20 18:33:52 -07:00 
			
		
		
		
	collector: refactor textfile collector to avoid looping defer
Signed-off-by: Matt Layher <mdlayher@gmail.com>
This commit is contained in:
		
							parent
							
								
									c4c5f1f062
								
							
						
					
					
						commit
						177ac7f50f
					
				|  | @ -162,98 +162,115 @@ func convertMetricFamily(metricFamily *dto.MetricFamily, ch chan<- prometheus.Me | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *textFileCollector) exportMTimes(mtimes map[string]time.Time, ch chan<- prometheus.Metric) { | func (c *textFileCollector) exportMTimes(mtimes map[string]time.Time, ch chan<- prometheus.Metric) { | ||||||
| 	// Export the mtimes of the successful files.
 | 	if len(mtimes) == 0 { | ||||||
| 	if len(mtimes) > 0 { | 		return | ||||||
| 		// Sorting is needed for predictable output comparison in tests.
 | 	} | ||||||
| 		filenames := make([]string, 0, len(mtimes)) |  | ||||||
| 		for filename := range mtimes { |  | ||||||
| 			filenames = append(filenames, filename) |  | ||||||
| 		} |  | ||||||
| 		sort.Strings(filenames) |  | ||||||
| 
 | 
 | ||||||
| 		for _, filename := range filenames { | 	// Export the mtimes of the successful files.
 | ||||||
| 			mtime := float64(mtimes[filename].UnixNano() / 1e9) | 	// Sorting is needed for predictable output comparison in tests.
 | ||||||
| 			if c.mtime != nil { | 	filenames := make([]string, 0, len(mtimes)) | ||||||
| 				mtime = *c.mtime | 	for filename := range mtimes { | ||||||
| 			} | 		filenames = append(filenames, filename) | ||||||
| 			ch <- prometheus.MustNewConstMetric(mtimeDesc, prometheus.GaugeValue, mtime, filename) | 	} | ||||||
|  | 	sort.Strings(filenames) | ||||||
|  | 
 | ||||||
|  | 	for _, filename := range filenames { | ||||||
|  | 		mtime := float64(mtimes[filename].UnixNano() / 1e9) | ||||||
|  | 		if c.mtime != nil { | ||||||
|  | 			mtime = *c.mtime | ||||||
| 		} | 		} | ||||||
|  | 		ch <- prometheus.MustNewConstMetric(mtimeDesc, prometheus.GaugeValue, mtime, filename) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Update implements the Collector interface.
 | // Update implements the Collector interface.
 | ||||||
| func (c *textFileCollector) Update(ch chan<- prometheus.Metric) error { | func (c *textFileCollector) Update(ch chan<- prometheus.Metric) error { | ||||||
| 	error := 0.0 | 	// Iterate over files and accumulate their metrics, but also track any
 | ||||||
| 	mtimes := map[string]time.Time{} | 	// parsing errors so an error metric can be reported.
 | ||||||
| 
 | 	var errored bool | ||||||
| 	// Iterate over files and accumulate their metrics.
 |  | ||||||
| 	files, err := ioutil.ReadDir(c.path) | 	files, err := ioutil.ReadDir(c.path) | ||||||
| 	if err != nil && c.path != "" { | 	if err != nil && c.path != "" { | ||||||
| 		log.Errorf("Error reading textfile collector directory %q: %s", c.path, err) | 		errored = true | ||||||
| 		error = 1.0 | 		log.Errorf("failed to read textfile collector directory %q: %v", c.path, err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	mtimes := make(map[string]time.Time, len(files)) | ||||||
| 	for _, f := range files { | 	for _, f := range files { | ||||||
| 		if !strings.HasSuffix(f.Name(), ".prom") { | 		if !strings.HasSuffix(f.Name(), ".prom") { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 		path := filepath.Join(c.path, f.Name()) | 
 | ||||||
| 		file, err := os.Open(path) | 		mtime, err := c.processFile(f.Name(), ch) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Errorf("Error opening %q: %v", path, err) | 			errored = true | ||||||
| 			error = 1.0 | 			log.Errorf("failed to collect textfile data from %q: %v", f.Name(), err) | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		defer file.Close() |  | ||||||
| 		var parser expfmt.TextParser |  | ||||||
| 		parsedFamilies, err := parser.TextToMetricFamilies(file) |  | ||||||
| 		if err != nil { |  | ||||||
| 			log.Errorf("Error parsing %q: %v", path, err) |  | ||||||
| 			error = 1.0 |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		if hasTimestamps(parsedFamilies) { |  | ||||||
| 			log.Errorf("Textfile %q contains unsupported client-side timestamps, skipping entire file", path) |  | ||||||
| 			error = 1.0 |  | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for _, mf := range parsedFamilies { | 		mtimes[f.Name()] = *mtime | ||||||
| 			if mf.Help == nil { |  | ||||||
| 				help := fmt.Sprintf("Metric read from %s", path) |  | ||||||
| 				mf.Help = &help |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		// Only set this once it has been parsed and validated, so that
 |  | ||||||
| 		// a failure does not appear fresh.
 |  | ||||||
| 		stat, err := file.Stat() |  | ||||||
| 		if err != nil { |  | ||||||
| 			log.Errorf("Error stat'ing %q: %v", path, err) |  | ||||||
| 			error = 1.0 |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		mtimes[f.Name()] = stat.ModTime() |  | ||||||
| 
 |  | ||||||
| 		for _, mf := range parsedFamilies { |  | ||||||
| 			convertMetricFamily(mf, ch) |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	c.exportMTimes(mtimes, ch) | 	c.exportMTimes(mtimes, ch) | ||||||
| 
 | 
 | ||||||
| 	// Export if there were errors.
 | 	// Export if there were errors.
 | ||||||
|  | 	var errVal float64 | ||||||
|  | 	if errored { | ||||||
|  | 		errVal = 1.0 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	ch <- prometheus.MustNewConstMetric( | 	ch <- prometheus.MustNewConstMetric( | ||||||
| 		prometheus.NewDesc( | 		prometheus.NewDesc( | ||||||
| 			"node_textfile_scrape_error", | 			"node_textfile_scrape_error", | ||||||
| 			"1 if there was an error opening or reading a file, 0 otherwise", | 			"1 if there was an error opening or reading a file, 0 otherwise", | ||||||
| 			nil, nil, | 			nil, nil, | ||||||
| 		), | 		), | ||||||
| 		prometheus.GaugeValue, error, | 		prometheus.GaugeValue, errVal, | ||||||
| 	) | 	) | ||||||
|  | 
 | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // processFile processes a single file, returning its modification time on success.
 | ||||||
|  | func (c *textFileCollector) processFile(name string, ch chan<- prometheus.Metric) (*time.Time, error) { | ||||||
|  | 	path := filepath.Join(c.path, name) | ||||||
|  | 	f, err := os.Open(path) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to open textfile data file %q: %v", path, err) | ||||||
|  | 	} | ||||||
|  | 	defer f.Close() | ||||||
|  | 
 | ||||||
|  | 	var parser expfmt.TextParser | ||||||
|  | 	families, err := parser.TextToMetricFamilies(f) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to parse textfile data from %q: %v", path, err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if hasTimestamps(families) { | ||||||
|  | 		return nil, fmt.Errorf("textfile %q contains unsupported client-side timestamps, skipping entire file", path) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for _, mf := range families { | ||||||
|  | 		if mf.Help == nil { | ||||||
|  | 			help := fmt.Sprintf("Metric read from %s", path) | ||||||
|  | 			mf.Help = &help | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for _, mf := range families { | ||||||
|  | 		convertMetricFamily(mf, ch) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Only stat the file once it has been parsed and validated, so that
 | ||||||
|  | 	// a failure does not appear fresh.
 | ||||||
|  | 	stat, err := f.Stat() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to stat %q: %v", path, err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	t := stat.ModTime() | ||||||
|  | 	return &t, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // hasTimestamps returns true when metrics contain unsupported timestamps.
 | // hasTimestamps returns true when metrics contain unsupported timestamps.
 | ||||||
| func hasTimestamps(parsedFamilies map[string]*dto.MetricFamily) bool { | func hasTimestamps(parsedFamilies map[string]*dto.MetricFamily) bool { | ||||||
| 	for _, mf := range parsedFamilies { | 	for _, mf := range parsedFamilies { | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue