utils.py 2.83 KB
Newer Older
1
from contextlib import contextmanager
2
from dogapi import dog_stats_api
3
import json
4 5
import logging
import requests
6
import settings
7 8
from time import time
from uuid import uuid4
9

10
log = logging.getLogger(__name__)
11

Calen Pennington committed
12

13
def strip_none(dic):
Rocky Duan committed
14 15
    return dict([(k, v) for k, v in dic.iteritems() if v is not None])

Calen Pennington committed
16

Rocky Duan committed
17 18 19 20
def strip_blank(dic):
    def _is_blank(v):
        return isinstance(v, str) and len(v.strip()) == 0
    return dict([(k, v) for k, v in dic.iteritems() if not _is_blank(v)])
21

Calen Pennington committed
22

23
def extract(dic, keys):
Rocky Duan committed
24 25 26 27
    if isinstance(keys, str):
        return strip_none({keys: dic.get(keys)})
    else:
        return strip_none({k: dic.get(k) for k in keys})
28

Calen Pennington committed
29

30 31
def merge_dict(dic1, dic2):
    return dict(dic1.items() + dic2.items())
32

Calen Pennington committed
33

34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51
@contextmanager
def request_timer(request_id, method, url):
    start = time()
    yield
    end = time()
    duration = end - start
    dog_stats_api.histogram('comment_client.request.time', duration, end)
    log.info(
        "comment_client_request_log: request_id={request_id}, method={method}, "
        "url={url}, duration={duration}".format(
            request_id=request_id,
            method=method,
            url=url,
            duration=duration
        )
    )


52
def perform_request(method, url, data_or_params=None, *args, **kwargs):
53 54
    if data_or_params is None:
        data_or_params = {}
55
    headers = {'X-Edx-Api-Key': settings.API_KEY}
56 57
    request_id = uuid4()
    request_id_dict = {'request_id': request_id}
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73

    if method in ['post', 'put', 'patch']:
        data = data_or_params
        params = request_id_dict
    else:
        data = None
        params = merge_dict(data_or_params, request_id_dict)
    with request_timer(request_id, method, url):
        response = requests.request(
            method,
            url,
            data=data,
            params=params,
            headers=headers,
            timeout=5
        )
74

75
    if 200 < response.status_code < 500:
76
        raise CommentClientRequestError(response.text, response.status_code)
77 78 79
    # Heroku returns a 503 when an application is in maintenance mode
    elif response.status_code == 503:
        raise CommentClientMaintenanceError(response.text)
80
    elif response.status_code == 500:
81
        raise CommentClient500Error(response.text)
82 83 84 85 86 87
    else:
        if kwargs.get("raw", False):
            return response.text
        else:
            return json.loads(response.text)

Calen Pennington committed
88

89 90 91 92 93 94 95
class CommentClientError(Exception):
    def __init__(self, msg):
        self.message = msg

    def __str__(self):
        return repr(self.message)

Calen Pennington committed
96

97 98 99 100 101 102 103
class CommentClientRequestError(CommentClientError):
    def __init__(self, msg, status_code=400):
        super(CommentClientRequestError, self).__init__(msg)
        self.status_code = status_code


class CommentClient500Error(CommentClientError):
104 105 106
    pass


107
class CommentClientMaintenanceError(CommentClientError):
108
    pass