from tempfile import TemporaryDirectory, NamedTemporaryFile
from pathlib import Path

from unittest import TestCase
from unittest.mock import patch
from contextlib import redirect_stdout
import io

from sohstationviewer.controller.processing import (
    load_data,
    read_mseed_channels,
    detect_data_type,
    get_data_type_from_file
)
from sohstationviewer.database.extract_data import get_signature_channels
from PySide2 import QtWidgets
from sohstationviewer.model.mseed.mseed import MSeed
from sohstationviewer.model.reftek.reftek import RT130

TEST_DATA_DIR = Path(__file__).resolve().parent.parent.joinpath('test_data')
rt130_dir = TEST_DATA_DIR.joinpath('RT130-sample/2017149.92EB/2017150')
q330_dir = TEST_DATA_DIR.joinpath('Q330-sample/day_vols_AX08')
centaur_dir = TEST_DATA_DIR.joinpath('Centaur-sample/SOH')
pegasus_dir = TEST_DATA_DIR.joinpath('Pegasus-sample/Pegasus_SVC4/soh')
mix_traces_dir = TEST_DATA_DIR.joinpath('Q330_mixed_traces')


class TestLoadDataAndReadChannels(TestCase):
    """Test suite for load_data and read_mseed_channels."""

    def setUp(self) -> None:
        """Set up test fixtures."""
        patcher = patch.object(QtWidgets, 'QTextBrowser')
        self.addCleanup(patcher.stop)
        self.widget_stub = patcher.start()

        # The actual value can be 'Q330', 'Centaur', or 'Pegasus'.
        # The code that uses this string only cares that it is not 'RT130',
        # though, so we are setting it to a stub value.
        self.mseed_dtype = 'MSeed'

    def test_load_data_rt130_good_dir(self):
        """
        Test basic functionality of load_data - the given directory can be
        loaded without issues. Test RT130.
        """
        self.assertIsInstance(
            load_data('RT130', self.widget_stub, [rt130_dir]),
            RT130
        )

    def test_load_data_mseed_q330_good_data_dir(self):
        """
        Test basic functionality of load_data - the given directory can be
        loaded without issues. Test MSeed.
        """
        self.assertIsInstance(
            load_data(self.mseed_dtype, self.widget_stub, [q330_dir]),
            MSeed
        )
        self.assertIsInstance(
            load_data(self.mseed_dtype, self.widget_stub, [centaur_dir]),
            MSeed
        )
        self.assertIsInstance(
            load_data(self.mseed_dtype, self.widget_stub, [pegasus_dir]),
            MSeed
        )

    def test_load_data_no_dir(self):
        """Test basic functionality of load_data - no directory was given."""
        no_dir_given = []
        self.assertIsNone(load_data('RT130', self.widget_stub, no_dir_given))
        self.assertIsNone(
            load_data(self.mseed_dtype, self.widget_stub, no_dir_given))

    def test_load_data_dir_does_not_exist(self):
        """
        Test basic functionality of load_data - the given directory does not
        exist.
        """
        empty_name_dir = ['']
        non_existent_dir = ['dir_that_does_not_exist']

        self.assertIsNone(
            load_data('RT130', self.widget_stub, empty_name_dir))
        self.assertIsNone(
            load_data('RT130', self.widget_stub, non_existent_dir))

        self.assertIsNone(
            load_data(self.mseed_dtype, self.widget_stub, empty_name_dir))
        self.assertIsNone(
            load_data(self.mseed_dtype, self.widget_stub, non_existent_dir))

    def test_load_data_empty_dir(self):
        """
        Test basic functionality of load_data - the given directory is empty.
        """
        with TemporaryDirectory() as empty_dir:
            self.assertIsNone(
                load_data('RT130', self.widget_stub, [empty_dir]))
            self.assertIsNone(
                load_data(self.mseed_dtype, self.widget_stub, [empty_dir]))

    def test_load_data_empty_data_dir(self):
        """
        Test basic functionality of load_data - the given directory
        contains a data folder but no data file.
        """
        with TemporaryDirectory() as outer_dir:
            with TemporaryDirectory(dir=outer_dir) as data_dir:
                self.assertIsNone(
                    load_data('RT130', self.widget_stub, [data_dir]))
                self.assertIsNone(
                    load_data(self.mseed_dtype, self.widget_stub, [outer_dir]))

    def test_load_data_data_type_mismatch(self):
        """
        Test basic functionality of load_data - the data type given does not
        match the type of the data contained in the given directory.
        """
        self.assertIsNone(
            load_data('RT130', self.widget_stub, [q330_dir]))
        self.assertIsNone(
            load_data(self.mseed_dtype, self.widget_stub, [rt130_dir]))

    def test_load_data_data_traceback_error(self):
        """
        Test basic functionality of load_data - when there is an error
        on loading data, the traceback info will be printed out
        """
        f = io.StringIO()
        with redirect_stdout(f):
            self.assertIsNone(load_data('RT130', None, [q330_dir]))
        output = f.getvalue()
        self.assertIn(
            f"Dir {q330_dir} "
            f"can't be read due to error: Traceback",
            output
        )
        with redirect_stdout(f):
            self.assertIsNone(
                load_data(self.mseed_dtype, None, [rt130_dir]))
        output = f.getvalue()
        self.assertIn(
            f"Dir {rt130_dir} "
            f"can't be read due to error: Traceback",
            output
        )

    def test_read_channels_mseed_dir(self):
        """
        Test basic functionality of load_data - the given directory contains
        MSeed data.
        """
        q330_soh_channels = sorted(['LOG', 'VKI'])
        q330_mass_pos_channels = ['VM1']
        q330_wf_channels = ['HHE', 'LHE']
        q330_spr_gt_1 = []
        ret = read_mseed_channels(self.widget_stub, [q330_dir], True)
        self.assertListEqual(ret[0], q330_soh_channels)
        self.assertListEqual(ret[1], q330_mass_pos_channels)
        self.assertListEqual(ret[2], q330_wf_channels)
        self.assertListEqual(ret[3], q330_spr_gt_1)

        centaur_soh_channels = sorted(
            ['VDT', 'EX3', 'GEL', 'VEC', 'EX2', 'LCE', 'EX1', 'GLA', 'LCQ',
             'GPL', 'GNS', 'GST', 'VCO', 'GAN', 'GLO', 'VPB', 'VEI'])
        centaur_mass_pos_channels = sorted(['VM1', 'VM2', 'VM3'])
        centaur_wf_channels = []
        centaur_spr_gt_1 = []
        ret = read_mseed_channels(self.widget_stub, [centaur_dir], True)
        self.assertListEqual(ret[0], centaur_soh_channels)
        self.assertListEqual(ret[1], centaur_mass_pos_channels)
        self.assertListEqual(ret[2], centaur_wf_channels)
        self.assertListEqual(ret[3], centaur_spr_gt_1)

        pegasus_soh_channels = sorted(['VDT', 'VE1'])
        pegasus_mass_pos_channels = sorted(['VM1'])
        pegasus_wf_channels = []
        pegasus_spr_gt_1 = []
        ret = read_mseed_channels(self.widget_stub, [pegasus_dir], True)
        self.assertListEqual(ret[0], pegasus_soh_channels)
        self.assertListEqual(ret[1], pegasus_mass_pos_channels)
        self.assertListEqual(ret[2], pegasus_wf_channels)
        self.assertListEqual(ret[3], pegasus_spr_gt_1)

        mix_traces_soh_channels = ['LOG']
        mix_traces_mass_pos_channels = []
        mix_traces_wf_channels = sorted(
            ['BH1', 'BH2', 'BH3', 'BH4', 'BH5', 'BH6',
             'EL1', 'EL2', 'EL4', 'EL5', 'EL6', 'ELZ'])
        mix_traces_spr_gt_1 = sorted(
            ['BS1', 'BS2', 'BS3', 'BS4', 'BS5', 'BS6',
             'ES1', 'ES2', 'ES3', 'ES4', 'ES5', 'ES6',
             'LS1', 'LS2', 'LS3', 'LS4', 'LS5', 'LS6',
             'SS1', 'SS2', 'SS3', 'SS4', 'SS5', 'SS6'])
        ret = read_mseed_channels(self.widget_stub, [mix_traces_dir], True)
        self.assertListEqual(ret[0], mix_traces_soh_channels)
        self.assertListEqual(ret[1], mix_traces_mass_pos_channels)
        self.assertListEqual(ret[2], mix_traces_wf_channels)
        self.assertListEqual(ret[3], mix_traces_spr_gt_1)

    def test_read_channels_rt130_dir(self):
        """
        Test basic functionality of load_data - the given directory contains
        RT130 data.
        """
        with self.assertRaises(Exception):
            read_mseed_channels(self.widget_stub, [rt130_dir], True)

    def test_read_mseed_channels_no_dir(self):
        """
        Test basic functionality of read_mseed_channels - no directory was
        given.
        """
        no_dir = []
        ret = read_mseed_channels(self.widget_stub, no_dir, True)
        self.assertEqual(ret, ([], [], [], []))

    def test_read_mseed_channels_dir_does_not_exist(self):
        """
        Test basic functionality of read_mseed_channels - the given directory
        does not exist.
        """
        empty_name_dir = ['']
        ret = read_mseed_channels(self.widget_stub, empty_name_dir, True)
        self.assertEqual(ret, ([], [], [], []))

        non_existent_dir = ['non_existent_dir']
        ret = read_mseed_channels(self.widget_stub, non_existent_dir, True)
        self.assertEqual(ret, ([], [], [], []))

    def test_read_mseed_channels_empty_dir(self):
        """
        Test basic functionality of read_mseed_channels - the given directory
        is empty.
        """
        with TemporaryDirectory() as empty_dir:
            ret = read_mseed_channels(self.widget_stub, [empty_dir], True)
            self.assertEqual(ret, ([], [], [], []))

    def test_read_mseed_channels_empty_data_dir(self):
        """
        Test basic functionality of read_mseed_channels - the given directory
        contains a data folder but no data file.
        """
        with TemporaryDirectory() as outer_dir:
            with TemporaryDirectory(dir=outer_dir):
                ret = read_mseed_channels(self.widget_stub, [outer_dir], True)
                self.assertEqual(ret, ([], [], [], []))


