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