Web/Spring
Micrometer Tracing
여성게
2024. 4. 30. 11:05
기본 사용법
// Span을 생성한다. 만약 현재 쓰레드에 스팬이 있다면, 현재 새로 생성한 newSpan의 부모가 된다.
Span newSpan = this.tracer.nextSpan().name("calculateTax");
// Span을 시작하고 Scope에 넣는다.(Scope에 넣는다는 의미는 Thread local에 스팬을 넣는다는 뜻)
try (Tracer.SpanInScope ws = this.tracer.withSpan(newSpan.start())) {
// 더 나은 디버깅을 위해서 Span에 key/value 쌍을 넣을 수 있다.
newSpan.tag("taxValue", taxValue);
// Span에 이벤트 로깅을 할 수 있다.(이벤트에는 타임스탬프가 찍힌다.)
newSpan.event("taxCalculated");
}
finally {
// Span을 닫는 것을 항상 기억해야 한다. Span을 닫아야 분산 트레이싱 시스템에 Span을 collecting된다.
newSpan.end();
}
새로운 스레드에서 트레이싱을 이어가고 싶을때
Span spanFromThreadX = this.tracer.nextSpan().name("calculateTax");
try (Tracer.SpanInScope ws = this.tracer.withSpan(spanFromThreadX.start())) {
executorService.submit(() -> {
// Pass the span from thread X
Span continuedSpan = spanFromThreadX;
// ...
// You can tag a span
continuedSpan.tag("taxValue", taxValue);
// ...
// You can log an event on a span
continuedSpan.event("taxCalculated");
}).get();
}
finally {
spanFromThreadX.end();
}
부모 Span을 알고 있을 때 명시적으로 자식 Span을 생성하는 방법
// let's assume that we're in a thread Y and we've received
// the `initialSpan` from thread X. `initialSpan` will be the parent
// of the `newSpan`
Span newSpan = this.tracer.nextSpan(initialSpan).name("calculateCommission");
// ...
// You can tag a span
newSpan.tag("commissionValue", commissionValue);
// ...
// You can log an event on a span
newSpan.event("commissionCalculated");
// Once done remember to end the span. This will allow collecting
// the span to send it to e.g. Zipkin. The tags and events set on the
// newSpan will not be present on the parent
newSpan.end();
Micrometer tracing Brave 세팅방법
# build.gradle.kts
// tracing
implementation("io.micrometer:micrometer-tracing-bridge-brave")
implementation("io.zipkin.reporter2:zipkin-reporter-brave")
// test tracing
testImplementation("io.micrometer:micrometer-tracing-test")
testImplementation("io.micrometer:micrometer-tracing-integration-test")
# application.yml
management:
tracing:
sampling:
probability: 1.0
propagation:
type: b3_multi
enabled: true
zipkin:
tracing:
endpoint: ~
Skip pattern
@Configuration
class TraceSkipPatternConfig {
@Bean
fun serverContextPredicate(): ObservationPredicate {
val skipRegularExpressions = "/health-check|/actuator/.*"
.split("|")
.map { it.toRegex() }
.toSet()
return ObservationPredicate { _, context ->
if (context !is ServerRequestObservationContext) {
return@ObservationPredicate true
}
val servletRequest = context.carrier
val path = servletRequest.requestURI
return@ObservationPredicate skipRegularExpressions.none { p -> path.matches(p) }
}
}
@Bean
fun noSpringSecurityObservations(): ObservationPredicate {
return ObservationPredicate { name, _ ->
!name.startsWith("spring.security.")
}
}
}