#!/usr/bin/env vpython3
# Copyright (c) 2020 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.

import os
import sys
import unittest
from unittest import mock

sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))

import gerrit_util
import owners_client

alice = 'alice@example.com'
bob = 'bob@example.com'
chris = 'chris@example.com'
dave = 'dave@example.com'
emily = 'emily@example.com'


class GerritClientTest(unittest.TestCase):
    def setUp(self):
        self.client = owners_client.GerritClient('host', 'project', 'branch')
        self.addCleanup(mock.patch.stopall)

    def testListOwners(self):
        mock.patch('gerrit_util.GetOwnersForFile',
                   return_value={
                       "code_owners": [{
                           "account": {
                               "email": 'approver@example.com'
                           }
                       }, {
                           "account": {
                               "email": 'reviewer@example.com'
                           },
                       }, {
                           "account": {
                               "email": 'missing@example.com'
                           },
                       }, {
                           "account": {},
                       }]
                   }).start()
        self.assertEqual([
            'approver@example.com', 'reviewer@example.com',
            'missing@example.com'
        ], self.client.ListOwners(os.path.join('bar', 'everyone', 'foo.txt')))

        # Result should be cached.
        self.assertEqual([
            'approver@example.com', 'reviewer@example.com',
            'missing@example.com'
        ], self.client.ListOwners(os.path.join('bar', 'everyone', 'foo.txt')))
        # Always use slashes as separators.
        gerrit_util.GetOwnersForFile.assert_called_once_with(
            'host',
            'project',
            'branch',
            'bar/everyone/foo.txt',
            resolve_all_users=False,
            highest_score_only=False,
            seed=mock.ANY)

    def testListOwnersOwnedByAll(self):
        mock.patch('gerrit_util.GetOwnersForFile',
                   side_effect=[
                       {
                           "code_owners": [
                               {
                                   "account": {
                                       "email": 'foo@example.com'
                                   },
                               },
                           ],
                           "owned_by_all_users":
                           True,
                       },
                       {
                           "code_owners": [
                               {
                                   "account": {
                                       "email": 'bar@example.com'
                                   },
                               },
                           ],
                           "owned_by_all_users":
                           False,
                       },
                   ]).start()
        self.assertEqual(['foo@example.com', self.client.EVERYONE],
                         self.client.ListOwners('foo.txt'))
        self.assertEqual(['bar@example.com'], self.client.ListOwners('bar.txt'))


class TestClient(owners_client.OwnersClient):
    def __init__(self, owners_by_path):
        super(TestClient, self).__init__()
        self.owners_by_path = owners_by_path

    def ListOwners(self, path):
        return self.owners_by_path[path]