class TestDetectDataType(TestCase):
    """
    Test suite for detect_data_type and get_data_type_from_file functions.
    """

    def setUp(self) -> None:
        """Set up text fixtures."""
        widget_patcher = patch.object(QtWidgets, 'QTextBrowser')
        self.addCleanup(widget_patcher.stop)
        self.widget_stub = widget_patcher.start()

        func_patcher = patch('sohstationviewer.controller.processing.'
                             'get_data_type_from_file')
        self.addCleanup(func_patcher.stop)
        self.mock_get_data_type_from_file = func_patcher.start()

        self.dir1 = TemporaryDirectory()
        self.dir2 = TemporaryDirectory()
        self.file1 = NamedTemporaryFile(dir=self.dir1.name)
        self.file2 = NamedTemporaryFile(dir=self.dir2.name)

    def tearDown(self) -> None:
        """Teardown text fixtures."""
        del self.file1, self.file2
        self.dir1.cleanup()
        self.dir2.cleanup()

    def test_one_directory_not_unknown_data_type(self):
        """
        Test basic functionality of detect_data_type - only one directory was
        given and the data type it contains can be detected.
        """
        expected_data_type = ('RT130', '_')
        self.mock_get_data_type_from_file.return_value = expected_data_type

        self.assertEqual(
            detect_data_type(self.widget_stub, [self.dir1.name]),
            expected_data_type[0]
        )

    def test_same_data_type_and_channel(self):
        """
        Test basic functionality of detect_data_type - the given directories
        contain the same data type and the data type was detected using the
        same channel.
        """
        expected_data_type = ('RT130', '_')
        self.mock_get_data_type_from_file.return_value = expected_data_type

        self.assertEqual(
            detect_data_type(
                self.widget_stub, [self.dir1.name, self.dir2.name]
            ),
            expected_data_type[0]
        )

    def test_same_data_type_different_channel(self):
        """
        Test basic functionality of detect_data_type - the given directories
        contain the same data type but the data type was detected using
        different channels.
        """
        returned_data_types = [('Q330', 'OCF'), ('Q330', 'VEP')]
        self.mock_get_data_type_from_file.side_effect = returned_data_types

        self.assertEqual(
            detect_data_type(
                self.widget_stub, [self.dir1.name, self.dir2.name]
            ),
            returned_data_types[0][0]
        )

    def test_different_data_types(self):
        """
        Test basic functionality of detect_data_type - the given directories
        contain different data types.
        """
        returned_data_types = [('RT130', '_'), ('Q330', 'VEP')]
        self.mock_get_data_type_from_file.side_effect = returned_data_types

        self.assertIsNone(
            detect_data_type(
                self.widget_stub, [self.dir1.name, self.dir2.name]
            )
        )

    def test_unknown_data_type(self):
        """
        Test basic functionality of detect_data_type - can't detect any data
        type.
        """
        unknown_data_type = ('Unknown', '_')
        self.mock_get_data_type_from_file.return_value = unknown_data_type

        self.assertIsNone(detect_data_type(self.widget_stub, [self.dir1.name]))


