com.netflix.server.context.BiDirectional Maven / Gradle / Ivy
/*
* Copyright 2014 Netflix, Inc.
*
* 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
*
* http://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 com.netflix.server.context;
import io.reactivex.netty.contexts.ContextKeySupplier;
import io.reactivex.netty.contexts.ContextsContainer;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* Any contexts added to {@link ContextsContainer#addContext(String, Object, ContextSerializer)} is unidirectional i.e.
* it always flows down the call chain, eg: In the below call graph
Service A ---> Service B ---> Service C
|
---> Service D
------------------ Context X Flows through the call graph ----------------->
* Any changes to Context "X" (which gets added by Service A) in Service B will only be available to Service C & D but not
* service A.
* This interface addresses the use-case where the update in context X must be available to Service A.
*
* By annotating their class with this annotation, the context implementers can guarantee that the context changes,
* flow bi-directionally in the call graph i.e. any code invoked after the change will get the updated context.
* These updates will be effective iff, the following steps are taken:
*
- Whenever the context gets updated, update the same with {@link ContextsContainer} by calling
{@link ContextsContainer#addContext(String, Object, ContextSerializer)}. Implementation may batch these updates if such a
batch can be made reliably.
- Before returning a response from the service, get the updated bidirectional contexts via
{@link ContextsContainer#getModifiedBidirectionalContexts()} and add it as response headers.
- On receiving a response, read & update bidirectional contexts by calling
{@link ContextsContainer#consumeBidirectionalContextsFromResponse(ContextKeySupplier)}
*
* NIWS infrastructure does the last two steps transparently. If you (or your dependencies) are not using NIWS you may
* have to do either one or both of these yourselves based on whether you are using client side, server side or none
* of the NIWS components.
*
* Mergeable contexts
* The infrastructure always over-writes the existing context when writing back a bi-directional context to the current
* thread. In the above example, when service B returns, it reads back the bi-directional context & update the context
* in Service A, at this point the default behavior is to over-write the existing context. However, in some cases, it
* may be desirable to merge the two contexts. If so, is the case with your context, you must implement the interface
* {@link MergeableContext} to indicate that the old context must be merged into the new context.
* For such a context, the infrastructure will call {@link MergeableContext#merge(MergeableContext)} on the old context
* with the new context as the argument.
*
* New Contexts added in Child service
* Any bi-directional contexts added fresh in a child service also flows back in the response.
*
* @author Nitesh Kant
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface BiDirectional {
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy