zoukankan      html  css  js  c++  java
  • python日志滚动-修复按天滚动bug

    python日志滚动-修复按天滚动bug

    一、问题描述

    python自带的logging库有一个问题,当日志滚动设置为24h时:

    1、程序启动后,连续运行时间超过24h

    日志滚动分割正常。

    2、程序启动后,间断运行(用完就关闭,之后再启动),连续运行时间不足24h

    日志不发生分割,直到连续运行超过24h,才可以发生日志文件的分割。

    问题原因参考:https://blog.csdn.net/weixin_38107388/article/details/90639151

    二、目的

    自定义类MyTimedRotatingFileHandler,继承logging的基础类BaseRotatingHandler,实现间断启动,日志也能按天滚动分割。

    同时实现每天的0点之后开始滚动。

    三、操作实现

    1、目录结构

    运行前

    .test01/
    |—— libs/
    |   └─  logRecord.py
    └─ t1.py
    

    运行后

    .test01/
    |—— libs/
    |   └─  logRecord.py
    ├─ log/
    |   └─ dd.log
    └─ t1.py
    

    第二天运行后

    .test01/
    |—— libs/
    |   └─  logRecord.py
    ├─ log/
    |   ├─ dd.log
    |   └─ dd.log.2021-08-08.log
    └─ t1.py
    

    2、编写代码文件

    (1)新建日志模块 logRecord.py

    .test01/
    |—— libs/
    |   └─  logRecord.py
    

    logRecord.py:

    import os
    import logging
    import logging.handlers
    
    from stat import ST_CTIME
    from logging.handlers import *
    
    _MIDNIGHT = 24 * 60 * 60  # number of seconds in a day
    
    
    # 自定义自己的TimedRotatingFileHandler类
    class MyTimedRotatingFileHandler(BaseRotatingHandler):
        """解决程序二次启动后无法按照天分割的问题。
        继承logging中的BaseRotatingHandler类,重写TimedRotatingFileHandler的init方法,其他复制。
        """
    
        def __init__(self, filename, when='h', interval=1, backupCount=0, encoding=None, delay=False, utc=False,
                     atTime=None):
            BaseRotatingHandler.__init__(self, filename, 'a', encoding, delay)
            self.when = when.upper()
            self.backupCount = backupCount
            self.utc = utc
            self.atTime = atTime
            # Calculate the real rollover interval, which is just the number of
            # seconds between rollovers.  Also set the filename suffix used when
            # a rollover occurs.  Current 'when' events supported:
            # S - Seconds
            # M - Minutes
            # H - Hours
            # D - Days
            # midnight - roll over at midnight
            # W{0-6} - roll over on a certain day; 0 - Monday
            #
            # Case of the 'when' specifier is not important; lower or upper case
            # will work.
            if self.when == 'S':
                self.interval = 1  # one second
                self.suffix = "%Y-%m-%d_%H-%M-%S"
                self.extMatch = r"^d{4}-d{2}-d{2}_d{2}-d{2}-d{2}(.w+)?$"
            elif self.when == 'M':
                self.interval = 60  # one minute
                self.suffix = "%Y-%m-%d_%H-%M"
                self.extMatch = r"^d{4}-d{2}-d{2}_d{2}-d{2}(.w+)?$"
            elif self.when == 'H':
                self.interval = 60 * 60  # one hour
                self.suffix = "%Y-%m-%d_%H"
                self.extMatch = r"^d{4}-d{2}-d{2}_d{2}(.w+)?$"
            elif self.when == 'D' or self.when == 'MIDNIGHT':
                self.interval = 60 * 60 * 24  # one day
                self.suffix = "%Y-%m-%d"
                self.extMatch = r"^d{4}-d{2}-d{2}(.w+)?$"
            elif self.when.startswith('W'):
                self.interval = 60 * 60 * 24 * 7  # one week
                if len(self.when) != 2:
                    raise ValueError("You must specify a day for weekly rollover from 0 to 6 (0 is Monday): %s" % self.when)
                if self.when[1] < '0' or self.when[1] > '6':
                    raise ValueError("Invalid day specified for weekly rollover: %s" % self.when)
                self.dayOfWeek = int(self.when[1])
                self.suffix = "%Y-%m-%d"
                self.extMatch = r"^d{4}-d{2}-d{2}(.w+)?$"
            else:
                raise ValueError("Invalid rollover interval specified: %s" % self.when)
    
            self.extMatch = re.compile(self.extMatch, re.ASCII)
            self.interval = self.interval * interval  # multiply by units requested
            # The following line added because the filename passed in could be a
            # path object (see Issue #27493), but self.baseFilename will be a string
            filename = self.baseFilename
            if os.path.exists(filename):
                t = os.stat(filename)[ST_CTIME]  # 我修改过的地方。ST_MTIME ==> ST_CTIME
            else:
                t = int(time.time())
            self.rolloverAt = self.computeRollover(t)
    
        def computeRollover(self, currentTime):
            """
            Work out the rollover time based on the specified time.
            """
            result = currentTime + self.interval
    
            if self.when == 'MIDNIGHT' or self.when.startswith('W'):
                # This could be done with less code, but I wanted it to be clear
                if self.utc:
                    t = time.gmtime(currentTime)
                else:
                    t = time.localtime(currentTime)
                currentHour = t[3]
                currentMinute = t[4]
                currentSecond = t[5]
                currentDay = t[6]
                # r is the number of seconds left between now and the next rotation
                if self.atTime is None:
                    rotate_ts = _MIDNIGHT
                else:
                    rotate_ts = ((self.atTime.hour * 60 + self.atTime.minute) * 60 +
                                 self.atTime.second)
    
                r = rotate_ts - ((currentHour * 60 + currentMinute) * 60 +
                                 currentSecond)
                if r < 0:
                    # Rotate time is before the current time (for example when
                    # self.rotateAt is 13:45 and it now 14:15), rotation is
                    # tomorrow.
                    r += _MIDNIGHT
                    currentDay = (currentDay + 1) % 7
                result = currentTime + r
    
                if self.when.startswith('W'):
                    day = currentDay  # 0 is Monday
                    if day != self.dayOfWeek:
                        if day < self.dayOfWeek:
                            daysToWait = self.dayOfWeek - day
                        else:
                            daysToWait = 6 - day + self.dayOfWeek + 1
                        newRolloverAt = result + (daysToWait * (60 * 60 * 24))
                        if not self.utc:
                            dstNow = t[-1]
                            dstAtRollover = time.localtime(newRolloverAt)[-1]
                            if dstNow != dstAtRollover:
                                if not dstNow:  # DST kicks in before next rollover, so we need to deduct an hour
                                    addend = -3600
                                else:  # DST bows out before next rollover, so we need to add an hour
                                    addend = 3600
                                newRolloverAt += addend
                        result = newRolloverAt
            return result
    
        def shouldRollover(self, record):
            """
            Determine if rollover should occur.
    
            record is not used, as we are just comparing times, but it is needed so
            the method signatures are the same
            """
            t = int(time.time())
            if t >= self.rolloverAt:
                return 1
            return 0
    
        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
    
        def doRollover(self):
            """
            do a rollover; in this case, a date/time stamp is appended to the filename
            when the rollover happens.  However, you want the file to be named for the
            start of the interval, not the current time.  If there is a backup count,
            then we have to get a list of matching filenames, sort them and remove
            the one with the oldest suffix.
            """
            if self.stream:
                self.stream.close()
                self.stream = None
            # get the time that this sequence started at and make it a TimeTuple
            currentTime = int(time.time())
            dstNow = time.localtime(currentTime)[-1]
            t = self.rolloverAt - self.interval
            if self.utc:
                timeTuple = time.gmtime(t)
            else:
                timeTuple = time.localtime(t)
                dstThen = timeTuple[-1]
                if dstNow != dstThen:
                    if dstNow:
                        addend = 3600
                    else:
                        addend = -3600
                    timeTuple = time.localtime(t + addend)
            dfn = self.rotation_filename(self.baseFilename + "." +
                                         time.strftime(self.suffix, timeTuple))
            if os.path.exists(dfn):
                os.remove(dfn)
            self.rotate(self.baseFilename, dfn)
            if self.backupCount > 0:
                for s in self.getFilesToDelete():
                    os.remove(s)
            if not self.delay:
                self.stream = self._open()
            newRolloverAt = self.computeRollover(currentTime)
            while newRolloverAt <= currentTime:
                newRolloverAt = newRolloverAt + self.interval
            # If DST changes and midnight or weekly rollover, adjust for this.
            if (self.when == 'MIDNIGHT' or self.when.startswith('W')) and not self.utc:
                dstAtRollover = time.localtime(newRolloverAt)[-1]
                if dstNow != dstAtRollover:
                    if not dstNow:  # DST kicks in before next rollover, so we need to deduct an hour
                        addend = -3600
                    else:  # DST bows out before next rollover, so we need to add an hour
                        addend = 3600
                    newRolloverAt += addend
            self.rolloverAt = newRolloverAt
    
    
    # 用字典保存输出格式
    format_dict = {
        1: logging.Formatter('%(asctime)s - %(filename)-9s - line:%(lineno)3d - %(levelname)-5s - %(message)s'),
        2: logging.Formatter(
            '%(asctime)s - %(name)s - %(filename)s - line:%(lineno)d - pid:%(process)d - %(levelname)s - %(message)s'),
        3: logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s'),
        4: logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s'),
        5: logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
    }
    
    # 日志文件配置
    LOG_DIR_NAME = 'log'  # 日志统一存放文件夹
    LOG_DIR_PATH = os.path.join(os.getcwd(), LOG_DIR_NAME)  # 日志统一存放完整路径
    
    if not os.path.exists(LOG_DIR_PATH):  # 日志统一存放路径不存在,则创建该路径
        os.makedirs(LOG_DIR_PATH)
    
    
    class Logger(object):
        def __init__(self, logfile, logname, logformat):
            '''
               指定保存日志的文件路径,日志级别,以及调用文件
               将日志存入到指定的文件中
            '''
    
            # 一、创建一个logger
            self.logger = logging.getLogger(logname)
            self.logger.setLevel(logging.DEBUG)
    
            # 二、定义日志格式样本
            # formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
            formatter = format_dict[int(logformat)]
    
            # 三、定义两类handler
    
            # 1、定义日志文件handler
    
            # 1-1 日志滚动功能。按时间,1d滚动一次,保留30个旧log文件。
            # 创建一个日志文件handler。
            # tfh = logging.handlers.TimedRotatingFileHandler(
            tfh = MyTimedRotatingFileHandler(  # 不一样的地方。用我自定义的类MyTimedRotatingFileHandler
                logfile,
                when='D',
                interval=1,
                backupCount=30,
                encoding="utf-8"
            )
    
            # 设置滚动后缀名称。如:app1.log.2021-08-03.log
            tfh.suffix = "%Y-%m-%d.log"
    
            # 设置日志最低输出级别
            tfh.setLevel(logging.DEBUG)
    
            # 定义handler的输出格式
            tfh.setFormatter(formatter)
    
            # 给logger添加这个类型的handler
            self.logger.addHandler(tfh)
    
            # 2、定义日志控制台handler
            # 创建一个handler,用于输出到控制台
            ch = logging.StreamHandler()
    
            # 设置日志最低输出级别
            ch.setLevel(logging.DEBUG)
    
            # 定义handler的输出格式
            ch.setFormatter(formatter)
    
            # 给logger添加这个类型的handler
            self.logger.addHandler(ch)
    
        def getlog(self):
            return self.logger
    
    
    if __name__ == '__main__':
        # print(LOG_DIR_PATH)
        # 定义日志记录器1
        logfile1 = os.path.join(LOG_DIR_PATH, "app1.log")
        logger1 = Logger(logfile=logfile1, logname="fox1", logformat=1).getlog()
        logger1.debug('i am debug')
        logger1.info('i am info')
        logger1.warning('i am warning')
    
        # 定义日志记录器2
        logfile2 = os.path.join(LOG_DIR_PATH, "app2.log")
        logger2 = Logger(logfile=logfile2, logname="fox2", logformat=2).getlog()
        logger2.debug('i am debug2')
        logger2.info('i am info2')
        logger2.warning('i am warning2')
    

    (2)新建启动文件 t1.py

    import time
    import os
    import sys
    
    print("当前的工作目录:", os.getcwd())
    sys.path.append(os.getcwd())  # 一定要把当前路径加入环境变量中,否则命令行运行python时会导包失败。
    sys.path.append(r"D:xxyy	est01")
    print("python搜索模块的路径集合", sys.path)
    from libs.logRecord import *
    
    if __name__ == '__main__':
        # 定义日志记录器
        logfile = os.path.join(LOG_DIR_PATH, "dd.log")
        logger = Logger(logfile=logfile, logname="log_main", logformat=1).getlog()
        while True:
            time.sleep(1)
            logger.debug("debug 123")
        pass
    
    

    3、运行查看

    (1)第1天:第一次运行 t1.py

    python t1.py
    
    .test01/
    |—— libs/
    |   └─  logRecord.py
    ├─ log/
    |   └─ dd.log
    └─ t1.py
    

    (2)第1天:第二次运行 t1.py

    python t1.py
    
    .test01/
    |—— libs/
    |   └─  logRecord.py
    ├─ log/
    |   └─ dd.log
    └─ t1.py
    

    本次运行产生的日志,会追加到dd.log中去。

    注意:如果日志文件不再log文件夹中,不会追加,日志会程序启动时,就直接滚动一次。原因不清楚。

    (3)第2天:第三次运行 t1.py

    python t1.py
    
    .test01/
    |—— libs/
    |   └─  logRecord.py
    ├─ log/
    |   ├─ dd.log
    |   └─ dd.log.2021-08-08.log
    └─ t1.py
    

    因为过了晚上0点,本次运行后,日志文件发生了滚动。2021-08-08为文件的创建日期。

    二、其他问题

    1、解决程序启动后,直接新建日志文件问题

        # 定义日志记录器
        logfile = "./log/dd.log"  # 日志文件夹一定要放在指定文件夹。否则会重写
        logger = Logger(logfile=logfile, logname="log_main", logformat=1).getlog()
    

    2、查看文件创建日期

    look_create_time.py

    # -*- coding: utf-8 -*-
    import os
    import sys
    import time
    from stat import ST_CTIME, ST_MTIME
    
    
    # 封装好的函数2.1:时间戳 转为 日期字符串。单位s,秒。
    def time2date(timeint=1565673941, format="%Y-%m-%d %H:%M:%S"):
        '''
        时间戳转为日期字串,单位s,秒
        :param timeint:时间戳
        :return:日期字串
        输出举例说明:
        (1565673941, "%Y-%m-%d %H:%M:%S")  输出  2019-08-13 13:25:41
        (1565673941, "%Y-%m-%d")  输出  2019-08-13
        (1565673941, "%Y%m%d")  输出  20190813
        '''
        local_time = time.localtime(timeint)
        data_head = time.strftime(format, local_time)
    
        return data_head
    
    
    if __name__ == '__main__':
    
        filename = "2.csv"
        print(sys.argv)
        if len(sys.argv) > 1:
            filename = sys.argv[1]
            
        r = os.stat(filename)
        print(r)
        ctime_int = os.stat(filename)[ST_CTIME]
        mtime_int = os.stat(filename)[ST_MTIME]
    
        # 文件的创建时间查询:严格来说,是文件的权限修改时间。元数据的修改
        print("文件的创建时间:
    %s  <== ctime_int:%s" % (time2date(ctime_int), ctime_int))
    
        print()
        # 文件的修改时间查询
        print("文件的修改时间:
    %s  <== mtime_int:%s" % (time2date(mtime_int), mtime_int))
        pass
    
    

    查看命令

    python  look_create_time.py  ./log/1.log
    

    3、设置每天的0点滚动

            tfh = MyTimedRotatingFileHandler(  # 不一样的地方。用我自定义的类MyTimedRotatingFileHandler
                logfile,
                # when='D',  # 从生成日志文件开始,24h后分割
                when='MIDNIGHT',  # 每天的0点开始分割
                interval=1,
                backupCount=30,
                encoding="utf-8"
            )
    
  • 相关阅读:
    钢铁雄心4陆战攻略
    装甲军团1跳出问题
    Git:解决报错:fatal: The remote end hung up unexpectedly
    保存网页到zotero研究
    面向服务软件工程
    华为matebook x pro监听耳机电流声
    战争游戏红龙修改手册
    manjaro20安装teamviewer出现sudo teamviewer –daemon start无响应
    manjaro20WPS缺少字体
    manjora20不小心卸载,重新安装terminal,软件商店/软件中心linux类似
  • 原文地址:https://www.cnblogs.com/andy9468/p/15104061.html
Copyright © 2011-2022 走看看