2019-10-21 15:52:14 -07:00
|
|
|
import logging
|
2024-04-19 10:36:40 +02:00
|
|
|
import re
|
2019-10-21 15:52:14 -07:00
|
|
|
|
|
|
|
import requests
|
|
|
|
|
2024-03-10 16:18:49 -07:00
|
|
|
import borgmatic.hooks.logs
|
2019-11-12 15:31:07 -08:00
|
|
|
from borgmatic.hooks import monitor
|
|
|
|
|
2019-10-21 15:52:14 -07:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2019-11-12 15:31:07 -08:00
|
|
|
MONITOR_STATE_TO_HEALTHCHECKS = {
|
|
|
|
monitor.State.START: 'start',
|
|
|
|
monitor.State.FINISH: None, # Healthchecks doesn't append to the URL for the finished state.
|
|
|
|
monitor.State.FAIL: 'fail',
|
2023-03-05 19:27:32 +05:30
|
|
|
monitor.State.LOG: 'log',
|
2019-11-12 15:31:07 -08:00
|
|
|
}
|
|
|
|
|
2024-03-10 16:18:49 -07:00
|
|
|
DEFAULT_PING_BODY_LIMIT_BYTES = 1500
|
|
|
|
HANDLER_IDENTIFIER = 'healthchecks'
|
2019-11-17 16:54:27 -08:00
|
|
|
|
2019-10-21 15:52:14 -07:00
|
|
|
|
2023-07-09 17:40:02 -07:00
|
|
|
def initialize_monitor(hook_config, config, config_filename, monitoring_log_level, dry_run):
|
2020-06-02 14:33:41 -07:00
|
|
|
'''
|
2022-05-24 14:44:33 -07:00
|
|
|
Add a handler to the root logger that stores in memory the most recent logs emitted. That way,
|
|
|
|
we can send them all to Healthchecks upon a finish or failure state. But skip this if the
|
|
|
|
"send_logs" option is false.
|
2020-06-02 14:33:41 -07:00
|
|
|
'''
|
2022-05-24 14:44:33 -07:00
|
|
|
if hook_config.get('send_logs') is False:
|
|
|
|
return
|
|
|
|
|
2022-05-24 12:23:38 -07:00
|
|
|
ping_body_limit = max(
|
|
|
|
hook_config.get('ping_body_limit', DEFAULT_PING_BODY_LIMIT_BYTES)
|
2024-03-10 16:18:49 -07:00
|
|
|
- len(borgmatic.hooks.logs.PAYLOAD_TRUNCATION_INDICATOR),
|
2022-05-24 12:23:38 -07:00
|
|
|
0,
|
|
|
|
)
|
|
|
|
|
2024-03-10 16:18:49 -07:00
|
|
|
borgmatic.hooks.logs.add_handler(
|
|
|
|
borgmatic.hooks.logs.Forgetful_buffering_handler(
|
|
|
|
HANDLER_IDENTIFIER, ping_body_limit, monitoring_log_level
|
|
|
|
)
|
2020-06-02 14:33:41 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-07-08 23:14:30 -07:00
|
|
|
def ping_monitor(hook_config, config, config_filename, state, monitoring_log_level, dry_run):
|
2019-10-21 15:52:14 -07:00
|
|
|
'''
|
2022-05-23 20:02:10 -07:00
|
|
|
Ping the configured Healthchecks URL or UUID, modified with the monitor.State. Use the given
|
2020-01-22 15:10:47 -08:00
|
|
|
configuration filename in any log entries, and log to Healthchecks with the giving log level.
|
|
|
|
If this is a dry run, then don't actually ping anything.
|
2019-10-21 15:52:14 -07:00
|
|
|
'''
|
|
|
|
ping_url = (
|
2022-05-23 20:02:10 -07:00
|
|
|
hook_config['ping_url']
|
|
|
|
if hook_config['ping_url'].startswith('http')
|
2023-03-23 23:11:14 -07:00
|
|
|
else f"https://hc-ping.com/{hook_config['ping_url']}"
|
2019-10-21 15:52:14 -07:00
|
|
|
)
|
|
|
|
dry_run_label = ' (dry run; not actually pinging)' if dry_run else ''
|
|
|
|
|
2022-05-24 14:09:42 -07:00
|
|
|
if 'states' in hook_config and state.name.lower() not in hook_config['states']:
|
2022-05-24 13:59:28 -07:00
|
|
|
logger.info(
|
2022-05-24 14:09:42 -07:00
|
|
|
f'{config_filename}: Skipping Healthchecks {state.name.lower()} ping due to configured states'
|
2022-05-24 13:59:28 -07:00
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2024-04-19 10:36:40 +02:00
|
|
|
ping_url_is_uuid = re.match(r'(\w{4}-?){4}$', hook_config['ping_url'])
|
|
|
|
|
2019-11-12 15:31:07 -08:00
|
|
|
healthchecks_state = MONITOR_STATE_TO_HEALTHCHECKS.get(state)
|
|
|
|
if healthchecks_state:
|
2023-03-23 23:11:14 -07:00
|
|
|
ping_url = f'{ping_url}/{healthchecks_state}'
|
2019-10-21 15:52:14 -07:00
|
|
|
|
2024-04-19 10:36:40 +02:00
|
|
|
if hook_config.get('create_slug') and not ping_url_is_uuid:
|
|
|
|
ping_url = f'{ping_url}?create=1'
|
|
|
|
|
2023-03-23 23:11:14 -07:00
|
|
|
logger.info(f'{config_filename}: Pinging Healthchecks {state.name.lower()}{dry_run_label}')
|
|
|
|
logger.debug(f'{config_filename}: Using Healthchecks ping URL {ping_url}')
|
2019-10-21 15:52:14 -07:00
|
|
|
|
2023-03-05 19:27:32 +05:30
|
|
|
if state in (monitor.State.FINISH, monitor.State.FAIL, monitor.State.LOG):
|
2024-03-10 16:18:49 -07:00
|
|
|
payload = borgmatic.hooks.logs.format_buffered_logs_for_payload(HANDLER_IDENTIFIER)
|
2020-06-02 14:33:41 -07:00
|
|
|
else:
|
|
|
|
payload = ''
|
2019-11-17 16:54:27 -08:00
|
|
|
|
2019-11-07 10:08:44 -08:00
|
|
|
if not dry_run:
|
|
|
|
logging.getLogger('urllib3').setLevel(logging.ERROR)
|
2022-05-24 15:50:04 -07:00
|
|
|
try:
|
2022-07-23 22:07:06 +02:00
|
|
|
response = requests.post(
|
|
|
|
ping_url, data=payload.encode('utf-8'), verify=hook_config.get('verify_tls', True)
|
|
|
|
)
|
2022-06-29 21:19:40 -07:00
|
|
|
if not response.ok:
|
|
|
|
response.raise_for_status()
|
2022-05-24 15:50:04 -07:00
|
|
|
except requests.exceptions.RequestException as error:
|
|
|
|
logger.warning(f'{config_filename}: Healthchecks error: {error}')
|
2020-06-23 11:01:03 -07:00
|
|
|
|
|
|
|
|
2023-07-09 17:40:02 -07:00
|
|
|
def destroy_monitor(hook_config, config, config_filename, monitoring_log_level, dry_run):
|
2020-06-23 11:01:03 -07:00
|
|
|
'''
|
|
|
|
Remove the monitor handler that was added to the root logger. This prevents the handler from
|
|
|
|
getting reused by other instances of this monitor.
|
|
|
|
'''
|
2024-03-10 16:18:49 -07:00
|
|
|
borgmatic.hooks.logs.remove_handler(HANDLER_IDENTIFIER)
|