Springboot 获取http数据、打印HTTP参数的4种方式 (便于生产排查问题)

这篇具有很好参考价值的文章主要介绍了Springboot 获取http数据、打印HTTP参数的4种方式 (便于生产排查问题)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Java的话本地打断点可以调试获取rest入参(http header),但是在生产环境可能我们获取入参(Http header/parameter)可能就没有那么的轻松了。我们可能在header中放置了很多自定的参数用来鉴权或者其他用途。如果排查问题的时候需要这些参数,我们有很多种选择去获取这些参数。

  1. 输出到应用日志中,比如使用logback,log.error(xxx)
  2. 借助nginx 输出到access.log日志中
  3. 借助Skywalking/zipkin等中间件输出到链路中
  4. 网关日志中输出

1. 输出到应用日志中

我们可以借助Springboot的拦截器在进入rest controller 之前将request header / param 输出出来,在rest controller调用结束之后将response header / param输出。

LogInterceptor
拦截器,注意拦截器和过滤器的区别,过滤器属于Tomcat/Jetty/… Servlet 容器的生命周期维护的,要早于拦截器。过滤器是Springboot维护的拦截,在handler mapping 映射之后先去调用拦截器之后在调用controller。

Springboot 获取http数据、打印HTTP参数的4种方式 (便于生产排查问题)

拦截器拦截的就是上图4的这部分。

@Component
@Slf4j
public class LogInterceptor extends HandlerInterceptorAdapter {

    private static ThreadLocal<Long> threadLocal = new ThreadLocal<>();

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        threadLocal.set(System.currentTimeMillis());
        log.info("Request uri = [{}], method is: [{}]", request.getRequestURI(), request.getMethod());
        log.info("Request header is : [{}]", parseRequestHeaders(request));
        log.info("Request param is : [{}]", parseParams(request));

        if (request instanceof RequestCustomWrapper) {
            RequestCustomWrapper requestCustomWrapper = (RequestCustomWrapper) request;
            byte[] body = requestCustomWrapper.getBody();
            log.info("Request body is : [{}]", new String(body));
        }

        return super.preHandle(request, response, handler);
    }

    public static String parseParams (HttpServletRequest request) {
        StringBuilder stringBuilder = new StringBuilder();
        Enumeration<String> parameterNames = request.getParameterNames();
        while (parameterNames.hasMoreElements()) {
            String name = parameterNames.nextElement();
            request.getParameter(name);
            stringBuilder.append(name).append("=").append(";");
        }
        return stringBuilder.toString();
    }

    public static String parseRequestHeaders (HttpServletRequest request) {
        StringBuilder stringBuilder = new StringBuilder();
        Enumeration<String> headerNames = request.getHeaderNames();
        while (headerNames.hasMoreElements()) {
            String name  = headerNames.nextElement();
            String value = request.getHeader(name);
            stringBuilder.append(name).append("=").append(value).append(";");
        }
        return stringBuilder.toString();
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        super.afterCompletion(request, response, handler, ex);
    }
}

RequestCustomWrapper
因为Springboot框架规定,Request getInputStream只能获取一次,获取第二次的时候就会报错。所以这个时候需要实现RequestWrapper去包裹Request重写getInputStream实现可重复获取Request Stream。

@Slf4j
public class RequestCustomWrapper extends HttpServletRequestWrapper {

    private byte[] body;

    public byte[] getBody() {
        return body;
    }

    public RequestCustomWrapper(HttpServletRequest request) {
        super(request);
        try {
            body = readBytes(request.getReader());
        } catch (IOException e) {
            log.error("读取request input stream失败..");
        }
    }

    @Override
    public BufferedReader getReader() throws IOException {
        return new BufferedReader(new InputStreamReader(getInputStream()));
    }

    @Override
    public ServletInputStream getInputStream() throws IOException {
        try (final ByteArrayInputStream bais = new ByteArrayInputStream(body)) {
            return new ServletInputStream() {
                @Override
                public boolean isFinished() {
                    return false;
                }

                @Override
                public boolean isReady() {
                    return false;
                }

                @Override
                public void setReadListener(ReadListener readListener) {

                }

                @Override
                public int read() throws IOException {
                    return bais.read();
                }
            };
        }

    }

    public byte[] readBytes (BufferedReader br) throws IOException {
        byte[] emptyBytes = new byte[0];
        String str;
        StringBuilder sb = new StringBuilder();
        while ((str = br.readLine()) != null) {
            sb.append(str);
        }

        if (StringUtils.isNotBlank(sb.toString())) {
            return sb.toString().getBytes(StandardCharsets.UTF_8);
        }

        return emptyBytes;
    }
}

RequestCustomFilter
全局过滤器到,在执行到RequestCustomFilter这一层的时候,将ServletRequest包裹替换成自己的Request,实现可重复获取Request Stream.

