Skip to content

Commit 9b3ecd4

Browse files
Dariusz Jędrzejczykrstoyanchev
Dariusz Jędrzejczyk
andauthored
Introducing ContextSnapshotFactory (#105)
This change addresses these two goals: * To allow clearing ThreadLocal values not present in source context when opening a scope. * To clean the API from accumulated static methods in ContextSnapshot and make the use more intuitive and comprehensible. The responsibility of ContextSnapshot has been reduced. ContextSnapshotFactory is introduced to capture the shared concerns: * ContextRegistry to use when capturing and setting ThreadLocal values. * Predicate for keys used when capturing state to a ContextSnpashot. * Flag whether to clear ThreadLocal values missing in the used context object. Multiple deprecations have been introduced with pointers on how to combine the new building blocks to achieve the desired effect. ContextSnapshotFactory contains a Builder interface with a default implementation provided. --------- Co-authored-by: Rossen Stoyanchev <[email protected]>
1 parent 4e9d852 commit 9b3ecd4

10 files changed

+982
-253
lines changed

context-propagation/src/main/java/io/micrometer/context/ContextExecutorService.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -154,9 +154,12 @@ public static ExecutorService wrap(ExecutorService service, Supplier<ContextSnap
154154
* Variant of {@link #wrap(ExecutorService, Supplier)} that uses
155155
* {@link ContextSnapshot#captureAll(Object...)} to create the context snapshot.
156156
* @param service the executorService to wrap
157+
* @deprecated use {@link #wrap(ExecutorService, Supplier)}
157158
*/
159+
@Deprecated
158160
public static ExecutorService wrap(ExecutorService service) {
159-
return new ContextExecutorService<>(service, ContextSnapshot::captureAll);
161+
return wrap(service,
162+
() -> DefaultContextSnapshotFactory.captureAll(ContextRegistry.getInstance(), key -> true, false));
160163
}
161164

162165
}

context-propagation/src/main/java/io/micrometer/context/ContextRegistry.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -122,7 +122,7 @@ public void setValue(V value) {
122122
}
123123

124124
@Override
125-
public void reset() {
125+
public void setValue() {
126126
resetTask.run();
127127
}
128128
});

context-propagation/src/main/java/io/micrometer/context/ContextScheduledExecutorService.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -79,9 +79,12 @@ public static ScheduledExecutorService wrap(ScheduledExecutorService service, Su
7979
* Variant of {@link #wrap(ScheduledExecutorService, Supplier)} that uses
8080
* {@link ContextSnapshot#captureAll(Object...)} to create the context snapshot.
8181
* @param service the executorService to wrap
82+
* @deprecated use {@link #wrap(ScheduledExecutorService, Supplier)}
8283
*/
84+
@Deprecated
8385
public static ScheduledExecutorService wrap(ScheduledExecutorService service) {
84-
return new ContextScheduledExecutorService(service, ContextSnapshot::captureAll);
86+
return wrap(service,
87+
() -> DefaultContextSnapshotFactory.captureAll(ContextRegistry.getInstance(), key -> true, false));
8588
}
8689

8790
}

context-propagation/src/main/java/io/micrometer/context/ContextSnapshot.java

+55-59
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,9 @@
2424
/**
2525
* Holds values extracted from {@link ThreadLocal} and other types of context and exposes
2626
* methods to propagate those values.
27-
*
27+
* <p>
28+
* Use {@link ContextSnapshotFactory#builder()} to configure a factory to work with
29+
* snapshots.
2830
* <p>
2931
* Implementations are disallowed to store {@code null} values. If a {@link ThreadLocal}
3032
* is not set, or it's value is {@code null}, there is no way of distinguishing one from
@@ -34,11 +36,9 @@
3436
* {@link ContextAccessor#readValues(Object, Predicate, Map)}, and should likewise ignore
3537
* {@code null} values from {@link ContextAccessor#readValue(Object, Object)}.
3638
*
37-
* <p>
38-
* Use static factory methods on this interface to create a snapshot.
39-
*
4039
* @author Rossen Stoyanchev
4140
* @author Brian Clozel
41+
* @author Dariusz Jędrzejczyk
4242
* @since 1.0.0
4343
*/
4444
public interface ContextSnapshot {
@@ -72,6 +72,7 @@ public interface ContextSnapshot {
7272
/**
7373
* Variant of {@link #setThreadLocals()} with a predicate to select context values by
7474
* key.
75+
* @param keyPredicate selects keys for use when setting {@link ThreadLocal} values
7576
* @return an object that can be used to reset {@link ThreadLocal} values at the end
7677
* of the context scope, either removing them or restoring their previous values, if
7778
* any.
@@ -82,6 +83,7 @@ public interface ContextSnapshot {
8283
* Return a new {@code Runnable} that sets {@code ThreadLocal} values from the
8384
* snapshot around the invocation of the given {@code Runnable}.
8485
* @param runnable the runnable to instrument
86+
* @return wrapped instance
8587
*/
8688
default Runnable wrap(Runnable runnable) {
8789
return () -> {
@@ -96,6 +98,7 @@ default Runnable wrap(Runnable runnable) {
9698
* snapshot around the invocation of the given {@code Callable}.
9799
* @param callable the callable to instrument
98100
* @param <T> the type of value produced by the {@code Callable}
101+
* @return wrapped instance
99102
*/
100103
default <T> Callable<T> wrap(Callable<T> callable) {
101104
return () -> {
@@ -110,6 +113,7 @@ default <T> Callable<T> wrap(Callable<T> callable) {
110113
* snapshot around the invocation of the given {@code Consumer}.
111114
* @param consumer the callable to instrument
112115
* @param <T> the type of value produced by the {@code Callable}
116+
* @return wrapped instance
113117
*/
114118
default <T> Consumer<T> wrap(Consumer<T> consumer) {
115119
return value -> {
@@ -123,6 +127,7 @@ default <T> Consumer<T> wrap(Consumer<T> consumer) {
123127
* Return a new {@code Executor} that sets {@code ThreadLocal} values from the
124128
* snapshot around the invocation of any executed, {@code Runnable}.
125129
* @param executor the executor to instrument
130+
* @return wrapped instance
126131
* @see ContextExecutorService
127132
* @see ContextScheduledExecutorService
128133
*/
@@ -139,7 +144,10 @@ default Executor wrapExecutor(Executor executor) {
139144
* instance.
140145
* @param contexts one more context objects to extract values from
141146
* @return a snapshot with saved context values
147+
* @deprecated use {@link ContextSnapshotFactory#captureAll(Object...)} on a factory
148+
* obtained via a {@link ContextSnapshotFactory#builder()}.
142149
*/
150+
@Deprecated
143151
static ContextSnapshot captureAll(Object... contexts) {
144152
return captureAll(ContextRegistry.getInstance(), contexts);
145153
}
@@ -151,7 +159,11 @@ static ContextSnapshot captureAll(Object... contexts) {
151159
* @param registry the registry to use
152160
* @param contexts one more context objects to extract values from
153161
* @return a snapshot with saved context values
162+
* @deprecated use {@link ContextSnapshotFactory#captureAll(Object...)} on a factory
163+
* obtained via a {@link ContextSnapshotFactory#builder()} combined with
164+
* {@link ContextSnapshotFactory.Builder#contextRegistry(ContextRegistry)}.
154165
*/
166+
@Deprecated
155167
static ContextSnapshot captureAll(ContextRegistry registry, Object... contexts) {
156168
return captureAllUsing(key -> true, registry, contexts);
157169
}
@@ -163,61 +175,23 @@ static ContextSnapshot captureAll(ContextRegistry registry, Object... contexts)
163175
* @param registry the registry with the accessors to use
164176
* @param contexts one more context objects to extract values from
165177
* @return a snapshot with saved context values
178+
* @deprecated use {@link ContextSnapshotFactory#captureAll(Object...)} on a factory
179+
* obtained via a {@link ContextSnapshotFactory#builder()} and configure
180+
* {@link ContextSnapshotFactory.Builder#captureKeyPredicate(Predicate)} and
181+
* {@link ContextSnapshotFactory.Builder#contextRegistry(ContextRegistry)}.
166182
*/
183+
@Deprecated
167184
static ContextSnapshot captureAllUsing(Predicate<Object> keyPredicate, ContextRegistry registry,
168185
Object... contexts) {
169-
170-
return DefaultContextSnapshot.captureAll(registry, keyPredicate, contexts);
171-
}
172-
173-
/**
174-
* Create a {@link ContextSnapshot} by reading values from the given context objects.
175-
* <p>
176-
* Values captured multiple times are overridden in the snapshot by the order of
177-
* contexts given as arguments.
178-
* @param contexts the contexts to read values from
179-
* @return the created {@link ContextSnapshot}
180-
* @since 1.0.3
181-
*/
182-
static ContextSnapshot captureFromContext(Object... contexts) {
183-
return DefaultContextSnapshot.captureFromContext(key -> true, ContextRegistry.getInstance(), null, contexts);
184-
}
185-
186-
/**
187-
* Create a {@link ContextSnapshot} by reading values from the given context objects.
188-
* <p>
189-
* Values captured multiple times are overridden in the snapshot by the order of
190-
* contexts given as arguments.
191-
* @param registry the registry to use
192-
* @param contexts the contexts to read values from
193-
* @return the created {@link ContextSnapshot}
194-
* @since 1.0.3
195-
*/
196-
static ContextSnapshot captureFromContext(ContextRegistry registry, Object... contexts) {
197-
return DefaultContextSnapshot.captureFromContext(key -> true, registry, null, contexts);
198-
}
199-
200-
/**
201-
* Create a {@link ContextSnapshot} by reading values from the given context objects.
202-
* <p>
203-
* Values captured multiple times are overridden in the snapshot by the order of
204-
* contexts given as arguments.
205-
* @param keyPredicate predicate for context value keys
206-
* @param registry the registry to use
207-
* @param contexts the contexts to read values from
208-
* @return the created {@link ContextSnapshot}
209-
* @since 1.0.3
210-
*/
211-
static ContextSnapshot captureFromContext(Predicate<Object> keyPredicate, ContextRegistry registry,
212-
Object... contexts) {
213-
return DefaultContextSnapshot.captureFromContext(keyPredicate, registry, null, contexts);
186+
return DefaultContextSnapshotFactory.captureAll(registry, keyPredicate, false, contexts);
214187
}
215188

216189
/**
217190
* Create a {@link ContextSnapshot} by reading values from the given context object.
218191
* @param context the context to read values from
219192
* @return the created {@link ContextSnapshot}
220-
* @deprecated as of 1.0.3 in favor of {@link #captureFromContext(Object...)}
193+
* @deprecated use {@link ContextSnapshotFactory#captureFrom(Object...)} on a factory
194+
* obtained via a {@link ContextSnapshotFactory#builder()}.
221195
*/
222196
@Deprecated
223197
static ContextSnapshot captureFrom(Object context) {
@@ -229,12 +203,13 @@ static ContextSnapshot captureFrom(Object context) {
229203
* @param context the context to read values from
230204
* @param registry the registry to use
231205
* @return the created {@link ContextSnapshot}
232-
* @deprecated as of 1.0.3 in favor of
233-
* {@link #captureFromContext(ContextRegistry, Object...)}
206+
* @deprecated use {@link ContextSnapshotFactory#captureFrom(Object...)} on a factory
207+
* obtained via a {@link ContextSnapshotFactory#builder()} combined with
208+
* {@link ContextSnapshotFactory.Builder#contextRegistry(ContextRegistry)}.
234209
*/
235210
@Deprecated
236211
static ContextSnapshot captureFrom(Object context, ContextRegistry registry) {
237-
return DefaultContextSnapshot.captureFromContext(key -> true, registry, null, context);
212+
return DefaultContextSnapshotFactory.captureFromContext(key -> true, false, registry, null, context);
238213
}
239214

240215
/**
@@ -243,12 +218,14 @@ static ContextSnapshot captureFrom(Object context, ContextRegistry registry) {
243218
* @param keyPredicate predicate for context value keys
244219
* @param registry the registry to use
245220
* @return the created {@link ContextSnapshot}
246-
* @deprecated as of 1.0.3 in favor of
247-
* {@link #captureFromContext(Predicate, ContextRegistry, Object...)}
221+
* @deprecated use {@link ContextSnapshotFactory#captureFrom(Object...)} on a factory
222+
* obtained via a {@link ContextSnapshotFactory#builder()} and configure
223+
* {@link ContextSnapshotFactory.Builder#captureKeyPredicate(Predicate)} and
224+
* {@link ContextSnapshotFactory.Builder#contextRegistry(ContextRegistry)}.
248225
*/
249226
@Deprecated
250227
static ContextSnapshot captureFrom(Object context, Predicate<Object> keyPredicate, ContextRegistry registry) {
251-
return DefaultContextSnapshot.captureFromContext(keyPredicate, registry, null, context);
228+
return DefaultContextSnapshotFactory.captureFromContext(keyPredicate, false, registry, null, context);
252229
}
253230

254231
/**
@@ -258,9 +235,14 @@ static ContextSnapshot captureFrom(Object context, Predicate<Object> keyPredicat
258235
* @return an object that can be used to reset {@link ThreadLocal} values at the end
259236
* of the context scope, either removing them or restoring their previous values, if
260237
* any.
238+
* @deprecated use
239+
* {@link ContextSnapshotFactory#setThreadLocalsFrom(Object, String...)} with no keys
240+
* on a factory obtained via a {@link ContextSnapshotFactory#builder()}.
261241
*/
242+
@Deprecated
262243
static Scope setAllThreadLocalsFrom(Object sourceContext) {
263-
return DefaultContextSnapshot.setAllThreadLocalsFrom(sourceContext, ContextRegistry.getInstance());
244+
return DefaultContextSnapshotFactory.setAllThreadLocalsFrom(sourceContext, ContextRegistry.getInstance(),
245+
false);
264246
}
265247

266248
/**
@@ -271,9 +253,14 @@ static Scope setAllThreadLocalsFrom(Object sourceContext) {
271253
* @return an object that can be used to reset {@link ThreadLocal} values at the end
272254
* of the context scope, either removing them or restoring their previous values, if
273255
* any.
256+
* @deprecated use
257+
* {@link ContextSnapshotFactory#setThreadLocalsFrom(Object, String...)} with no keys
258+
* on a factory obtained via a {@link ContextSnapshotFactory#builder()} combined with
259+
* {@link ContextSnapshotFactory.Builder#contextRegistry(ContextRegistry)}.
274260
*/
261+
@Deprecated
275262
static Scope setAllThreadLocalsFrom(Object sourceContext, ContextRegistry contextRegistry) {
276-
return DefaultContextSnapshot.setAllThreadLocalsFrom(sourceContext, contextRegistry);
263+
return DefaultContextSnapshotFactory.setAllThreadLocalsFrom(sourceContext, contextRegistry, false);
277264
}
278265

279266
/**
@@ -286,7 +273,11 @@ static Scope setAllThreadLocalsFrom(Object sourceContext, ContextRegistry contex
286273
* @return an object that can be used to reset {@link ThreadLocal} values at the end
287274
* of the context scope, either removing them or restoring their previous values, if
288275
* any.
276+
* @deprecated use
277+
* {@link ContextSnapshotFactory#setThreadLocalsFrom(Object, String...)} on a factory
278+
* obtained via a {@link ContextSnapshotFactory#builder()}.
289279
*/
280+
@Deprecated
290281
static Scope setThreadLocalsFrom(Object sourceContext, String... keys) {
291282
return setThreadLocalsFrom(sourceContext, ContextRegistry.getInstance(), keys);
292283
}
@@ -300,9 +291,14 @@ static Scope setThreadLocalsFrom(Object sourceContext, String... keys) {
300291
* @return an object that can be used to reset {@link ThreadLocal} values at the end
301292
* of the context scope, either removing them or restoring their previous values, if
302293
* any.
294+
* @deprecated use
295+
* {@link ContextSnapshotFactory#setThreadLocalsFrom(Object, String...)} on a factory
296+
* obtained via a {@link ContextSnapshotFactory#builder()} combined with
297+
* {@link ContextSnapshotFactory.Builder#contextRegistry(ContextRegistry)}.
303298
*/
299+
@Deprecated
304300
static Scope setThreadLocalsFrom(Object sourceContext, ContextRegistry contextRegistry, String... keys) {
305-
return DefaultContextSnapshot.setThreadLocalsFrom(sourceContext, contextRegistry, keys);
301+
return DefaultContextSnapshotFactory.setThreadLocalsFrom(sourceContext, contextRegistry, false, keys);
306302
}
307303

308304
/**

0 commit comments

Comments
 (0)