Files @ b990071f0a51
Branch filter:

Location: FVDE/ennstatus/ennstatus/status/functions.py

Dennis Fink
Don't fail on wrong written json files
import os
import os.path
import json

from collections import defaultdict
from datetime import datetime

from flask import current_app
from flask_mail import Mail, Message


mail = Mail()


def _send_mail(server_name, last_updated):

    current_app.logger.info('Try sending mail')
    try:
        msg = Message('[Ennstatus] %s went offline' % (server_name),
                      sender='ennstatus@enn.lu')
        msg.add_recipient(current_app.config['SERVER_ADMINS'][server_name])
        msg.body = ('%s went offline. I received the last update at %s'
                    % (server_name, last_updated))
        mail.send(msg)
    except KeyError:
        current_app.logger.error('Admin for %s not found!' % server_name)
    except AssertionError:
        pass
    except Exception as e:
        current_app.logger.error('Unexpected error: %s' % e,
                                 exc_info=True)
    current_app.logger.info('Finished trying!')


def _check_server(data):

    server_name = data['server_name']
    last_updated = data['last_updated']
    current_status = data['server_status']
    filename = os.path.join('data', '.'.join([server_name.lower(), 'json']))

    date = datetime.strptime(last_updated, '%d-%m-%Y %H:%M:%S')
    now = datetime.utcnow()
    delta = now - date

    if delta.days >= 7:
        os.remove(filename)
        current_app.logger.error('%s was removed!' % server_name)
        return False
    elif current_status == 'Offline':
        return data

    if delta.seconds >= 3600:
        status = 'Offline'
    elif delta.seconds >= 1200:
        status = 'Unknown'
    else:
        status = None

    if status is not None:
        current_app.logger.error('%s is set to %s' % (server_name,
                                 status))

        if (current_status == 'Online' or current_status == 'Unknown') and \
           status == 'Offline':
            _send_mail(server_name, last_updated)

        for key in ('server_status', 'tor_status'):
            data[key] = status

        with open(filename, mode='w', encoding='utf-8') as file_object:
            json.dump(data, file_object)

    return data


def _load_single_server(filename):

    current_app.logger.info('Loading {}'.format(filename))
    try:
        with open(filename, encoding='utf-8') as f:
            server = json.load(f)
    except (IOError, ValueError):
        return False

    server = _check_server(server)
    return server


def single_server(name):

    filename = ''.join(['data/', name, '.json'])
    return _load_single_server(filename)


def _get_json_files(root, files):

    for f in files:
        if f.endswith('.json'):
            yield os.path.join(root, f)


def all_servers():

    for root, _, files in os.walk('data'):
        for f in _get_json_files(root, files):
            yield _load_single_server(f)


def all_servers_by_type(type):

    for server in all_servers():
        try:
            if server['server_type'] == type:
                yield server
        except TypeError:
            continue


def split_all_servers_to_types():

    servers = defaultdict(list)

    for server in all_servers():
        try:
            servers[server['server_type']].append(server)
        except TypeError:
            continue

    return servers