Add Splunk MultiMetric support (#6640)
This commit is contained in:
		
							parent
							
								
									169ba2ecc4
								
							
						
					
					
						commit
						bc8769ba24
					
				|  | @ -1952,6 +1952,18 @@ func buildSerializer(name string, tbl *ast.Table) (serializers.Serializer, error | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if node, ok := tbl.Fields["splunkmetric_multimetric"]; ok { | ||||||
|  | 		if kv, ok := node.(*ast.KeyValue); ok { | ||||||
|  | 			if b, ok := kv.Value.(*ast.Boolean); ok { | ||||||
|  | 				var err error | ||||||
|  | 				c.SplunkmetricMultiMetric, err = b.Boolean() | ||||||
|  | 				if err != nil { | ||||||
|  | 					return nil, err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	if node, ok := tbl.Fields["wavefront_source_override"]; ok { | 	if node, ok := tbl.Fields["wavefront_source_override"]; ok { | ||||||
| 		if kv, ok := node.(*ast.KeyValue); ok { | 		if kv, ok := node.(*ast.KeyValue); ok { | ||||||
| 			if ary, ok := kv.Value.(*ast.Array); ok { | 			if ary, ok := kv.Value.(*ast.Array); ok { | ||||||
|  | @ -1985,6 +1997,7 @@ func buildSerializer(name string, tbl *ast.Table) (serializers.Serializer, error | ||||||
| 	delete(tbl.Fields, "template") | 	delete(tbl.Fields, "template") | ||||||
| 	delete(tbl.Fields, "json_timestamp_units") | 	delete(tbl.Fields, "json_timestamp_units") | ||||||
| 	delete(tbl.Fields, "splunkmetric_hec_routing") | 	delete(tbl.Fields, "splunkmetric_hec_routing") | ||||||
|  | 	delete(tbl.Fields, "splunkmetric_multimetric") | ||||||
| 	delete(tbl.Fields, "wavefront_source_override") | 	delete(tbl.Fields, "wavefront_source_override") | ||||||
| 	delete(tbl.Fields, "wavefront_use_strict") | 	delete(tbl.Fields, "wavefront_use_strict") | ||||||
| 	return serializers.NewSerializer(c) | 	return serializers.NewSerializer(c) | ||||||
|  |  | ||||||
|  | @ -73,6 +73,9 @@ type Config struct { | ||||||
| 	// Include HEC routing fields for splunkmetric output
 | 	// Include HEC routing fields for splunkmetric output
 | ||||||
| 	HecRouting bool | 	HecRouting bool | ||||||
| 
 | 
 | ||||||
|  | 	// Enable Splunk MultiMetric output (Splunk 8.0+)
 | ||||||
|  | 	SplunkmetricMultiMetric bool | ||||||
|  | 
 | ||||||
| 	// Point tags to use as the source name for Wavefront (if none found, host will be used).
 | 	// Point tags to use as the source name for Wavefront (if none found, host will be used).
 | ||||||
| 	WavefrontSourceOverride []string | 	WavefrontSourceOverride []string | ||||||
| 
 | 
 | ||||||
|  | @ -93,7 +96,7 @@ func NewSerializer(config *Config) (Serializer, error) { | ||||||
| 	case "json": | 	case "json": | ||||||
| 		serializer, err = NewJsonSerializer(config.TimestampUnits) | 		serializer, err = NewJsonSerializer(config.TimestampUnits) | ||||||
| 	case "splunkmetric": | 	case "splunkmetric": | ||||||
| 		serializer, err = NewSplunkmetricSerializer(config.HecRouting) | 		serializer, err = NewSplunkmetricSerializer(config.HecRouting, config.SplunkmetricMultiMetric) | ||||||
| 	case "nowmetric": | 	case "nowmetric": | ||||||
| 		serializer, err = NewNowSerializer() | 		serializer, err = NewNowSerializer() | ||||||
| 	case "carbon2": | 	case "carbon2": | ||||||
|  | @ -118,8 +121,8 @@ func NewCarbon2Serializer() (Serializer, error) { | ||||||
| 	return carbon2.NewSerializer() | 	return carbon2.NewSerializer() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func NewSplunkmetricSerializer(splunkmetric_hec_routing bool) (Serializer, error) { | func NewSplunkmetricSerializer(splunkmetric_hec_routing bool, splunkmetric_multimetric bool) (Serializer, error) { | ||||||
| 	return splunkmetric.NewSerializer(splunkmetric_hec_routing) | 	return splunkmetric.NewSerializer(splunkmetric_hec_routing, splunkmetric_multimetric) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func NewNowSerializer() (Serializer, error) { | func NewNowSerializer() (Serializer, error) { | ||||||
|  |  | ||||||
|  | @ -27,6 +27,36 @@ In the above snippet, the following keys are dimensions: | ||||||
| * dc | * dc | ||||||
| * user | * user | ||||||
| 
 | 
 | ||||||
|  | ## Using Multimetric output | ||||||
|  | 
 | ||||||
|  | Starting with Splunk Enterprise and Splunk Cloud 8.0, you can now send multiple metric values in one payload. This means, for example, that | ||||||
|  | you can send all of your CPU stats in one JSON struct, an example event looks like: | ||||||
|  | 
 | ||||||
|  | ```javascript | ||||||
|  | { | ||||||
|  |   "time": 1572469920, | ||||||
|  |   "event": "metric", | ||||||
|  |   "host": "mono.local", | ||||||
|  |   "fields": { | ||||||
|  |     "_config_hecRouting": false, | ||||||
|  |     "_config_multiMetric": true, | ||||||
|  |     "class": "osx", | ||||||
|  |     "cpu": "cpu0", | ||||||
|  |     "metric_name:telegraf.cpu.usage_guest": 0, | ||||||
|  |     "metric_name:telegraf.cpu.usage_guest_nice": 0, | ||||||
|  |     "metric_name:telegraf.cpu.usage_idle": 65.1, | ||||||
|  |     "metric_name:telegraf.cpu.usage_iowait": 0, | ||||||
|  |     "metric_name:telegraf.cpu.usage_irq": 0, | ||||||
|  |     "metric_name:telegraf.cpu.usage_nice": 0, | ||||||
|  |     "metric_name:telegraf.cpu.usage_softirq": 0, | ||||||
|  |     "metric_name:telegraf.cpu.usage_steal": 0, | ||||||
|  |     "metric_name:telegraf.cpu.usage_system": 10.2, | ||||||
|  |     "metric_name:telegraf.cpu.usage_user": 24.7, | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  | In order to enable this mode, there's a new option `splunkmetric_multimetric` that you set in the appropriate output module you plan on using. | ||||||
|  | 
 | ||||||
| ## Using with the HTTP output | ## Using with the HTTP output | ||||||
| 
 | 
 | ||||||
| To send this data to a Splunk HEC, you can use the HTTP output, there are some custom headers that you need to add | To send this data to a Splunk HEC, you can use the HTTP output, there are some custom headers that you need to add | ||||||
|  | @ -61,6 +91,7 @@ to manage the HEC authorization, here's a sample config for an HTTP output: | ||||||
|    data_format = "splunkmetric" |    data_format = "splunkmetric" | ||||||
|     ## Provides time, index, source overrides for the HEC |     ## Provides time, index, source overrides for the HEC | ||||||
|    splunkmetric_hec_routing = true |    splunkmetric_hec_routing = true | ||||||
|  |    # splunkmentric_multimetric = true | ||||||
| 
 | 
 | ||||||
|    ## Additional HTTP headers |    ## Additional HTTP headers | ||||||
|     [outputs.http.headers] |     [outputs.http.headers] | ||||||
|  | @ -118,7 +149,6 @@ disabled = false | ||||||
| INDEXED_EXTRACTIONS = json | INDEXED_EXTRACTIONS = json | ||||||
| KV_MODE = none | KV_MODE = none | ||||||
| TIMESTAMP_FIELDS = time | TIMESTAMP_FIELDS = time | ||||||
| TIME_FORMAT = %s.%3N |  | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| An example configuration of a file based output is: | An example configuration of a file based output is: | ||||||
|  | @ -134,5 +164,6 @@ An example configuration of a file based output is: | ||||||
|    ## more about them here: |    ## more about them here: | ||||||
|    ## https://github.com/influxdata/telegraf/blob/master/docs/DATA_FORMATS_OUTPUT.md |    ## https://github.com/influxdata/telegraf/blob/master/docs/DATA_FORMATS_OUTPUT.md | ||||||
|    data_format = "splunkmetric" |    data_format = "splunkmetric" | ||||||
|    hec_routing = false |    splunkmetric_hec_routing = false | ||||||
|  |    splunkmetric_multimetric = true | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | @ -10,11 +10,32 @@ import ( | ||||||
| 
 | 
 | ||||||
| type serializer struct { | type serializer struct { | ||||||
| 	HecRouting              bool | 	HecRouting              bool | ||||||
|  | 	SplunkmetricMultiMetric bool | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func NewSerializer(splunkmetric_hec_routing bool) (*serializer, error) { | type CommonTags struct { | ||||||
|  | 	Time   float64 | ||||||
|  | 	Host   string | ||||||
|  | 	Index  string | ||||||
|  | 	Source string | ||||||
|  | 	Fields map[string]interface{} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type HECTimeSeries struct { | ||||||
|  | 	Time   float64                `json:"time"` | ||||||
|  | 	Event  string                 `json:"event"` | ||||||
|  | 	Host   string                 `json:"host,omitempty"` | ||||||
|  | 	Index  string                 `json:"index,omitempty"` | ||||||
|  | 	Source string                 `json:"source,omitempty"` | ||||||
|  | 	Fields map[string]interface{} `json:"fields"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // NewSerializer Setup our new serializer
 | ||||||
|  | func NewSerializer(splunkmetric_hec_routing bool, splunkmetric_multimetric bool) (*serializer, error) { | ||||||
|  | 	/*	Define output params */ | ||||||
| 	s := &serializer{ | 	s := &serializer{ | ||||||
| 		HecRouting:              splunkmetric_hec_routing, | 		HecRouting:              splunkmetric_hec_routing, | ||||||
|  | 		SplunkmetricMultiMetric: splunkmetric_multimetric, | ||||||
| 	} | 	} | ||||||
| 	return s, nil | 	return s, nil | ||||||
| } | } | ||||||
|  | @ -45,26 +66,61 @@ func (s *serializer) SerializeBatch(metrics []telegraf.Metric) ([]byte, error) { | ||||||
| 	return serialized, nil | 	return serialized, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *serializer) createObject(metric telegraf.Metric) (metricGroup []byte, err error) { | func (s *serializer) createMulti(metric telegraf.Metric, dataGroup HECTimeSeries, commonTags CommonTags) (metricGroup []byte, err error) { | ||||||
| 
 | 	/* When splunkmetric_multimetric is true, then we can write out multiple name=value pairs as part of the same | ||||||
| 	/*  Splunk supports one metric json object, and does _not_ support an array of JSON objects. | 	** event payload. This only works when the time, host, and dimensions are the same for every name=value pair | ||||||
| 	     ** Splunk has the following required names for the metric store: | 	** in the timeseries data. | ||||||
| 		 ** metric_name: The name of the metric | 	** | ||||||
| 		 ** _value:      The value for the metric | 	** The format for multimetric data is 'metric_name:nameOfMetric = valueOfMetric' | ||||||
| 		 ** time:       The timestamp for the metric |  | ||||||
| 		 ** All other index fields become dimensions. |  | ||||||
| 	 */ | 	 */ | ||||||
| 	type HECTimeSeries struct { | 	var metricJSON []byte | ||||||
| 		Time   float64                `json:"time"` | 
 | ||||||
| 		Event  string                 `json:"event"` | 	// Set the event data from the commonTags above.
 | ||||||
| 		Host   string                 `json:"host,omitempty"` | 	dataGroup.Event = "metric" | ||||||
| 		Index  string                 `json:"index,omitempty"` | 	dataGroup.Time = commonTags.Time | ||||||
| 		Source string                 `json:"source,omitempty"` | 	dataGroup.Host = commonTags.Host | ||||||
| 		Fields map[string]interface{} `json:"fields"` | 	dataGroup.Index = commonTags.Index | ||||||
|  | 	dataGroup.Source = commonTags.Source | ||||||
|  | 	dataGroup.Fields = commonTags.Fields | ||||||
|  | 
 | ||||||
|  | 	// Stuff the metrid data into the structure.
 | ||||||
|  | 	for _, field := range metric.FieldList() { | ||||||
|  | 		value, valid := verifyValue(field.Value) | ||||||
|  | 
 | ||||||
|  | 		if !valid { | ||||||
|  | 			log.Printf("D! Can not parse value: %v for key: %v", field.Value, field.Key) | ||||||
|  | 			continue | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 	dataGroup := HECTimeSeries{} | 		dataGroup.Fields["metric_name:"+metric.Name()+"."+field.Key] = value | ||||||
| 	var metricJson []byte | 	} | ||||||
|  | 
 | ||||||
|  | 	// Manage the rest of the event details based upon HEC routing rules
 | ||||||
|  | 	switch s.HecRouting { | ||||||
|  | 	case true: | ||||||
|  | 		// Output the data as a fields array and host,index,time,source overrides for the HEC.
 | ||||||
|  | 		metricJSON, err = json.Marshal(dataGroup) | ||||||
|  | 	default: | ||||||
|  | 		// Just output the data and the time, useful for file based outuputs
 | ||||||
|  | 		dataGroup.Fields["time"] = dataGroup.Time | ||||||
|  | 		metricJSON, err = json.Marshal(dataGroup.Fields) | ||||||
|  | 	} | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	// Let the JSON fall through to the return below
 | ||||||
|  | 	metricGroup = metricJSON | ||||||
|  | 
 | ||||||
|  | 	return metricGroup, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *serializer) createSingle(metric telegraf.Metric, dataGroup HECTimeSeries, commonTags CommonTags) (metricGroup []byte, err error) { | ||||||
|  | 	/* The default mode is to generate one JSON entitiy per metric (required for pre-8.0 Splunks) | ||||||
|  | 	** | ||||||
|  | 	** The format for single metric is 'nameOfMetric = valueOfMetric' | ||||||
|  | 	 */ | ||||||
|  | 
 | ||||||
|  | 	var metricJSON []byte | ||||||
| 
 | 
 | ||||||
| 	for _, field := range metric.FieldList() { | 	for _, field := range metric.FieldList() { | ||||||
| 
 | 
 | ||||||
|  | @ -75,39 +131,30 @@ func (s *serializer) createObject(metric telegraf.Metric) (metricGroup []byte, e | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		obj := map[string]interface{}{} |  | ||||||
| 		obj["metric_name"] = metric.Name() + "." + field.Key |  | ||||||
| 		obj["_value"] = value |  | ||||||
| 
 |  | ||||||
| 		dataGroup.Event = "metric" | 		dataGroup.Event = "metric" | ||||||
| 		// Convert ns to float seconds since epoch.
 |  | ||||||
| 		dataGroup.Time = float64(metric.Time().UnixNano()) / float64(1000000000) |  | ||||||
| 		dataGroup.Fields = obj |  | ||||||
| 
 | 
 | ||||||
| 		// Break tags out into key(n)=value(t) pairs
 | 		dataGroup.Time = commonTags.Time | ||||||
| 		for n, t := range metric.Tags() { | 
 | ||||||
| 			if n == "host" { | 		// Apply the common tags from above to every record.
 | ||||||
| 				dataGroup.Host = t | 		dataGroup.Host = commonTags.Host | ||||||
| 			} else if n == "index" { | 		dataGroup.Index = commonTags.Index | ||||||
| 				dataGroup.Index = t | 		dataGroup.Source = commonTags.Source | ||||||
| 			} else if n == "source" { | 		dataGroup.Fields = commonTags.Fields | ||||||
| 				dataGroup.Source = t | 
 | ||||||
| 			} else { | 		dataGroup.Fields["metric_name"] = metric.Name() + "." + field.Key | ||||||
| 				dataGroup.Fields[n] = t | 		dataGroup.Fields["_value"] = value | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		switch s.HecRouting { | 		switch s.HecRouting { | ||||||
| 		case true: | 		case true: | ||||||
| 			// Output the data as a fields array and host,index,time,source overrides for the HEC.
 | 			// Output the data as a fields array and host,index,time,source overrides for the HEC.
 | ||||||
| 			metricJson, err = json.Marshal(dataGroup) | 			metricJSON, err = json.Marshal(dataGroup) | ||||||
| 		default: | 		default: | ||||||
| 			// Just output the data and the time, useful for file based outuputs
 | 			// Just output the data and the time, useful for file based outuputs
 | ||||||
| 			dataGroup.Fields["time"] = dataGroup.Time | 			dataGroup.Fields["time"] = dataGroup.Time | ||||||
| 			metricJson, err = json.Marshal(dataGroup.Fields) | 			metricJSON, err = json.Marshal(dataGroup.Fields) | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		metricGroup = append(metricGroup, metricJson...) | 		metricGroup = append(metricGroup, metricJSON...) | ||||||
| 
 | 
 | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
|  | @ -117,6 +164,52 @@ func (s *serializer) createObject(metric telegraf.Metric) (metricGroup []byte, e | ||||||
| 	return metricGroup, nil | 	return metricGroup, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (s *serializer) createObject(metric telegraf.Metric) (metricGroup []byte, err error) { | ||||||
|  | 
 | ||||||
|  | 	/*  Splunk supports one metric json object, and does _not_ support an array of JSON objects. | ||||||
|  | 	     ** Splunk has the following required names for the metric store: | ||||||
|  | 		 ** metric_name: The name of the metric | ||||||
|  | 		 ** _value:      The value for the metric | ||||||
|  | 		 ** time:       The timestamp for the metric | ||||||
|  | 		 ** All other index fields become dimensions. | ||||||
|  | 	*/ | ||||||
|  | 
 | ||||||
|  | 	dataGroup := HECTimeSeries{} | ||||||
|  | 
 | ||||||
|  | 	// The tags are common to all events in this timeseries
 | ||||||
|  | 	commonTags := CommonTags{} | ||||||
|  | 
 | ||||||
|  | 	commonObj := map[string]interface{}{} | ||||||
|  | 
 | ||||||
|  | 	commonObj["config:hecRouting"] = s.HecRouting | ||||||
|  | 	commonObj["config:multiMetric"] = s.SplunkmetricMultiMetric | ||||||
|  | 
 | ||||||
|  | 	commonTags.Fields = commonObj | ||||||
|  | 
 | ||||||
|  | 	// Break tags out into key(n)=value(t) pairs
 | ||||||
|  | 	for n, t := range metric.Tags() { | ||||||
|  | 		if n == "host" { | ||||||
|  | 			commonTags.Host = t | ||||||
|  | 		} else if n == "index" { | ||||||
|  | 			commonTags.Index = t | ||||||
|  | 		} else if n == "source" { | ||||||
|  | 			commonTags.Source = t | ||||||
|  | 		} else { | ||||||
|  | 			commonTags.Fields[n] = t | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	commonTags.Time = float64(metric.Time().UnixNano()) / float64(1000000000) | ||||||
|  | 	switch s.SplunkmetricMultiMetric { | ||||||
|  | 	case true: | ||||||
|  | 		metricGroup, _ = s.createMulti(metric, dataGroup, commonTags) | ||||||
|  | 	default: | ||||||
|  | 		metricGroup, _ = s.createSingle(metric, dataGroup, commonTags) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Return the metric group regardless of if it's multimetric or single metric.
 | ||||||
|  | 	return metricGroup, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func verifyValue(v interface{}) (value interface{}, valid bool) { | func verifyValue(v interface{}) (value interface{}, valid bool) { | ||||||
| 	switch v.(type) { | 	switch v.(type) { | ||||||
| 	case string: | 	case string: | ||||||
|  |  | ||||||
|  | @ -29,11 +29,11 @@ func TestSerializeMetricFloat(t *testing.T) { | ||||||
| 	m, err := metric.New("cpu", tags, fields, now) | 	m, err := metric.New("cpu", tags, fields, now) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	s, _ := NewSerializer(false) | 	s, _ := NewSerializer(false, false) | ||||||
| 	var buf []byte | 	var buf []byte | ||||||
| 	buf, err = s.Serialize(m) | 	buf, err = s.Serialize(m) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	expS := `{"_value":91.5,"cpu":"cpu0","metric_name":"cpu.usage_idle","time":1529875740.819}` | 	expS := `{"_value":91.5,"config:hecRouting":false,"config:multiMetric":false,"cpu":"cpu0","metric_name":"cpu.usage_idle","time":1529875740.819}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -49,11 +49,11 @@ func TestSerializeMetricFloatHec(t *testing.T) { | ||||||
| 	m, err := metric.New("cpu", tags, fields, now) | 	m, err := metric.New("cpu", tags, fields, now) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	s, _ := NewSerializer(true) | 	s, _ := NewSerializer(true, false) | ||||||
| 	var buf []byte | 	var buf []byte | ||||||
| 	buf, err = s.Serialize(m) | 	buf, err = s.Serialize(m) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	expS := `{"time":1529875740.819,"event":"metric","fields":{"_value":91.5,"cpu":"cpu0","metric_name":"cpu.usage_idle"}}` | 	expS := `{"time":1529875740.819,"event":"metric","fields":{"_value":91.5,"config:hecRouting":true,"config:multiMetric":false,"cpu":"cpu0","metric_name":"cpu.usage_idle"}}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -68,12 +68,12 @@ func TestSerializeMetricInt(t *testing.T) { | ||||||
| 	m, err := metric.New("cpu", tags, fields, now) | 	m, err := metric.New("cpu", tags, fields, now) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	s, _ := NewSerializer(false) | 	s, _ := NewSerializer(false, false) | ||||||
| 	var buf []byte | 	var buf []byte | ||||||
| 	buf, err = s.Serialize(m) | 	buf, err = s.Serialize(m) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	expS := `{"_value":90,"cpu":"cpu0","metric_name":"cpu.usage_idle","time":0}` | 	expS := `{"_value":90,"config:hecRouting":false,"config:multiMetric":false,"cpu":"cpu0","metric_name":"cpu.usage_idle","time":0}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -88,12 +88,12 @@ func TestSerializeMetricIntHec(t *testing.T) { | ||||||
| 	m, err := metric.New("cpu", tags, fields, now) | 	m, err := metric.New("cpu", tags, fields, now) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	s, _ := NewSerializer(true) | 	s, _ := NewSerializer(true, false) | ||||||
| 	var buf []byte | 	var buf []byte | ||||||
| 	buf, err = s.Serialize(m) | 	buf, err = s.Serialize(m) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	expS := `{"time":0,"event":"metric","fields":{"_value":90,"cpu":"cpu0","metric_name":"cpu.usage_idle"}}` | 	expS := `{"time":0,"event":"metric","fields":{"_value":90,"config:hecRouting":true,"config:multiMetric":false,"cpu":"cpu0","metric_name":"cpu.usage_idle"}}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -108,12 +108,12 @@ func TestSerializeMetricBool(t *testing.T) { | ||||||
| 	m, err := metric.New("docker", tags, fields, now) | 	m, err := metric.New("docker", tags, fields, now) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	s, _ := NewSerializer(false) | 	s, _ := NewSerializer(false, false) | ||||||
| 	var buf []byte | 	var buf []byte | ||||||
| 	buf, err = s.Serialize(m) | 	buf, err = s.Serialize(m) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	expS := `{"_value":1,"container-name":"telegraf-test","metric_name":"docker.oomkiller","time":0}` | 	expS := `{"_value":1,"config:hecRouting":false,"config:multiMetric":false,"container-name":"telegraf-test","metric_name":"docker.oomkiller","time":0}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -128,12 +128,12 @@ func TestSerializeMetricBoolHec(t *testing.T) { | ||||||
| 	m, err := metric.New("docker", tags, fields, now) | 	m, err := metric.New("docker", tags, fields, now) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	s, _ := NewSerializer(true) | 	s, _ := NewSerializer(true, false) | ||||||
| 	var buf []byte | 	var buf []byte | ||||||
| 	buf, err = s.Serialize(m) | 	buf, err = s.Serialize(m) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	expS := `{"time":0,"event":"metric","fields":{"_value":0,"container-name":"telegraf-test","metric_name":"docker.oomkiller"}}` | 	expS := `{"time":0,"event":"metric","fields":{"_value":0,"config:hecRouting":true,"config:multiMetric":false,"container-name":"telegraf-test","metric_name":"docker.oomkiller"}}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -149,12 +149,12 @@ func TestSerializeMetricString(t *testing.T) { | ||||||
| 	m, err := metric.New("cpu", tags, fields, now) | 	m, err := metric.New("cpu", tags, fields, now) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	s, _ := NewSerializer(false) | 	s, _ := NewSerializer(false, false) | ||||||
| 	var buf []byte | 	var buf []byte | ||||||
| 	buf, err = s.Serialize(m) | 	buf, err = s.Serialize(m) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	expS := `{"_value":5,"cpu":"cpu0","metric_name":"cpu.usage_idle","time":0}` | 	expS := `{"_value":5,"config:hecRouting":false,"config:multiMetric":false,"cpu":"cpu0","metric_name":"cpu.usage_idle","time":0}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| } | } | ||||||
|  | @ -182,11 +182,33 @@ func TestSerializeBatch(t *testing.T) { | ||||||
| 	) | 	) | ||||||
| 
 | 
 | ||||||
| 	metrics := []telegraf.Metric{m, n} | 	metrics := []telegraf.Metric{m, n} | ||||||
| 	s, _ := NewSerializer(false) | 	s, _ := NewSerializer(false, false) | ||||||
| 	buf, err := s.SerializeBatch(metrics) | 	buf, err := s.SerializeBatch(metrics) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	expS := `{"_value":42,"metric_name":"cpu.value","time":0}` + `{"_value":92,"metric_name":"cpu.value","time":0}` | 	expS := `{"_value":42,"config:hecRouting":false,"config:multiMetric":false,"metric_name":"cpu.value","time":0}{"_value":92,"config:hecRouting":false,"config:multiMetric":false,"metric_name":"cpu.value","time":0}` | ||||||
|  | 	assert.Equal(t, string(expS), string(buf)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestSerializeMulti(t *testing.T) { | ||||||
|  | 	m := MustMetric( | ||||||
|  | 		metric.New( | ||||||
|  | 			"cpu", | ||||||
|  | 			map[string]string{}, | ||||||
|  | 			map[string]interface{}{ | ||||||
|  | 				"user":   42.0, | ||||||
|  | 				"system": 8.0, | ||||||
|  | 			}, | ||||||
|  | 			time.Unix(0, 0), | ||||||
|  | 		), | ||||||
|  | 	) | ||||||
|  | 
 | ||||||
|  | 	metrics := []telegraf.Metric{m} | ||||||
|  | 	s, _ := NewSerializer(false, true) | ||||||
|  | 	buf, err := s.SerializeBatch(metrics) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 
 | ||||||
|  | 	expS := `{"config:hecRouting":false,"config:multiMetric":true,"metric_name:cpu.system":8,"metric_name:cpu.user":42,"time":0}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -213,10 +235,32 @@ func TestSerializeBatchHec(t *testing.T) { | ||||||
| 	) | 	) | ||||||
| 
 | 
 | ||||||
| 	metrics := []telegraf.Metric{m, n} | 	metrics := []telegraf.Metric{m, n} | ||||||
| 	s, _ := NewSerializer(true) | 	s, _ := NewSerializer(true, false) | ||||||
| 	buf, err := s.SerializeBatch(metrics) | 	buf, err := s.SerializeBatch(metrics) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 
 | 
 | ||||||
| 	expS := `{"time":0,"event":"metric","fields":{"_value":42,"metric_name":"cpu.value"}}` + `{"time":0,"event":"metric","fields":{"_value":92,"metric_name":"cpu.value"}}` | 	expS := `{"time":0,"event":"metric","fields":{"_value":42,"config:hecRouting":true,"config:multiMetric":false,"metric_name":"cpu.value"}}{"time":0,"event":"metric","fields":{"_value":92,"config:hecRouting":true,"config:multiMetric":false,"metric_name":"cpu.value"}}` | ||||||
|  | 	assert.Equal(t, string(expS), string(buf)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestSerializeMultiHec(t *testing.T) { | ||||||
|  | 	m := MustMetric( | ||||||
|  | 		metric.New( | ||||||
|  | 			"cpu", | ||||||
|  | 			map[string]string{}, | ||||||
|  | 			map[string]interface{}{ | ||||||
|  | 				"usage":  42.0, | ||||||
|  | 				"system": 8.0, | ||||||
|  | 			}, | ||||||
|  | 			time.Unix(0, 0), | ||||||
|  | 		), | ||||||
|  | 	) | ||||||
|  | 
 | ||||||
|  | 	metrics := []telegraf.Metric{m} | ||||||
|  | 	s, _ := NewSerializer(true, true) | ||||||
|  | 	buf, err := s.SerializeBatch(metrics) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 
 | ||||||
|  | 	expS := `{"time":0,"event":"metric","fields":{"config:hecRouting":true,"config:multiMetric":true,"metric_name:cpu.system":8,"metric_name:cpu.usage":42}}` | ||||||
| 	assert.Equal(t, string(expS), string(buf)) | 	assert.Equal(t, string(expS), string(buf)) | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue