-
Notifications
You must be signed in to change notification settings - Fork 235
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support for set type #184
Support for set type #184
Conversation
still todo: - flush interval option - support strings, not just floats - check for leaking goroutines during configuration reload Signed-off-by: Bryan Larsen <[email protected]>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a great start, thank you!
Please also remember to add tests for set events, including mappings with match_type: set
.
} | ||
|
||
case *SetEvent: | ||
store, found := b.Sets.store[metricName] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
what happens if the event has dogstatsd tags attached? I think we need to keep separate sets for each of them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
not familiar with dogstatsd, but that makes sense. Are those stored in Event.Labels()?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yes
@@ -366,6 +398,35 @@ func (b *Exporter) handleEvent(event Event) { | |||
conflictingEventStats.WithLabelValues("counter").Inc() | |||
} | |||
|
|||
case *FlushEvent: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure about handling the flush in-line with incoming statsd events. It's fundamentally different (internal, not user-generated. Would it make sense to break it out into its own loop, or does that get us into locking/mutex hell?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
if we keep it inline, please add a comment explaining why it's here
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure how much locking/mutex work there would be, I'm not super familiar with your code. This mechanism let me completely avoid the question. :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
you would need to mutex-protect all the map accesses. it's fine this way, but I think we can't safely circle back into handleEvent
because that would cause double-mapping.
store, found := b.Sets.store[metricName] | ||
if found { | ||
_, have := store[event.Value()] | ||
if !have { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
it's customary in Go to collapse these lines to
if _, ok := store[event.Value()]; !ok {
store[event.Value()] = true
}
so that it can go out of scope when no longer needed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I also see the variable usually called ok
– you're calling it both have
and found
, I'd rather be consistent with those where possible.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Did it that way to avoid aliasing. Is aliasing generally not frowned on in Go code?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
it's probably better not to – although my instinct is that if you run into aliasing, you're nesting the code too deeply. One thing that took me a while to get used to in Go style is that we prefer
if … {
… do stuff …
return
}
… do other stuff …
rather than using an else
. In most cases this means there is only one "ok" to work with. I don't see how to do that neatly here though, so I'm okay with leaving the names as they are.
for k, v := range b.Sets.store { | ||
setevent := b.Sets.events[k] | ||
|
||
b.handleEvent(&GaugeEvent{ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
won't this cause strange side effects? we allow matching on the metric type, so if I have a mapping that explicitly only applies to sets, it won't affect this event, but others will.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
One thing you could to, instead of sending a GaugeEvent through the event handling again, is to maintain and set prometheus Gauge metrics directly, based on the mapping of the set event.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I found myself duplicating much of the handleEvent function (to get mapping, help, labels), code which I don't completely understand yet. Doing it this way let me avoid the duplication.
As far as I'm concerned, a set is a type of gauge; anything you'd want to do to a gauge (mapping, histograms as discussed, et cetera) probably would be useful on a set. So I also rationalize my laziness that way. It'd have to be called out in the documentation if so.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
but these have all been already determined when mapping the set event. the mappings act on statsd types, not Prometheus types, so even if the output is a gauge we cannot treat |s
events as the same.
I think the correct way is to preserve whatever is needed from the first event that is handled, and then get the Prometheus metric objects the same way case *GaugeEvent
does.
I did a quick look, but I didn't find a test to copy & modify to get me started. Can you point me at one? |
There is no test to copy&paste because we don't have this kind of time-delayed behavior yet. At the moment, we test two pieces separately: the mapping, and on a higher level the [event handling](https://github.com/prometheus/statsd_exporter/blob/master/bridge_test. When you teach |
@bryanlarsen are you still interested in working on this? |
We're not using this capability at the moment, so I haven't been working on it. It'd be nice to use it, but at this point you certainly can't count on me to finish this PR. |
Thanks, in that case I'll close it for now – if you or anyone else wants to pick it up, I'm happy to reopen it. |
fixes #183
still todo: