import pytest

from tgm.exceptions import EventOrderedConversionError
from tgm.timedelta import TGB_TIME_DELTAS, TimeDeltaDG


@pytest.fixture(params=['Y', 'M', 'W', 'D', 'h', 's', 'ms', 'us', 'ns'])
def time_granularity(request):
    return request.param


def test_init_default_value(time_granularity):
    td = TimeDeltaDG(time_granularity)
    assert td.unit == time_granularity
    assert td.value == 1
    assert td.is_time_ordered


def test_init_non_default_value(time_granularity):
    value = 5
    td = TimeDeltaDG(time_granularity, value)
    assert td.unit == time_granularity
    assert td.value == value
    assert td.is_time_ordered


def test_init_event_ordered():
    time_granularity = 'r'
    td = TimeDeltaDG(time_granularity)
    assert td.unit == time_granularity
    assert td.value == 1
    assert td.is_event_ordered


@pytest.mark.parametrize('bad_unit', ['mock'])
def test_init_bad_unit(bad_unit):
    with pytest.raises(ValueError):
        _ = TimeDeltaDG(bad_unit)


@pytest.mark.parametrize('bad_value', [-1, 0])
def test_init_bad_value(time_granularity, bad_value):
    with pytest.raises(ValueError):
        _ = TimeDeltaDG(time_granularity, bad_value)


@pytest.mark.parametrize('bad_value', [-1, 0, 2])
def test_init_event_ordered_bad_value(bad_value):
    # Note: Only '1' accepted as the value for an event-ordered type
    with pytest.raises(ValueError):
        _ = TimeDeltaDG('r', bad_value)


def test_convert_between_same_units(time_granularity):
    td1 = TimeDeltaDG(time_granularity, 2)
    td2 = TimeDeltaDG(time_granularity, 3)
    assert td1.convert(td2) == 2 / 3
    assert td2.convert(td1) == 3 / 2


def test_convert_between_different_units():
    value_a = 5
    value_b = 3

    td_a = TimeDeltaDG('ns', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == (value_a / value_b) / (1000 * 1000 * 1000)
    assert td_b.convert(td_a) == (1000 * 1000 * 1000) * (value_b / value_a)

    td_a = TimeDeltaDG('us', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == (value_a / value_b) / (1000 * 1000)
    assert td_b.convert(td_a) == (1000 * 1000) * (value_b / value_a)

    td_a = TimeDeltaDG('ms', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == (value_a / value_b) / (1000)
    assert td_b.convert(td_a) == (1000) * (value_b / value_a)

    td_a = TimeDeltaDG('s', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == 1 * (value_a / value_b)
    assert td_b.convert(td_a) == (value_b / value_a) / 1

    td_a = TimeDeltaDG('m', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == 60 * (value_a / value_b)
    assert td_b.convert(td_a) == (value_b / value_a) / 60

    td_a = TimeDeltaDG('h', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == 60 * 60 * (value_a / value_b)
    assert td_b.convert(td_a) == (value_b / value_a) / (60 * 60)

    td_a = TimeDeltaDG('D', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == 24 * 60 * 60 * (value_a / value_b)
    assert td_b.convert(td_a) == (value_b / value_a) / (24 * 60 * 60)

    td_a = TimeDeltaDG('W', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == 7 * 24 * 60 * 60 * (value_a / value_b)
    assert td_b.convert(td_a) == (value_b / value_a) / (7 * 24 * 60 * 60)

    td_a = TimeDeltaDG('M', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == 30 * 24 * 60 * 60 * (value_a / value_b)
    assert td_b.convert(td_a) == (value_b / value_a) / (30 * 24 * 60 * 60)

    td_a = TimeDeltaDG('Y', value_a)
    td_b = TimeDeltaDG('s', value_b)
    assert td_a.convert(td_b) == 365 * 24 * 60 * 60 * (value_a / value_b)
    assert td_b.convert(td_a) == (value_b / value_a) / (365 * 24 * 60 * 60)


def test_bad_convert_from_event_ordered():
    td1 = TimeDeltaDG('r')
    td2 = TimeDeltaDG('Y', 1)

    with pytest.raises(EventOrderedConversionError):
        _ = td2.convert(td1)


def test_bad_convert_to_event_ordered():
    td1 = TimeDeltaDG('Y', 1)
    td2 = TimeDeltaDG('r')

    with pytest.raises(EventOrderedConversionError):
        _ = td1.convert(td2)


def test_time_delta_is_coarser_than():
    td1 = TimeDeltaDG('s', value=30)
    td2 = TimeDeltaDG('m')
    assert td2.is_coarser_than(td1)
    assert not td1.is_coarser_than('m')

    # Check that comparison is strict
    td1 = TimeDeltaDG('s')
    assert not td1.is_coarser_than(td1)
    assert not td1.is_coarser_than('s')

    # Check that the 'value' is taken into account (not just unit)
    td1 = TimeDeltaDG('s', value=60 * 5)
    td2 = TimeDeltaDG('m', value=3)
    assert td1.is_coarser_than(td2)
    assert not td2.is_coarser_than(td1)


def test_time_delta_is_coarser_than_due_to_value():
    # Testing that granularity is not just based on the unit.
    # In this case, 1 minute should be more granular than 80 seconds
    # even though seconds are more granular than minutes.
    td1 = TimeDeltaDG('s', value=80)
    td2 = TimeDeltaDG('m')
    assert not td2.is_coarser_than(td1)
    assert td1.is_coarser_than(td2)


def test_time_delta_is_coarser_try_compare_event_ordered():
    td1 = TimeDeltaDG('r')
    td2 = TimeDeltaDG('s')

    with pytest.raises(EventOrderedConversionError):
        td1.is_coarser_than(td2)

    with pytest.raises(EventOrderedConversionError):
        td2.is_coarser_than(td1)

    with pytest.raises(EventOrderedConversionError):
        td1.is_coarser_than('r')


def test_tgb_native_time_deltas():
    exp_dict = {
        'tgbl-wiki': TimeDeltaDG('s'),
        'tgbl-subreddit': TimeDeltaDG('s'),
        'tgbl-lastfm': TimeDeltaDG('s'),
        'tgbl-review': TimeDeltaDG('s'),
        'tgbl-coin': TimeDeltaDG('s'),
        'tgbl-flight': TimeDeltaDG('s'),
        'tgbl-comment': TimeDeltaDG('s'),
        'tgbn-trade': TimeDeltaDG('Y'),
        'tgbn-genre': TimeDeltaDG('s'),
        'tgbn-reddit': TimeDeltaDG('s'),
        'tgbn-token': TimeDeltaDG('s'),
    }
    assert TGB_TIME_DELTAS == exp_dict
