xref: /OK3568_Linux_fs/yocto/poky/scripts/lib/resulttool/store.py (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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