Skip to content

Latest commit

 

History

History
728 lines (688 loc) · 35.1 KB

Event_log-zh.md

File metadata and controls

728 lines (688 loc) · 35.1 KB

event_log 模块

模块介绍

event_log模块是log模块的扩展功能模块,目的是结合qlog规范定义的事件类型规范化日志打印。 qlog规范:https://datatracker.ietf.org/doc/draft-marx-qlog-event-definitions-quic-h3/

模块使用

开启event_log功能

编译时

在cmake编译xquic时添加“-DXQC_ENABLE_EVENT_LOG=1”将编译event_log相关功能,默认状态不编译。

运行时

通过xqc_engine_create函数创建engine对象时,根据传入的xqc_config_t对象的cfg_log_event参数选择是否启用event_log功能,当cfg_log_event>0时开启。

使用event_log功能

调用事件日志打印

在事件发生的位置调用xqc_log_event函数

xqc_log_event(log, 事件类型, 事件相关参数);

其中log为xqc_log_t对象,编译时会根据事件类型检查传入的事件相关参数与事件的回调函数格式是否符合。

现有qlog事件相关参数如下(详见xqc_log_event_callback.h):

事件类型

事件相关参数

CON_CONNECTION_STARTED

xqc_connection_t *conn, xqc_int_t local

CON_CONNECTION_CLOSED

xqc_connection_t *conn

CON_CONNECTION_ID_UPDATED

xqc_connection_t *conn

CON_CONNECTION_STATE_UPDATED

xqc_connection_t *conn

SEC_KEY_UPDATED

xqc_engine_ssl_config_t ssl_config, xqc_int_t local

TRA_VERSION_INFORMATION

uint32_t local_count, uint32_t *local_version, uint32_t remote_count, uint32_t *remote_version, uint32_t choose

TRA_ALPN_INFORMATION

size_t local_count, uint8_t *local_alpn, size_t remote_count, const uint8_t *remote_alpn, size_t alpn_len, const unsigned char *alpn

TRA_PARAMETERS_SET

xqc_connection_t *conn, xqc_int_t local

TRA_PACKET_SENT

xqc_packet_out_t *packet_out

TRA_PACKET_RECEIVED

xqc_packet_in_t *packet_in

TRA_PACKET_BUFFERED

xqc_packet_in_t *packet_in

TRA_PACKETS_ACKED

xqc_packet_in_t *packet_in, xqc_packet_number_t high, xqc_packet_number_t low

TRA_DATAGRAMS_SENT

ssize_t size

TRA_DATAGRAMS_RECEIVED

ssize_t size

TRA_STREAM_STATE_UPDATED

xqc_stream_t *stream, xqc_int_t stream_type, xqc_int_t state

TRA_FRAMES_PROCESSED

根据frame类型不同,传入参数不同

REC_PARAMETERS_SET

xqc_send_ctl_t *ctl

REC_METRICS_UPDATED

xqc_send_ctl_t *ctl

REC_CONGESTION_STATE_UPDATED

char *new_state

REC_LOSS_TIMER_UPDATED

xqc_send_ctl_t *ctl,

xqc_usec_t inter_time, xqc_int_t type, xqc_int_t event

REC_PACKET_LOST

xqc_packet_out_t *packet_out

HTTP_PARAMETERS_SET

xqc_h3_conn_t *h3_conn, xqc_int_t local

HTTP_PARAMETERS_RESTORED

xqc_h3_conn_t *h3_conn

HTTP_STREAM_TYPE_SET

xqc_h3_stream_t *h3_stream, xqc_int_t local

HTTP_FRAME_CREATED

根据frame类型不同,传入参数不同

HTTP_FRAME_PARSED

xqc_h3_stream_t *h3_stream

HTTP_SETTING_PARSED

uint64_t identifier, uint64_t value

QPACK_STATE_UPDATED

根据类型属于编码器/解码器,传入参数不同

QPACK_STREAM_STATE_UPDATED

xqc_h3_stream_t *h3_stream

