利用Filter输出HTTP请求报文

本文介绍如何使用Tomcat内置工具记录HTTP请求日志。针对不同版本的Tomcat,提供了RequestDumperValve和RequestDumperFilter的具体配置方法,并附带自定义过滤器的源码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

有时候开发需要抓取HTTP请求报文进行数据分析,从浏览器发出的请求还比较好操作,比较工具比较多,从移动客户端上来的请求,除了数据利用网络监听等工具之外,还可以通过request对象进行报文输出分析。
tomcat6为我们提供了org.apache.catalina.valves.RequestDumperValve工具,可以将请求报文输出,在在server.xml中将

 <Valve className="org.apache.catalina.valves.RequestDumperValve"/>

取消注释即可(来源:http://www.tuicool.com/articles/Nj6RN3)。
tomcat7中,提供了org.apache.catalina.filters.RequestDumperFilter过滤器,可见官方文档:https://tomcat.apache.org/tomcat-7.0-doc/config/filter.html#Request_Dumper_Filter
我这里直接翻到RequestDumperFilter的源码:源码,仿照源码写一份自己的dumperfilter:


import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.sql.Timestamp;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Enumeration;
import java.util.Locale;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;

public final class OutputRequestFilter implements Filter {

    public void destroy() {
    }

    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        // Render the generic servlet request properties
        StringBuilder writer = new StringBuilder();
        writer.append(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date())+":");
        writer.append("\n");
        writer.append("Request Received at " + (new Timestamp(System.currentTimeMillis())));
        writer.append("\n");
        writer.append(" characterEncoding=" + request.getCharacterEncoding());
        writer.append("\n");
        writer.append("     contentLength=" + request.getContentLength());
        writer.append("\n");
        writer.append("       contentType=" + request.getContentType());
        writer.append("\n");
        writer.append("            locale=" + request.getLocale());
        writer.append("\n");
        writer.append("           locales=");
        Enumeration locales = request.getLocales();
        boolean first = true;
        while (locales.hasMoreElements()) {
            Locale locale = (Locale) locales.nextElement();
            if (first)
                first = false;
            else
                writer.append(", ");
            writer.append(locale.toString());
        }
        writer.append("\n");
        Enumeration names = request.getParameterNames();
        while (names.hasMoreElements()) {
            String name = (String) names.nextElement();
            writer.append("         parameter=" + name + "=");
            String values[] = request.getParameterValues(name);
            for (int i = 0; i < values.length; i++) {
                if (i > 0)
                    writer.append(", ");
                writer.append(values[i]);
            }
            writer.append("\n");
        }
        writer.append("          protocol=" + request.getProtocol());
        writer.append("\n");
        writer.append("        remoteAddr=" + request.getRemoteAddr());
        writer.append("\n");
        writer.append("        remoteHost=" + request.getRemoteHost());
        writer.append("\n");
        writer.append("            scheme=" + request.getScheme());
        writer.append("\n");
        writer.append("        serverName=" + request.getServerName());
        writer.append("\n");
        writer.append("        serverPort=" + request.getServerPort());
        writer.append("\n");
        writer.append("          isSecure=" + request.isSecure());
        writer.append("\n");

        // Render the HTTP servlet request properties
        if (request instanceof HttpServletRequest) {
            writer.append("---------------------------------------------");
            HttpServletRequest hrequest = (HttpServletRequest) request;
            writer.append("       contextPath=" + hrequest.getContextPath());
            writer.append("\n");
            Cookie cookies[] = hrequest.getCookies();
            if (cookies == null)
                cookies = new Cookie[0];
            for (int i = 0; i < cookies.length; i++) {
                writer.append("            cookie=" + cookies[i].getName() + "=" + cookies[i].getValue());
                writer.append("\n");
            }
            names = hrequest.getHeaderNames();
            while (names.hasMoreElements()) {
                String name = (String) names.nextElement();
                String value = hrequest.getHeader(name);
                writer.append("            header=" + name + "=" + value);
                writer.append("\n");
            }
            writer.append("            method=" + hrequest.getMethod());
            writer.append("\n");
            writer.append("          pathInfo=" + hrequest.getPathInfo());
            writer.append("\n");
            writer.append("       queryString=" + hrequest.getQueryString());
            writer.append("\n");
            writer.append("        remoteUser=" + hrequest.getRemoteUser());
            writer.append("\n");
            writer.append("requestedSessionId=" + hrequest.getRequestedSessionId());
            writer.append("\n");
            writer.append("        requestURI=" + hrequest.getRequestURI());
            writer.append("\n");
            writer.append("       servletPath=" + hrequest.getServletPath());
            writer.append("\n");

            if(hrequest.getMethod().equals("POST")){
                writer.append("====================Body===================");
                String s=null;
                BufferedReader br = new BufferedReader(new InputStreamReader(hrequest.getInputStream()));
                while((s=br.readLine())!=null){
                    writer.append(s);
                }
                writer.append("\n");
                writer.append("====================Body End===================");
            }
        }

        writer.append("\n");
        writer.append("=============================================\n\n\n");

        System.out.println(writer);
        PrintWriter pw = new PrintWriter(new File("/usr/tomcat/apache-tomcat-7.0.75/requestDumper.txt"));
        pw.append(writer);
        pw.flush();
        pw.close();

        // Pass control on to the next filter
        chain.doFilter(request, response);

    }


    public void init(FilterConfig filterConfig) throws ServletException {

    }

}

