import sys
from collections import namedtuple
from pathlib import Path

import pytest
from _pytest.legacypath import Testdir
from _pytest.pytester import LineMatcher

pytest_plugins = ('pytester',)

THIS = Path(__file__)
STORAGE = THIS.with_name('test_storage')


@pytest.fixture
def testdir(testdir, monkeypatch):
    return namedtuple('testdir', ['makepyfile', 'runpytest_subprocess', 'tmpdir', 'run'])(
        testdir.makepyfile,
        testdir.runpytest_subprocess,
        testdir.tmpdir,
        lambda bin, *args: testdir.run(bin + '.exe' if sys.platform == 'win32' else bin, *args),
    )


def test_help(testdir):
    result = testdir.run('py.test-benchmark', '--help')
    result.stdout.fnmatch_lines(
        [
            'usage: py.test-benchmark *',
            '                         {help,list,compare} ...',
            '',
            "pytest_benchmark's management commands.",
            '',
            'option*:',
            '  -h [COMMAND], --help [COMMAND]',
            '                        Display help and exit.',
            '  --storage URI, -s URI',
            '                        Specify a path to store the runs as uri in form',
            '                        file://path or elasticsearch+http[s]://host1,host2/[in',
            '                        dex/doctype?project_name=Project] (when --benchmark-',
            '                        save or --benchmark-autosave are used). For backwards',
            '                        compatibility unexpected values are converted to',
            "                        file://<value>. Default: 'file://./.benchmarks'.",
            '  --verbose, -v         Dump diagnostic and progress information.',
            '',
            'commands:',
            '  {help,list,compare}',
            '    help                Display help and exit.',
            '    list                List saved runs.',
            '    compare             Compare saved runs.',
        ]
    )
    assert result.ret == 0


def test_help_command(testdir):
    result = testdir.run('py.test-benchmark', 'help')
    result.stdout.fnmatch_lines(
        [
            'usage: py.test-benchmark help [-h] [command]',
            '',
            'Display help and exit.',
            '',
            'positional arguments:',
            '  command',
            '',
            'option*:',
            '  -h, --help  show this help message and exit',
        ]
    )


@pytest.mark.parametrize('args', ['list --help', 'help list'])
def test_help_list(testdir, args):
    result = testdir.run('py.test-benchmark', *args.split())
    result.stdout.fnmatch_lines(
        [
            'usage: py.test-benchmark list [-h]',
            '',
            'List saved runs.',
            '',
            'option*:',
            '  -h, --help  show this help message and exit',
        ]
    )
    assert result.ret == 0


@pytest.mark.parametrize('args', ['compare --help', 'help compare'])
def test_help_compare(testdir, args):
    result = testdir.run('py.test-benchmark', *args.split())
    result.stdout.fnmatch_lines(
        [
            'usage: py.test-benchmark compare [-h] [--sort COL] [--group-by LABEL]',
            '                                 [--columns LABELS] [--name FORMAT]',
            '                                 [--histogram [FILENAME-PREFIX]]',
            '                                 [--csv [FILENAME]]',
            '                                 [[]glob_or_file *[]]',
            '',
            'Compare saved runs.',
            '',
            'positional arguments:',
            '  glob_or_file          Glob or exact path for json files. If not specified',
            '                        all runs are loaded.',
            '',
            'option*:',
            '  -h, --help            show this help message and exit',
            "  --sort COL            Column to sort on. Can be one of: 'min', 'max',",
            "                        'mean', 'stddev', 'name', 'fullname'. Default: 'min'",
            "  --group-by LABEL      How to group tests. Can be one of: 'group', 'name',",
            "                        'fullname', 'func', 'fullfunc', 'param' or",
            "                        'param:NAME', where NAME is the name passed to",
            "                        @pytest.parametrize. Default: 'group'",
            '  --columns LABELS      Comma-separated list of columns to show in the result',
            "                        table. Default: 'min, max, mean, stddev, median, iqr,",
            "                        outliers, ops, rounds, iterations'",
            "  --name FORMAT         How to format names in results. Can be one of 'short',",
            "                        'normal', 'long', or 'trial'. Default: 'normal'",
            '  --histogram [FILENAME-PREFIX]',
            '                        Plot graphs of min/max/avg/stddev over time in',
            '                        FILENAME-PREFIX-test_name.svg. If FILENAME-PREFIX',
            "                        contains slashes ('/') then directories will be",
            "                        created. Default: 'benchmark_*'",
            "  --csv [FILENAME]      Save a csv report. If FILENAME contains slashes ('/')",
            '                        then directories will be created. Default:',
            "                        'benchmark_*'",
            '',
            'examples:',
            '',
            "    pytest-benchmark compare 'Linux-CPython-3.5-64bit/*'",
            '',
            "        Loads all benchmarks ran with that interpreter. Note the special quoting that disables your shell's " 'glob',
            '        expansion.',
            '',
            '    pytest-benchmark compare 0001',
            '',
            '        Loads first run from all the interpreters.',
            '',
            '    pytest-benchmark compare /foo/bar/0001_abc.json /lorem/ipsum/0001_sir_dolor.json',
            '',
            '        Loads runs from exactly those files.',
        ]
    )
    assert result.ret == 0


