fuzidage
专注嵌入式、linux驱动 、arm裸机研究

导航

 

一. syslogd简介

syslogd不仅仅是记录kernel log的服务,还能记录user space中的日志。

syslogd是Linux下的一个记录日志文件服务。新版本叫做rsyslogd。

syslogd有一系列的子服务,例如mail、auth、cron、kern等等,这些子服务提供日志记录的功能,。当程序要记录log时,可以直接调用这些子服务将日志记录到设定的地方。

syslogd是一个守护进程,配置这整个守护进程以及其子服务的地方就是/etc/syslog.conf这个文件。可以从https://www.rsyslog.com/doc/master/获取官方文档。

1.1 日志的output格式

1. 如果配置好并运行了 syslogd 或 klogd,一般所有 log的信息也会追加到 /var/log/messages。并且kernel log信息被记录在/val/log/kern.log。

可以看到基本日志格式包含以下四列:

1 事件产生的时间(Jan 1 08:00:09)

2 发生事件的服务器的主机名 (xxxxxx)

3 产生事件的服务名或程序名 (kernel or local5)

4 事件的具体信息(…cif a0c2000.cif:..)

2.当开启rsyslogd后,不能透过/proc/kmsg来查看kernel log.

1.2 配置/etc/syslog.conf服务

  1. /etc/rsyslog.conf 是rsyslog服务的总配置文件
  2. /etc/rsyslog.d 该目录是单独配置的rsyslog配置文件

1.2.1 总配置/etc/syslog.conf

rsyslog记录哪些日志,到底记录了什么样的日志,是通过这个/etc/rsyslog.conf配置文件来决定的,先分析一下rsyslogd的配置文件:

默认规则会定义在/etc/rsyslog.d/50-default.conf中。

#################
#### MODULES ####
#################
module(load="imuxsock") # provides support for local system logging   ;加载提供对本地系统日志的支持
module(load="imklog")   # provides kernel logging support;加载读取内核消息模块
#module(load="immark")  # provides --MARK-- message capability
# provides UDP syslog reception  (接收使用UDP 协议转发过来的日志,这里#注释掉了表示不启用)
#module(load="imudp")
#input(type="imudp" port="514")  (允许514端口接收)
# provides TCP syslog reception (接收使用UDP 协议转发过来的日志)
#module(load="imtcp")
#input(type="imtcp" port="514")(允许514端口接收)
# Enable non-kernel facility klog messages
$KLogPermitNonKernelFacility on
###########################
#### GLOBAL DIRECTIVES ####
###########################
#
# Use traditional timestamp format.
# To enable high precision timestamps, comment out the following line.
#
$ActionFileDefaultTemplate RSYSLOG_TraditionalFileFormat
# Filter duplicated messages $RepeatedMsgReduction on
# # Set the default permissions for all log files. # $FileOwner syslog $FileGroup adm $FileCreateMode 0640 $DirCreateMode 0755 $Umask 0022 $PrivDropToUser syslog $PrivDropToGroup syslog # # Where to place spool and state files # $WorkDirectory /var/spool/rsyslog (记录所有日志类型的info级别以及大于info级别的信息到/var/log/messages,但是mail邮件信息,authpriv验证方面的信息和cron时间任务相关的信息除外) #*.info;mail.none;authpriv.none;cron.none /var/log/messages # Include all config files in /etc/rsyslog.d/ # $IncludeConfig /etc/rsyslog.d/*.conf (表示加载该目录中的所有配置)