QPACK_DYNAMIC_TABLE_UPDATED

根据动态表插入/删除,传入参数不同

QPACK_HEADERS_ENCODED

根据prefixheader,传入参数不同

QPACK_HEADERS_DECODED

根据prefixheader,传入参数不同

QPACK_INSTRUCTION_CREATED

根据指令不同,传入参数不同

QPACK_INSTRUCTION_PARSED

根据指令不同,传入参数不同

GEN_REPORT

report级别通用日志打印,类似printf传入格式化字符串和参数

GEN_FATAL

fatal级别通用日志打印,类似printf传入格式化字符串和参数

GEN_ERROR

error级别通用日志打印,类似printf传入格式化字符串和参数

GEN_WARN

warn级别通用日志打印,类似printf传入格式化字符串和参数

GEN_STATS

stats级别通用日志打印,类似printf传入格式化字符串和参数

GEN_INFO

info级别通用日志打印,类似printf传入格式化字符串和参数

GEN_DEBUG

debug级别通用日志打印,类似printf传入格式化字符串和参数

注册新事件

在已实现的qlog事件基础上可根据实现特点和需求注册自定义的事件,注册新事件方法如下:

  • 在xqc_log.h的xqc_log_type_t枚举类中添加事件名
typedef enum {
    /* connectivity event */
    CON_SERVER_LISTENING,
    CON_CONNECTION_STARTED,
    CON_CONNECTION_CLOSED,
    CON_CONNECTION_ID_UPDATED,
    CON_SPIN_BIM_UPDATED,
    CON_CONNECTION_STATE_UPDATED,

    ... ...
} xqc_log_type_t;
  • 在xqc_log.c的xqc_log_type_str函数中添加事件对应的事件名表示
const char *
xqc_log_type_str(xqc_log_type_t type)
{
    static const char *event_type2str[] = {
            [CON_SERVER_LISTENING]              = "server_listening",
            [CON_CONNECTION_STARTED]            = "connection_started",
            [CON_CONNECTION_CLOSED]             = "connection_closed",
            [CON_CONNECTION_ID_UPDATED]         = "connection_id_updated",
            [CON_SPIN_BIM_UPDATED]              = "spin_bin_updated",
            [CON_CONNECTION_STATE_UPDATED]      = "connection_state_updated",
            
            ... ...
    };
    return event_type2str[type];
}

形如:[事件类型] = "事件名"

  • 在xqc_log.c的xqc_log_type_2_level函数中设置事件的日志级别
xqc_log_level_t
xqc_log_type_2_level(xqc_log_type_t type)
{
    switch (type) {
    case GEN_REPORT:
        return XQC_LOG_REPORT;
    case GEN_FATAL:
        return XQC_LOG_FATAL;
    case GEN_ERROR:
        return XQC_LOG_ERROR;
    case GEN_WARN:
        return XQC_LOG_WARN;
        
    ... ...
}
  • 在xqc_log_event_callback.h中定义日志的回调函数,回调函数格式为
xqc_log_事件类型_callback(xqc_log_t *log, const char *func, ...)

传参的log和func为固定参数,可选参数部分为事件相关的参数

  • 在xqc_log_event_callback.c中实现日志的回调函数,根据事件类型规范日志格式并调用xqc_log_implement函数(定义在xqc_log.h中,使用方式类似printf函数)打印
xqc_log_implement(log, 事件类型, func, 日志格式, 打印参数);

event_log日志分析

日志格式

[时间] [事件名] |scid:所属连接的scid|打印日志的函数名|打印日志内容|

事件的具体打印内容和格式详见xqc_log_event_callback.c文件

常用grep:

过滤特定事件:grep "[事件名]"

过滤特定连接:grep "|scid:所属连接的scid|"

后处理将日志转换为qlog格式

通过python等脚本语言将日志格式转变为qlog规范中定义的格式(.qlog) --- todo

借助 qvis (一个针对qlog规范的可视化工具)解析.qlog文件,方便使用者分析学习xquic的传输过程。