python 日志模块的使用

日志

软件开发中通过日志记录程序的运行情况是一个开发的好习惯,对于错误排查和系统运维都有很大帮助。
python标准库自带了强大的 logging 日志模块,在各种python模块中得到广泛应用。

一、简单实用

1. 入门小案例

import logging
logging.basicConfig(level=logging.DEBUG, # 设置级别,根据等级显示
                    format='%(asctime)s-[%(filename)s-->line:%(lineno)d]-%(levelname)s:%(message)s')#设置输出格式
logging.debug('This is a debug log')
logging.info('This is a info log')
logging.warning('This is a warning log')
logging.error('This is a error log')
logging.critical('This is a critical log')
2021-11-15 12:03:34,790-[3279673026.py-->line:4]-DEBUG:This is a debug log
2021-11-15 12:03:34,791-[3279673026.py-->line:5]-INFO:This is a info log
2021-11-15 12:03:34,792-[3279673026.py-->line:6]-WARNING:This is a warning log
2021-11-15 12:03:34,792-[3279673026.py-->line:7]-ERROR:This is a error log
2021-11-15 12:03:34,794-[3279673026.py-->line:8]-CRITICAL:This is a critical log

直接导入logging就可以直接调用各种级别的日志方法。

2. 日志级别

根据不同情况设置了五种日志等级,不同情况输出不同等级的日志。

日志等级(level) 描述
DEBUG 调试信息,通常在诊断问题的时候用得着
INFO 普通信息,确认程序按照预期运行
WARNING 警告信息,表示发生意想不到的事情,或者指示接下来可能会出现一些问题,但是程序还是继续运行
ERROR 错误信息,程序运行中出现了一些问题,程序某些功能不能执行
CRITICAL 危险信息,一个严重的错误,导致程序无法继续运行

日志器设置的级别会过滤掉低于这个级别的日志

import logging
logging.basicConfig(level=logging.ERROR, # 设置级别,根据等级显示
                    format='%(asctime)s-[%(filename)s-->line:%(lineno)d]-%(levelname)s:%(message)s')#设置输出格式
logging.debug('This is a debug log')
logging.info('This is a info log')
logging.warning('This is a warning log')
logging.error('This is a error log')
logging.critical('This is a critical log')
2021-11-15 13:50:16,235-[4267381777.py-->line:4]-ERROR:This is a error log
2021-11-15 13:50:16,237-[4267381777.py-->line:5]-CRITICAL:This is a critical log

basicConfig方法支持以下关键字参数进行配置

参数 描述
filename 使用指定的文件名而不是 StreamHandler 创建 FileHandler。
filemode 如果指定了 filename,则用此 模式 打开该文件。 默认模式为 'a'。
format 处理器使用的指定格式字符串。
datefmt 使用指定的日期/时间格式,与 time.strftime() 所接受的格式相同。
style 如果指定了 format,将为格式字符串使用此风格。 '%', '{' 或 '$' 分别对应于 printf 风格,str.format() 或 string.Template。 默认为 '%'。
level 设置根记录器级别去指定 level.
stream 使用指定的流初始化 StreamHandler。 请注意此参数与 filename 是不兼容的 - 如果两者同时存在,则会引发 ValueError。
handlers 如果指定,这应为一个包含要加入根日志记录器的已创建处理程序的可迭代对象。 任何尚未设置格式描述符的处理程序将被设置为在此函数中创建的默认格式描述符。 请注意此参数与 filename 或 stream 不兼容 —— 如果两者同时存在,则会引发 ValueError。
force 如果将此关键字参数指定为 true,则在执行其他参数指定的配置之前,将移除并关闭附加到根记录器的所有现有处理器。

4. 格式化规则

日志的输出格式可以通过下面格式自由组合输出

规则 描述
%(asctime) 日志事件发生的时间
%(levelname) 该日志记录的日志级别
%(message) 日志记录的文本内容
%(name)s 所使用的日志器名称,默认是'root'
%(pathname)s 调用日志记录函数的文件的全路径
%(filename)s 调用日志记录函数的文件
%(module)s 模块 (filename 的名称部分)。
%(funcName)s 调用日志记录函数的函数名
%(lineno)d 调用日志记录函数的代码所在的行号

常用格式:%(asctime)s-[%(filename)s-->line:%(lineno)d]-%(levelname)s:%(message)s

5. 日志写到文件

只需要配置filename参数即可

import logging
logging.basicConfig(level=logging.WARNING,  # 设置级别,根据等级显示
                    filename='example.log',
                    format='%(asctime)s-[%(filename)s-->line:%(lineno)d]-%(levelname)s:%(message)s')# 设置输出格式
logging.debug('This is a debug log')
logging.info('This is a info log')
logging.warning('This is a warning log')
logging.error('This is a error log')
logging.critical('This is a critical log')

注意,配置filename后,日志将不会输出在控制台。

二、 高级用法

简单的代码通过logging直接使用即可,如果要深入使用需要按照面向对象的方式使用logging。

1. 日志组件

logging模块包含以下几个组件。

组件 说明
Loggers(日志记录器) 提供程序直接使用的接口
Handlers(日志处理器) 将记录的日志发送到指定的位置
Filters(日志过滤器) 用于过滤特定的日志记录
Formatters(日志格式器) 用于控制日志信息的输出格式

2. 步骤

2.1 创建日志记录器

import logging
第一步创建一个logger,用来产生日志

