com.netflix.hystrix.strategy.concurrency.HystrixRequestVariableHolder Maven / Gradle / Ivy
/**
* Copyright 2012 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.hystrix.strategy.concurrency;
import java.util.concurrent.ConcurrentHashMap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Factory that encompasses functionality of {@link HystrixRequestVariable} for internal Hystrix code.
*
* This is used as a layer between the actual {@link HystrixRequestVariable} and calling code to allow injected implementations of {@link HystrixConcurrencyStrategy}.
*
* Typically a {@link HystrixRequestVariable} would be statically referenced (similar to a ThreadLocal) but to allow dynamic injection we instead statically reference this class which can then
* dynamically fetch the correct implementation and statically retain an instance across threads within a context (such as {@link HystrixRequestContext}.
*
* @param
*
* @ExcludeFromJavadoc
*/
public class HystrixRequestVariableHolder {
static final Logger logger = LoggerFactory.getLogger(HystrixRequestVariableHolder.class);
private static ConcurrentHashMap> requestVariableInstance = new ConcurrentHashMap>();
private final HystrixRequestVariableLifecycle lifeCycleMethods;
public HystrixRequestVariableHolder(HystrixRequestVariableLifecycle lifeCycleMethods) {
this.lifeCycleMethods = lifeCycleMethods;
}
@SuppressWarnings("unchecked")
public T get(HystrixConcurrencyStrategy concurrencyStrategy) {
/*
* 1) Fetch RequestVariable implementation from cache.
* 2) If no implementation is found in cache then construct from factory.
* 3) Cache implementation from factory as each object instance needs to be statically cached to be relevant across threads.
*/
RVCacheKey key = new RVCacheKey(this, concurrencyStrategy);
HystrixRequestVariable> rvInstance = requestVariableInstance.get(key);
if (rvInstance == null) {
requestVariableInstance.putIfAbsent(key, concurrencyStrategy.getRequestVariable(lifeCycleMethods));
/*
* A safety check to help debug problems if someone starts injecting dynamically created HystrixConcurrencyStrategy instances - which should not be done and has no good reason to be done.
*
* The 100 value is arbitrary ... just a number far higher than we should see.
*/
if (requestVariableInstance.size() > 100) {
logger.warn("Over 100 instances of HystrixRequestVariable are being stored. This is likely the sign of a memory leak caused by using unique instances of HystrixConcurrencyStrategy instead of a single instance.");
}
}
return (T) requestVariableInstance.get(key).get();
}
private static class RVCacheKey {
private final HystrixRequestVariableHolder> rvHolder;
private final HystrixConcurrencyStrategy concurrencyStrategy;
private RVCacheKey(HystrixRequestVariableHolder> rvHolder, HystrixConcurrencyStrategy concurrencyStrategy) {
this.rvHolder = rvHolder;
this.concurrencyStrategy = concurrencyStrategy;
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + ((concurrencyStrategy == null) ? 0 : concurrencyStrategy.hashCode());
result = prime * result + ((rvHolder == null) ? 0 : rvHolder.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
RVCacheKey other = (RVCacheKey) obj;
if (concurrencyStrategy == null) {
if (other.concurrencyStrategy != null)
return false;
} else if (!concurrencyStrategy.equals(other.concurrencyStrategy))
return false;
if (rvHolder == null) {
if (other.rvHolder != null)
return false;
} else if (!rvHolder.equals(other.rvHolder))
return false;
return true;
}
}
}