class OwnersClientTest(unittest.TestCase):
    def setUp(self):
        self.owners = {}
        self.client = TestClient(self.owners)

    def testGetFilesApprovalStatus(self):
        self.client.owners_by_path = {
            'approved': ['approver@example.com'],
            'pending': ['reviewer@example.com'],
            'insufficient': ['insufficient@example.com'],
            'everyone': [owners_client.OwnersClient.EVERYONE],
        }
        self.assertEqual(
            self.client.GetFilesApprovalStatus(
                ['approved', 'pending', 'insufficient'],
                ['approver@example.com'], ['reviewer@example.com']), {
                    'approved': owners_client.OwnersClient.APPROVED,
                    'pending': owners_client.OwnersClient.PENDING,
                    'insufficient':
                    owners_client.OwnersClient.INSUFFICIENT_REVIEWERS,
                })
        self.assertEqual(
            self.client.GetFilesApprovalStatus(['everyone'],
                                               ['anyone@example.com'], []),
            {'everyone': owners_client.OwnersClient.APPROVED})
        self.assertEqual(
            self.client.GetFilesApprovalStatus(['everyone'], [],
                                               ['anyone@example.com']),
            {'everyone': owners_client.OwnersClient.PENDING})
        self.assertEqual(
            self.client.GetFilesApprovalStatus(['everyone'], [], []),
            {'everyone': owners_client.OwnersClient.INSUFFICIENT_REVIEWERS})

    def testScoreOwners(self):
        self.client.owners_by_path = {'a': [alice, bob, chris]}
        self.assertEqual(
            self.client.ScoreOwners(self.client.owners_by_path.keys()),
            [alice, bob, chris])

        self.client.owners_by_path = {
            'a': [alice, bob],
            'b': [bob],
            'c': [bob, chris]
        }
        self.assertEqual(
            self.client.ScoreOwners(self.client.owners_by_path.keys()),
            [alice, bob, chris])

        self.client.owners_by_path = {
            'a': [alice, bob],
            'b': [bob],
            'c': [bob, chris]
        }
        self.assertEqual(
            self.client.ScoreOwners(self.client.owners_by_path.keys(),
                                    exclude=[chris]),
            [alice, bob],
        )

        self.client.owners_by_path = {
            'a': [alice, bob, chris, dave],
            'b': [chris, bob, dave],
            'c': [chris, dave],
            'd': [alice, chris, dave]
        }
        self.assertEqual(
            self.client.ScoreOwners(self.client.owners_by_path.keys()),
            [alice, chris, bob, dave])

    def assertSuggestsOwners(self, owners_by_path, exclude=None):
        self.client.owners_by_path = owners_by_path
        suggested = self.client.SuggestOwners(owners_by_path.keys(),
                                              exclude=exclude)

        # Owners should appear only once
        self.assertEqual(len(suggested), len(set(suggested)))

        # All paths should be covered.
        suggested = set(suggested)
        for owners in owners_by_path.values():
            self.assertTrue(suggested & set(owners))

        # No excluded owners should be present.
        if exclude:
            for owner in suggested:
                self.assertNotIn(owner, exclude)

    def testSuggestOwners(self):
        self.assertSuggestsOwners({})
        self.assertSuggestsOwners({'a': [alice]})
        self.assertSuggestsOwners({'abcd': [alice, bob, chris, dave]})
        self.assertSuggestsOwners({'abcd': [alice, bob, chris, dave]},
                                  exclude=[alice, bob])
        self.assertSuggestsOwners({
            'ae': [alice, emily],
            'be': [bob, emily],
            'ce': [chris, emily],
            'de': [dave, emily]
        })
        self.assertSuggestsOwners({
            'ad': [alice, dave],
            'cad': [chris, alice, dave],
            'ead': [emily, alice, dave],
            'bd': [bob, dave]
        })
        self.assertSuggestsOwners({
            'a': [alice],
            'b': [bob],
            'c': [chris],
            'ad': [alice, dave]
        })
        self.assertSuggestsOwners({
            'abc': [alice, bob, chris],
            'acb': [alice, chris, bob],
            'bac': [bob, alice, chris],
            'bca': [bob, chris, alice],
            'cab': [chris, alice, bob],
            'cba': [chris, bob, alice]
        })

        # Check that we can handle a large amount of files with unrelated
        # owners.
        self.assertSuggestsOwners({str(x): [str(x)] for x in range(100)})

    def testBatchListOwners(self):
        self.client.owners_by_path = {
            'bar/everyone/foo.txt': [alice, bob],
            'bar/everyone/bar.txt': [bob],
            'bar/foo/': [bob, chris]
        }

        self.assertEqual(
            {
                'bar/everyone/foo.txt': [alice, bob],
                'bar/everyone/bar.txt': [bob],
                'bar/foo/': [bob, chris]
            },
            self.client.BatchListOwners(
                ['bar/everyone/foo.txt', 'bar/everyone/bar.txt', 'bar/foo/']))


class GetCodeOwnersClientTest(unittest.TestCase):
    def setUp(self):
        mock.patch('gerrit_util.IsCodeOwnersEnabledOnHost').start()
        self.addCleanup(mock.patch.stopall)

    def testGetCodeOwnersClient_CodeOwnersEnabled(self):
        gerrit_util.IsCodeOwnersEnabledOnHost.return_value = True
        self.assertIsInstance(
            owners_client.GetCodeOwnersClient('host', 'project', 'branch'),
            owners_client.GerritClient)

    def testGetCodeOwnersClient_CodeOwnersDisabled(self):
        gerrit_util.IsCodeOwnersEnabledOnHost.return_value = False
        with self.assertRaises(Exception):
            owners_client.GetCodeOwnersClient('', '', '')


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