Skip to content
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

Delete stale metric labels from BPF maps #99

Open
wants to merge 5 commits into
base: main
Choose a base branch
from
Open

Delete stale metric labels from BPF maps #99

wants to merge 5 commits into from

Conversation

tjons
Copy link

@tjons tjons commented Feb 26, 2023

When an entry is removed from a map, we need to delete the associated metric.

@EItanya EItanya changed the title Fix metrics Delete stale histogram metrics from BPF maps Feb 27, 2023
@EItanya EItanya changed the title Delete stale histogram metrics from BPF maps Delete stale metric labels from BPF maps Feb 27, 2023
@@ -367,6 +367,7 @@ func (l *loader) startHashMap(
select {
case <-ticker.C:
mapIter := liveMap.Iterate()
labels := make([]map[string]string, 0)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does the map have a Len function to use here?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

very unfortunately it does not

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okie dokie

Copy link
Member

@EItanya EItanya left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM just 1 nit

Copy link
Collaborator

@lgadban lgadban left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for taking this on @tjons!

Code looks good to me, just curious how you tested?

@tjons
Copy link
Author

tjons commented Feb 27, 2023

No problem @lgadban. I ran tcpconnect to verify that the code didn't break anything but would like to hold off merging this until I finish adding another example program that performs map deletion. I'm working on it today/this evening.

@@ -80,6 +80,7 @@ exit_tcp_connect(struct pt_regs *ctx, int ret)
val = 1;
}
else {
bpf_map_delete_elem(&sockets, &tid);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this a permanent change and is it required? For the long term, I think it would be nice if we would only alter the upstream examples as little as absolutely needed for maintainability purposes.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@krisztianfekete no this is just for testing, I'm working off a VM in gcloud so have been pushing changes for tests. Will remove before merge.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@tjons this PR looks good. Can you clean up so we can merge?

Copy link
Collaborator

@lgadban lgadban left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Adding this so we don't forget, we need to test this change before merging

@tjons
Copy link
Author

tjons commented May 23, 2023

@lgadban is correct, we need to do some testing. I'm working on other issues but will make some time this week to revisit this again.

@andrea-tomassi
Copy link

We cherry picked this commit and tested it.
Unfortunately there is some issue:

With the new code data you get from prometheus endpoint is not aligned with the one from the ebpf map.

The ebpf map shows the correct data (you can look into it both from cli gui or inspecting the pinned map under (/sys/fs/bpf folder), on the other hand the prometheus endpoint is cleared every second. It shows only the delta from the previous reading (1 second tick). If you use an hash counter, it shows the count for each key incremented in the last second or so.
In the actual bpf map you get the right count (total, not just the delta).

@lgadban
Copy link
Collaborator

lgadban commented May 24, 2023

@andrea-tomassi this is super helpful, thanks for the feedback!

We will take a closer look at the implementation and rework it as needed

@andrea-tomassi
Copy link

andrea-tomassi commented May 29, 2023

@lgadban We did some more accurate testing on that, and we isolated the bug in a more precise way I think.
I'm shareing with you our findings so that you can focus on a more precise testing.

It looks like the actual problem is having the char array as a filed of the struct we use as a key for the bpf map.

In fact we use the following:

struct sl_process_t {

	u32 pid;
	u32 ppid;
	u32 mntns;
	char glcomm[TASK_COMM_LEN];

} __attribute__((packed));
  • We declared an "sl_process_map" (type=BPF_MAP_TYPE_HASH, key=struct sl_process_t, value u64).
  • run bee with --pin-maps flag.

Output (both into /sys/fs/bpf/sl_process_map and CLI the following:

{5239,5163,4026532694,['e','t','c','d',],}: 1
{5239,5163,4026532694,['e','t','c','d',],}: 1
{26301,25647,4026534797,['n','o','d','e',],}: 1
{5239,5163,4026532694,['e','t','c','d',],}: 1
{33303,33074,4026533145,['j','a','v','a',],}: 1
{33303,33074,4026533145,['j','a','v','a',],}: 1

As you can see some of the keys are identical, and this is not working as intended for an hash map.

Now, if we remove the char glcomm[TASK_COMM_LEN]; form struct the key duplication disappear and all seems to works properly (our test code both inserts and deletes keys).

We did not tested a u32[] array, so I cannot say if the problem is either the array itself or the "char" data type.
Maybe something related to wrong '\0' management? Just a guess, really don't know.

Hope this helps...

Andrea T

@andrea-tomassi
Copy link

Sorry, I didn't specified some important detail:

this both works the same way into all lines of codes. We specifically tested the "master" branch and the keys deletion in case you don't use char[] is working there.
We dismissed the "cherry picked" branch because this is not solving the above issue, but it adds the issue I wrote previously.

On the Prometheus endpoint side we still get an issue also without the char[] field: in fact the keys are not properly deleted.
The number of the keys you get calling host:9091/metrics is higher that the set you actually can find into /sys/fs/bpf/sl_process_map.

It seems that keys are not deleted (maybe a part of) properly into userspace.
We collect data from SEC("tracepoint/raw_syscalls/sys_enter") that iworks at high frequencies, so this should be tested under an high load to be sure it works properly.

@andrea-tomassi
Copy link

Maybe this can help to summarize, sorry for splitting the information across several posts...

image

@lgadban
Copy link
Collaborator

lgadban commented Jun 1, 2023

@andrea-tomassi thank you so much for the detailed report! We will use this to recreate this issue

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants