Skip to content

Commit

Permalink
Merge branch '3.0.x' into 3.1.x
Browse files Browse the repository at this point in the history
Closes gh-37156
  • Loading branch information
mhalbritter committed Aug 31, 2023
2 parents 7f56fe8 + 61739bd commit b86937b
Show file tree
Hide file tree
Showing 4 changed files with 187 additions and 11 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,9 @@
import brave.propagation.CurrentTraceContext;
import brave.propagation.CurrentTraceContext.ScopeDecorator;
import brave.propagation.CurrentTraceContextCustomizer;
import brave.propagation.Propagation;
import brave.propagation.Propagation.Factory;
import brave.propagation.Propagation.KeyFactory;
import brave.propagation.ThreadLocalCurrentTraceContext;
import brave.sampler.Sampler;
import io.micrometer.tracing.brave.bridge.BraveBaggageManager;
Expand Down Expand Up @@ -188,7 +190,7 @@ static class BraveNoBaggageConfiguration {
@Bean
@ConditionalOnMissingBean
Factory propagationFactory(TracingProperties properties) {
return CompositePropagationFactory.create(properties.getPropagation(), null);
return CompositePropagationFactory.create(properties.getPropagation());
}

}
Expand All @@ -207,13 +209,31 @@ static class BraveBaggageConfiguration {
@ConditionalOnMissingBean
BaggagePropagation.FactoryBuilder propagationFactoryBuilder(
ObjectProvider<BaggagePropagationCustomizer> baggagePropagationCustomizers) {
CompositePropagationFactory factory = CompositePropagationFactory
.create(this.tracingProperties.getPropagation(), BRAVE_BAGGAGE_MANAGER);
FactoryBuilder builder = BaggagePropagation.newFactoryBuilder(factory);
baggagePropagationCustomizers.orderedStream().forEach((customizer) -> customizer.customize(builder));
// There's a chicken-and-egg problem here: to create a builder, we need a
// factory. But the CompositePropagationFactory needs data from the builder.
// We create a throw-away builder with a throw-away factory, and then copy the
// config to the real builder
FactoryBuilder throwAwayBuilder = BaggagePropagation.newFactoryBuilder(createThrowAwayFactory());
baggagePropagationCustomizers.orderedStream()
.forEach((customizer) -> customizer.customize(throwAwayBuilder));
CompositePropagationFactory propagationFactory = CompositePropagationFactory.create(
this.tracingProperties.getPropagation(), BRAVE_BAGGAGE_MANAGER,
LocalBaggageFields.extractFrom(throwAwayBuilder));
FactoryBuilder builder = BaggagePropagation.newFactoryBuilder(propagationFactory);
throwAwayBuilder.configs().forEach(builder::add);
return builder;
}

@SuppressWarnings("deprecation")
private Factory createThrowAwayFactory() {
return new Factory() {
@Override
public <K> Propagation<K> create(KeyFactory<K> keyFactory) {
return null;
}
};
}

@Bean
@Order(0)
BaggagePropagationCustomizer remoteFieldsBaggagePropagationCustomizer() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,6 @@
package org.springframework.boot.actuate.autoconfigure.tracing;

import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.function.Predicate;
import java.util.stream.Stream;
Expand Down Expand Up @@ -85,15 +84,27 @@ public TraceContext decorate(TraceContext context) {
.orElse(context);
}

/**
* Creates a new {@link CompositePropagationFactory}, which uses the given
* {@code injectionTypes} for injection and {@code extractionTypes} for extraction.
* @param properties the propagation properties
* @return the {@link CompositePropagationFactory}
*/
static CompositePropagationFactory create(TracingProperties.Propagation properties) {
return create(properties, null, null);
}

/**
* Creates a new {@link CompositePropagationFactory}, which uses the given
* {@code injectionTypes} for injection and {@code extractionTypes} for extraction.
* @param properties the propagation properties
* @param baggageManager the baggage manager to use, or {@code null}
* @param localFields the local fields, or {@code null}
* @return the {@link CompositePropagationFactory}
*/
static CompositePropagationFactory create(TracingProperties.Propagation properties, BaggageManager baggageManager) {
PropagationFactoryMapper mapper = new PropagationFactoryMapper(baggageManager);
static CompositePropagationFactory create(TracingProperties.Propagation properties, BaggageManager baggageManager,
LocalBaggageFields localFields) {
PropagationFactoryMapper mapper = new PropagationFactoryMapper(baggageManager, localFields);
List<Factory> injectors = properties.getEffectiveProducedTypes().stream().map(mapper::map).toList();
List<Factory> extractors = properties.getEffectiveConsumedTypes().stream().map(mapper::map).toList();
return new CompositePropagationFactory(injectors, extractors);
Expand All @@ -107,8 +118,11 @@ private static class PropagationFactoryMapper {

private final BaggageManager baggageManager;

PropagationFactoryMapper(BaggageManager baggageManager) {
private final LocalBaggageFields localFields;

PropagationFactoryMapper(BaggageManager baggageManager, LocalBaggageFields localFields) {
this.baggageManager = baggageManager;
this.localFields = (localFields != null) ? localFields : LocalBaggageFields.empty();
}

Propagation.Factory map(PropagationType type) {
Expand Down Expand Up @@ -140,8 +154,10 @@ private Propagation.Factory b3Multi() {
* @return the W3C propagation factory
*/
private Propagation.Factory w3c() {
return (this.baggageManager != null) ? new W3CPropagation(this.baggageManager, Collections.emptyList())
: new W3CPropagation();
if (this.baggageManager == null) {
return new W3CPropagation();
}
return new W3CPropagation(this.baggageManager, this.localFields.asList());
}

}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
/*
* Copyright 2012-2023 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package org.springframework.boot.actuate.autoconfigure.tracing;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import brave.baggage.BaggagePropagation;
import brave.baggage.BaggagePropagationConfig;
import brave.baggage.BaggagePropagationConfig.SingleBaggageField;

import org.springframework.util.Assert;
import org.springframework.util.CollectionUtils;

/**
* Local baggage fields.
*
* @author Moritz Halbritter
*/
class LocalBaggageFields {

private final List<String> fields;

LocalBaggageFields(List<String> fields) {
Assert.notNull(fields, "fields must not be null");
this.fields = fields;
}

/**
* Returns the local fields as a list.
* @return the list
*/
List<String> asList() {
return Collections.unmodifiableList(this.fields);
}

/**
* Extracts the local fields from the given propagation factory builder.
* @param builder the propagation factory builder to extract the local fields from
* @return the local fields
*/
static LocalBaggageFields extractFrom(BaggagePropagation.FactoryBuilder builder) {
List<String> localFields = new ArrayList<>();
for (BaggagePropagationConfig config : builder.configs()) {
if (config instanceof SingleBaggageField field) {
if (CollectionUtils.isEmpty(field.keyNames())) {
localFields.add(field.field().name());
}
}
}
return new LocalBaggageFields(localFields);
}

/**
* Creates empty local fields.
* @return the empty local fields
*/
static LocalBaggageFields empty() {
return new LocalBaggageFields(Collections.emptyList());
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
/*
* Copyright 2012-2023 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package org.springframework.boot.actuate.autoconfigure.tracing;

import brave.baggage.BaggageField;
import brave.baggage.BaggagePropagation;
import brave.baggage.BaggagePropagation.FactoryBuilder;
import brave.baggage.BaggagePropagationConfig;
import brave.propagation.Propagation;
import brave.propagation.Propagation.Factory;
import brave.propagation.Propagation.KeyFactory;
import org.junit.jupiter.api.Test;

import static org.assertj.core.api.Assertions.assertThat;

/**
* Tests for {@link LocalBaggageFields}.
*
* @author Moritz Halbritter
*/
class LocalBaggageFieldsTests {

@Test
void extractFromBuilder() {
FactoryBuilder builder = createBuilder();
builder.add(BaggagePropagationConfig.SingleBaggageField.remote(BaggageField.create("remote-field-1")));
builder.add(BaggagePropagationConfig.SingleBaggageField.remote(BaggageField.create("remote-field-2")));
builder.add(BaggagePropagationConfig.SingleBaggageField.local(BaggageField.create("local-field-1")));
builder.add(BaggagePropagationConfig.SingleBaggageField.local(BaggageField.create("local-field-2")));
LocalBaggageFields fields = LocalBaggageFields.extractFrom(builder);
assertThat(fields.asList()).containsExactlyInAnyOrder("local-field-1", "local-field-2");
}

@Test
void empty() {
assertThat(LocalBaggageFields.empty().asList()).isEmpty();
}

@SuppressWarnings("deprecation")
private static FactoryBuilder createBuilder() {
return BaggagePropagation.newFactoryBuilder(new Factory() {
@Override
public <K> Propagation<K> create(KeyFactory<K> keyFactory) {
return null;
}
});
}

}

0 comments on commit b86937b

Please sign in to comment.