# Copyright 2018 - The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Tests for list."""

import unittest

from unittest import mock

from acloud import errors
from acloud.internal.lib import cvd_runtime_config
from acloud.internal.lib import driver_test_lib
from acloud.internal.lib import utils
from acloud.list import list as list_instance
from acloud.list import instance


class InstanceObject:
    """Mock to store data of instance."""

    def __init__(self, name):
        self.name = name


class ListTest(driver_test_lib.BaseDriverTest):
    """Test list."""

    def testGetInstancesFromInstanceNames(self):
        """test get instances from instance names."""
        cfg = mock.MagicMock()
        instance_names = ["alive_instance1", "alive_local_instance"]

        alive_instance1 = InstanceObject("alive_instance1")
        alive_instance2 = InstanceObject("alive_instance2")
        alive_local_instance = InstanceObject("alive_local_instance")

        self.Patch(list_instance, "GetLocalInstancesByNames",
                   return_value=[alive_local_instance])
        self.Patch(list_instance, "GetRemoteInstances",
                   return_value=[alive_instance1, alive_instance2])
        instances_list = list_instance.GetInstancesFromInstanceNames(cfg, instance_names)
        instances_name_in_list = [instance_object.name for instance_object in instances_list]
        self.assertEqual(instances_name_in_list.sort(), instance_names.sort())

        instance_names = ["alive_instance1", "alive_local_instance", "alive_local_instance"]
        instances_list = list_instance.GetInstancesFromInstanceNames(cfg, instance_names)
        instances_name_in_list = [instance_object.name for instance_object in instances_list]
        self.assertEqual(instances_name_in_list.sort(), instance_names.sort())

        # test get instance from instance name error with invalid input.
        instance_names = ["miss2_local_instance", "alive_instance1"]
        miss_instance_names = ["miss2_local_instance"]
        self.assertRaisesRegex(
            errors.NoInstancesFound,
            "Did not find the following instances: %s" % ' '.join(miss_instance_names),
            list_instance.GetInstancesFromInstanceNames,
            cfg=cfg,
            instance_names=instance_names)

    def testChooseOneRemoteInstance(self):
        """test choose one remote instance from instance names."""
        cfg = mock.MagicMock()

        # Test only one instance case
        instance_names = ["cf_instance1"]
        self.Patch(list_instance, "GetCFRemoteInstances", return_value=instance_names)
        expected_instance = "cf_instance1"
        self.assertEqual(list_instance.ChooseOneRemoteInstance(cfg), expected_instance)

        # Test no instance case
        self.Patch(list_instance, "GetCFRemoteInstances", return_value=[])
        with self.assertRaises(errors.NoInstancesFound):
            list_instance.ChooseOneRemoteInstance(cfg)

        # Test two instances case.
        instance_names = ["cf_instance1", "cf_instance2"]
        choose_instance = ["cf_instance2"]
        self.Patch(list_instance, "GetCFRemoteInstances", return_value=instance_names)
        self.Patch(utils, "GetAnswerFromList", return_value=choose_instance)
        expected_instance = "cf_instance2"
        self.assertEqual(list_instance.ChooseOneRemoteInstance(cfg), expected_instance)

    def testGetLocalInstancesByNames(self):
        """test GetLocalInstancesByNames."""
        self.Patch(
            instance, "GetLocalInstanceIdByName",
            side_effect=lambda name: 1 if name == "local-instance-1" else None)
        self.Patch(instance, "GetLocalInstanceConfig",
                   return_value="path1")
        self.Patch(instance, "GetDefaultCuttlefishConfig",
                   return_value="path2")
        mock_cf_ins = mock.Mock()
        mock_cf_ins.name = "local-instance-1"
        mock_get_cf = self.Patch(list_instance,
                                 "_GetLocalCuttlefishInstances",
                                 return_value=[mock_cf_ins])
        mock_gf_ins = mock.Mock()
        mock_gf_ins.name = "local-goldfish-instance-1"
        self.Patch(instance.LocalGoldfishInstance, "GetExistingInstances",
                   return_value=[mock_gf_ins])

        ins_list = list_instance.GetLocalInstancesByNames([
            mock_cf_ins.name, "local-instance-6", mock_gf_ins.name])
        self.assertEqual([mock_cf_ins, mock_gf_ins], ins_list)
        mock_get_cf.assert_called_with([(1, "path1"), (1, "path2")])

    # pylint: disable=attribute-defined-outside-init
    def testFilterInstancesByAdbPort(self):
        """test FilterInstancesByAdbPort."""
        alive_instance1 = InstanceObject("alive_instance1")
        alive_instance1.adb_port = 1111
        alive_instance1.fullname = "device serial: 127.0.0.1:1111 alive_instance1"
        expected_instance = [alive_instance1]
        # Test to find instance by adb port number.
        self.assertEqual(
            expected_instance,
            list_instance.FilterInstancesByAdbPort(expected_instance, 1111))
        # Test for instance can't be found by adb port number.
        with self.assertRaises(errors.NoInstancesFound):
            list_instance.FilterInstancesByAdbPort(expected_instance, 2222)

    # pylint: disable=protected-access
    def testGetLocalCuttlefishInstances(self):
        """test _GetLocalCuttlefishInstances."""
        # Test getting two instance case
        id_cfg_pairs = [(1, "fake_path1"), (2, "fake_path2")]
        mock_isfile = self.Patch(list_instance.os.path, "isfile",
                                 return_value=True)

        mock_lock = mock.Mock()
        mock_lock.Lock.return_value = True
        self.Patch(instance, "GetLocalInstanceLock", return_value=mock_lock)

        local_ins = mock.MagicMock()
        local_ins.CvdStatus.return_value = True
        self.Patch(instance, "LocalInstance", return_value=local_ins)

        ins_list = list_instance._GetLocalCuttlefishInstances(id_cfg_pairs)
        self.assertEqual(2, len(ins_list))
        mock_isfile.assert_called()
        local_ins.CvdStatus.assert_called()
        self.assertEqual(2, mock_lock.Lock.call_count)
        self.assertEqual(2, mock_lock.Unlock.call_count)

        local_ins.CvdStatus.reset_mock()
        mock_lock.Lock.reset_mock()
        mock_lock.Lock.return_value = False
        mock_lock.Unlock.reset_mock()
        ins_list = list_instance._GetLocalCuttlefishInstances(id_cfg_pairs)
        self.assertEqual(0, len(ins_list))
        local_ins.CvdStatus.assert_not_called()
        self.assertEqual(2, mock_lock.Lock.call_count)
        mock_lock.Unlock.assert_not_called()

        mock_lock.Lock.reset_mock()
        mock_lock.Lock.return_value = True
        local_ins.CvdStatus.return_value = False
        ins_list = list_instance._GetLocalCuttlefishInstances(id_cfg_pairs)
        self.assertEqual(0, len(ins_list))
        self.assertEqual(2, mock_lock.Lock.call_count)
        self.assertEqual(2, mock_lock.Unlock.call_count)

    # pylint: disable=no-member
    def testPrintInstancesDetails(self):
        """test PrintInstancesDetails."""
        # Test instance Summary should be called if verbose
        self.Patch(instance.Instance, "Summary")
        cf_config = mock.MagicMock(
            x_res=728,
            y_res=728,
            dpi=240,
            instance_dir="fake_dir",
            adb_ip_port="127.0.0.1:6520"
        )
        self.Patch(cvd_runtime_config, "CvdRuntimeConfig",
                   return_value=cf_config)

        ins = instance.LocalInstance("fake_cf_path")
        list_instance.PrintInstancesDetails([ins], verbose=True)
        instance.Instance.Summary.assert_called_once()

        # Test Summary shouldn't be called if not verbose
        self.Patch(instance.Instance, "Summary")
        list_instance.PrintInstancesDetails([ins], verbose=False)
        instance.Instance.Summary.assert_not_called()

        # Test Summary shouldn't be called if no instance found.
        list_instance.PrintInstancesDetails([], verbose=True)
        instance.Instance.Summary.assert_not_called()


if __name__ == "__main__":
    unittest.main()