-
Notifications
You must be signed in to change notification settings - Fork 626
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Example for Java profiling using Grafana Alloy in Kubernetes (#3603)
* example for java profiling using alloy in kubernetes * Split config into files * Add linux capabilities to the example
- Loading branch information
Showing
12 changed files
with
380 additions
and
3 deletions.
There are no files selected for viewing
File renamed without changes.
2 changes: 1 addition & 1 deletion
2
...agent-auto-instrumentation/java/README.md → ...uto-instrumentation/java/docker/README.md
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
File renamed without changes.
File renamed without changes.
File renamed without changes.
47 changes: 47 additions & 0 deletions
47
examples/grafana-agent-auto-instrumentation/java/kubernetes/README.md
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
# Grafana Alloy Java profiling via auto-instrumentation in Kubernetes | ||
|
||
This repository provides a practical demonstration of leveraging Grafana Alloy for continuous Java application profiling using Pyroscope in Kubernetes. | ||
It illustrates a seamless approach to profiling Java processes, aiding in performance optimization. | ||
|
||
## Overview | ||
|
||
Grafana Alloy automates Java process discovery for profiling, streamlining the setup for applications. It enables precise and targeted profiling configurations through the Grafana Alloy settings. | ||
|
||
Java profiling via Grafana Alloy is based on a few components: | ||
- `discovery.process` for process discovery | ||
- `discovery.kubernetes` for adding Kubernetes labels (namespace, pod, and more) | ||
- `discovery.relabel` for detecting java processes and setting up labels | ||
- `pyroscope.java` for enabling profiling for specific applications | ||
- `pyroscope.write` for writing the profiles data to a remote endpoint | ||
|
||
Refer to the [official documentation](https://grafana.com/docs/pyroscope/latest/configure-client/grafana-agent/java/) for an in-depth understanding and additional configuration options for Java profiling with Grafana Alloy. | ||
Also, check the [Grafana Alloy Components reference](https://grafana.com/docs/alloy/latest/reference/components/) for more details on each used component. | ||
|
||
### async-profiler | ||
|
||
The `pyroscope.java` agent component internally uses the [async-profiler](https://github.com/async-profiler/async-profiler) library. | ||
This approach offers a key advantage over other instrumentation mechanisms in that you can profile applications that are already running without interruptions (code changes, config changes or restarts). | ||
|
||
Under the hood, this is achieved by attaching to the application at a process level and issuing commands to control profiling. | ||
|
||
## Getting started | ||
|
||
To use this example: | ||
|
||
1. Set up a local kubernetes cluster using Kind or a similar tool. | ||
2. Clone this repository and navigate to this example's directory. | ||
3. Create a `pyroscope-java` namespace: | ||
```shell | ||
kubectl create namespace pyroscope-java | ||
``` | ||
4. Deploy the manifests: | ||
```shell | ||
kubectl apply -n pyroscope-java -f . | ||
``` | ||
|
||
After the deployment is operational, the Grafana Alloy will profile the Java application using the defined configuration. | ||
The example will deploy a Grafana instance in the same cluster, available via the `grafana` service at port 3000. | ||
|
||
## Documentation | ||
|
||
Refer to the [official documentation](https://grafana.com/docs/pyroscope/latest/configure-client/grafana-agent/java/) for an in-depth understanding and additional configuration options for Java profiling with Grafana Alloy. |
183 changes: 183 additions & 0 deletions
183
examples/grafana-agent-auto-instrumentation/java/kubernetes/grafana-alloy.yaml
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,183 @@ | ||
--- | ||
|
||
apiVersion: rbac.authorization.k8s.io/v1 | ||
kind: ClusterRole # needed for the discovery.kubernetes alloy component | ||
metadata: | ||
name: grafana-alloy-role | ||
rules: | ||
- apiGroups: [""] | ||
resources: ["pods"] | ||
verbs: ["list", "watch"] | ||
|
||
--- | ||
|
||
apiVersion: v1 | ||
kind: ServiceAccount | ||
metadata: | ||
name: grafana-alloy | ||
|
||
--- | ||
|
||
apiVersion: rbac.authorization.k8s.io/v1 | ||
kind: ClusterRoleBinding | ||
metadata: | ||
name: grafana-alloy-binding | ||
roleRef: | ||
apiGroup: rbac.authorization.k8s.io | ||
kind: ClusterRole | ||
name: grafana-alloy-role | ||
subjects: | ||
- kind: ServiceAccount | ||
name: grafana-alloy | ||
namespace: pyroscope-java | ||
|
||
--- | ||
|
||
apiVersion: apps/v1 | ||
kind: DaemonSet | ||
metadata: | ||
name: grafana-alloy | ||
spec: | ||
selector: | ||
matchLabels: | ||
app: grafana-alloy | ||
template: | ||
metadata: | ||
labels: | ||
app: grafana-alloy | ||
spec: | ||
serviceAccountName: grafana-alloy | ||
containers: | ||
- name: grafana-alloy | ||
image: grafana/alloy | ||
command: | ||
- /bin/alloy | ||
- run | ||
- /etc/agent-config/config.river | ||
- --server.http.listen-addr=0.0.0.0:12345 | ||
env: | ||
- name: HOSTNAME | ||
valueFrom: | ||
fieldRef: | ||
fieldPath: spec.nodeName | ||
- name: AGENT_MODE | ||
value: flow | ||
ports: | ||
- containerPort: 12345 | ||
volumeMounts: | ||
- name: agent-config | ||
mountPath: /etc/agent-config | ||
securityContext: | ||
privileged: true | ||
runAsGroup: 0 | ||
runAsUser: 0 | ||
capabilities: | ||
add: | ||
- PERFMON | ||
- SYS_PTRACE | ||
- SYS_RESOURCE | ||
- SYS_ADMIN | ||
volumes: | ||
- name: agent-config | ||
configMap: | ||
name: agent-config | ||
hostPID: true | ||
|
||
--- | ||
|
||
apiVersion: v1 | ||
kind: ConfigMap | ||
metadata: | ||
name: agent-config | ||
data: | ||
config.river: | | ||
logging { | ||
level = "debug" | ||
format = "logfmt" | ||
} | ||
// Discovers all kubernetes pods. | ||
// Relies on serviceAccountName=grafana-alloy in the pod spec for permissions. | ||
discovery.kubernetes "pods" { | ||
role = "pod" | ||
} | ||
// Discovers all processes running on the node. | ||
// Relies on a security context with elevated permissions for the alloy container (running as root). | ||
// Relies on hostPID=true on the pod spec, to be able to see processes from other pods. | ||
discovery.process "all" { | ||
// Merges kubernetes and process data (using container_id), to attach kubernetes labels to discovered processes. | ||
join = discovery.kubernetes.pods.targets | ||
} | ||
// Drops non-java processes and adjusts labels. | ||
discovery.relabel "java" { | ||
targets = discovery.process.all.targets | ||
// Drops non-java processes. | ||
rule { | ||
source_labels = ["__meta_process_exe"] | ||
action = "keep" | ||
regex = ".*/java$" | ||
} | ||
// Sets up the service_name using the namespace and container names. | ||
rule { | ||
source_labels = ["__meta_kubernetes_namespace", "__meta_kubernetes_pod_container_name"] | ||
target_label = "service_name" | ||
separator = "/" | ||
} | ||
// Sets up kubernetes labels (labels with the __ prefix are ultimately dropped). | ||
rule { | ||
action = "replace" | ||
source_labels = ["__meta_kubernetes_pod_node_name"] | ||
target_label = "node" | ||
} | ||
rule { | ||
action = "replace" | ||
source_labels = ["__meta_kubernetes_namespace"] | ||
target_label = "namespace" | ||
} | ||
rule { | ||
action = "replace" | ||
source_labels = ["__meta_kubernetes_pod_name"] | ||
target_label = "pod" | ||
} | ||
rule { | ||
action = "replace" | ||
source_labels = ["__meta_kubernetes_pod_container_name"] | ||
target_label = "container" | ||
} | ||
// Sets up the cluster label. | ||
// Relies on a pod-level annotation with the "cluster_name" name. | ||
// Alternatively it can be set up using external_labels in pyroscope.write. | ||
rule { | ||
action = "replace" | ||
source_labels = ["__meta_kubernetes_pod_annotation_cluster_name"] | ||
target_label = "cluster" | ||
} | ||
} | ||
// Attaches the Pyroscope profiler to the processes returned by the discovery.relabel component. | ||
// Relies on a security context with elevated permissions for the alloy container (running as root). | ||
// Relies on hostPID=true on the pod spec, to be able to access processes from other pods. | ||
pyroscope.java "java" { | ||
profiling_config { | ||
interval = "15s" | ||
alloc = "512k" | ||
cpu = true | ||
lock = "10ms" | ||
sample_rate = 100 | ||
} | ||
forward_to = [pyroscope.write.local.receiver] | ||
targets = discovery.relabel.java.output | ||
} | ||
pyroscope.write "local" { | ||
// Send metrics to the locally running Pyroscope instance. | ||
endpoint { | ||
url = "http://pyroscope:4040" | ||
} | ||
external_labels = { | ||
"static_label" = "static_label_value", | ||
} | ||
} | ||
--- |
75 changes: 75 additions & 0 deletions
75
examples/grafana-agent-auto-instrumentation/java/kubernetes/grafana.yaml
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,75 @@ | ||
apiVersion: apps/v1 | ||
kind: Deployment | ||
metadata: | ||
name: grafana | ||
spec: | ||
replicas: 1 | ||
selector: | ||
matchLabels: | ||
app: grafana | ||
template: | ||
metadata: | ||
labels: | ||
app: grafana | ||
spec: | ||
containers: | ||
- name: grafana | ||
image: grafana/grafana:latest | ||
env: | ||
- name: GF_INSTALL_PLUGINS | ||
value: grafana-pyroscope-app | ||
- name: GF_AUTH_ANONYMOUS_ENABLED | ||
value: "true" | ||
- name: GF_AUTH_ANONYMOUS_ORG_ROLE | ||
value: Admin | ||
- name: GF_AUTH_DISABLE_LOGIN_FORM | ||
value: "true" | ||
ports: | ||
- containerPort: 3000 | ||
volumeMounts: | ||
- name: grafana-provisioning | ||
mountPath: /etc/grafana/provisioning | ||
volumes: | ||
- name: grafana-provisioning | ||
configMap: | ||
name: grafana-provisioning | ||
items: | ||
- key: datasources | ||
path: datasources/datasources.yaml | ||
- key: plugins | ||
path: plugins/plugins.yaml | ||
--- | ||
apiVersion: v1 | ||
kind: Service | ||
metadata: | ||
name: grafana | ||
spec: | ||
selector: | ||
app: grafana | ||
ports: | ||
- protocol: TCP | ||
port: 3000 | ||
targetPort: 3000 | ||
--- | ||
apiVersion: v1 | ||
kind: ConfigMap | ||
metadata: | ||
name: grafana-provisioning | ||
data: | ||
"datasources": | | ||
apiVersion: 1 | ||
datasources: | ||
- uid: local-pyroscope | ||
type: grafana-pyroscope-datasource | ||
name: Pyroscope | ||
url: http://pyroscope:4040 | ||
jsonData: | ||
keepCookies: [pyroscope_git_session] | ||
"plugins": | | ||
apiVersion: 1 | ||
apps: | ||
- type: grafana-pyroscope-app | ||
jsonData: | ||
backendUrl: http://pyroscope:4040 | ||
secureJsonData: |
40 changes: 40 additions & 0 deletions
40
examples/grafana-agent-auto-instrumentation/java/kubernetes/java-fast-slow.yaml
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,40 @@ | ||
apiVersion: v1 | ||
kind: ConfigMap | ||
metadata: | ||
name: java-app-jar | ||
binaryData: # holds FastSlow.jar from the sibling "Docker" directory, as a base64 encoded binary | ||
jar: | | ||
UEsDBAoAAAgAAD1iQlkAAAAAAAAAAAAAAAAJAAQATUVUQS1JTkYv/soAAFBLAwQUAAgICAA9YkJZAAAAAAAAAAAAAAAAFAAAAE1FVEEtSU5GL01BTklGRVNULk1G803My0xLLS7RDUstKs7Mz7NSMNQz4OXyTczM03XOSSwutlJwSywuCc7JL+flci5KTSxJTdF1qrRSMAKq0zNU0HBNzsksKE5VcEzJLyjJLM3V5OXi5QIAUEsHCJiPQHxXAAAAVgAAAFBLAwQUAAgICAASYkJZAAAAAAAAAAAAAAAADgAAAEZhc3RTbG93LmNsYXNznVTLUhNBFD0dkgwMAxke8ohPMEgCSkDxGYRSCkqsoItQaJWrzqSBgclMaqaHxwf4I25csXAlsvADLD/HtXp7QggWgpSpSvfcvvece27fW/3t5+FXAM8wpyOGFg1xAwkkGcxNvs3zDnfX86/Lm8KSDMkZ27XlLENLNreqoxVtGnQD7TAYWhd5IEuOt0PeNbvMEM8u5V7q6ERKg2mgC90M1yLOUNpO3vJcK/R94cr8wq6wQun5AUO/K3ZKtrvuiJUNX/BKw8WQzeaK56JLwt+2LVHopmp6DVxCH2mwuOMwDJ2FnSc3Lzui0I4BDGpIG7iMKwyZi2Si+wjCctWmi5nN/oP/jPyLoQx9USCh816FCFNF2xWvwmpZ+CsKx9BRktzaWua1Izte5bbL0Jd9V2z2pyR9urNCbpVBX9i1RE3anhtoyDCMnlMJBR1Ha7jVaE9EueRK4fthTYrKcQypcXi1XOEZJSIzydDbuNgTY1LQMMbQ0zw+gddLXuhbYtGOSmtMzISKpYF77nkykD6vLQu54VUCE9lUEpNq0KYMjEbWPR3TuK/hgYGHeMQw3Mxju9velsgXI4lEwde4Rd3aY2ivnrS+Z4unQPWML7hbcUSQKXreVlgrnL7hs4ArezXxf856yvOxudNeNVglW6rRMZZcV/jzDg8CQT2nk5ELlafh6R8d/1sojXg9GEPUhBjULw6mHghar5J1lXZGe2LsM9gn+iBOWpPRYQva1ONwFDoXQYG+dOIDkun3++l45QBaOqFWvk+eWITtJJxiSEBTTwyuR0xMfd4gHbGIbIp2pafrAB3F8UP0AF/QH8ObpoY6Twdl7YSJXgxHrDHcxIipq3E6opo+KixlpknIW3NArR9//TgWpEcBJoG7GmIQsal6cmSOR6G3cYd2Ve1dOptAnv6PIzjDE8xgFoO/AVBLBwipPrnIuQIAAG0FAABQSwECCgAKAAAIAAA9YkJZAAAAAAAAAAAAAAAACQAEAAAAAAAAAAAAAAAAAAAATUVUQS1JTkYv/soAAFBLAQIUABQACAgIAD1iQlmYj0B8VwAAAFYAAAAUAAAAAAAAAAAAAAAAACsAAABNRVRBLUlORi9NQU5JRkVTVC5NRlBLAQIUABQACAgIABJiQlmpPrnIuQIAAG0FAAAOAAAAAAAAAAAAAAAAAMQAAABGYXN0U2xvdy5jbGFzc1BLBQYAAAAAAwADALkAAAC5AwAAAAA= | ||
--- | ||
apiVersion: apps/v1 | ||
kind: Deployment | ||
metadata: | ||
name: java-fast-slow | ||
spec: | ||
replicas: 1 | ||
selector: | ||
matchLabels: | ||
app: java-fast-slow | ||
template: | ||
metadata: | ||
annotations: | ||
cluster-name: dev-us-east-1 | ||
labels: | ||
app: java-fast-slow | ||
spec: | ||
containers: | ||
- name: java-fast-slow | ||
image: openjdk:21-jdk-slim | ||
imagePullPolicy: IfNotPresent | ||
command: [ "java" ] | ||
args: [ "-jar", "/app/FastSlow.jar" ] | ||
volumeMounts: | ||
- name: app-jar | ||
mountPath: /app | ||
volumes: | ||
- name: app-jar | ||
configMap: | ||
name: java-app-jar | ||
items: | ||
- key: jar | ||
path: FastSlow.jar |
Oops, something went wrong.