#!/usr/bin/env python3
# -*- coding: utf-8 -*-

# Copyright (c) 2023 Huawei Device Co., Ltd.
# 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.

import os
import sys
import time
import json
import stat
import types
import argparse
import subprocess
import multiprocessing
import xml.etree.ElementTree as ET
from datetime import datetime


def build_target_name(xml_file_path):
    last_backslash_index = xml_file_path.rfind('/')
    last_dot_index = xml_file_path.rfind('.')
    if last_backslash_index != -1 and last_dot_index != -1 and last_backslash_index < last_dot_index:
        result = xml_file_path[last_backslash_index + 1:last_dot_index]
        return result
    else:
        return "Build Target Not Find"

def parse_xml(xml_file_path):
    """
    Parse the XML file of the execution output of the use case
    """
    test_module_name = build_target_name(xml_file_path)
    tree = ET.parse(xml_file_path)
    root = tree.getroot()
    tests = root.attrib.get("tests")
    failures = root.attrib.get("failures")
    disabled = root.attrib.get("disabled")
    common_info = types.SimpleNamespace()
    common_info.total = int(tests)
    common_info.disabled = int(disabled)
    failed_info = {
        "test_module_name": test_module_name,
        "total_count": tests,
        "failed_count": failures,
        "failed_testcase_name": []
    }
    passed_info = {
        "test_module_name": test_module_name,
        "total_count": tests,
        "passed_count": 0,
        "passed_testcase_name": []
    }
    passed_count = 0
    for testsuite in root.findall(".//testsuite"):
        testsuite_name = testsuite.attrib.get("name")
        testsuite_failures = testsuite.attrib.get("failures")
        for testcase in testsuite.findall(".//testcase"):
            testcase_status = testcase.attrib.get("status")
            testcase_result = testcase.attrib.get("result")
            if testcase_status != "run" or testcase_result != "completed":
                continue
            testcase_name = testcase.attrib.get("name")
            failure = testcase.find("failure")
            if failure is not None:
                failed_info["failed_testcase_name"].append("{}#{}".format(testsuite_name, testcase_name))
            else:
                passed_info["passed_testcase_name"].append("{}#{}".format(testsuite_name, testcase_name))
                passed_count = passed_count+1
    passed_info["passed_count"] = str(passed_count)
    return common_info, failed_info, passed_info


def run_command(test_binary_path: str, alter_cmds: list = None):
    """
    Run a gtest test binary.
    """
    default_cmds = []
    default_cmds.append(test_binary_path)
    default_cmds.append("--gtest_output=xml:{}.xml".format(test_binary_path))
    default_cmds.append("--gtest_print_time=0")
    default_cmds.append("--gtest_brief=1")
    if alter_cmds is not None:
        default_cmds.extend(alter_cmds)
    try:
        subprocess.run(default_cmds, timeout=30, stderr=subprocess.DEVNULL)
    except Exception:
        pass

def run_single_test(tests_path, test_suite_name):
    """
    Run a gtest test suite
    """
    test_suite_path = None
    for root, _, files in os.walk(tests_path):
        for file in files:
            if file.endswith(test_suite_name):
                test_suite_path =  os.path.join(root, test_suite_name)
    if test_suite_path is not None:
        run_command(test_suite_path)
    else:
        print("TestSuite {} did not compile successfully.".format(test_suite_name))