logger = logging.getLogger('%s_log'%name)
logger.setLevel(logging.DEBUG) # 设置日志等级通过getLogger这个方法可以创建一个日志记录器,注意要给名字否则返回根日志记录器。  

通过setLevel设置日志记录器的等级。

2.2 创建日志处理器

创建一个文本处理器用来将日志写入到文件

file_handler = logging.FileHandler(filename='py45.log',encoding='utf-8')file_handler.setLevel('WARNING') # 设置处理器的日志等级 # 创建一个控制台处理器用来将日志输出到控制台
console_handler = logging.StreamHandler()
console_handler.setLevel('INFO') # 设置控制台处理器的日志等级日志处理器就是将日志发送到指定的位置。

FileHandler 将日志发送到文件 StreaHandler 将它可将日志记录输出发送到数据流例如 sys.stdout, sys.stderr 或任何文件类对象默认 sys.stdout 即控制台。 RotatingFileHandler 支持根据日志文件大小进行轮转 TimedRotatingFileHandler 支持根据时间进行轮转日志文件

2.3 创建格式化器

formatter = logging.Formatter(fmt='%(levelname)s %(asctime)s [%(filename)s-->line:%(lineno)d]:%(message)s')格式化器需要设置到处理器上

file_handler.setFormatter(formatter)
console_handler.setFormatter(formatter)  

2.4 创建过滤器

过滤器用来过滤指定日志。具体使用略,一般用不到。

2.5 将处理器添加到记录器上

logger.addHandler(file_handler)
logger.addHandler(console_handler)

2.6 记录日志

logger.info('This is a info')  
2020-09-11 22:22:44,095-[-->line:1]-INFO:This is a info

logger.warning('This is a warning')  
2020-09-11 22:23:20,337-[-->line:1]-WARNING:This is a warning

三、日志模块封装

1. 功能分析

  1. 能够自定义日志器名
  2. 能够自定义日志文件名和路径
  3. 能够自定义日志文件编码方式
  4. 能够自定义日志格式
  5. 使用时间轮转处理器,并能够配置

2.封装成函数

在 common 目录下创建模块 log_handler.py 在其中创建如下函数。

import logging
from logging.handlers import TimedRotatingFileHandler

def get_logger(name, filename, encoding='utf-8', fmt=None, when='d', interval=1, backup_count=7, debug=False):
"""

:param name: 日志器的名字
:param filename: 日志文件名(包含路径)
:param encoding: 字符编码
:param fmt: 日志格式
:param when: 日志轮转时间单位
:param interval: 间隔
:param backup_count: 日志文件个数
:param debug: 调试模式
:return:
"""
logger = logging.getLogger(name)
logger.setLevel(logging.DEBUG)

文件处理器的等级一般情况一定比控制台要高

if debug:
file_level = logging.DEBUG
console_level = logging.DEBUG
else:
file_level = logging.WARNING
console_level = logging.INFO

if fmt is None:
fmt = '%(levelname)s %(asctime)s [%(filename)s-->line:%(lineno)d]:%(message)s'

file_handler = TimedRotatingFileHandler(
filename=filename, when=when, interval=interval, backupCount=backup_count, encoding=encoding)
file_handler.setLevel(file_level)

console_handler = logging.StreamHandler()
console_handler.setLevel(console_level)

formatter = logging.Formatter(fmt=fmt)
file_handler.setFormatter(formatter)
console_handler.setFormatter(formatter)

logger.addHandler(file_handler)
logger.addHandler(console_handler)

return logger

if name == 'main':
log = get_logger(name='py41', filename='py41.log', debug=True, when='s')
log.info('我是普通信息')
import time
time.sleep(3)
log.warning('我是警告信息')

四、应用到项目中

1. 导入

日志器生成函数的导入不能像 Excel 数据读取函数一样,每个用例模块里都导入一遍。因为它返回一个日志器对象,当多次调用日志器生成函数,且日志器名称相同时,会给同一个日志器添加多个日志处理器,从而出现重复记录日志器的问题。

为了解决上面的问题,在 common 文件夹下创建一个名为 init.py 的文件,在 common 模块被导入时会自动执行这个文件里的代码,且只会执行一次。

init.py 文件编写如下代码:

from .log_handler import get_logger
logger = get_logger('py41', 'py38.log')那么在项目中的其他模块中就可以通过如下代码导入

from common import logger 从而可以保证在项目执行过程中,get_logger 方法只会执行一遍。

2. 记录日志

日志的作用是记录程序的运行状态和当程序出现问题时能提供定位分析错误的依据。

什么时候需要记录日志,记录什么日志,根据每个人对程序的理解,以及经验。

我们的项目中,在用例执行的过程是核心,所以我们的日志也是围绕着用例的执行。

使用日志记录每个用例的测试数据,和测试结果,代码如下:

...
@list_data(*cases)
def test_login(self, case):
"""
登陆测试
"""
logger.info('测试用例【{}】开始测试'.format(case['title']))

1. 测试数据

传入进来的 case 参数

logger.info('测试用例【{}】的测试数据是:{}'.format(case['title'], case))

2. 测试步骤

res = login_check(case['username'], case['password'])
logger.info('测试用例【{}】的测试结果是:{}'.format(case['title'], res))

3. 断言

try:
self.assertEqual(res, case['expect'])
except AssertionError as e:
logger.error('测试用例【{}】断言失败'.format(case['title']))
raise e
else:
logger.info('测试用例【{}】断言成功'.format(case['title']))
finally:
logger.info('测试用例【{}】测试结束')
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容