【发布时间】:2016-06-22 06:37:43
【问题描述】:
我使用本地线程来存储用户请求的特定功能(例如浏览器代理),它过去在 JAVA 7 上可以正常工作,但现在升级到 JAVA 8 后在某些情况下,我看到来自 android 浏览器的请求被处理得好像它的来自 iOS 浏览器,即使它被正确检测为 android 浏览器,但后来在处理请求时,它被另一个线程本地值替换!我不确定这里缺少什么有人可以帮助我吗?我的环境设置(之前/之后)升级是:
- tomcat 8 之前和之后。
- JAVA 从 7 升级到 8。
- Spring 从 4.1.7 升级到 4.2.5
- Spring 安全性从 3.2.3 升级到 4.03
我有一个看起来像这样的安全过滤器:
import java.io.IOException;
import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import org.springframework.security.authentication.UsernamePasswordAuthenticationToken;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.security.web.authentication.WebAuthenticationDetailsSource;
import org.springframework.web.filter.GenericFilterBean;
public class AuthenticationTokenProcessingFilter extends GenericFilterBean {
private final IdentityService identityService;
public AuthenticationTokenProcessingFilter(IdentityService userService) {
this.identityService = userService;
}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
SecurityManager.manager().clearManager();
HttpServletRequest httpRequest = this.getAsHttpRequest(request);
String agent = httpRequest.getHeader("User-Agent");
SecurityManager.manager().setAgent(agent);
...
chain.doFilter(request, response);
}
}
安全管理器如下所示:
import com.appseleon.platform.web.shared.CrossAppConstants;
public class SecurityManager {
private static SecurityManager manager;
private final ThreadLocal<String> agentContext = new ThreadLocal<String>();
private SecurityManager() {
manager = this;
}
public void clearManager() {
agentContext.set(null);
}
public static SecurityManager manager() {
return manager;
}
public String getAgent() {
String os = agentContext.get();
if (os == null) {
os = CrossAppConstants.DEFAULT_OS;
}
return os;
}
public void setAgent(String agent) {
System.out.println("### os detected: " + agent);
agentContext.set(agent);
}
}
最后,在设置代理后,在我的代码的各个区域,我调用 SecurityManager 来获取当前的用户代理:
SecurityManager.manager().getAgent()
谁能帮我找出这个问题的原因,或者找到更可靠的替代方法来解决这个问题?
提前致谢:)
【问题讨论】:
标签: java spring tomcat thread-local