特此以记。

### 使用Wireshark捕获和分析Profinet PN报文 #### 工具准备 在开始之前,确保已安装最新版本的 Wireshark 软件。由于 Profinet 是基于以太网的一种工业通信协议,因此需要了解其基本结构以及特定字段的意义。 #### 数据捕获设置 为了有效捕获 Profinet 报文,可以按照以下方法进行操作: 1. **启动数据捕获** 打开 Wireshark 并选择目标网络接口卡(NIC)。如果计算机连接至交换机,则可能需要启用端口镜像功能或将交换机替换为集线器以便获取完整的流量[^3]。 2. **应用过滤条件** 鉴于 Profinet 流量通常涉及多种类型的帧(如 DCP、ARP 或 RT 业务),建议先定义初步显示过滤器 `profinet` 来隔离无关的数据流。对于更具体的场景,例如仅关注设备发现阶段中的动态配置协议 (DCP),可改用更为精确的表达式 `"pn_dcp"`。 #### 分析典型消息类型 一旦成功截取到相关样本,就可以着手解读各个部分的内容了: - **DCP 帧**: 这些用于初始化期间或者运行时修改参数。例如,在执行自动寻址过程中会观察到一对交互——请求方发出询问并附带期望属性列表;响应则携带实际值返回给查询者。 - **报警与诊断信息**: 当检测到异常情况(如同步丢失或 IO 控制失败)时,系统会产生相应的通知事件。这些记录有助于快速定位潜在隐患所在位置[^2]。 - **周期性 I/O 数据传输**: 此类事务构成了大部分正常运作下的负载成分。它们遵循固定的调度模式,并且承载着来自现场级传感器/执行机构的状态更新或是命令指令集合[^1]。 #### 故障排除技巧 当遇到诸如 “Real-time communication has been aborted due to protocol error or watchdog timeout” 的提示时,应着重审查以下几个方面: - 确认所有参与节点的时间同步机制处于良好状态; - 检查是否存在过多延迟引入因素影响整体性能表现; - 对比当前使用的固件版本号同官方推荐清单之间是否有差异; - 利用高级统计视图辅助判断哪些环节出现了瓶颈现象。 ```python # 示例 Python 脚本片段展示如何自动化提取某些字段值 import pyshark def parse_profinet_packets(pcap_file): cap = pyshark.FileCapture(pcap_file, display_filter='profinet') results = [] for packet in cap: try: src_addr = packet.ip.src dst_addr = packet.ip.dst io_data = getattr(packet.profinet, 'io_data', None) result_entry = { 'source': src_addr, 'destination': dst_addr, 'payload': io_data } results.append(result_entry) except AttributeError as e: continue return results if __name__ == "__main__": packets_info = parse_profinet_packets('example.pcapng') print(packets_info[:5]) # 输出前五个条目作为示例 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值