Add TTL field to ping input (#5556)
This commit is contained in:
		
							parent
							
								
									7da57fe5f5
								
							
						
					
					
						commit
						80155029c4
					
				|  | @ -70,6 +70,7 @@ LimitNOFILE=4096 | ||||||
|     - packets_transmitted (integer) |     - packets_transmitted (integer) | ||||||
|     - packets_received (integer) |     - packets_received (integer) | ||||||
|     - percent_packets_loss (float) |     - percent_packets_loss (float) | ||||||
|  |     - ttl (integer, Not available on Windows) | ||||||
|     - average_response_ms (integer) |     - average_response_ms (integer) | ||||||
|     - minimum_response_ms (integer) |     - minimum_response_ms (integer) | ||||||
|     - maximum_response_ms (integer) |     - maximum_response_ms (integer) | ||||||
|  | @ -92,5 +93,5 @@ ping,url=example.org result_code=0i,average_response_ms=7i,maximum_response_ms=9 | ||||||
| 
 | 
 | ||||||
| **Linux:** | **Linux:** | ||||||
| ``` | ``` | ||||||
| ping,url=example.org average_response_ms=23.066,maximum_response_ms=24.64,minimum_response_ms=22.451,packets_received=5i,packets_transmitted=5i,percent_packet_loss=0,result_code=0i,standard_deviation_ms=0.809 1535747258000000000 | ping,url=example.org average_response_ms=23.066,ttl=63,maximum_response_ms=24.64,minimum_response_ms=22.451,packets_received=5i,packets_transmitted=5i,percent_packet_loss=0,result_code=0i,standard_deviation_ms=0.809 1535747258000000000 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | @ -7,6 +7,7 @@ import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net" | 	"net" | ||||||
| 	"os/exec" | 	"os/exec" | ||||||
|  | 	"regexp" | ||||||
| 	"runtime" | 	"runtime" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
|  | @ -151,7 +152,7 @@ func (p *Ping) pingToURL(u string, acc telegraf.Accumulator) { | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	trans, rec, min, avg, max, stddev, err := processPingOutput(out) | 	trans, rec, ttl, min, avg, max, stddev, err := processPingOutput(out) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		// fatal error
 | 		// fatal error
 | ||||||
| 		acc.AddError(fmt.Errorf("%s: %s", err, u)) | 		acc.AddError(fmt.Errorf("%s: %s", err, u)) | ||||||
|  | @ -164,6 +165,9 @@ func (p *Ping) pingToURL(u string, acc telegraf.Accumulator) { | ||||||
| 	fields["packets_transmitted"] = trans | 	fields["packets_transmitted"] = trans | ||||||
| 	fields["packets_received"] = rec | 	fields["packets_received"] = rec | ||||||
| 	fields["percent_packet_loss"] = loss | 	fields["percent_packet_loss"] = loss | ||||||
|  | 	if ttl >= 0 { | ||||||
|  | 		fields["ttl"] = ttl | ||||||
|  | 	} | ||||||
| 	if min >= 0 { | 	if min >= 0 { | ||||||
| 		fields["minimum_response_ms"] = min | 		fields["minimum_response_ms"] = min | ||||||
| 	} | 	} | ||||||
|  | @ -253,50 +257,74 @@ func (p *Ping) args(url string, system string) []string { | ||||||
| //     round-trip min/avg/max/stddev = 34.843/43.508/52.172/8.664 ms
 | //     round-trip min/avg/max/stddev = 34.843/43.508/52.172/8.664 ms
 | ||||||
| //
 | //
 | ||||||
| // It returns (<transmitted packets>, <received packets>, <average response>)
 | // It returns (<transmitted packets>, <received packets>, <average response>)
 | ||||||
| func processPingOutput(out string) (int, int, float64, float64, float64, float64, error) { | func processPingOutput(out string) (int, int, int, float64, float64, float64, float64, error) { | ||||||
| 	var trans, recv int | 	var trans, recv, ttl int = 0, 0, -1 | ||||||
| 	var min, avg, max, stddev float64 = -1.0, -1.0, -1.0, -1.0 | 	var min, avg, max, stddev float64 = -1.0, -1.0, -1.0, -1.0 | ||||||
| 	// Set this error to nil if we find a 'transmitted' line
 | 	// Set this error to nil if we find a 'transmitted' line
 | ||||||
| 	err := errors.New("Fatal error processing ping output") | 	err := errors.New("Fatal error processing ping output") | ||||||
| 	lines := strings.Split(out, "\n") | 	lines := strings.Split(out, "\n") | ||||||
| 	for _, line := range lines { | 	for _, line := range lines { | ||||||
| 		if strings.Contains(line, "transmitted") && | 		// Reading only first TTL, ignoring other TTL messages
 | ||||||
|  | 		if ttl == -1 && strings.Contains(line, "ttl=") { | ||||||
|  | 			ttl, err = getTTL(line) | ||||||
|  | 		} else if strings.Contains(line, "transmitted") && | ||||||
| 			strings.Contains(line, "received") { | 			strings.Contains(line, "received") { | ||||||
|  | 			trans, recv, err = getPacketStats(line, trans, recv) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return trans, recv, ttl, min, avg, max, stddev, err | ||||||
|  | 			} | ||||||
|  | 		} else if strings.Contains(line, "min/avg/max") { | ||||||
|  | 			min, avg, max, stddev, err = checkRoundTripTimeStats(line, min, avg, max, stddev) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return trans, recv, ttl, min, avg, max, stddev, err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return trans, recv, ttl, min, avg, max, stddev, err | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func getPacketStats(line string, trans, recv int) (int, int, error) { | ||||||
| 	stats := strings.Split(line, ", ") | 	stats := strings.Split(line, ", ") | ||||||
| 	// Transmitted packets
 | 	// Transmitted packets
 | ||||||
| 			trans, err = strconv.Atoi(strings.Split(stats[0], " ")[0]) | 	trans, err := strconv.Atoi(strings.Split(stats[0], " ")[0]) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 				return trans, recv, min, avg, max, stddev, err | 		return trans, recv, err | ||||||
| 	} | 	} | ||||||
| 	// Received packets
 | 	// Received packets
 | ||||||
| 	recv, err = strconv.Atoi(strings.Split(stats[1], " ")[0]) | 	recv, err = strconv.Atoi(strings.Split(stats[1], " ")[0]) | ||||||
| 			if err != nil { | 	return trans, recv, err | ||||||
| 				return trans, recv, min, avg, max, stddev, err |  | ||||||
| } | } | ||||||
| 		} else if strings.Contains(line, "min/avg/max") { | 
 | ||||||
|  | func getTTL(line string) (int, error) { | ||||||
|  | 	ttlLine := regexp.MustCompile(`ttl=(\d+)`) | ||||||
|  | 	ttlMatch := ttlLine.FindStringSubmatch(line) | ||||||
|  | 	return strconv.Atoi(ttlMatch[1]) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func checkRoundTripTimeStats(line string, min, avg, max, | ||||||
|  | 	stddev float64) (float64, float64, float64, float64, error) { | ||||||
| 	stats := strings.Split(line, " ")[3] | 	stats := strings.Split(line, " ")[3] | ||||||
| 	data := strings.Split(stats, "/") | 	data := strings.Split(stats, "/") | ||||||
| 			min, err = strconv.ParseFloat(data[0], 64) | 
 | ||||||
|  | 	min, err := strconv.ParseFloat(data[0], 64) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 				return trans, recv, min, avg, max, stddev, err | 		return min, avg, max, stddev, err | ||||||
| 	} | 	} | ||||||
| 	avg, err = strconv.ParseFloat(data[1], 64) | 	avg, err = strconv.ParseFloat(data[1], 64) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 				return trans, recv, min, avg, max, stddev, err | 		return min, avg, max, stddev, err | ||||||
| 	} | 	} | ||||||
| 	max, err = strconv.ParseFloat(data[2], 64) | 	max, err = strconv.ParseFloat(data[2], 64) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 				return trans, recv, min, avg, max, stddev, err | 		return min, avg, max, stddev, err | ||||||
| 	} | 	} | ||||||
| 	if len(data) == 4 { | 	if len(data) == 4 { | ||||||
| 		stddev, err = strconv.ParseFloat(data[3], 64) | 		stddev, err = strconv.ParseFloat(data[3], 64) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 					return trans, recv, min, avg, max, stddev, err | 			return min, avg, max, stddev, err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 		} | 	return min, avg, max, stddev, err | ||||||
| 	} |  | ||||||
| 	return trans, recv, min, avg, max, stddev, err |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func init() { | func init() { | ||||||
|  |  | ||||||
|  | @ -61,8 +61,9 @@ ping: -i interval too short: Operation not permitted | ||||||
| 
 | 
 | ||||||
| // Test that ping command output is processed properly
 | // Test that ping command output is processed properly
 | ||||||
| func TestProcessPingOutput(t *testing.T) { | func TestProcessPingOutput(t *testing.T) { | ||||||
| 	trans, rec, min, avg, max, stddev, err := processPingOutput(bsdPingOutput) | 	trans, rec, ttl, min, avg, max, stddev, err := processPingOutput(bsdPingOutput) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, 55, ttl, "ttl value is 55") | ||||||
| 	assert.Equal(t, 5, trans, "5 packets were transmitted") | 	assert.Equal(t, 5, trans, "5 packets were transmitted") | ||||||
| 	assert.Equal(t, 5, rec, "5 packets were transmitted") | 	assert.Equal(t, 5, rec, "5 packets were transmitted") | ||||||
| 	assert.InDelta(t, 15.087, min, 0.001) | 	assert.InDelta(t, 15.087, min, 0.001) | ||||||
|  | @ -70,8 +71,9 @@ func TestProcessPingOutput(t *testing.T) { | ||||||
| 	assert.InDelta(t, 27.263, max, 0.001) | 	assert.InDelta(t, 27.263, max, 0.001) | ||||||
| 	assert.InDelta(t, 4.076, stddev, 0.001) | 	assert.InDelta(t, 4.076, stddev, 0.001) | ||||||
| 
 | 
 | ||||||
| 	trans, rec, min, avg, max, stddev, err = processPingOutput(linuxPingOutput) | 	trans, rec, ttl, min, avg, max, stddev, err = processPingOutput(linuxPingOutput) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, 63, ttl, "ttl value is 63") | ||||||
| 	assert.Equal(t, 5, trans, "5 packets were transmitted") | 	assert.Equal(t, 5, trans, "5 packets were transmitted") | ||||||
| 	assert.Equal(t, 5, rec, "5 packets were transmitted") | 	assert.Equal(t, 5, rec, "5 packets were transmitted") | ||||||
| 	assert.InDelta(t, 35.225, min, 0.001) | 	assert.InDelta(t, 35.225, min, 0.001) | ||||||
|  | @ -79,8 +81,9 @@ func TestProcessPingOutput(t *testing.T) { | ||||||
| 	assert.InDelta(t, 51.806, max, 0.001) | 	assert.InDelta(t, 51.806, max, 0.001) | ||||||
| 	assert.InDelta(t, 5.325, stddev, 0.001) | 	assert.InDelta(t, 5.325, stddev, 0.001) | ||||||
| 
 | 
 | ||||||
| 	trans, rec, min, avg, max, stddev, err = processPingOutput(busyBoxPingOutput) | 	trans, rec, ttl, min, avg, max, stddev, err = processPingOutput(busyBoxPingOutput) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, 56, ttl, "ttl value is 56") | ||||||
| 	assert.Equal(t, 4, trans, "4 packets were transmitted") | 	assert.Equal(t, 4, trans, "4 packets were transmitted") | ||||||
| 	assert.Equal(t, 4, rec, "4 packets were transmitted") | 	assert.Equal(t, 4, rec, "4 packets were transmitted") | ||||||
| 	assert.InDelta(t, 15.810, min, 0.001) | 	assert.InDelta(t, 15.810, min, 0.001) | ||||||
|  | @ -89,10 +92,37 @@ func TestProcessPingOutput(t *testing.T) { | ||||||
| 	assert.InDelta(t, -1.0, stddev, 0.001) | 	assert.InDelta(t, -1.0, stddev, 0.001) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Linux ping output with varying TTL
 | ||||||
|  | var linuxPingOutputWithVaryingTTL = ` | ||||||
|  | PING www.google.com (216.58.218.164) 56(84) bytes of data. | ||||||
|  | 64 bytes from host.net (216.58.218.164): icmp_seq=1 ttl=63 time=35.2 ms | ||||||
|  | 64 bytes from host.net (216.58.218.164): icmp_seq=2 ttl=255 time=42.3 ms | ||||||
|  | 64 bytes from host.net (216.58.218.164): icmp_seq=3 ttl=64 time=45.1 ms | ||||||
|  | 64 bytes from host.net (216.58.218.164): icmp_seq=4 ttl=64 time=43.5 ms | ||||||
|  | 64 bytes from host.net (216.58.218.164): icmp_seq=5 ttl=255 time=51.8 ms | ||||||
|  | 
 | ||||||
|  | --- www.google.com ping statistics --- | ||||||
|  | 5 packets transmitted, 5 received, 0% packet loss, time 4010ms | ||||||
|  | rtt min/avg/max/mdev = 35.225/43.628/51.806/5.325 ms | ||||||
|  | ` | ||||||
|  | 
 | ||||||
|  | // Test that ping command output is processed properly
 | ||||||
|  | func TestProcessPingOutputWithVaryingTTL(t *testing.T) { | ||||||
|  | 	trans, rec, ttl, min, avg, max, stddev, err := processPingOutput(linuxPingOutputWithVaryingTTL) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, 63, ttl, "ttl value is 63") | ||||||
|  | 	assert.Equal(t, 5, trans, "5 packets were transmitted") | ||||||
|  | 	assert.Equal(t, 5, rec, "5 packets were transmitted") | ||||||
|  | 	assert.InDelta(t, 35.225, min, 0.001) | ||||||
|  | 	assert.InDelta(t, 43.628, avg, 0.001) | ||||||
|  | 	assert.InDelta(t, 51.806, max, 0.001) | ||||||
|  | 	assert.InDelta(t, 5.325, stddev, 0.001) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Test that processPingOutput returns an error when 'ping' fails to run, such
 | // Test that processPingOutput returns an error when 'ping' fails to run, such
 | ||||||
| // as when an invalid argument is provided
 | // as when an invalid argument is provided
 | ||||||
| func TestErrorProcessPingOutput(t *testing.T) { | func TestErrorProcessPingOutput(t *testing.T) { | ||||||
| 	_, _, _, _, _, _, err := processPingOutput(fatalPingOutput) | 	_, _, _, _, _, _, _, err := processPingOutput(fatalPingOutput) | ||||||
| 	assert.Error(t, err, "Error was expected from processPingOutput") | 	assert.Error(t, err, "Error was expected from processPingOutput") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -160,6 +190,7 @@ func TestPingGather(t *testing.T) { | ||||||
| 		"packets_transmitted":   5, | 		"packets_transmitted":   5, | ||||||
| 		"packets_received":      5, | 		"packets_received":      5, | ||||||
| 		"percent_packet_loss":   0.0, | 		"percent_packet_loss":   0.0, | ||||||
|  | 		"ttl":                   63, | ||||||
| 		"minimum_response_ms":   35.225, | 		"minimum_response_ms":   35.225, | ||||||
| 		"average_response_ms":   43.628, | 		"average_response_ms":   43.628, | ||||||
| 		"maximum_response_ms":   51.806, | 		"maximum_response_ms":   51.806, | ||||||
|  | @ -201,6 +232,7 @@ func TestLossyPingGather(t *testing.T) { | ||||||
| 		"packets_transmitted":   5, | 		"packets_transmitted":   5, | ||||||
| 		"packets_received":      3, | 		"packets_received":      3, | ||||||
| 		"percent_packet_loss":   40.0, | 		"percent_packet_loss":   40.0, | ||||||
|  | 		"ttl":                   63, | ||||||
| 		"minimum_response_ms":   35.225, | 		"minimum_response_ms":   35.225, | ||||||
| 		"average_response_ms":   44.033, | 		"average_response_ms":   44.033, | ||||||
| 		"maximum_response_ms":   51.806, | 		"maximum_response_ms":   51.806, | ||||||
|  | @ -262,6 +294,8 @@ func TestFatalPingGather(t *testing.T) { | ||||||
| 		"Fatal ping should not have packet measurements") | 		"Fatal ping should not have packet measurements") | ||||||
| 	assert.False(t, acc.HasMeasurement("percent_packet_loss"), | 	assert.False(t, acc.HasMeasurement("percent_packet_loss"), | ||||||
| 		"Fatal ping should not have packet measurements") | 		"Fatal ping should not have packet measurements") | ||||||
|  | 	assert.False(t, acc.HasMeasurement("ttl"), | ||||||
|  | 		"Fatal ping should not have packet measurements") | ||||||
| 	assert.False(t, acc.HasMeasurement("minimum_response_ms"), | 	assert.False(t, acc.HasMeasurement("minimum_response_ms"), | ||||||
| 		"Fatal ping should not have packet measurements") | 		"Fatal ping should not have packet measurements") | ||||||
| 	assert.False(t, acc.HasMeasurement("average_response_ms"), | 	assert.False(t, acc.HasMeasurement("average_response_ms"), | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue