From 7d0a7181e85b16f450798800ef590d3356c3fc4b Mon Sep 17 00:00:00 2001 From: Mark Paluch Date: Wed, 21 Aug 2024 10:54:49 +0200 Subject: [PATCH] Allow configuration of `CassandraObservationConvention`. Closes #1490 --- .../observability/CassandraObservation.java | 2 +- .../CqlSessionObservationInterceptor.java | 9 ++-- ...DefaultCassandraObservationConvention.java | 42 +++++++++-------- .../ObservableCqlSessionFactory.java | 24 ++++++++-- .../ObservableCqlSessionFactoryBean.java | 46 +++++++++++------- .../ObservableReactiveSession.java | 27 +++++++++-- .../ObservableReactiveSessionFactory.java | 20 +++++++- .../ObservableReactiveSessionFactoryBean.java | 47 ++++++++++++------- .../ObservableCqlSessionFactoryUnitTests.java | 3 +- ...leReactiveSessionFactoryBeanUnitTests.java | 19 ++++++++ .../modules/ROOT/pages/observability.adoc | 2 + 11 files changed, 169 insertions(+), 72 deletions(-) diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CassandraObservation.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CassandraObservation.java index f0d83ba7b..30e24891f 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CassandraObservation.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CassandraObservation.java @@ -27,7 +27,7 @@ * @author Greg Turnquist * @since 4.0 */ -enum CassandraObservation implements ObservationDocumentation { +public enum CassandraObservation implements ObservationDocumentation { /** * Create an {@link io.micrometer.observation.Observation} for Cassandra-based queries. diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CqlSessionObservationInterceptor.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CqlSessionObservationInterceptor.java index b9a081645..428d3a536 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CqlSessionObservationInterceptor.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/CqlSessionObservationInterceptor.java @@ -50,13 +50,14 @@ final class CqlSessionObservationInterceptor implements MethodInterceptor { private final ObservationRegistry observationRegistry; - private final CassandraObservationConvention observationConvention = new DefaultCassandraObservationConvention(); + private final CassandraObservationConvention convention; CqlSessionObservationInterceptor(CqlSession delegate, String remoteServiceName, - ObservationRegistry observationRegistry) { + CassandraObservationConvention convention, ObservationRegistry observationRegistry) { this.delegate = delegate; this.remoteServiceName = remoteServiceName; + this.convention = convention; this.observationRegistry = observationRegistry; } @@ -140,7 +141,7 @@ private static Statement createStatement(Object[] args) { return SimpleStatement.newInstance((String) args[0]); } - if (args[0]instanceof String query && args.length == 2) { + if (args[0] instanceof String query && args.length == 2) { return args[1] instanceof Map // ? SimpleStatement.newInstance(query, (Map) args[1]) // : SimpleStatement.newInstance(query, (Object[]) args[1]); @@ -179,7 +180,7 @@ private Observation startObservation(Statement statement, boolean prepare, St delegate.getContext().getSessionName(), delegate.getKeyspace().map(CqlIdentifier::asInternal).orElse("system")), observationRegistry) - .observationConvention(observationConvention); + .observationConvention(convention); if (currentObservation != null) { observation.parentObservation(currentObservation); diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/DefaultCassandraObservationConvention.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/DefaultCassandraObservationConvention.java index 0fb726529..936cf60bf 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/DefaultCassandraObservationConvention.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/DefaultCassandraObservationConvention.java @@ -41,7 +41,9 @@ * @author Mark Paluch * @since 4.0 */ -class DefaultCassandraObservationConvention implements CassandraObservationConvention { +public class DefaultCassandraObservationConvention implements CassandraObservationConvention { + + public static final CassandraObservationConvention INSTANCE = new DefaultCassandraObservationConvention(); @Override public KeyValues getLowCardinalityKeyValues(CassandraObservationContext context) { @@ -104,10 +106,10 @@ public KeyValues getHighCardinalityKeyValues(CassandraObservationContext context } @Nullable - private InetSocketAddress tryGetSocketAddress(EndPoint endPoint) { + protected InetSocketAddress tryGetSocketAddress(EndPoint endPoint) { try { - if (endPoint.resolve()instanceof InetSocketAddress inet) { + if (endPoint.resolve() instanceof InetSocketAddress inet) { return inet; } @@ -121,13 +123,28 @@ public String getContextualName(CassandraObservationContext context) { return (context.isPrepare() ? "PREPARE: " : "") + getOperationName(getCql(context.getStatement()), ""); } + /** + * Tries to parse the CQL query or provides the default name. + * + * @param defaultName if there's no query + * @return span name + */ + public String getOperationName(String cql, String defaultName) { + + if (StringUtils.hasText(cql) && cql.indexOf(' ') > -1) { + return cql.substring(0, cql.indexOf(' ')); + } + + return defaultName; + } + /** * Extract the CQL query from the delegate {@link Statement}. * * @return string-based CQL of the delegate * @param statement */ - private static String getCql(Statement statement) { + protected static String getCql(Statement statement) { String query = ""; @@ -155,7 +172,7 @@ private static String getCql(Statement statement) { * @param statement * @return query */ - private static String getQuery(Statement statement) { + protected static String getQuery(Statement statement) { if (statement instanceof SimpleStatement) { return ((SimpleStatement) statement).getQuery(); @@ -167,19 +184,4 @@ private static String getQuery(Statement statement) { return ""; } - - /** - * Tries to parse the CQL query or provides the default name. - * - * @param defaultName if there's no query - * @return span name - */ - public String getOperationName(String cql, String defaultName) { - - if (StringUtils.hasText(cql) && cql.indexOf(' ') > -1) { - return cql.substring(0, cql.indexOf(' ')); - } - - return defaultName; - } } diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactory.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactory.java index c505c896c..172485e3e 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactory.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactory.java @@ -17,8 +17,6 @@ import io.micrometer.observation.ObservationRegistry; -import org.springframework.aop.RawTargetAccess; -import org.springframework.aop.TargetSource; import org.springframework.aop.framework.ProxyFactory; import org.springframework.data.cassandra.observability.CqlSessionObservationInterceptor.ObservationDecoratedProxy; import org.springframework.util.Assert; @@ -58,20 +56,36 @@ public static CqlSession wrap(CqlSession session, ObservationRegistry observatio * @return */ public static CqlSession wrap(CqlSession session, String remoteServiceName, ObservationRegistry observationRegistry) { + return wrap(session, remoteServiceName, DefaultCassandraObservationConvention.INSTANCE, observationRegistry); + } + + /** + * Wrap the {@link CqlSession} with a {@link CqlSessionObservationInterceptor}. + * + * @param session must not be {@literal null}. + * @param remoteServiceName must not be {@literal null}. + * @param convention the observation convention. + * @param observationRegistry must not be {@literal null}. + * @return + * @since 4.3.4 + */ + public static CqlSession wrap(CqlSession session, String remoteServiceName, CassandraObservationConvention convention, + ObservationRegistry observationRegistry) { Assert.notNull(session, "CqlSession must not be null"); - Assert.notNull(remoteServiceName, "CqlSessionObservationConvention must not be null"); + Assert.notNull(remoteServiceName, "Remote service name must not be null"); + Assert.notNull(convention, "CassandraObservationConvention must not be null"); Assert.notNull(observationRegistry, "ObservationRegistry must not be null"); ProxyFactory proxyFactory = new ProxyFactory(); proxyFactory.setTarget(session); - proxyFactory.addAdvice(new CqlSessionObservationInterceptor(session, remoteServiceName, observationRegistry)); + proxyFactory + .addAdvice(new CqlSessionObservationInterceptor(session, remoteServiceName, convention, observationRegistry)); proxyFactory.addInterface(CqlSession.class); proxyFactory.addInterface(ObservationDecoratedProxy.class); return (CqlSession) proxyFactory.getProxy(); } - } diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryBean.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryBean.java index 4a86f39d4..200f78c87 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryBean.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryBean.java @@ -43,6 +43,8 @@ public class ObservableCqlSessionFactoryBean extends AbstractFactoryBean getObjectType() { return CqlSession.class; } - @Nullable - public String getRemoteServiceName() { - return remoteServiceName; - } - - /** - * Set the remote service name. - * - * @param remoteServiceName - */ - public void setRemoteServiceName(@Nullable String remoteServiceName) { - this.remoteServiceName = remoteServiceName; - } } diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSession.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSession.java index 5ac661ee6..248e4f6e6 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSession.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSession.java @@ -50,12 +50,13 @@ public class ObservableReactiveSession implements ReactiveSession { private final ObservationRegistry observationRegistry; - private final CassandraObservationConvention convention = new DefaultCassandraObservationConvention(); + private final CassandraObservationConvention convention; ObservableReactiveSession(ReactiveSession delegate, String remoteServiceName, - ObservationRegistry observationRegistry) { + CassandraObservationConvention convention, ObservationRegistry observationRegistry) { this.delegate = delegate; this.remoteServiceName = remoteServiceName; + this.convention = convention; this.observationRegistry = observationRegistry; } @@ -67,19 +68,37 @@ public class ObservableReactiveSession implements ReactiveSession { * @return traced representation of a {@link ReactiveSession}. */ public static ReactiveSession create(ReactiveSession session, ObservationRegistry observationRegistry) { - return new ObservableReactiveSession(session, "Cassandra", observationRegistry); + return new ObservableReactiveSession(session, "Cassandra", DefaultCassandraObservationConvention.INSTANCE, + observationRegistry); } /** * Factory method for creation of a {@link ObservableReactiveSession}. * * @param session reactive session. + * @param remoteServiceName the remote service name. * @param observationRegistry observation registry. * @return traced representation of a {@link ReactiveSession}. */ public static ReactiveSession create(ReactiveSession session, String remoteServiceName, ObservationRegistry observationRegistry) { - return new ObservableReactiveSession(session, remoteServiceName, observationRegistry); + return new ObservableReactiveSession(session, remoteServiceName, DefaultCassandraObservationConvention.INSTANCE, + observationRegistry); + } + + /** + * Factory method for creation of a {@link ObservableReactiveSession}. + * + * @param session reactive session. + * @param remoteServiceName the remote service name. + * @param convention the observation convention. + * @param observationRegistry observation registry. + * @return traced representation of a {@link ReactiveSession}. + * @since 4.3.4 + */ + public static ReactiveSession create(ReactiveSession session, String remoteServiceName, + CassandraObservationConvention convention, ObservationRegistry observationRegistry) { + return new ObservableReactiveSession(session, remoteServiceName, convention, observationRegistry); } @Override diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactory.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactory.java index 651de14a2..3a860e5a2 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactory.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactory.java @@ -55,11 +55,27 @@ public static ReactiveSession wrap(ReactiveSession session, ObservationRegistry */ public static ReactiveSession wrap(ReactiveSession session, String remoteServiceName, ObservationRegistry observationRegistry) { + return wrap(session, remoteServiceName, DefaultCassandraObservationConvention.INSTANCE, observationRegistry); + } + + /** + * Wrap the {@link CqlSession} with a {@link CqlSessionObservationInterceptor}. + * + * @param session must not be {@literal null}. + * @param remoteServiceName must not be {@literal null}. + * @param convention the observation convention. + * @param observationRegistry must not be {@literal null}. + * @return + * @since 4.3.4 + */ + public static ReactiveSession wrap(ReactiveSession session, String remoteServiceName, + CassandraObservationConvention convention, ObservationRegistry observationRegistry) { Assert.notNull(session, "CqlSession must not be null"); - Assert.notNull(remoteServiceName, "CqlSessionObservationConvention must not be null"); + Assert.notNull(remoteServiceName, "Remote service name must not be null"); + Assert.notNull(convention, "CassandraObservationConvention must not be null"); Assert.notNull(observationRegistry, "ObservationRegistry must not be null"); - return ObservableReactiveSession.create(session, remoteServiceName, observationRegistry); + return ObservableReactiveSession.create(session, remoteServiceName, convention, observationRegistry); } } diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBean.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBean.java index 3060d5ac8..50a6a5180 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBean.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBean.java @@ -50,6 +50,8 @@ public class ObservableReactiveSessionFactoryBean extends AbstractFactoryBean getObjectType() { return ReactiveSession.class; } - @Nullable - public String getRemoteServiceName() { - return remoteServiceName; - } - @Override public void destroy() { @@ -114,12 +133,4 @@ public void destroy() { } } - /** - * Set the remote service name. - * - * @param remoteServiceName - */ - public void setRemoteServiceName(@Nullable String remoteServiceName) { - this.remoteServiceName = remoteServiceName; - } } diff --git a/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryUnitTests.java b/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryUnitTests.java index 779016e11..c34101075 100644 --- a/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryUnitTests.java +++ b/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableCqlSessionFactoryUnitTests.java @@ -21,6 +21,7 @@ import io.micrometer.observation.ObservationRegistry; import org.junit.jupiter.api.Test; + import org.springframework.aop.framework.AopProxyUtils; import org.springframework.aop.support.AopUtils; @@ -34,7 +35,7 @@ class ObservableCqlSessionFactoryUnitTests { @Test // GH-1426 - void sessionFactoryBeanUnwrapsObservationProxy() throws Exception { + void sessionFactoryBeanUnwrapsObservationProxy() { CqlSession session = mock(CqlSession.class); ObservationRegistry registry = ObservationRegistry.NOOP; diff --git a/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBeanUnitTests.java b/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBeanUnitTests.java index dfa55bbc4..5fb524236 100644 --- a/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBeanUnitTests.java +++ b/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/observability/ObservableReactiveSessionFactoryBeanUnitTests.java @@ -96,4 +96,23 @@ void doesNotCloseCqlSession() throws Exception { verifyNoInteractions(sessionMock); } + + @Test // GH-1490 + void considersConvention() throws Exception { + + CqlSession sessionMock = mock(CqlSession.class); + CassandraObservationConvention conventionMock = mock(CassandraObservationConvention.class); + ObservationRegistry registry = ObservationRegistry.NOOP; + + CqlSession wrapped = ObservableCqlSessionFactory.wrap(sessionMock, registry); + + ObservableReactiveSessionFactoryBean bean = new ObservableReactiveSessionFactoryBean(wrapped, registry); + bean.setConvention(conventionMock); + bean.afterPropertiesSet(); + + ReactiveSession object = bean.getObject(); + Object usedConvention = ReflectionTestUtils.getField(object, "convention"); + + assertThat(usedConvention).isSameAs(conventionMock); + } } diff --git a/src/main/antora/modules/ROOT/pages/observability.adoc b/src/main/antora/modules/ROOT/pages/observability.adoc index 2ea651da6..a3a334c4e 100644 --- a/src/main/antora/modules/ROOT/pages/observability.adoc +++ b/src/main/antora/modules/ROOT/pages/observability.adoc @@ -32,6 +32,8 @@ Also, registers `ObservationRequestTracker.INSTANCE` with the `CqlSessionBuilder <2> Wraps a CQL session object to observe reactive Cassandra statement execution. ==== +Both, javadoc:org.springframework.data.cassandra.observability.ObservableCqlSessionFactoryBean[] and javadoc:org.springframework.data.cassandra.observability.ObservableReactiveSessionFactoryBean[] support configuration of javadoc:org.springframework.data.cassandra.observability.CassandraObservationConvention[]. + See also https://opentelemetry.io/docs/reference/specification/trace/semantic_conventions/database/#cassandra[OpenTelemetry Semantic Conventions] for further reference. include::observability/conventions.adoc[leveloffset=+1]