def run_tests_parallel(test_directory, process_number: int, path, output):
    """
    Run all gtest test binaries in parallel.
    """
    test_binaries = []
    search_dir = os.path.join(test_directory, path) if path else test_directory
    for root, _, files in os.walk(search_dir):
        for file in files:
            test_suite_path = os.path.join(root, file)
            name, ext = os.path.splitext(file)
            if ext == "":
                test_binaries.append(test_suite_path)
    for test_binary in test_binaries:
        xml_file_path = "{}.xml".format(test_binary)
        if os.path.exists(xml_file_path):
            os.remove(xml_file_path)
    start = time.time()
    with multiprocessing.Pool(processes=process_number) as pool:
        pool.map(run_command, iter(test_binaries))
    end = time.time()
    test_result = {
        "time_stamp": str(datetime.now()),
        "execute_time": 0,
        "total_execute_tests": 0,
        "failed_tests_count": 0,
        "disabled_tests_count": 0,
        "passed_tests_count": 0,
        "crashed_tests_count": 0,
        "unavailable": [],
        "failed": [],
        "passed": []
    }
    total_tests_count = 0
    failed_tests_count = 0
    passed_tests_count = 0
    disabled_tests_count = 0
    for test_binary in test_binaries:
        xml_file_path = "{}.xml".format(test_binary)
        if os.path.exists(xml_file_path):
            common_info, failed_info, passed_info= parse_xml(xml_file_path)
            total_tests_count += common_info.total
            disabled_tests_count += common_info.disabled
            failed_tests_count = failed_tests_count + int(failed_info.get('failed_count', '0'))
            passed_tests_count = passed_tests_count + int(passed_info.get('passed_count', '0'))
            if int(failed_info.get('failed_count', '0')):
                test_result['failed'].append(failed_info)
            test_result['passed'].append(passed_info)
        else:
            test_result["unavailable"].append(test_binary.split('/')[-1])
    test_result["execute_time"] = "{} seconds".format(round(end - start, 2))
    test_result['total_execute_tests'] = total_tests_count
    test_result['failed_tests_count'] = failed_tests_count
    test_result['passed_tests_count'] = passed_tests_count
    test_result['disabled_tests_count'] = disabled_tests_count
    test_result['crashed_tests_count'] = len(test_result["unavailable"])
    json_file_path = output if output else os.path.join(test_directory, "test_result.json")
    flags = os.O_CREAT | os.O_WRONLY | os.O_TRUNC
    mode = stat.S_IRUSR | stat.S_IWUSR
    with os.fdopen(os.open(json_file_path, flags, mode), 'w') as json_file:
        json.dump(test_result, json_file, indent=2)

    print("The test results have been generated, path is {}".format(os.path.abspath(json_file_path)))
    return failed_tests_count + len(test_result["unavailable"])


def get_tests_out_path(debug):
    """
    Obtain the output directory of test cases
    """
    base_path = os.path.dirname(os.path.realpath(__file__))
    for _ in range(6):
        base_path = os.path.dirname(base_path)
    json_config_path =  os.path.join(base_path,"out/ohos_config.json")
    if not os.path.exists(json_config_path):
        print("{} not exist, please build linux_unittest first.".format(json_config_path))
        code_path = os.path.join(base_path, "out/rk3568/clang_x64")
    else:
        with open(json_config_path, 'r', encoding='utf-8') as file:
            data = json.load(file)
            code_path = os.path.join(data["out_path"], "clang_x64")

    if debug:
        code_path = os.path.join(code_path, "exe.unstripped/clang_x64")
        os.environ["ASAN_SYMBOLIZER_PATH"] = os.path.join(base_path,
            "prebuilts/clang/ohos/linux-x86_64/llvm/bin/llvm-symbolizer")
    code_path = os.path.join(code_path, "tests/unittest/ace_engine")
    return code_path


def main():
    """
    Add unitest case execution parameters
    """
    parser = argparse.ArgumentParser()
    parser.add_argument("-t", "--target", nargs='+', type=str, default=None)
    parser.add_argument("-j", "--process", type=int, default=64)
    parser.add_argument("-p", "--path", type=str, default=None)
    parser.add_argument("-o", "--output", type=str, default=None)
    parser.add_argument("-d", "--debug", action='store_true')
    args = parser.parse_args()
    tests_out_path = get_tests_out_path(args.debug)
    targets = args.target
    if targets is not None:
        for target in targets:
            run_single_test(tests_out_path, target)
    else:
        return run_tests_parallel(tests_out_path, args.process, args.path, args.output)


if __name__ == "__main__":
    sys.exit(main())