python經過TimedRotatingFileHandler按時間切割日誌

經過TimedRotatingFileHandler按時間切割日誌


線上跑了一個定時腳本,天天生成的日誌文件都寫在了一個文件中。可是日誌信息不可能輸出到單一的一個文件中。
緣由有二:1.日誌文件愈來愈大會影響系統的性能。2.日誌文件格式不夠清晰,好比我想看今天的日誌,不太方便找到的今天的日誌信息(即便對日誌輸出作了時間提示)
經過設置TimedRotatingFileHandler進行日誌按周(W)、天(D)、時(H)、分(M)、秒(S)切割。
先看一個簡單例子:python

import time
import logging
import os
from logging import handlers


def _logging(**kwargs):
    level = kwargs.pop('level', None)
    filename = kwargs.pop('filename', None)
    datefmt = kwargs.pop('datefmt', None)
    format = kwargs.pop('format', None)
    if level is None:
        level = logging.DEBUG
    if filename is None:
        filename = 'default.log'
    if datefmt is None:
        datefmt = '%Y-%m-%d %H:%M:%S'
    if format is None:
        format = '%(asctime)s [%(module)s] %(levelname)s [%(lineno)d] %(message)s'

    log = logging.getLogger(filename)
    format_str = logging.Formatter(format, datefmt)
    # backupCount 保存日誌的數量,過時自動刪除
    # when 按什麼日期格式切分(這裏方便測試使用的秒)
    th = handlers.TimedRotatingFileHandler(filename=filename, when='S', backupCount=3, encoding='utf-8')
    th.setFormatter(format_str)
    th.setLevel(logging.INFO)
    log.addHandler(th)
    log.setLevel(level)
    return log

os.makedirs("./logs", exist_ok=True)
logger = _logging(filename='./logs/default.log')



if __name__ == '__main__':
    while True:
        time.sleep(0.1)
        logger.info('哈哈哈')

結果以下:
6cbd34660f2e89833322d78c77b4811aapp

上述代碼能夠正常運行,並且也能夠生成固定的日誌個數,可是有一個問題,生成的日誌文件格式是你的文件名+時間的格式,沒有設置時間的話默認設置到了秒(這裏是按秒切割)
修改日誌格式後綴名稱:性能

# 在上述代碼中加入
def namer(filename):
    return filename.split('default.')
th.namer = namer
# 設置爲S,默認的suffix爲 Y-%m-%d_%H-%M-%S
th.suffix = "%Y-%m-%d_%H-%M-%S.log"

# 爲了看的更視覺效果,能夠顯示在控制檯答應
cmd = logging.StreamHandler()
cmd.setFormatter(format_str)
cmd.setLevel(level)
log.addHandler(cmd)

運行結果:
f26d7f27295b762d8d3322c9c12a1c85測試

名字好像能夠了,可是日誌好像沒有起到自動刪除的目的啊,並且也沒在以前的log文件夾了。
來看看源碼:debug

def getFilesToDelete(self):
        """
        Determine the files to delete when rolling over.

        More specific than the earlier method, which just used glob.glob().
        """
        dirName, baseName = os.path.split(self.baseFilename)
        fileNames = os.listdir(dirName)
        result = []
        prefix = baseName + "."
        plen = len(prefix)
        for fileName in fileNames:
            if fileName[:plen] == prefix:
                suffix = fileName[plen:]
                if self.extMatch.match(suffix):
                    result.append(os.path.join(dirName, fileName))
        if len(result) < self.backupCount:
            result = []
        else:
            result.sort()
            result = result[:len(result) - self.backupCount]
        return result

這是它的刪除邏輯,關鍵是經過.前面的字段判斷是否重複,當有特定的重複數後開始刪除。
因此問題來了,要麼本身去重寫源碼,要麼就只能用default.日期.log這種格式了。
附上平時使用的日誌代碼日誌

import logging
import os
from logging import handlers


def _logging(**kwargs):
    level = kwargs.pop('level', None)
    filename = kwargs.pop('filename', None)
    datefmt = kwargs.pop('datefmt', None)
    format = kwargs.pop('format', None)
    if level is None:
        level = logging.DEBUG
    if filename is None:
        filename = 'default.log'
    if datefmt is None:
        datefmt = '%Y-%m-%d %H:%M:%S'
    if format is None:
        format = '%(asctime)s [%(module)s] %(levelname)s [%(lineno)d] %(message)s'

    log = logging.getLogger(filename)
    format_str = logging.Formatter(format, datefmt)

    def namer(filename):
        return filename.split('default.')[1]

    # cmd = logging.StreamHandler()
    # cmd.setFormatter(format_str)
    # cmd.setLevel(level)
    # log.addHandler(cmd)

    os.makedirs("./debug/logs", exist_ok=True)
    th_debug = handlers.TimedRotatingFileHandler(filename="./debug/" + filename, when='D', backupCount=3,
                                                 encoding='utf-8')
    # th_debug.namer = namer
    th_debug.suffix = "%Y-%m-%d.log"
    th_debug.setFormatter(format_str)
    th_debug.setLevel(logging.DEBUG)
    log.addHandler(th_debug)

    th = handlers.TimedRotatingFileHandler(filename=filename, when='D', backupCount=3, encoding='utf-8')
    # th.namer = namer
    th.suffix = "%Y-%m-%d.log"
    th.setFormatter(format_str)
    th.setLevel(logging.INFO)
    log.addHandler(th)
    log.setLevel(level)
    return log


os.makedirs('./logs', exist_ok=True)
logger = _logging(filename='./logs/default')
相關文章
相關標籤/搜索