public class RequestCustomFilter implements Filter {
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        if (servletRequest instanceof HttpServletRequest) {
            ServletRequest requestWrapper = new RequestCustomWrapper((HttpServletRequest) servletRequest);
            filterChain.doFilter(requestWrapper, servletResponse);
        } else {
            filterChain.doFilter(servletRequest, servletResponse);
        }
    }
}

WebMvcConfig
注册过滤器和拦截器

@Configuration
public class WebMvcConfig implements WebMvcConfigurer {

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new LogInterceptor())
                .addPathPatterns("/**");
    }

    @Bean
    public FilterRegistrationBean<RequestCustomFilter> requestCustomFilter () {
        FilterRegistrationBean<RequestCustomFilter> registrationBean = new FilterRegistrationBean<>();
        RequestCustomFilter requestCustomFilter = new RequestCustomFilter();
        registrationBean.setFilter(requestCustomFilter);
        registrationBean.addUrlPatterns("/*");
        registrationBean.setOrder(1);
        return registrationBean;
    }
}

2.Nginx 配置输出Log

配置方式:
在nginx的配置文件中有个变量:$http_cookie来获取cookie的信息。配置方式很简单,只需要在nginx配置文件的http段,新添加一个log_format就可以了:
nginx.conf

log_format  sendfile  '$remote_addr - $remote_user [$time_local] "$request" '
              '$status $body_bytes_sent "$http_referer" '
              '"$http_user_agent" "$http_x_forwarded_for" "$http_cookie"';

在server.conf中加入

access_log  /var/log/php/access.log sendfile;

配置层级结构
Springboot 获取http数据、打印HTTP参数的4种方式 (便于生产排查问题)
sendfile 就是上面定义的log_format 的名字,只要acces_log 后面带这个名字的日志,就会按照定义的格式输出日志。

reload一下nginx就可以在日志里面看到cookie信息

nginx  -s  reload

Nginx 变量参考

  1. $remote_addr #存放了客户端的地址,注意是客户端的公⽹IP
  2. $args #变量中存放了URL中的指令http://www.magedu.net/main/index.do?id=090&partner=search以上:id=090&partner=search 即为 $args
  3. $document_root #保存了针对当前资源的请求的系统根⽬录,如/apps/nginx/html
  4. $cookie_name #表⽰key为 name 的cookie值
  5. $document_uri #保存了当前请求中不包含指令的URI,注意是不包含请求的指令,如http://www.magedu.net/main/index.do?id=090&partner=search会被定义为/main/index.do
  6. $host;#存放了请求的host名称
  7. $http_user_agent #客户端浏览器的详细信息
  8. $http_cookie #客户端的cookie信息
  9. $limit_rate #如果nginx服务器使⽤limit_rate配置了显⽰⽹络速率,则会显⽰,如果没有设置, 则显⽰0
  10. $remote_port #客户端请求Nginx服务器时客户端随机打开的端⼝
  11. $remote_user #已经经过Auth Basic Module验证的⽤户名
  12. $request_body_file #做反向代理时发给后端服务器的本地资源的名称
  13. $request_method #请求资源的⽅式,GET/PUT/DELETE等
  14. $request_filename #当前请求的资源⽂件的路径名称,由root或alias指令与URI请求⽣成的⽂件绝对路径,

3. 借助Skywalking/zipkin等中间件输出到链路中

请关注我,后续会单独出一期Skywalking的教程以及Skywalking原理讲解。

4. 网关日志中输出

这里只简单贴一下代码介绍网管如何打印Http信息,后续会单独出一期介绍网关的文章。请关注我!!!文章来源地址https://www.toymoban.com/news/detail-422813.html

@Component
@Slf4j
public class LoggingFilter implements GlobalFilter, Ordered {

    private static final String START_TIME = "START_TIME";

    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        ServerHttpRequest request = exchange.getRequest();
        String info = String.format("Method: {%s} Host: {%s} Path: {%s} Query: {%s}",
                request.getMethod().name(),
                request.getURI().getHost(),
                request.getURI().getPath(),
                request.getQueryParams());
        log.info(info);
        exchange.getAttributes().put(START_TIME, System.currentTimeMillis());
        return chain.filter(exchange).then(Mono.fromRunnable(() -> {
            Long start = exchange.getAttribute(START_TIME);
            if (start != null) {
                long executeTime = System.currentTimeMillis() - start;
                log.info(exchange.getRequest().getURI().getRawPath() + ":" + executeTime + "ms");
            }
        }));
    }

    @Override
    public int getOrder() {
        return Ordered.LOWEST_PRECEDENCE;
    }
}