1.2.2 rsyslog规则(/etc/rsyslog.d/*.conf)

rsyslog规则配置文件一般由以下3部分组成,每一行表示一个项目,格式为:facility.level action,分别表示日志类型,日志等级,日志输出路径。一般系统默认的规则定义在/etc/rsyslog.d/50-default.conf,

 

一般所有日志类型都会被追加在/val/log/messages。如:

*.info;mail.none;authpriv.none;cron.none    /var/log/messages  

facility日志类型

  • kern: 内核信息
  • user: 用户进程相关信息
  • mail: 电子邮件相关信息
  • Local0- local7: 为本地使用预留的服务
  • daemon: 后台进程相关信息
  • syslog: 系统日志信息

level(按严重程度由低到高排序):

  • none: 没有重要级
  • debug: 调试信息
  • info: 打印的信息
  • notice: 具有重要信息的普通条件
  • warning: 警告信息
  • err: 错误信息
  • crit: 阻止某些工具或子系统功能实现的错误条件
  • alert: 需要立即被修改的条件
  • emerg: 该系统不可用

action(表示log保存的位置)

那下面我也抄过来一份比较全面的规则定义示例供大家参考:

# 记录mail日志等级为error及以上日志
mail.err                                                        /var/log/mail_err.log
# 记录mail所有等级为warn级别的日志(仅记录warn级别)
mail.=warn                                                        /var/log/mail_err.log
# 记录kern所有日志
kern.*                                                                /var/log/kern.log
# 将mail的所有信息,除了info以外,其他的都写入/var/adm/mail
mail.*;mail.!=info   /var/adm/mail
# 将日志等级为crit或更高的内核消息定向到远程主机finlandia
# 如果主机崩溃,磁盘出现不可修复的错误,可能无法读取存储的消息。如果有日志在远程主机上,可以尝试找出崩溃的原因。
kern.crit                                                     @finlandia
# 记录所有类型的warning等级及以上日志
*.warning                                                        /var/log/syslog_warn.log
# 记录mail的warning日志和kern的error日志,其他所有的info日志
*.info;mail.warning;kern.error                /var/log/messages
# 记录kernel的info到warning日志
kern.info;kern.!err   /var/adm/kernel-info
# 将mail和news的info级别日志写入/var/adminfo
mail,news.=info    /var/adm/info
# 将所有系统中所有类型的info日志和notice日志存入/var/log/massages,mail的所有日志除外。
*.=info;*.=notice;\
mail.none /var/log/messages
# 紧急消息(emerg级别)将使用wall显示给当前所有登录的用户,这里用等号表示只对emerg日志级别有效
*.=emerg                   *
# 该规则将所有alert以及更高级别的消息定向到操作员的终端,即登录的用户“root”和“joey”的终端。
*.alert      root,joey

1.3  程序如何配置syslog子服务(如何自定义rsyslog规则) 

问题:进程如何发送消息给rsyslog守护进程,rsyslog守护进程是如何对各种日志区分开来的?

像/usr/sbin/sshd、/usr/bin/login、/usr/bin/su这些进程,它们是调用一个叫syslog的系统调用,syslog系统调用是一个用于向rsyslog守护进程发送消息的的系统函数。

/usr/sbin/sshd,/usr/bin/login、/usr/bin/su这些进程专门执行登录验证时,它们在调用syslog系统函数会一般会调用LOG_AUTH这个常量,

而/usr/bin/crond和/usr/bin/at这些在调用syslog系统调用会传入LOG_CRON这个常量(具体请看syslog()函数),日志归类规则如下:

所以如果用LOG_AUTH的syslog()函数调用,那么会归类到了/val/log/secure,

如果用LOG_CRON的syslog()调用则归类到了/val/log/cron。而kernel等其他log被记录在了/val/log/messages中。

那么我们可以自定义规则如下:

 

syslogfacility-text:表示facility日志类型

syslogseverity-text:表示level日志级别

这里对aisdk.conf用local7日志类型产生的大于warn日志级别的log都将记录到/val/log/aisdk,

对kern.conf用kern类型产生的所有级别日志都记录到/val/log/kern,

Local5类型的日志记录到/val/log/middleware, 并且当日志级别等于或高于4(warn)时也会追加到console.

注意busybox要开启使能syslog.conf解析:/etc/syslog.conf解析,CONFIG_FEATURE_SYSLOGD_CFG=y

 

二. syslog()函数

用户空间也可以用syslog()函数来记录自己的进程的日志,所以用户进程可以自定义日志规则。

调用openlog是可选择的。如果不调用openlog,则在第一次调用syslog时,会自动调用openlog。

syslog的相关函数和宏定义一般在toolchain中都会有定义:

 

2.1 openlog函数

第1个参数为ident,该参数常用来表示信息的来源。ident信息会被固定地添加在每行日志的前面。

第2个参数 option控制标志

LOG_CONS

如果将信息发送给syslogd守护进程时发生错误,直接将相关信息输出到终端

LOG_PID

每条日志信息中都包括进程号

第3个参数为facility:

facility参数

syslog.conf中对应的facility取值

LOG_KERN

kern

LOG_USER

user

LOG_MAIL

mail

LOG_DAEMON

daemon

LOG_AUTH

auth

LOG_SYSLOG

syslog

LOG_LPR

lpr

LOG_NEWS

news

LOG_UUCP

uucp

LOG_CRON

cron

LOG_AUTHPRIV

authpriv

LOG_FTP

ftp

LOG_LOCAL0~LOG_LOCAL7

local0~local7

 2.2 syslog函数

第一个参数priority表示日志级别

priority参数

syslog.conf中对应的level取值

LOG_EMERG

emerg

LOG_ALERT

alert

LOG_CRIT

crit

LOG_ERR

err

LOG_WARNING

warning

LOG_NOTICE

notice

LOG_INFO

info

LOG_DEBUG

debug

 

下面是具体的例子:

这里Printf("%m")等价于printf("%s",strerror(errno));它表示把errno用string形式打印出来。

由于我这里facility为user时,是记录在/val/log/syslog中的,

 

因此打印log如下:

2.3 重定向log

那我们也可以把log定向到自己想要的地方,

方法1:修改rsyslog.conf:

 

将facility=user时的所有level级别的log重定向到/val/log/user.log, 重启rsyslog服务,

 

此时log将被写入到新配置的位置/val/log/user.log, 当然/val/log/syslog也会保留一份.(因为也符合/val/log/syslog这条规则)

 

方法2:修改code中的facility:

 

那这里的facility被设置成了local0, 那也会记录在/val/log/syslog.

 

2.4 设置log等级

1. 这里新增一个app.conf,然后自定义log路径:

 

当然还可以类似于这样子写, syslogfacility-text和syslogseverity-text是rsyslog自带的系统变量。

 

2. 重启rsyslog服务

 

 这里切成4个文件,每个文件记录1024k。

3. 运行程序,查看log如下:

 

 

4. 那现在修改log等级为warn, 表示只有大于等于该等级的log才会记录。 

5. 再次重启rsyslog服务,运行程序,可以看到"log debug"不再打印. 

2.5 重定向log到console

 

再次重启rsyslog服务,运行程序,那么可以看到err级别的log打印在了console上,但是低于err级别还是会记录在/val/log/app。

 

3. dup函数介绍

 用来将标准输出重定向到文件。

static int dup_fd;
static int dup_fd_bak = 1000;

static int dup_fd(void)  {
    dup_fd = open( "./printf_dup_log.txt ", O_CREAT | O_RDWR | O_TRUNC);
    dup2(STDOUT_FILENO, dup_fd_bak);/*backup stdout*/
    dup2(dup_fd, STDOUT_FILENO);
    return 0;
}
static int rst_fd(void)  {
    dup2(dup_fd_bak, fileno(stdout));/*recover stdout*/
    close(dup_fd);
    return 0;
}

 

posted on 2021-05-20 11:26  fuzidage  阅读(8162)  评论(0编辑  收藏  举报