historical/toontown-classic.git/panda/direct/directnotify/RotatingLog.py
2024-01-16 11:20:27 -06:00

144 lines
4.5 KiB
Python

import os
import time
class RotatingLog:
"""
A file() (or open()) replacement that will automatically open and write
to a new file if the prior file is too large or after a time interval.
"""
def __init__(self, path="./log_file", hourInterval=24, megabyteLimit=1024):
"""
path is a full or partial path with file name.
hourInterval is the number of hours at which to rotate the file.
megabyteLimit is the number of megabytes of file size the log
may grow to, after which the log is rotated. Note: The log
file may get a bit larger than limit do to writing out whole
lines (last line may exceed megabyteLimit or "megabyteGuidline").
"""
self.path=path
self.timeInterval=None
self.timeLimit=None
self.sizeLimit=None
if hourInterval is not None:
self.timeInterval=hourInterval*60*60
self.timeLimit=time.time()+self.timeInterval
if megabyteLimit is not None:
self.sizeLimit=megabyteLimit*1024*1024
def __del__(self):
self.close()
def close(self):
if hasattr(self, "file"):
self.file.flush()
self.file.close()
self.closed = self.file.closed
del self.file
else:
self.closed = 1
def shouldRotate(self):
"""
Returns a bool about whether a new log file should
be created and written to (while at the same time
stopping output to the old log file and closing it).
"""
if not hasattr(self, "file"):
return 1
if self.timeLimit is not None and time.time() > self.timeLimit:
return 1
if self.sizeLimit is not None and self.file.tell() > self.sizeLimit:
return 1
return 0
def filePath(self):
dateString=time.strftime("%Y_%m_%d_%H", time.localtime())
for i in range(26):
path="%s_%s_%s.log"%(self.path, dateString, chr(i+97))
if not os.path.exists(path) or os.stat(path)[6] < self.sizeLimit:
return path
# Hmm, 26 files are full? throw the rest in z:
# Maybe we should clear the self.sizeLimit here... maybe.
return path
def rotate(self):
"""
Rotate the log now. You normally shouldn't need to call this.
See write().
"""
path=self.filePath()
file=open(path, "a")
if file:
self.close()
# This should be redundant with "a" open() mode,
# but on some platforms tell() will return 0
# until the first write:
file.seek(0, 2)
self.file=file
# Some of these data members may be expected by some of our clients:
self.closed = self.file.closed
self.mode = self.file.mode
self.name = self.file.name
self.softspace = self.file.softspace
#self.encoding = self.file.encoding # Python 2.3
#self.newlines = self.file.newlines # Python 2.3, maybe
if self.timeLimit is not None and time.time() > self.timeLimit:
self.timeLimit=time.time()+self.timeInterval
else:
# We'll keep writing to the old file, if available.
print("RotatingLog error: Unable to open new log file \"%s\"." % (path,))
def write(self, data):
"""
Write the data to either the current log or a new one,
depending on the return of shouldRotate() and whether
the new file can be opened.
"""
if self.shouldRotate():
self.rotate()
if hasattr(self, "file"):
r = self.file.write(data)
self.file.flush()
return r
def flush(self):
return self.file.flush()
def fileno(self):
return self.file.fileno()
def isatty(self):
return self.file.isatty()
def __next__(self):
return next(self.file)
next = __next__
def read(self, size):
return self.file.read(size)
def readline(self, size):
return self.file.readline(size)
def readlines(self, sizehint):
return self.file.readlines(sizehint)
def xreadlines(self):
return self.file.xreadlines()
def seek(self, offset, whence=0):
return self.file.seek(offset, whence)
def tell(self):
return self.file.tell()
def truncate(self, size):
return self.file.truncate(size)
def writelines(self, sequence):
return self.file.writelines(sequence)