class TestGetDataTypeFromFile(TestCase):
    """Test suite for get_data_type_from_file"""
    def test_rt130_data(self):
        """
        Test basic functionality of get_data_type_from_file - given file
        contains RT130 data.
        """
        rt130_file = Path(rt130_dir).joinpath(
            '92EB/0/000000000_00000000')
        expected_data_type = ('RT130', '_')
        self.assertTupleEqual(
            get_data_type_from_file(rt130_file, get_signature_channels()),
            expected_data_type
        )

    def test_cannot_detect_data_type(self):
        """
        Test basic functionality of get_data_type_from_file - cannot detect
        data type contained in given file.
        """
        test_file = NamedTemporaryFile()
        self.assertIsNone(
            get_data_type_from_file(test_file.name, get_signature_channels()))

    def test_mseed_data(self):
        """
        Test basic functionality of get_data_type_from_file - given file
        contains MSeed data.
        """
        q330_file = q330_dir.joinpath('AX08.XA..VKI.2021.186')
        centaur_file = centaur_dir.joinpath(
            'XX.3734.SOH.centaur-3_3734..20180817_000000.miniseed.miniseed')
        pegasus_file = pegasus_dir.joinpath(
            '2020/XX/KC01/VE1.D/XX.KC01..VE1.D.2020.129')
        q330_data_type = ('Q330', 'VKI')
        centaur_data_type = ('Centaur', 'GEL')
        pegasus_data_type = ('Pegasus', 'VE1')

        sig_chan = get_signature_channels()

        self.assertTupleEqual(get_data_type_from_file(q330_file, sig_chan),
                              q330_data_type)
        self.assertTupleEqual(get_data_type_from_file(centaur_file, sig_chan),
                              centaur_data_type)
        self.assertTupleEqual(get_data_type_from_file(pegasus_file, sig_chan),
                              pegasus_data_type)

    def test_file_does_not_exist(self):
        """
        Test basic functionality of get_data_type_from_file - given file does
        not exist.
        """
        empty_name_file = ''
        non_existent_file = 'non_existent_dir'
        with self.assertRaises(FileNotFoundError):
            get_data_type_from_file(empty_name_file, get_signature_channels())
        with self.assertRaises(FileNotFoundError):
            get_data_type_from_file(non_existent_file,
                                    get_signature_channels())