mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 1 | #!/usr/bin/python -u |
mbligh | c251454 | 2008-02-19 15:54:26 +0000 | [diff] [blame] | 2 | |
mbligh | b33e53e | 2008-06-17 19:41:26 +0000 | [diff] [blame] | 3 | import os, sys, optparse, fcntl, errno, traceback, socket |
mbligh | bb7b891 | 2006-10-08 03:59:02 +0000 | [diff] [blame] | 4 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 5 | import common |
jadmanski | 6e8bf75 | 2008-05-14 00:17:48 +0000 | [diff] [blame] | 6 | from autotest_lib.client.common_lib import mail, utils |
| 7 | from autotest_lib.tko import db as tko_db, utils as tko_utils, status_lib |
mbligh | 74fc046 | 2007-11-05 20:24:17 +0000 | [diff] [blame] | 8 | |
| 9 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 10 | def parse_args(): |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 11 | # build up our options parser and parse sys.argv |
| 12 | parser = optparse.OptionParser() |
| 13 | parser.add_option("-m", help="Send mail for FAILED tests", |
| 14 | dest="mailit", action="store_true") |
| 15 | parser.add_option("-r", help="Reparse the results of a job", |
| 16 | dest="reparse", action="store_true") |
| 17 | parser.add_option("-o", help="Parse a single results directory", |
| 18 | dest="singledir", action="store_true") |
| 19 | parser.add_option("-l", help=("Levels of subdirectories to include " |
| 20 | "in the job name"), |
| 21 | type="int", dest="level", default=1) |
| 22 | parser.add_option("-n", help="No blocking on an existing parse", |
| 23 | dest="noblock", action="store_true") |
| 24 | parser.add_option("-s", help="Database server hostname", |
| 25 | dest="db_host", action="store") |
| 26 | parser.add_option("-u", help="Database username", dest="db_user", |
| 27 | action="store") |
| 28 | parser.add_option("-p", help="Database password", dest="db_pass", |
| 29 | action="store") |
| 30 | parser.add_option("-d", help="Database name", dest="db_name", |
| 31 | action="store") |
| 32 | options, args = parser.parse_args() |
mbligh | 74fc046 | 2007-11-05 20:24:17 +0000 | [diff] [blame] | 33 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 34 | # we need a results directory |
| 35 | if len(args) == 0: |
| 36 | tko_utils.dprint("ERROR: at least one results directory must " |
| 37 | "be provided") |
| 38 | parser.print_help() |
| 39 | sys.exit(1) |
mbligh | 74fc046 | 2007-11-05 20:24:17 +0000 | [diff] [blame] | 40 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 41 | # pass the options back |
| 42 | return options, args |
mbligh | 74fc046 | 2007-11-05 20:24:17 +0000 | [diff] [blame] | 43 | |
| 44 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 45 | def format_failure_message(jobname, kernel, testname, status, reason): |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 46 | format_string = "%-12s %-20s %-12s %-10s %s" |
| 47 | return format_string % (jobname, kernel, testname, status, reason) |
mbligh | b85e6b0 | 2006-10-08 17:20:56 +0000 | [diff] [blame] | 48 | |
mbligh | bb7b891 | 2006-10-08 03:59:02 +0000 | [diff] [blame] | 49 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 50 | def mailfailure(jobname, job, message): |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 51 | message_lines = [""] |
| 52 | message_lines.append("The following tests FAILED for this job") |
| 53 | message_lines.append("http://%s/results/%s" % |
| 54 | (socket.gethostname(), jobname)) |
| 55 | message_lines.append("") |
| 56 | message_lines.append(format_failure_message("Job name", "Kernel", |
| 57 | "Test name", "FAIL/WARN", |
| 58 | "Failure reason")) |
| 59 | message_lines.append(format_failure_message("=" * 8, "=" * 6, "=" * 8, |
| 60 | "=" * 8, "=" * 14)) |
| 61 | message_header = "\n".join(message_lines) |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 62 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 63 | subject = "AUTOTEST: FAILED tests from job %s" % jobname |
| 64 | mail.send("", job.user, "", subject, message_header + message) |
mbligh | 006f230 | 2007-09-13 20:46:46 +0000 | [diff] [blame] | 65 | |
| 66 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 67 | def parse_one(db, jobname, path, reparse, mail_on_failure): |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 68 | """ |
| 69 | Parse a single job. Optionally send email on failure. |
| 70 | """ |
| 71 | tko_utils.dprint("\nScanning %s (%s)" % (jobname, path)) |
| 72 | if reparse and db.find_job(jobname): |
| 73 | tko_utils.dprint("! Deleting old copy of job results to " |
| 74 | "reparse it") |
| 75 | db.delete_job(jobname) |
| 76 | if db.find_job(jobname): |
| 77 | tko_utils.dprint("! Job is already parsed, done") |
| 78 | return |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 79 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 80 | # look up the status version |
| 81 | try: |
| 82 | job_keyval = utils.read_keyval(path) |
| 83 | except IOError, e: |
| 84 | if e.errno == errno.ENOENT: |
| 85 | status_version = 0 |
| 86 | else: |
| 87 | raise |
| 88 | else: |
| 89 | status_version = job_keyval.get("status_version", 0) |
jadmanski | 6e8bf75 | 2008-05-14 00:17:48 +0000 | [diff] [blame] | 90 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 91 | # parse out the job |
| 92 | parser = status_lib.parser(status_version) |
| 93 | job = parser.make_job(path) |
| 94 | status_log = os.path.join(path, "status.log") |
| 95 | if not os.path.exists(status_log): |
| 96 | status_log = os.path.join(path, "status") |
| 97 | if not os.path.exists(status_log): |
| 98 | tko_utils.dprint("! Unable to parse job, no status file") |
| 99 | return |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 100 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 101 | # parse the status logs |
| 102 | tko_utils.dprint("+ Parsing dir=%s, jobname=%s" % (path, jobname)) |
| 103 | status_lines = open(status_log).readlines() |
| 104 | parser.start(job) |
| 105 | tests = parser.end(status_lines) |
| 106 | job.tests = tests |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 107 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 108 | # check for failures |
| 109 | message_lines = [""] |
| 110 | for test in job.tests: |
| 111 | if not test.subdir: |
| 112 | continue |
| 113 | tko_utils.dprint("* testname, status, reason: %s %s %s" |
| 114 | % (test.subdir, test.status, test.reason)) |
| 115 | if test.status in ("FAIL", "WARN"): |
| 116 | message_lines.append(format_failure_message( |
| 117 | jobname, test.kernel.base, test.subdir, |
| 118 | test.status, test.reason)) |
| 119 | message = "\n".join(message_lines) |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 120 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 121 | # send out a email report of failure |
| 122 | if len(message) > 2 and mail_on_failure: |
| 123 | tko_utils.dprint("Sending email report of failure on %s to %s" |
| 124 | % (jobname, job.user)) |
| 125 | mailfailure(jobname, job, message) |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 126 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 127 | # write the job into the database |
| 128 | db.insert_job(jobname, job) |
| 129 | db.commit() |
mbligh | 26b992b | 2008-02-19 15:46:21 +0000 | [diff] [blame] | 130 | |
| 131 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 132 | def parse_path(db, path, level, reparse, mail_on_failure): |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 133 | machine_list = os.path.join(path, ".machines") |
| 134 | if os.path.exists(machine_list): |
| 135 | # multi-machine job |
| 136 | for m in file(machine_list): |
| 137 | machine = m.rstrip() |
| 138 | if not machine: |
| 139 | continue |
| 140 | jobpath = os.path.join(path, machine) |
| 141 | jobname = "%s/%s" % (os.path.basename(path), machine) |
| 142 | try: |
| 143 | db.run_with_retry(parse_one, db, jobname, |
jadmanski | 63bce5f | 2008-06-20 14:10:55 +0000 | [diff] [blame] | 144 | jobpath, reparse, |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 145 | mail_on_failure) |
| 146 | except Exception: |
| 147 | traceback.print_exc() |
| 148 | continue |
| 149 | else: |
| 150 | # single machine job |
| 151 | job_elements = path.split("/")[-level:] |
| 152 | jobname = "/".join(job_elements) |
| 153 | try: |
| 154 | db.run_with_retry(parse_one, db, jobname, path, |
| 155 | reparse, mail_on_failure) |
| 156 | except Exception: |
| 157 | traceback.print_exc() |
mbligh | bb7b891 | 2006-10-08 03:59:02 +0000 | [diff] [blame] | 158 | |
| 159 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 160 | def main(): |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 161 | options, args = parse_args() |
| 162 | results_dir = os.path.abspath(args[0]) |
| 163 | assert os.path.exists(results_dir) |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 164 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 165 | # build up the list of job dirs to parse |
| 166 | if options.singledir: |
| 167 | jobs_list = [results_dir] |
| 168 | else: |
| 169 | jobs_list = [os.path.join(results_dir, subdir) |
| 170 | for subdir in os.listdir(results_dir)] |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 171 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 172 | # build up the database |
| 173 | db = tko_db.db(autocommit=False, host=options.db_host, |
| 174 | user=options.db_user, password=options.db_pass, |
| 175 | database=options.db_name) |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 176 | |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 177 | # parse all the jobs |
| 178 | for path in jobs_list: |
| 179 | lockfile = open(os.path.join(path, ".parse.lock"), "w") |
| 180 | flags = fcntl.LOCK_EX |
| 181 | if options.noblock: |
| 182 | flags != fcntl.LOCK_NB |
| 183 | try: |
| 184 | fcntl.flock(lockfile, flags) |
| 185 | except IOError, e: |
| 186 | # was this because the lock is unavailable? |
| 187 | if e.errno == errno.EWOULDBLOCK: |
| 188 | lockfile.close() |
| 189 | continue |
| 190 | else: |
| 191 | raise # something unexpected happened |
| 192 | try: |
| 193 | parse_path(db, path, options.level, options.reparse, |
| 194 | options.mailit) |
| 195 | finally: |
| 196 | fcntl.flock(lockfile, fcntl.LOCK_UN) |
| 197 | lockfile.close() |
mbligh | 71d340d | 2008-03-05 15:51:16 +0000 | [diff] [blame] | 198 | |
mbligh | 532cb27 | 2007-11-26 18:54:20 +0000 | [diff] [blame] | 199 | |
mbligh | 96cf051 | 2008-04-17 15:25:38 +0000 | [diff] [blame] | 200 | if __name__ == "__main__": |
jadmanski | 0afbb63 | 2008-06-06 21:10:57 +0000 | [diff] [blame] | 201 | main() |