Filter
クラスを拡張するクラスがあり、次のようになっています。
@Component
@Order(Ordered.HIGHEST_PRECEDENCE)
public class CorsFilter implements Filter {
@Override
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
System.out.println("--------------------------------------");
HttpServletRequest request = (HttpServletRequest) req;
req.setAttribute("test", "test");
final HttpServletResponse response = (HttpServletResponse) res;
response.setHeader("Access-Control-Allow-Origin", "*");
response.setHeader("Access-Control-Allow-Methods", "POST, PUT, GET, OPTIONS, DELETE");
response.setHeader("Access-Control-Allow-Headers", "Authorization, Content-Type");
response.setHeader("Access-Control-Max-Age", "3600");
if ("OPTIONS".equalsIgnoreCase(((HttpServletRequest) req).getMethod())) {
response.setStatus(HttpServletResponse.SC_OK);
} else {
chain.doFilter(req, res);
}
}
問題は、カスタムheader
をrequest
に追加できるかどうかです。 request
とreq
内で使用できるすべてのメソッドを実行しましたが、方法が見つかりませんでした。しかし、debug
モードでは、CoyoteRequest
内にheaders
のリストがあることがわかりました。カスタムheader
をそこに追加するにはどうすればよいですか?
HttpServletRequest
objectにヘッダーを設定することはできませんが、ラッパークラスを使用できます。
このガイドを見てください: http://wilddiary.com/adding-custom-headers-Java-httpservletrequest/
将来のためにリンクが無効になる場合に備えて:
final class MutableHttpServletRequest extends HttpServletRequestWrapper {
// holds custom header and value mapping
private final Map<String, String> customHeaders;
public MutableHttpServletRequest(HttpServletRequest request){
super(request);
this.customHeaders = new HashMap<String, String>();
}
public void putHeader(String name, String value){
this.customHeaders.put(name, value);
}
public String getHeader(String name) {
// check the custom headers first
String headerValue = customHeaders.get(name);
if (headerValue != null){
return headerValue;
}
// else return from into the original wrapped object
return ((HttpServletRequest) getRequest()).getHeader(name);
}
public Enumeration<String> getHeaderNames() {
// create a set of the custom header names
Set<String> set = new HashSet<String>(customHeaders.keySet());
// now add the headers from the wrapped request object
@SuppressWarnings("unchecked")
Enumeration<String> e = ((HttpServletRequest) getRequest()).getHeaderNames();
while (e.hasMoreElements()) {
// add the names of the request headers into the list
String n = e.nextElement();
set.add(n);
}
// create an enumeration from the set and return
return Collections.enumeration(set);
}
}
用法:
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
HttpServletRequest req = (HttpServletRequest) request;
MutableHttpServletRequest mutableRequest = new MutableHttpServletRequest(req);
...
mutableRequest.putHeader("x-custom-header", "custom value");
chain.doFilter(mutableRequest, response);
}
Guy Smorodinskyの答えは問題ありませんが、たとえば、Springの場合、_@RequestHeader
_アノテーションでヘッダー値を取得するときにこれを使用するため、Enumeration<String> getHeaders(String name)
などの追加のメソッドをオーバーライドすることができます。
コード例は次のようになります。
_ @Override
public Enumeration<String> getHeaders(String name) {
Set<String> headerValues = new HashSet<>();
headerValues.add(this.headers.get(name));
Enumeration<String> underlyingHeaderValues = ((HttpServletRequest) getRequest()).getHeaders(name);
while (underlyingHeaderValues.hasMoreElements()) {
headerValues.add(underlyingHeaderValues.nextElement());
}
return Collections.enumeration(headerValues);
}
_
Springのフィルターを介してOAuth2Authenticationの値からカスタムヘッダーX-Loginにユーザー名を設定し、コントローラーで引数の1つとして使用する方法の例@RequestHeader(X_LOGIN) String login
import static Java.util.Collections.enumeration;
import static Java.util.Collections.singleton;
@Component
public class HeaderLoginFilter extends GenericFilterBean {
public static final String X_LOGIN = "X-Login";
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
OAuth2Authentication auth = (OAuth2Authentication) SecurityContextHolder.getContext().getAuthentication();
if (auth.getUserAuthentication() == null) {
//ignore, header value is set in my case
chain.doFilter(request, response);
return;
}
//filling custom header with value from auth
HttpServletRequestWrapper wrapper = new HttpServletRequestWrapper((HttpServletRequest) request) {
@Override
public Enumeration<String> getHeaders(String name) {
if (X_CLIENT_LOGIN.equals(name))
return enumeration(singleton(auth
.getUserAuthentication()
.getName()));
return super.getHeaders(name);
}
};
chain.doFilter(wrapper, response);
}
}