kl个人博客 首页>>quarkus>>Quarkus中过滤器filter和跨域cors问题(4)

Quarkus中过滤器filter和跨域cors问题(4)

Quarkus中过滤器filter和跨域cors问题(4)

前言

Quarkus中的web模块是基于java标准web规范jax-rs构建的,实现则选用了jboss的resteasy。这部分只是请求路由转发部分实现。真正的请求接收则使用了eclipse开源的vert.x框架,底层也是基于netty的一个响应式开发框架。Quarkus将vert.x和resteasy集成在了一起,所以支持响应式和非响应式应用混合开发,这也是Quarkus的一大卖点。基于以上的认知,我们来看看在Quarkus中,怎么写过滤器和解决跨域的问题

Quarkus技术交流QQ群:871808563

resteasy4.4.5开发文档:https://docs.jboss.org/resteasy/docs/4.5.5.Final

vert'x开发文档:https://vertx.io/docs/vertx-web/java/

web依赖

        <dependency>
            <groupId>io.quarkus</groupId>
            <artifactId>quarkus-resteasy-jsonb</artifactId>
        </dependency>

过滤器filter开发

resteasy的filter

/**
 * @author kl : http://kailing.pub
 * @version 1.0
 * @date 2020/7/9 15:34
 */
@Priority(Priorities.USER + 1)
@Provider
public class MyFilter implements ContainerRequestFilter, ContainerResponseFilter {

    private volatile CurrentVertxRequest currentVertxRequest;

    CurrentVertxRequest currentVertxRequest() {
        if (currentVertxRequest == null) {
            currentVertxRequest = CDI.current().select(CurrentVertxRequest.class).get();
        }
        return currentVertxRequest;
    }

    @Override
    public void filter(ContainerRequestContext requestContext) throws IOException {
        RoutingContext httpServerRequest = this.currentVertxRequest().getCurrent();
        String str = httpServerRequest.getBodyAsString();
        JsonObject jsonObject = httpServerRequest.getBodyAsJson();
        RequestImpl request = (RequestImpl) requestContext.getRequest();
        System.out.println("拦截到请求了");
    }

    @Override
    public void filter(ContainerRequestContext requestContext, ContainerResponseContext responseContext) throws IOException {
        System.out.println("拦截到响应了");
    }
}

实现ContainerRequestFilter、ContainerResponseFilter接口,可以分别拦截请求和响应。最后使用@Provider注解标记,@Priority注解用于表明优先级,值越大,优先级越高。前面已经说过,Quarkus虽然使用了resteasy,但是请求是使用vert'x来接收的,所以在拦截器实现里,可以通过上下文信息拿到vert'x的路由信息RoutingContext

vertx的filter

/**
 * @author kl : http://kailing.pub
 * @version 1.0
 * @date 2020/7/9 18:15
 */
@ApplicationScoped
public class MyFilter {

    public void initfilter(@Observes Filters filters) {
        filters.register(routingContext -> {
            HttpServerRequest httpServerRequest = routingContext.request();
            ForkJoinPool.commonPool().submit(()->{
                System.out.println("进入vertx拦截器,下面是header参数:");
            });
            httpServerRequest.headers().forEach(stringStringEntry -> {
                System.out.println("key:"+stringStringEntry.getKey() +",value:"+stringStringEntry.getValue());
            });
            routingContext.next();//这一句不能漏掉,让拦截器继续往下走的逻辑
        }, 100);
    }
}

Quarkus中的跨域

1、Quarkus中解决跨域问题,可以从两个层面来分析,一个是resteasy的角度。resteasy中内置了CorsFilter过滤器,我们只需要激活它即可解决跨域问题。如:

/**
 * @author kl : http://kailing.pub
 * @version 1.0
 * @date 2020/7/9 16:46
 */
@Provider
public class CorsFilter extends org.jboss.resteasy.plugins.interceptors.CorsFilter {
    public CorsFilter() {
        super.setAllowedMethods("OPTIONS, GET, POST, DELETE, PUT, PATCH");
        super.setAllowedHeaders("*");
        super.getAllowedOrigins().add("*");
    }
}
2、Quarkus本身也做了跨域的解决方案,是基于vert't的角度来实现的,代码见io.quarkus.vertx.http.runtime.cors.CORSFilter。从vertx的handler就拦截到了请求并做了跨域处理,但是跨域功能默认不是开启的,我们可以基于以下的配置来激活它,并进行相关的设置:
quarkus.http.cors=true
quarkus.http.cors.origins=*
quarkus.http.cors.headers=accept, authorization, content-type, x-requested-with
quarkus.http.cors.methods=GET, OPTIONS


kl个人博客