springboot通过过滤器实现将部分请求处理后转发到第三方平台

前言

这几天在搞一个项目,主要是将部分请求进行请求转发到第三方平台。主要是两个操作。请求转发和第三方请求。

过滤器

自定义过滤器 ReqResFilter 必须实现 javax.servlet.Filter
然后添加注解 @WebFilter(javax.servlet.annotation.WebFilter),urlPatterns 过滤器要过滤的URL规则配置,filterName 过滤器的名称。

请求转发

request.getRequestDispatcher(newUri).forward(request,response);

http3

OkHttpClient

OkHttpClient.Builder OBuilder = new OkHttpClient().newBuilder()
		.connectTimeout(10000L, TimeUnit.SECONDS)  // 最大连接时间
		.readTimeout(10000L, TimeUnit.SECONDS)     // 最大读取时间
		.writeTimeout(10000L, TimeUnit.SECONDS);   // 最大写入时间
if (targetUrl.startsWith("https")) {
	// 添加SSL证书信任     【SSLSocketClient】
	builder = builder.sslSocketFactory(SSLSocketClient.getSSLSocketFactory(), SSLSocketClient.getX509TrustManager());
}
OkHttpClient client = builder.build();

Request

// POST
MediaType mediaType = MediaType.parse("application/json");
RequestBody body = RequestBody.create(mediaType, data);
Request.Builder rBudlder= new Request.Builder().url(url).post(body);
// GET
String urlParameter = request.getQueryString();//网址中的参数
Request.Builder rBudlder= new Request.Builder().url(url + "?" + urlParameter).method("GET", null);
// 其他
MediaType mediaType = MediaType.parse("application/json");
RequestBody body = RequestBody.create(mediaType, data);
Request.Builder rBudlder= new Request.Builder().url(url).method(request.getMethod(), body);
// 设置来源 [可忽略]
rBudlder.addHeader("Origin", "https://XXX");
rBudlder.addHeader("Referer", "https://XXX/XX");
Request request = rBudlder.build();

Response

Request req = rBudlder.build();
Response resp = client.newCall(req).execute();

请求

String res = resp.body().string();

参考代码

