2013-10-04 19:10:15 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
# Copyright 2013 Red Hat, Inc.
|
|
|
|
# All Rights Reserved.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
2013-10-10 01:31:32 +00:00
|
|
|
import argparse
|
|
|
|
import gzip
|
|
|
|
import os
|
|
|
|
import re
|
2013-10-04 19:10:15 +00:00
|
|
|
import sys
|
2014-06-09 22:37:19 +00:00
|
|
|
|
2017-02-10 09:05:26 +00:00
|
|
|
import six
|
|
|
|
import six.moves.urllib.request as urlreq
|
2013-10-10 01:31:32 +00:00
|
|
|
import yaml
|
|
|
|
|
2014-09-03 19:29:03 +00:00
|
|
|
# DEVSTACK_GATE_GRENADE is either unset if grenade is not running
|
|
|
|
# or a string describing what type of grenade run to perform.
|
|
|
|
is_grenade = os.environ.get('DEVSTACK_GATE_GRENADE') is not None
|
2013-12-30 17:04:17 +00:00
|
|
|
dump_all_errors = True
|
2013-11-27 15:53:54 +00:00
|
|
|
|
2014-10-31 12:56:36 +00:00
|
|
|
# As logs are made clean, remove from this set
|
2014-03-18 18:31:05 +00:00
|
|
|
allowed_dirty = set([
|
|
|
|
'c-api',
|
|
|
|
'ceilometer-acentral',
|
|
|
|
'ceilometer-acompute',
|
|
|
|
'ceilometer-alarm-evaluator',
|
|
|
|
'ceilometer-anotification',
|
|
|
|
'ceilometer-api',
|
2014-03-26 19:39:05 +00:00
|
|
|
'ceilometer-collector',
|
2014-03-18 18:31:05 +00:00
|
|
|
'c-vol',
|
|
|
|
'g-api',
|
|
|
|
'h-api',
|
|
|
|
'h-eng',
|
|
|
|
'ir-cond',
|
|
|
|
'n-api',
|
|
|
|
'n-cpu',
|
|
|
|
'n-net',
|
|
|
|
'q-agt',
|
|
|
|
'q-dhcp',
|
|
|
|
'q-lbaas',
|
|
|
|
'q-meta',
|
|
|
|
'q-metering',
|
|
|
|
'q-svc',
|
|
|
|
's-proxy'])
|
2014-02-27 20:23:35 +00:00
|
|
|
|
2013-11-27 15:53:54 +00:00
|
|
|
|
2013-10-10 01:31:32 +00:00
|
|
|
def process_files(file_specs, url_specs, whitelists):
|
2014-02-20 22:53:02 +00:00
|
|
|
regexp = re.compile(r"^.* (ERROR|CRITICAL|TRACE) .*\[.*\-.*\]")
|
2014-02-27 20:23:35 +00:00
|
|
|
logs_with_errors = []
|
2013-10-10 01:31:32 +00:00
|
|
|
for (name, filename) in file_specs:
|
|
|
|
whitelist = whitelists.get(name, [])
|
|
|
|
with open(filename) as content:
|
2018-01-29 03:36:54 +00:00
|
|
|
if scan_content(content, regexp, whitelist):
|
2014-02-27 20:23:35 +00:00
|
|
|
logs_with_errors.append(name)
|
2013-10-10 01:31:32 +00:00
|
|
|
for (name, url) in url_specs:
|
|
|
|
whitelist = whitelists.get(name, [])
|
2016-05-30 17:15:58 +00:00
|
|
|
req = urlreq.Request(url)
|
2013-10-10 01:31:32 +00:00
|
|
|
req.add_header('Accept-Encoding', 'gzip')
|
2016-05-30 17:15:58 +00:00
|
|
|
page = urlreq.urlopen(req)
|
2015-12-22 17:24:26 +00:00
|
|
|
buf = six.StringIO(page.read())
|
2013-10-10 01:31:32 +00:00
|
|
|
f = gzip.GzipFile(fileobj=buf)
|
2018-01-29 03:36:54 +00:00
|
|
|
if scan_content(f.read().splitlines(), regexp, whitelist):
|
2014-02-27 20:23:35 +00:00
|
|
|
logs_with_errors.append(name)
|
|
|
|
return logs_with_errors
|
2013-10-10 01:31:32 +00:00
|
|
|
|
|
|
|
|
2018-01-29 03:36:54 +00:00
|
|
|
def scan_content(content, regexp, whitelist):
|
2013-10-10 01:31:32 +00:00
|
|
|
had_errors = False
|
|
|
|
for line in content:
|
|
|
|
if not line.startswith("Stderr:") and regexp.match(line):
|
|
|
|
whitelisted = False
|
|
|
|
for w in whitelist:
|
|
|
|
pat = ".*%s.*%s.*" % (w['module'].replace('.', '\\.'),
|
|
|
|
w['message'])
|
|
|
|
if re.match(pat, line):
|
|
|
|
whitelisted = True
|
|
|
|
break
|
2013-11-27 15:53:54 +00:00
|
|
|
if not whitelisted or dump_all_errors:
|
|
|
|
if not whitelisted:
|
|
|
|
had_errors = True
|
2013-10-10 01:31:32 +00:00
|
|
|
return had_errors
|
|
|
|
|
|
|
|
|
|
|
|
def collect_url_logs(url):
|
2016-05-30 17:15:58 +00:00
|
|
|
page = urlreq.urlopen(url)
|
2013-10-10 01:31:32 +00:00
|
|
|
content = page.read()
|
2018-07-06 12:58:21 +00:00
|
|
|
logs = re.findall(r'(screen-[\w-]+\.txt\.gz)</a>', content)
|
2013-10-10 01:31:32 +00:00
|
|
|
return logs
|
|
|
|
|
|
|
|
|
|
|
|
def main(opts):
|
|
|
|
if opts.directory and opts.url or not (opts.directory or opts.url):
|
|
|
|
print("Must provide exactly one of -d or -u")
|
2016-01-25 08:45:21 +00:00
|
|
|
return 1
|
2013-10-10 01:31:32 +00:00
|
|
|
print("Checking logs...")
|
|
|
|
WHITELIST_FILE = os.path.join(
|
|
|
|
os.path.abspath(os.path.dirname(os.path.dirname(__file__))),
|
|
|
|
"etc", "whitelist.yaml")
|
|
|
|
|
|
|
|
file_matcher = re.compile(r".*screen-([\w-]+)\.log")
|
|
|
|
files = []
|
|
|
|
if opts.directory:
|
|
|
|
d = opts.directory
|
|
|
|
for f in os.listdir(d):
|
|
|
|
files.append(os.path.join(d, f))
|
|
|
|
files_to_process = []
|
|
|
|
for f in files:
|
|
|
|
m = file_matcher.match(f)
|
|
|
|
if m:
|
|
|
|
files_to_process.append((m.group(1), f))
|
|
|
|
|
|
|
|
url_matcher = re.compile(r".*screen-([\w-]+)\.txt\.gz")
|
|
|
|
urls = []
|
|
|
|
if opts.url:
|
|
|
|
for logfile in collect_url_logs(opts.url):
|
|
|
|
urls.append("%s/%s" % (opts.url, logfile))
|
|
|
|
urls_to_process = []
|
|
|
|
for u in urls:
|
|
|
|
m = url_matcher.match(u)
|
|
|
|
if m:
|
|
|
|
urls_to_process.append((m.group(1), u))
|
|
|
|
|
|
|
|
whitelists = {}
|
|
|
|
with open(WHITELIST_FILE) as stream:
|
|
|
|
loaded = yaml.safe_load(stream)
|
|
|
|
if loaded:
|
2017-06-11 15:20:43 +00:00
|
|
|
for (name, l) in six.iteritems(loaded):
|
2013-10-10 01:31:32 +00:00
|
|
|
for w in l:
|
|
|
|
assert 'module' in w, 'no module in %s' % name
|
|
|
|
assert 'message' in w, 'no message in %s' % name
|
|
|
|
whitelists = loaded
|
2014-02-27 20:23:35 +00:00
|
|
|
logs_with_errors = process_files(files_to_process, urls_to_process,
|
|
|
|
whitelists)
|
2014-09-03 18:53:16 +00:00
|
|
|
|
2014-02-27 20:23:35 +00:00
|
|
|
failed = False
|
2014-09-03 18:53:16 +00:00
|
|
|
if logs_with_errors:
|
|
|
|
log_files = set(logs_with_errors)
|
|
|
|
for log in log_files:
|
|
|
|
msg = '%s log file has errors' % log
|
|
|
|
if log not in allowed_dirty:
|
|
|
|
msg += ' and is not allowed to have them'
|
|
|
|
failed = True
|
|
|
|
print(msg)
|
|
|
|
print("\nPlease check the respective log files to see the errors")
|
2014-02-27 20:23:35 +00:00
|
|
|
if failed:
|
2014-09-03 18:53:16 +00:00
|
|
|
if is_grenade:
|
|
|
|
print("Currently not failing grenade runs with errors")
|
|
|
|
return 0
|
2014-02-27 20:23:35 +00:00
|
|
|
return 1
|
|
|
|
print("ok")
|
|
|
|
return 0
|
2013-10-10 01:31:32 +00:00
|
|
|
|
2018-07-06 12:58:21 +00:00
|
|
|
|
2013-10-10 01:31:32 +00:00
|
|
|
usage = """
|
|
|
|
Find non-white-listed log errors in log files from a devstack-gate run.
|
|
|
|
Log files will be searched for ERROR or CRITICAL messages. If any
|
|
|
|
error messages do not match any of the whitelist entries contained in
|
|
|
|
etc/whitelist.yaml, those messages will be printed to the console and
|
|
|
|
failure will be returned. A file directory containing logs or a url to the
|
|
|
|
log files of an OpenStack gate job can be provided.
|
|
|
|
|
|
|
|
The whitelist yaml looks like:
|
|
|
|
|
|
|
|
log-name:
|
|
|
|
- module: "a.b.c"
|
|
|
|
message: "regexp"
|
|
|
|
- module: "a.b.c"
|
|
|
|
message: "regexp"
|
|
|
|
|
|
|
|
repeated for each log file with a whitelist.
|
|
|
|
"""
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(description=usage)
|
|
|
|
parser.add_argument('-d', '--directory',
|
|
|
|
help="Directory containing log files")
|
|
|
|
parser.add_argument('-u', '--url',
|
|
|
|
help="url containing logs from an OpenStack gate job")
|
2013-10-04 19:10:15 +00:00
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2013-10-10 01:31:32 +00:00
|
|
|
try:
|
|
|
|
sys.exit(main(parser.parse_args()))
|
|
|
|
except Exception as e:
|
|
|
|
print("Failure in script: %s" % e)
|
|
|
|
# Don't fail if there is a problem with the script.
|
|
|
|
sys.exit(0)
|