到了这里,关于Springboot 获取http数据、打印HTTP参数的4种方式 (便于生产排查问题)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • HTTP四大参数类型及请求参数的方式和如何接收

    请求头参数顾名思义,是存放在请求头中发送给服务器的参数,服务器通过解析请求头获取参数内容。通常会存放本次请求的基本设置,以帮助服务器理解并解析本次请求的body体。 参数形式如: 在我们的SpringBoot中,通过 @RequestHeader 注解可以获取到。 Spring Boot 读取http head

    2024年02月03日
    浏览(30)
  • Spring Boot学习笔记(十三)获取HTTP请求参数

    SpringBoot获取参数代码 User类: (1)路径参数 使用@PathVariable获取 (2)URL参数 1、使用@RequestParam获取 2、使用request的getParameter方法获取 3、直接在controller里使用同名的方法参数获取 4、使用实体类的同名属性接收参数 (3)表单参数 获取方式Url参数一样获取。GET请求只能拿到

    2024年02月04日
    浏览(41)
  • RTPEngine 通过 HTTP 获取指标的方式

    RTPEngine 是常用的媒体代理服务器,通常被集成到 SIP 代理服务器中以减小代理服务器媒体传输的压力,其架构如下图所示。这种使用方式相当于将 RTPEngine 隐藏在 SIP 代理服务器后面,我们虽然可以借助 SIP 代理服务器提供的相关接口有限操作 RTPEngine,但是对 RTPEngine 实例的相

    2024年02月10日
    浏览(31)
  • http之GET请求和POST请求的传递参数的方式

    get: 情况1:接口正常返回 code,data,message ,且无需传递参数 (vu3) 情况2: 需要将参数拼接到url地址上 (vue3)   情况3 需要传递参数,query 传参 (vue3) 情况4 既有query传参也有path传参 post 情况1 接口返回的数据没有使用data包裹,这种情况一般取不到数据,我们需要用 ext

    2024年02月16日
    浏览(32)
  • java业务代码发送http请求(Post方式:请求参数为JSON格式;Get方式)

    实际开发中,可能需要发送http请求到第三方服务获取数据,于是就有以下应用: 依赖: 假设我需要在我的业务代码中调用该地址: url:http://xx.xx:xxxx/user/count 请求方法:post 内容类型:application/json 请求参数:id, username 返回参数:code 响应结果 int类型                  

    2024年02月12日
    浏览(45)
  • 【SpringBoot】7 种实现 HTTP 调用的方式

    作者在工作中,遇到一些需要调用三方接口的任务,就需要用到 HTTP 调用工具。这里,我总结了一下 实现 HTTP 调用的方式,共有 7 种(后续会继续新增),分别如下: HttpClient OkHttp OKHttpUtil Jodd HTTP Hutool HTTP RestTemplate forest 使用: 使用: 在 Java 的世界中,Http 客户端之前一直

    2024年02月13日
    浏览(27)
  • 【SpringBoot】6 种实现 HTTP 调用的方式

    作者在工作中,遇到一些需要调用三方接口的任务,就需要用到 HTTP 调用工具。这里,我总结了一下 实现 HTTP 调用的方式,共有 7 种(后续会继续新增),分别如下: HttpClient OkHttp OKHttpUtil Jodd HTTP Hutool HTTP RestTemplate forest 使用: 使用: 在 Java 的世界中,Http 客户端之前一直

    2024年02月10日
    浏览(26)
  • 31、springboot 配置HTTP服务端口及如何通过WebServer实例动态获取项目中的HTTP端口

    代码示例 就是在yml配置文件中配置端口号 代码演示 需求:在项目中获取服务器的动态端口,通过获取WebServer实例来获取动态端口号。 **获取WebServer实例的方法1:**通过WebServerApplicationContext来获取WebServer 获取WebServer实例的方法2: 实现一个监听器接口:ApplicationListener 来获取

    2024年02月12日
    浏览(28)
  • el-upload实现自定义携带参数上传文件( :http-request 方式)

    1. el-upload组件 使用 :http-request 自定义上传方法,action仍然要有,随便起个名字即可, 注意使用 :http-request 之后, :on-success, :on-error 指令是不会触发的 自定义上传 函数为  uploadFile 2. 封装上传方法(定义传输请求头,传输格式) 在main.js中将封装好的方法加入全局,后面可直接

    2024年02月11日
    浏览(35)
  • Springboot GET和POST请求的常用参数获取方式

    可以在控制器方法的参数上使用@RequestParam注解来获取请求中的参数值。例如: 可以为@RequestParam注解的参数提供默认值,以处理参数缺失的情况。例如: 可以使用@RequestParam MapString, String来获取所有的请求参数键值对。例如: 如果参数是作为路径的一部分传递的,可以使用

    2024年02月10日
    浏览(29)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包