@Slf4j
@Component
@WebFilter(filterName = "reqFilter", urlPatterns = "/*")
public class ReqForwardFilter implements Filter {
    @Autowired
    private SmartHomeConfigService smartHomeConfigService;
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        HttpServletRequest req = (HttpServletRequest) request;
        HttpServletResponse resp = (HttpServletResponse) response;
        //可能需要请求转发到新的地址
        boolean next = handllerReq(req, resp);
        if (next) {
            chain.doFilter(req, resp);//交给下一个过滤器或servlet处理
        }
    }
    @Override
    public void destroy() {

    }
    public boolean handllerReq(HttpServletRequest request, HttpServletResponse response) {
        String uri = request.getRequestURI();
        if (StringUtil.isNotBlank(uri) && (uri.startsWith("/restapi/") || uri.startsWith("/api/") )) {
        	String token = JToken.accessToken;
            boolean flagTime = JToken.createTime + (JToken.expires * 1000) < System.currentTimeMillis();
            if(StringUtil.isBlank(token)||flagTime){
                boolean falgLogin = getToken();
               if(!falgLogin){
                   // 登录失败
                   try {
                       PrintWriter writer = response.getWriter();
                       writer.write("登录失败");
                       writer.close();
                       return false;
                   }catch (IOException e){
                       log.error("e:"+e);
                   }
               }
            }
            if(uri.startsWith("/restapi/")){
                try {
                    String data = IOUtils.toString(request.getInputStream());
                    log.info("过滤器请求参数 data :"+data);
                    log.info("过滤器请求路径 uri :"+uri);
                    String newUri = "/"+uri+"?uri="+uri;
                    // 传参
                    request.setAttribute("json",data);
                    request.getRequestDispatcher(newUri).forward(request,response);
                    return false;
                }catch (Exception e){
                    log.error("转发失败"+e.toString());
                }

            }else if(uri.startsWith("/api/")){
                try {
                    String data = IOUtils.toString(request.getInputStream());
                    request.setAttribute("json",data);
                    String newUri = "/testFilter/api?uri="+uri;
                    request.getRequestDispatcher(newUri).forward(request,response);
                    return false;
                }catch (Exception e){
                    log.error("转发失败"+e.toString());
                }
            }
        }
        return true;
    }

    public boolean getToken(){
        OkHttpClient.Builder okBuilder = new OkHttpClient().newBuilder()
                .connectTimeout(10000L, TimeUnit.SECONDS)
                .readTimeout(10000L, TimeUnit.SECONDS)
                .writeTimeout(10000L, TimeUnit.SECONDS);
        // https ssl授权
        JSONObject json = new JSONObject();
        ConfigEntity sConfig = ConfigService.selectOne(new EntityWrapper<ConfigEntity>());
        YZYJToken.id = sConfig.getUid();
        json.put("identifier",sConfig.getIdentifier());
        json.put("credential",sConfig.getCredential());
        json.put("type",sConfig.getAuthType());
        String data = json.toString();

        okBuilder = okBuilder.sslSocketFactory(SSLSocketClient.getSSLSocketFactory(), SSLSocketClient.getX509TrustManager());
        OkHttpClient okHttpClient = okBuilder.build();

        MediaType mediaType = MediaType.parse("application/json");
        RequestBody body = RequestBody.create(mediaType, data);
        Request.Builder rBudlder= new Request.Builder().url("https://url/restapi/tokens").post(body);

        if(StringUtil.isNotBlank(YZYJToken.refreshToken)){
            rBudlder.addHeader("authorization", "Bearer "+JToken.refresh);
        }
        rBudlder.addHeader("authority", "url");
        rBudlder.addHeader("content-type", "application/json");

        Request request = rBudlder.build();
        try {
            Response response = okHttpClient.newCall(request).execute();
            String reslut = response.body().string();
            log.info("返回结果"+reslut);

            if(reslut.indexOf("token")>-1){
                JSONObject resJson = JSONObject.parseObject(reslut);
                JToken.type = resJson.getString("type");
                JToken.token = resJson.getString("token");
                JToken.expires = resJson.getInteger("expires");
                JToken.refresh = resJson.getString("refresh");
                JToken.createTime = System.currentTimeMillis();
                return true;
            }
        }catch (IOException e){
            log.error("e:"+e);
        }
        return false;
    }
}
  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: 在 Spring Boot实现过滤器有两种方法: 1. 通过编写过滤器类并使用注解 @WebFilter 将其注册为过滤器。例如: ``` @WebFilter(urlPatterns = "/*") public class MyFilter implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { // 处理过滤逻辑 chain.doFilter(request, response); } } ``` 2. 通过编写过滤器类并通过编写配置类来注册过滤器。例如: ``` @Configuration public class WebConfiguration { @Bean public FilterRegistrationBean<MyFilter> filterRegistrationBean() { FilterRegistrationBean<MyFilter> registrationBean = new FilterRegistrationBean<>(); registrationBean.setFilter(new MyFilter()); registrationBean.addUrlPatterns("/*"); return registrationBean; } } ``` 注意:在 Spring Boot 2.3 及更高版本中,使用注解 @WebFilter 注册过滤器时,需要添加以下依赖: ``` <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> </dependency> ``` 在 Spring Boot 2.2 及更低版本中,不需要添加此依赖。 ### 回答2: Spring Boot可以通过使用过滤器(Filter)实现对于应用程序的请求进行预处理或后处理的功能。在Spring Boot中,实现过滤器非常简单。 首先,我们需要创建一个实现javax.servlet.Filter接口的过滤器类。我们可以在过滤器类中实现请求的预处理或后处理逻辑。例如,我们可以在过滤器中对请求进行日志记录、授权验证、防止跨站点请求伪造(CSRF)等操作。 接下来,我们需要在Spring Boot应用程序的配置文件(例如application.properties或application.yml)中配置过滤器。我们可以使用Spring Boot提供的特定属性来配置过滤器的顺序、URL匹配模式和其他属性。例如,我们可以使用以下配置来注册并配置过滤器spring.servlet.filter.order=1 spring.servlet.filter.dispatcher-types=REQUEST spring.servlet.filter.url-pattern=/api/* spring.servlet.filter.init-param.debug=true 最后,我们需要在Spring Boot应用程序的启动类(例如带有@SpringBootApplication注解的主类)中添加一个注解@ServletComponentScan,以扫描并注册我们创建的过滤器类。这将使Spring Boot自动识别并加载该过滤器实现过滤器后,当应用程序收到请求时,过滤器将自动拦截请求并执行预处理或后处理逻辑。过滤器的执行顺序可以通过配置文件中的spring.servlet.filter.order属性进行调整。 通过以上步骤,我们就可以在Spring Boot应用程序中实现一个过滤器过滤器可以帮助我们在请求到达控制器之前或返回给客户端之前对请求进行额外处理,从而实现请求的自定义处理逻辑。 ### 回答3: Spring Boot提供了一种简单且易于实现的方式来实现过滤器。在Spring Boot中,我们可以通过创建一个实现javax.servlet.Filter接口的类来实现过滤器。 首先,我们需要在Spring Boot应用程序的代码中创建一个新的类并实现Filter接口。该类需要实现doFilter方法,该方法的目的是在请求进入Web应用程序之前或之后执行一些操作。该方法接受三个参数,分别是ServletRequest、ServletResponse和FilterChain。 在doFilter方法中,我们可以编写我们的逻辑代码来进行过滤操作。例如,我们可以检查请求的URL,如果满足我们的条件,则可以进行一些特定的处理,比如记录日志、验证权限等。然后,我们需要调用FilterChain对象的doFilter方法,以便将请求传递给下一个过滤器或Web应用程序。 接下来,我们还需要在Spring Boot应用程序的配置类中注册我们的过滤器。可以通过使用@Bean注解将过滤器类声明为一个bean,并将其添加到Spring Boot应用程序的过滤器链中。这将确保在请求到达控制器之前,我们的过滤器将被调用。 最后,我们可以通过在配置类中使用@Order注解来指定过滤器的执行顺序。如果有多个过滤器,根据需要可以使用不同的顺序来执行它们。 总的来说,Spring Boot通过实现javax.servlet.Filter接口和在配置类中注册过滤器的方式来实现过滤器。通过编写逻辑代码并将过滤器添加到过滤器链中,我们可以在请求进入Web应用程序之前或之后进行一些自定义操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值