162 lines
6.2 KiB
Python
162 lines
6.2 KiB
Python
|
import os, time, shutil
|
||
|
from threading import Thread
|
||
|
|
||
|
class log(object):
|
||
|
def __init__(self, workdir):
|
||
|
'''
|
||
|
Init vars and threads.
|
||
|
'''
|
||
|
self.FATAL = "FATAL"
|
||
|
self.ERROR = "ERROR"
|
||
|
self.WARN = "WARN"
|
||
|
self.INFO = "INFO"
|
||
|
self.__levelMap = {
|
||
|
'FATAL': self.FATAL,
|
||
|
'ERROR': self.ERROR,
|
||
|
'WARN': self.WARN,
|
||
|
'INFO': self.INFO
|
||
|
}
|
||
|
if type(workdir) != str:
|
||
|
raise TypeError("Wrong type for param @workdir")
|
||
|
self.workdir = workdir
|
||
|
self.__logs = []
|
||
|
if os.path.exists("{}\\logs".format(workdir)) and os.path.isfile("{}\\logs".format(workdir)):
|
||
|
guard = Thread(target = self.__daemon)
|
||
|
guard.start()
|
||
|
else:
|
||
|
try:
|
||
|
os.makedirs("{}\\logs".format(workdir), exist_ok = True)
|
||
|
daemon = Thread(target = self.__daemon)
|
||
|
daemon.daemon = True
|
||
|
daemon.start()
|
||
|
except:
|
||
|
raise PermissionError("Failed to operate the log file! ")
|
||
|
def __daemon(self):
|
||
|
'''
|
||
|
Guard thread for recording logs and archiving.
|
||
|
'''
|
||
|
workdir = "{}\\logs".format(self.workdir)
|
||
|
self.__suicide = False
|
||
|
def archive(workdir):
|
||
|
timeStamp = time.time()
|
||
|
logsList = []
|
||
|
for fname in os.listdir(workdir):
|
||
|
if os.path.isfile("{}\\{}".format(workdir, fname)):
|
||
|
logsList.append("{}\\{}".format(workdir, fname))
|
||
|
archiveList = []
|
||
|
today = time.strftime("%Y-%m-%d", time.localtime(timeStamp))
|
||
|
# yesterday = time.strftime("%Y-%m-%d", time.localtime(timeStamp - 86400))
|
||
|
flag = False
|
||
|
for fname in logsList:
|
||
|
if os.path.basename(fname) != "{}.logs".format(today) and os.path.splitext(fname)[-1] == ".logs":
|
||
|
archiveList.append(fname)
|
||
|
flag = True
|
||
|
if flag:
|
||
|
self.info("Archiving logs.")
|
||
|
'''
|
||
|
#Archive all logs before today in a single zip file
|
||
|
os.makedirs("{}\\{}".format(workdir,yesterday),exist_ok=True)
|
||
|
for fname in archiveList:
|
||
|
shutil.move(fname,"{}\\{}".format(workdir,yesterday))
|
||
|
shutil.make_archive("{}\\{}".format(workdir,yesterday),"zip",workdir,yesterday)
|
||
|
shutil.rmtree("{}\\{}".format(workdir,yesterday))
|
||
|
'''
|
||
|
# Archive each log in independence zip file
|
||
|
for fname in archiveList:
|
||
|
basename = os.path.basename(fname)
|
||
|
shutil.make_archive("{}\\{}".format(workdir, os.path.splitext(basename)[0]), "zip", workdir, basename)
|
||
|
try:
|
||
|
os.remove("{}\\{}".format(workdir, basename))
|
||
|
except:
|
||
|
pass
|
||
|
|
||
|
Thread(target = archive, args = (workdir,)).start()
|
||
|
while True:
|
||
|
timeStamp = time.time()
|
||
|
today = time.strftime("%Y-%m-%d", time.localtime(timeStamp))
|
||
|
while True:
|
||
|
with open("{}\\{}.logs".format(workdir,today),"a+") as fb:
|
||
|
data, self.__logs = self.__logs, []
|
||
|
for log in data:
|
||
|
fb.write("[{}] [{}] {}\n".format(time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(log[0])), log[1], log[2]))
|
||
|
if self.__suicide or today != time.strftime("%Y-%m-%d", time.localtime()):
|
||
|
break
|
||
|
archive(workdir)
|
||
|
if self.__suicide:
|
||
|
break
|
||
|
def suicide(self):
|
||
|
'''
|
||
|
Suicide the guard thread.
|
||
|
Before close the program, You have to exec this function.
|
||
|
'''
|
||
|
self.__suicide = True
|
||
|
def record(self, level = "INFO", *content, split = " ", output = True):
|
||
|
'''
|
||
|
Record logs.
|
||
|
@*content: The content for your log
|
||
|
@split: The split text for your each log
|
||
|
@level: The level for your log level
|
||
|
@output: Print the log or not
|
||
|
'''
|
||
|
temp = []
|
||
|
for con in content:
|
||
|
temp.append(str(con))
|
||
|
content = split.join(temp)
|
||
|
if type(level) != str or level not in self.__levelMap.values():
|
||
|
raise TypeError("Wrong type for param @level")
|
||
|
if type(output) != bool:
|
||
|
raise TypeError("Wrong type for param @output")
|
||
|
timeStamp = time.time()
|
||
|
self.__logs.append((timeStamp, level.upper(), content))
|
||
|
if output:
|
||
|
print("[{}] [{}] {}".format(time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(timeStamp)), level.upper(), content))
|
||
|
def info(self, *content, split = " ", output = True):
|
||
|
'''
|
||
|
Record logs for info level.
|
||
|
@*content: The content for your log
|
||
|
@split: The split text for your each log
|
||
|
@output: Print the log or not
|
||
|
'''
|
||
|
temp = []
|
||
|
for con in content:
|
||
|
temp.append(str(con))
|
||
|
content = split.join(temp)
|
||
|
self.record("INFO", content, split = split, output = output)
|
||
|
def warn(self, *content, split = " ", output = True):
|
||
|
'''
|
||
|
Record logs for warn level.
|
||
|
@*content: The content for your log
|
||
|
@split: The split text for your each log
|
||
|
@output: Print the log or not
|
||
|
'''
|
||
|
temp = []
|
||
|
for con in content:
|
||
|
temp.append(str(con))
|
||
|
content = split.join(temp)
|
||
|
self.record("WARN", content, split = split, output = output)
|
||
|
def error(self, *content, split = " ", output = True):
|
||
|
'''
|
||
|
Record logs for error level.
|
||
|
@*content: The content for your log
|
||
|
@split: The split text for your each log
|
||
|
@output: Print the log or not
|
||
|
'''
|
||
|
temp = []
|
||
|
for con in content:
|
||
|
temp.append(str(con))
|
||
|
content = split.join(temp)
|
||
|
self.record("ERROR", content, split = split, output = output)
|
||
|
def fatal(self, *content, split = " ", output = True):
|
||
|
'''
|
||
|
Record logs for fatal level.
|
||
|
@*content: The content for your log
|
||
|
@split: The split text for your each log
|
||
|
@output: Print the log or not
|
||
|
'''
|
||
|
temp = []
|
||
|
for con in content:
|
||
|
temp.append(str(con))
|
||
|
content = split.join(temp)
|
||
|
self.record("FATAL", content, split = split, output = output)
|
||
|
self.__suicide = True
|
||
|
raise Exception(content)
|