def test_hooks(testdir: Testdir):
    testdir.makepyfile(
        conftest="""
def pytest_benchmark_scale_unit(config, unit, benchmarks, best, worst, sort):
    return '', 1
    """
    )
    result = testdir.run('py.test-benchmark', '--storage', STORAGE, 'list')
    assert result.stderr.lines == []
    result.stdout.fnmatch_lines(
        [
            '*0001_*.json',
            '*0002_*.json',
            '*0003_*.json',
            '*0004_*.json',
            '*0005_*.json',
            '*0006_*.json',
            '*0007_*.json',
            '*0008_*.json',
            '*0009_*.json',
            '*0010_*.json',
            '*0011_*.json',
            '*0012_*.json',
            '*0013_*.json',
            '*0014_*.json',
            '*0015_*.json',
            '*0016_*.json',
            '*0017_*.json',
            '*0018_*.json',
            '*0019_*.json',
            '*0020_*.json',
            '*0021_*.json',
            '*0022_*.json',
            '*0023_*.json',
            '*0024_*.json',
            '*0025_*.json',
            '*0026_*.json',
            '*0027_*.json',
            '*0028_*.json',
            '*0029_*.json',
            '*0030_*.json',
        ]
    )
    assert result.ret == 0


def test_list(testdir):
    result = testdir.run('py.test-benchmark', '--storage', STORAGE, 'list')
    assert result.stderr.lines == []
    result.stdout.fnmatch_lines(
        [
            '*0001_*.json',
            '*0002_*.json',
            '*0003_*.json',
            '*0004_*.json',
            '*0005_*.json',
            '*0006_*.json',
            '*0007_*.json',
            '*0008_*.json',
            '*0009_*.json',
            '*0010_*.json',
            '*0011_*.json',
            '*0012_*.json',
            '*0013_*.json',
            '*0014_*.json',
            '*0015_*.json',
            '*0016_*.json',
            '*0017_*.json',
            '*0018_*.json',
            '*0019_*.json',
            '*0020_*.json',
            '*0021_*.json',
            '*0022_*.json',
            '*0023_*.json',
            '*0024_*.json',
            '*0025_*.json',
            '*0026_*.json',
            '*0027_*.json',
            '*0028_*.json',
            '*0029_*.json',
            '*0030_*.json',
        ]
    )
    assert result.ret == 0


@pytest.mark.parametrize(
    ('name', 'name_pattern_generator'),
    [
        ('short', lambda n: '*xfast_parametrized[[]0[]] ' '(%.4d*)' % n),
        ('long', lambda n: '*xfast_parametrized[[]0[]] ' '(%.4d*)' % n),
        ('normal', lambda n: '*xfast_parametrized[[]0[]] ' '(%.4d*)' % n),
        ('trial', lambda n: '%.4d*' % n),
    ],
)
def test_compare(testdir, name, name_pattern_generator):
    result = testdir.run(
        'py.test-benchmark',
        '--storage',
        STORAGE,
        'compare',
        '0001',
        '0002',
        '0003',
        '--sort',
        'min',
        '--columns',
        'min,max',
        '--name',
        name,
        '--histogram',
        'foobar',
        '--csv',
        'foobar',
    )
    result.stderr.fnmatch_lines(['Generated csv: *foobar.csv'])
    LineMatcher(testdir.tmpdir.join('foobar.csv').readlines(cr=0)).fnmatch_lines(
        [
            'name,min,max',
            'tests/test_normal.py::test_xfast_parametrized[[]0[]],2.15628567*e-07,1.03186158*e-05',
            'tests/test_normal.py::test_xfast_parametrized[[]0[]],2.16902756*e-07,7.73929968*e-06',
            'tests/test_normal.py::test_xfast_parametrized[[]0[]],2.17314542*e-07,1.14473891*e-05',
            '',
        ]
    )
    result.stdout.fnmatch_lines(
        [
            '---*--- benchmark: 3 tests ---*---',
            'Name (time in ns) * Min * Max          ',
            '---*---',
            f'{name_pattern_generator(3)} * 215.6286 (1.0)      10*318.6159 (1.33)   ',
            f'{name_pattern_generator(2)} * 216.9028 (1.01)      7*739.2997 (1.0)    ',
            f'{name_pattern_generator(1)} * 217.3145 (1.01)     11*447.3891 (1.48)   ',
            '---*---',
            '',
            'Legend:',
            '  Outliers: 1 Standard Deviation from Mean; 1.5 IQR (InterQuartile Range) from 1st Quartile and 3rd Quartile.',
        ]
    )
    assert result.ret == 0


def test_compare_csv(testdir):
    test = testdir.makepyfile("""
    import pytest
    @pytest.mark.parametrize("arg", ["foo", "bar"])
    def test1(benchmark, arg):
        def func():
            print(arg)
        benchmark(func)
    def test2(benchmark):
        def func():
            print("foo")
        benchmark(func)
    """)
    result = testdir.runpytest_subprocess('--benchmark-autosave', test)
    result.stderr.fnmatch_lines(
        [
            'Saved benchmark data in: *',
        ]
    )
    result = testdir.run('py.test-benchmark', 'compare', '--csv')
    result.stderr.fnmatch_lines(['Generated csv: *.csv'])
