1*4882a593Smuzhiyun# resulttool - store test results 2*4882a593Smuzhiyun# 3*4882a593Smuzhiyun# Copyright (c) 2019, Intel Corporation. 4*4882a593Smuzhiyun# Copyright (c) 2019, Linux Foundation 5*4882a593Smuzhiyun# 6*4882a593Smuzhiyun# SPDX-License-Identifier: GPL-2.0-only 7*4882a593Smuzhiyun# 8*4882a593Smuzhiyun 9*4882a593Smuzhiyunimport tempfile 10*4882a593Smuzhiyunimport os 11*4882a593Smuzhiyunimport subprocess 12*4882a593Smuzhiyunimport json 13*4882a593Smuzhiyunimport shutil 14*4882a593Smuzhiyunimport scriptpath 15*4882a593Smuzhiyunscriptpath.add_bitbake_lib_path() 16*4882a593Smuzhiyunscriptpath.add_oe_lib_path() 17*4882a593Smuzhiyunimport resulttool.resultutils as resultutils 18*4882a593Smuzhiyunimport oeqa.utils.gitarchive as gitarchive 19*4882a593Smuzhiyun 20*4882a593Smuzhiyun 21*4882a593Smuzhiyundef store(args, logger): 22*4882a593Smuzhiyun tempdir = tempfile.mkdtemp(prefix='testresults.') 23*4882a593Smuzhiyun try: 24*4882a593Smuzhiyun configvars = resultutils.extra_configvars.copy() 25*4882a593Smuzhiyun if args.executed_by: 26*4882a593Smuzhiyun configvars['EXECUTED_BY'] = args.executed_by 27*4882a593Smuzhiyun if args.extra_test_env: 28*4882a593Smuzhiyun configvars['EXTRA_TEST_ENV'] = args.extra_test_env 29*4882a593Smuzhiyun results = {} 30*4882a593Smuzhiyun logger.info('Reading files from %s' % args.source) 31*4882a593Smuzhiyun if resultutils.is_url(args.source) or os.path.isfile(args.source): 32*4882a593Smuzhiyun resultutils.append_resultsdata(results, args.source, configvars=configvars) 33*4882a593Smuzhiyun else: 34*4882a593Smuzhiyun for root, dirs, files in os.walk(args.source): 35*4882a593Smuzhiyun for name in files: 36*4882a593Smuzhiyun f = os.path.join(root, name) 37*4882a593Smuzhiyun if name == "testresults.json": 38*4882a593Smuzhiyun resultutils.append_resultsdata(results, f, configvars=configvars) 39*4882a593Smuzhiyun elif args.all: 40*4882a593Smuzhiyun dst = f.replace(args.source, tempdir + "/") 41*4882a593Smuzhiyun os.makedirs(os.path.dirname(dst), exist_ok=True) 42*4882a593Smuzhiyun shutil.copyfile(f, dst) 43*4882a593Smuzhiyun 44*4882a593Smuzhiyun revisions = {} 45*4882a593Smuzhiyun 46*4882a593Smuzhiyun if not results and not args.all: 47*4882a593Smuzhiyun if args.allow_empty: 48*4882a593Smuzhiyun logger.info("No results found to store") 49*4882a593Smuzhiyun return 0 50*4882a593Smuzhiyun logger.error("No results found to store") 51*4882a593Smuzhiyun return 1 52*4882a593Smuzhiyun 53*4882a593Smuzhiyun # Find the branch/commit/commit_count and ensure they all match 54*4882a593Smuzhiyun for suite in results: 55*4882a593Smuzhiyun for result in results[suite]: 56*4882a593Smuzhiyun config = results[suite][result]['configuration']['LAYERS']['meta'] 57*4882a593Smuzhiyun revision = (config['commit'], config['branch'], str(config['commit_count'])) 58*4882a593Smuzhiyun if revision not in revisions: 59*4882a593Smuzhiyun revisions[revision] = {} 60*4882a593Smuzhiyun if suite not in revisions[revision]: 61*4882a593Smuzhiyun revisions[revision][suite] = {} 62*4882a593Smuzhiyun revisions[revision][suite][result] = results[suite][result] 63*4882a593Smuzhiyun 64*4882a593Smuzhiyun logger.info("Found %d revisions to store" % len(revisions)) 65*4882a593Smuzhiyun 66*4882a593Smuzhiyun for r in revisions: 67*4882a593Smuzhiyun results = revisions[r] 68*4882a593Smuzhiyun keywords = {'commit': r[0], 'branch': r[1], "commit_count": r[2]} 69*4882a593Smuzhiyun subprocess.check_call(["find", tempdir, "!", "-path", "./.git/*", "-delete"]) 70*4882a593Smuzhiyun resultutils.save_resultsdata(results, tempdir, ptestlogs=True) 71*4882a593Smuzhiyun 72*4882a593Smuzhiyun logger.info('Storing test result into git repository %s' % args.git_dir) 73*4882a593Smuzhiyun 74*4882a593Smuzhiyun gitarchive.gitarchive(tempdir, args.git_dir, False, False, 75*4882a593Smuzhiyun "Results of {branch}:{commit}", "branch: {branch}\ncommit: {commit}", "{branch}", 76*4882a593Smuzhiyun False, "{branch}/{commit_count}-g{commit}/{tag_number}", 77*4882a593Smuzhiyun 'Test run #{tag_number} of {branch}:{commit}', '', 78*4882a593Smuzhiyun [], [], False, keywords, logger) 79*4882a593Smuzhiyun 80*4882a593Smuzhiyun finally: 81*4882a593Smuzhiyun subprocess.check_call(["rm", "-rf", tempdir]) 82*4882a593Smuzhiyun 83*4882a593Smuzhiyun return 0 84*4882a593Smuzhiyun 85*4882a593Smuzhiyundef register_commands(subparsers): 86*4882a593Smuzhiyun """Register subcommands from this plugin""" 87*4882a593Smuzhiyun parser_build = subparsers.add_parser('store', help='store test results into a git repository', 88*4882a593Smuzhiyun description='takes a results file or directory of results files and stores ' 89*4882a593Smuzhiyun 'them into the destination git repository, splitting out the results ' 90*4882a593Smuzhiyun 'files as configured', 91*4882a593Smuzhiyun group='setup') 92*4882a593Smuzhiyun parser_build.set_defaults(func=store) 93*4882a593Smuzhiyun parser_build.add_argument('source', 94*4882a593Smuzhiyun help='source file/directory/URL that contain the test result files to be stored') 95*4882a593Smuzhiyun parser_build.add_argument('git_dir', 96*4882a593Smuzhiyun help='the location of the git repository to store the results in') 97*4882a593Smuzhiyun parser_build.add_argument('-a', '--all', action='store_true', 98*4882a593Smuzhiyun help='include all files, not just testresults.json files') 99*4882a593Smuzhiyun parser_build.add_argument('-e', '--allow-empty', action='store_true', 100*4882a593Smuzhiyun help='don\'t error if no results to store are found') 101*4882a593Smuzhiyun parser_build.add_argument('-x', '--executed-by', default='', 102*4882a593Smuzhiyun help='add executed-by configuration to each result file') 103*4882a593Smuzhiyun parser_build.add_argument('-t', '--extra-test-env', default='', 104*4882a593Smuzhiyun help='add extra test environment data to each result file configuration') 105