forked from ginking/archimedes-1
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlogs.py
157 lines (121 loc) · 5.06 KB
/
logs.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
# -*- coding: utf-8 -*-
from __future__ import unicode_literals
from backoff import expo
from backoff import on_exception
from google.cloud import error_reporting
from google.cloud import logging
from logging import Formatter
from logging import getLogger
from logging import DEBUG
from logging.handlers import RotatingFileHandler
from sys import exc_info
from traceback import format_exception
# The format for local logs.
LOGS_FORMAT = ("%(asctime)s "
"%(name)s "
"%(process)d "
"%(thread)d "
"%(levelname)s "
"%(message)s")
# The path to the log file for local logging.
LOG_FILE = "/tmp/trump2cash.log"
# The path to the log file for the local fallback of cloud logging.
FALLBACK_LOG_FILE = "/tmp/trump2cash-fallback.log"
# The maximum size in bytes for each local log file.
MAX_LOG_BYTES = 10 * 1024 * 1024
class Logs:
"""A helper for logging locally or in the cloud."""
def __init__(self, name, to_cloud=True):
self.to_cloud = to_cloud
if self.to_cloud:
# Initialize the Stackdriver logging and error reporting clients.
self.cloud_logger = logging.Client().logger(name)
self.error_client = error_reporting.Client()
# Initialize the local fallback logger.
self.fallback_logger, fallback_handler = self.get_local_logger(
name, FALLBACK_LOG_FILE)
# Redirect the backoff logs to the local fallback handler.
backoff_logger = getLogger("backoff")
backoff_logger.setLevel(DEBUG)
backoff_logger.handlers = [fallback_handler]
else:
# Initialize the local file logger.
self.local_logger, _ = self.get_local_logger(name, LOG_FILE)
def get_local_logger(self, name, log_file):
"""Returns a local logger with a file handler."""
handler = RotatingFileHandler(log_file, maxBytes=MAX_LOG_BYTES)
handler.setFormatter(Formatter(LOGS_FORMAT))
handler.setLevel(DEBUG)
logger = getLogger(name)
logger.setLevel(DEBUG)
logger.handlers = [handler]
return (logger, handler)
def debug(self, text):
"""Logs at the DEBUG level."""
if self.to_cloud:
self.safe_cloud_log_text(text, severity="DEBUG")
else:
self.local_logger.debug(text)
def info(self, text):
"""Logs at the INFO level."""
if self.to_cloud:
self.safe_cloud_log_text(text, severity="INFO")
else:
self.local_logger.info(text)
def warn(self, text):
"""Logs at the WARNING level."""
if self.to_cloud:
self.safe_cloud_log_text(text, severity="WARNING")
else:
self.local_logger.warning(text)
def error(self, text):
"""Logs at the ERROR level."""
if self.to_cloud:
self.safe_cloud_log_text(text, severity="ERROR")
else:
self.local_logger.error(text)
def catch(self):
"""Logs the latest exception."""
exception_str = self.format_exception()
if self.to_cloud:
self.safe_report_exception(exception_str)
self.safe_cloud_log_text(exception_str, severity="CRITICAL")
else:
self.local_logger.critical(exception_str)
def safe_cloud_log_text(self, text, severity):
"""Logs to the cloud, retries if necessary, and eventually fails over
to local logs.
"""
try:
self.retry_cloud_log_text(text, severity)
except Exception:
exception_str = self.format_exception()
self.fallback_logger.error("Failed to log to cloud: %s %s\n%s" %
(severity, text, exception_str))
@on_exception(expo, Exception, max_tries=8)
def retry_cloud_log_text(self, text, severity):
"""Logs to the cloud and retries up to 10 times with exponential backoff
(51.2 seconds max total) if the upload fails.
"""
self.cloud_logger.log_text(text, severity=severity)
def safe_report_exception(self, exception_str):
"""Reports the exception, retries if necessary, and eventually fails
over to local logs.
"""
try:
self.retry_report_exception(exception_str)
except Exception:
meta_exception_str = self.format_exception()
self.fallback_logger.error("Failed to report exception: %s\n%s" %
(exception_str, meta_exception_str))
@on_exception(expo, Exception, max_tries=8)
def retry_report_exception(self, exception_str):
"""Reports the exception and retries up to 10 times with exponential
backoff (51.2 seconds max total) if the upload fails.
"""
self.error_client.report(exception_str)
def format_exception(self):
"""Grabs the latest exception and formats it."""
exc_type, exc_value, exc_traceback = exc_info()
exc_format = format_exception(exc_type, exc_value, exc_traceback)
return "".join(exc_format).strip()