blob: f34fb0610d778a2bf4e6682c7c5c81cc592a637e [file] [log] [blame]
Tobin Ehlis35308dd2016-10-31 13:27:36 -06001#!/usr/bin/env python3
2# Copyright (c) 2015-2016 The Khronos Group Inc.
3# Copyright (c) 2015-2016 Valve Corporation
4# Copyright (c) 2015-2016 LunarG, Inc.
5# Copyright (c) 2015-2016 Google Inc.
6#
7# Licensed under the Apache License, Version 2.0 (the "License");
8# you may not use this file except in compliance with the License.
9# You may obtain a copy of the License at
10#
11# http://www.apache.org/licenses/LICENSE-2.0
12#
13# Unless required by applicable law or agreed to in writing, software
14# distributed under the License is distributed on an "AS IS" BASIS,
15# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16# See the License for the specific language governing permissions and
17# limitations under the License.
18#
19# Author: Tobin Ehlis <tobine@google.com>
20
21import argparse
22import os
23import sys
24import platform
25
26# vk_validation_stats.py overview
27# This script is intended to generate statistics on the state of validation code
28# based on information parsed from the source files and the database file
29# Here's what it currently does:
30# 1. Parse vk_validation_error_database.txt to store claimed state of validation checks
31# 2. Parse vk_validation_error_messages.h to verify the actual checks in header vs. the
32# claimed state of the checks
33# 3. Parse source files to identify which checks are implemented and verify that this
34# exactly matches the list of checks claimed to be implemented in the database
35# 4. Parse test file(s) and verify that reported tests exist
36# 5. Report out stats on number of checks, implemented checks, and duplicated checks
37#
Tobin Ehlis20e32582016-12-05 14:50:03 -070038# If a mis-match is found during steps 2, 3, or 4, then the script exits w/ a non-zero error code
39# otherwise, the script will exit(0)
40#
Tobin Ehlis35308dd2016-10-31 13:27:36 -060041# TODO:
42# 1. Would also like to report out number of existing checks that don't yet use new, unique enum
43# 2. Could use notes to store custom fields (like TODO) and print those out here
44# 3. Update test code to check if tests use new, unique enums to check for errors instead of strings
45
46db_file = 'vk_validation_error_database.txt'
47layer_source_files = [
48'core_validation.cpp',
49'descriptor_sets.cpp',
50'parameter_validation.cpp',
51'object_tracker.cpp',
Mike Weiblen6a27de52016-12-09 17:36:28 -070052'image.cpp',
Mark Lobodzinskibfab4a62017-01-27 15:34:37 -070053'buffer_validation.cpp',
Mike Weiblen6a27de52016-12-09 17:36:28 -070054'swapchain.cpp'
Tobin Ehlis35308dd2016-10-31 13:27:36 -060055]
56header_file = 'vk_validation_error_messages.h'
57# TODO : Don't hardcode linux path format if we want this to run on windows
58test_file = '../tests/layer_validation_tests.cpp'
Tobin Ehlis225b59c2016-12-22 13:59:42 -070059# List of enums that are allowed to be used more than once so don't warn on their duplicates
60duplicate_exceptions = [
Tobin Ehlis1fd9eaf2016-12-22 14:31:01 -070061'VALIDATION_ERROR_00018', # This covers the broad case that all child objects must be destroyed at DestroyInstance time
62'VALIDATION_ERROR_00049', # This covers the broad case that all child objects must be destroyed at DestroyDevice time
Tobin Ehlis8ef54c92017-01-04 08:11:01 -070063'VALIDATION_ERROR_00112', # Obj tracker check makes sure non-null framebuffer is valid & CV check makes sure it's compatible w/ renderpass framebuffer
64'VALIDATION_ERROR_00324', # This is an aliasing error that we report twice, for each of the two allocations that are aliasing
Tobin Ehlis2cb8eb22017-01-03 14:09:57 -070065'VALIDATION_ERROR_00515', # Covers valid shader module handle for both Compute & Graphics pipelines
Tobin Ehlise664a2b2016-12-22 14:10:21 -070066'VALIDATION_ERROR_00648', # This is a case for VkMappedMemoryRange struct that is used by both Flush & Invalidate MappedMemoryRange
Tobin Ehlis1fd9eaf2016-12-22 14:31:01 -070067'VALIDATION_ERROR_00741', # This is a blanket case for all invalid image aspect bit errors. The spec link has appropriate details for all separate cases.
68'VALIDATION_ERROR_00768', # This case covers two separate checks which are done independently
69'VALIDATION_ERROR_00769', # This case covers two separate checks which are done independently
70'VALIDATION_ERROR_00942', # This is a descriptor set write update error that we use for a couple copy cases as well
Tobin Ehlis2cb8eb22017-01-03 14:09:57 -070071'VALIDATION_ERROR_00988', # Single error for mis-matched stageFlags of vkCmdPushConstants() that is flagged for no stage flags & mis-matched flags
72'VALIDATION_ERROR_01088', # Handles both depth/stencil & compressed image errors for vkCmdClearColorImage()
73'VALIDATION_ERROR_01223', # Used for the mipLevel check of both dst & src images on vkCmdCopyImage call
74'VALIDATION_ERROR_01224', # Used for the arraySize check of both dst & src images on vkCmdCopyImage call
75'VALIDATION_ERROR_01450', # Used for both x & y bounds of viewport
76'VALIDATION_ERROR_01489', # Used for both x & y value of scissors to make sure they're not negative
77'VALIDATION_ERROR_01926', # Surface of VkSwapchainCreateInfoKHR must be valid when creating both single or shared swapchains
Tobin Ehlis8ef54c92017-01-04 08:11:01 -070078'VALIDATION_ERROR_01935', # oldSwapchain of VkSwapchainCreateInfoKHR must be valid when creating both single or shared swapchains
Tobin Ehlis2cb8eb22017-01-03 14:09:57 -070079'VALIDATION_ERROR_02333', # Single error for both imageFormat & imageColorSpace requirements when creating swapchain
Tobin Ehlis1fd9eaf2016-12-22 14:31:01 -070080'VALIDATION_ERROR_02525', # Used twice for the same error codepath as both a param & to set a variable, so not really a duplicate
Tobin Ehlis225b59c2016-12-22 13:59:42 -070081]
Tobin Ehlis35308dd2016-10-31 13:27:36 -060082
83class ValidationDatabase:
84 def __init__(self, filename=db_file):
85 self.db_file = filename
86 self.delimiter = '~^~'
87 self.db_dict = {} # complete dict of all db values per error enum
88 # specialized data structs with slices of complete dict
89 self.db_implemented_enums = [] # list of all error enums claiming to be implemented in database file
Tobin Ehlis2bedc242017-01-12 13:45:55 -070090 self.db_unimplemented_implicit = [] # list of all implicit checks that aren't marked implemented
Tobin Ehlis35308dd2016-10-31 13:27:36 -060091 self.db_enum_to_tests = {} # dict where enum is key to lookup list of tests implementing the enum
92 #self.src_implemented_enums
93 def read(self):
94 """Read a database file into internal data structures, format of each line is <enum><implemented Y|N?><testname><api><errormsg><notes>"""
95 #db_dict = {} # This is a simple db of just enum->errormsg, the same as is created from spec
96 #max_id = 0
97 with open(self.db_file, "r") as infile:
98 for line in infile:
99 line = line.strip()
100 if line.startswith('#') or '' == line:
101 continue
102 db_line = line.split(self.delimiter)
103 if len(db_line) != 6:
Tobin Ehlis027f3212016-12-09 12:15:26 -0700104 print("ERROR: Bad database line doesn't have 6 elements: %s" % (line))
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600105 error_enum = db_line[0]
106 implemented = db_line[1]
107 testname = db_line[2]
108 api = db_line[3]
109 error_str = db_line[4]
110 note = db_line[5]
111 # Read complete database contents into our class var for later use
112 self.db_dict[error_enum] = {}
113 self.db_dict[error_enum]['check_implemented'] = implemented
114 self.db_dict[error_enum]['testname'] = testname
115 self.db_dict[error_enum]['api'] = api
116 self.db_dict[error_enum]['error_string'] = error_str
117 self.db_dict[error_enum]['note'] = note
118 # Now build custom data structs
119 if 'Y' == implemented:
120 self.db_implemented_enums.append(error_enum)
Tobin Ehlis2bedc242017-01-12 13:45:55 -0700121 elif 'implicit' in note: # only make note of non-implemented implicit checks
122 self.db_unimplemented_implicit.append(error_enum)
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600123 if testname.lower() not in ['unknown', 'none']:
124 self.db_enum_to_tests[error_enum] = testname.split(',')
125 #if len(self.db_enum_to_tests[error_enum]) > 1:
126 # print "Found check %s that has multiple tests: %s" % (error_enum, self.db_enum_to_tests[error_enum])
127 #else:
128 # print "Check %s has single test: %s" % (error_enum, self.db_enum_to_tests[error_enum])
129 #unique_id = int(db_line[0].split('_')[-1])
130 #if unique_id > max_id:
131 # max_id = unique_id
132 #print "Found %d total enums in database" % (len(self.db_dict.keys()))
133 #print "Found %d enums claiming to be implemented in source" % (len(self.db_implemented_enums))
134 #print "Found %d enums claiming to have tests implemented" % (len(self.db_enum_to_tests.keys()))
135
136class ValidationHeader:
137 def __init__(self, filename=header_file):
138 self.filename = header_file
139 self.enums = []
140 def read(self):
141 """Read unique error enum header file into internal data structures"""
142 grab_enums = False
143 with open(self.filename, "r") as infile:
144 for line in infile:
145 line = line.strip()
146 if 'enum UNIQUE_VALIDATION_ERROR_CODE {' in line:
147 grab_enums = True
148 continue
149 if grab_enums:
150 if 'VALIDATION_ERROR_MAX_ENUM' in line:
151 grab_enums = False
152 break # done
Tobin Ehlisf53eac32016-12-09 14:10:47 -0700153 elif 'VALIDATION_ERROR_UNDEFINED' in line:
154 continue
155 elif 'VALIDATION_ERROR_' in line:
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600156 enum = line.split(' = ')[0]
157 self.enums.append(enum)
158 #print "Found %d error enums. First is %s and last is %s." % (len(self.enums), self.enums[0], self.enums[-1])
159
160class ValidationSource:
161 def __init__(self, source_file_list):
162 self.source_files = source_file_list
Tobin Ehlis3d1f2bd2016-12-22 11:19:15 -0700163 self.enum_count_dict = {} # dict of enum values to the count of how much they're used, and location of where they're used
Tobin Ehlis3d9dd942016-11-23 13:08:01 -0700164 # 1790 is a special case that provides an exception when an extension is enabled. No specific error is flagged, but the exception is handled so add it here
Tobin Ehlis3d1f2bd2016-12-22 11:19:15 -0700165 self.enum_count_dict['VALIDATION_ERROR_01790'] = {}
166 self.enum_count_dict['VALIDATION_ERROR_01790']['count'] = 1
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600167 def parse(self):
168 duplicate_checks = 0
169 for sf in self.source_files:
Tobin Ehlis3d1f2bd2016-12-22 11:19:15 -0700170 line_num = 0
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600171 with open(sf) as f:
172 for line in f:
Tobin Ehlis3d1f2bd2016-12-22 11:19:15 -0700173 line_num = line_num + 1
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600174 if True in [line.strip().startswith(comment) for comment in ['//', '/*']]:
175 continue
176 # Find enums
177 #if 'VALIDATION_ERROR_' in line and True not in [ignore in line for ignore in ['[VALIDATION_ERROR_', 'UNIQUE_VALIDATION_ERROR_CODE']]:
Tobin Ehlisf53eac32016-12-09 14:10:47 -0700178 if ' VALIDATION_ERROR_' in line:
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600179 # Need to isolate the validation error enum
180 #print("Line has check:%s" % (line))
181 line_list = line.split()
Tobin Ehlis928742e2016-12-09 17:11:13 -0700182 enum_list = []
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600183 for str in line_list:
Tobin Ehlisf53eac32016-12-09 14:10:47 -0700184 if 'VALIDATION_ERROR_' in str and True not in [ignore_str in str for ignore_str in ['[VALIDATION_ERROR_', 'VALIDATION_ERROR_UNDEFINED', 'UNIQUE_VALIDATION_ERROR_CODE']]:
Tobin Ehlis928742e2016-12-09 17:11:13 -0700185 enum_list.append(str.strip(',);'))
186 #break
187 for enum in enum_list:
188 if enum != '':
189 if enum not in self.enum_count_dict:
Tobin Ehlis3d1f2bd2016-12-22 11:19:15 -0700190 self.enum_count_dict[enum] = {}
191 self.enum_count_dict[enum]['count'] = 1
192 self.enum_count_dict[enum]['file_line'] = []
193 self.enum_count_dict[enum]['file_line'].append('%s,%d' % (sf, line_num))
Tobin Ehlis928742e2016-12-09 17:11:13 -0700194 #print "Found enum %s implemented for first time in file %s" % (enum, sf)
195 else:
Tobin Ehlis3d1f2bd2016-12-22 11:19:15 -0700196 self.enum_count_dict[enum]['count'] = self.enum_count_dict[enum]['count'] + 1
197 self.enum_count_dict[enum]['file_line'].append('%s,%d' % (sf, line_num))
Tobin Ehlis928742e2016-12-09 17:11:13 -0700198 #print "Found enum %s implemented for %d time in file %s" % (enum, self.enum_count_dict[enum], sf)
199 duplicate_checks = duplicate_checks + 1
200 #else:
201 #print("Didn't find actual check in line:%s" % (line))
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600202 #print "Found %d unique implemented checks and %d are duplicated at least once" % (len(self.enum_count_dict.keys()), duplicate_checks)
203
204# Class to parse the validation layer test source and store testnames
205# TODO: Enhance class to detect use of unique error enums in the test
206class TestParser:
207 def __init__(self, test_file_list, test_group_name=['VkLayerTest', 'VkPositiveLayerTest', 'VkWsiEnabledLayerTest']):
208 self.test_files = test_file_list
Tobin Ehlis9a68c982016-12-29 14:51:17 -0700209 self.test_to_errors = {} # Dict where testname maps to list of error enums found in that test
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600210 self.test_trigger_txt_list = []
211 for tg in test_group_name:
212 self.test_trigger_txt_list.append('TEST_F(%s' % tg)
213 #print('Test trigger test list: %s' % (self.test_trigger_txt_list))
214
215 # Parse test files into internal data struct
216 def parse(self):
217 # For each test file, parse test names into set
218 grab_next_line = False # handle testname on separate line than wildcard
Tobin Ehlis9a68c982016-12-29 14:51:17 -0700219 testname = ''
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600220 for test_file in self.test_files:
221 with open(test_file) as tf:
222 for line in tf:
223 if True in [line.strip().startswith(comment) for comment in ['//', '/*']]:
224 continue
225
226 if True in [ttt in line for ttt in self.test_trigger_txt_list]:
227 #print('Test wildcard in line: %s' % (line))
228 testname = line.split(',')[-1]
229 testname = testname.strip().strip(' {)')
230 #print('Inserting test: "%s"' % (testname))
231 if ('' == testname):
232 grab_next_line = True
233 continue
Tobin Ehlis9a68c982016-12-29 14:51:17 -0700234 self.test_to_errors[testname] = []
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600235 if grab_next_line: # test name on its own line
236 grab_next_line = False
237 testname = testname.strip().strip(' {)')
Tobin Ehlis9a68c982016-12-29 14:51:17 -0700238 self.test_to_errors[testname] = []
239 if ' VALIDATION_ERROR_' in line:
240 line_list = line.split()
Tobin Ehlis71f38c12017-01-12 14:26:56 -0700241 for sub_str in line_list:
242 if 'VALIDATION_ERROR_' in sub_str and True not in [ignore_str in sub_str for ignore_str in ['VALIDATION_ERROR_UNDEFINED', 'UNIQUE_VALIDATION_ERROR_CODE', 'VALIDATION_ERROR_MAX_ENUM']]:
243 #print("Trying to add enums for line: %s" % ())
244 #print("Adding enum %s to test %s" % (sub_str.strip(',);'), testname))
245 self.test_to_errors[testname].append(sub_str.strip(',);'))
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600246
247# Little helper class for coloring cmd line output
248class bcolors:
249
250 def __init__(self):
251 self.GREEN = '\033[0;32m'
252 self.RED = '\033[0;31m'
253 self.YELLOW = '\033[1;33m'
254 self.ENDC = '\033[0m'
255 if 'Linux' != platform.system():
256 self.GREEN = ''
257 self.RED = ''
258 self.YELLOW = ''
259 self.ENDC = ''
260
261 def green(self):
262 return self.GREEN
263
264 def red(self):
265 return self.RED
266
267 def yellow(self):
268 return self.YELLOW
269
270 def endc(self):
271 return self.ENDC
272
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600273def main(argv=None):
Tobin Ehlis20e32582016-12-05 14:50:03 -0700274 result = 0 # Non-zero result indicates an error case
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600275 # parse db
276 val_db = ValidationDatabase()
277 val_db.read()
278 # parse header
279 val_header = ValidationHeader()
280 val_header.read()
281 # Create parser for layer files
282 val_source = ValidationSource(layer_source_files)
283 val_source.parse()
284 # Parse test files
285 test_parser = TestParser([test_file, ])
286 test_parser.parse()
287
288 # Process stats - Just doing this inline in main, could make a fancy class to handle
289 # all the processing of data and then get results from that
290 txt_color = bcolors()
291 print("Validation Statistics")
292 # First give number of checks in db & header and report any discrepancies
293 db_enums = len(val_db.db_dict.keys())
294 hdr_enums = len(val_header.enums)
295 print(" Database file includes %d unique checks" % (db_enums))
296 print(" Header file declares %d unique checks" % (hdr_enums))
297 tmp_db_dict = val_db.db_dict
298 db_missing = []
299 for enum in val_header.enums:
300 if not tmp_db_dict.pop(enum, False):
301 db_missing.append(enum)
302 if db_enums == hdr_enums and len(db_missing) == 0 and len(tmp_db_dict.keys()) == 0:
303 print(txt_color.green() + " Database and Header match, GREAT!" + txt_color.endc())
304 else:
305 print(txt_color.red() + " Uh oh, Database doesn't match Header :(" + txt_color.endc())
Tobin Ehlis20e32582016-12-05 14:50:03 -0700306 result = 1
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600307 if len(db_missing) != 0:
308 print(txt_color.red() + " The following checks are in header but missing from database:" + txt_color.endc())
309 for missing_enum in db_missing:
310 print(txt_color.red() + " %s" % (missing_enum) + txt_color.endc())
311 if len(tmp_db_dict.keys()) != 0:
312 print(txt_color.red() + " The following checks are in database but haven't been declared in the header:" + txt_color.endc())
313 for extra_enum in tmp_db_dict:
314 print(txt_color.red() + " %s" % (extra_enum) + txt_color.endc())
315 # Report out claimed implemented checks vs. found actual implemented checks
316 imp_not_found = [] # Checks claimed to implemented in DB file but no source found
317 imp_not_claimed = [] # Checks found implemented but not claimed to be in DB
318 multiple_uses = False # Flag if any enums are used multiple times
319 for db_imp in val_db.db_implemented_enums:
320 if db_imp not in val_source.enum_count_dict:
321 imp_not_found.append(db_imp)
322 for src_enum in val_source.enum_count_dict:
Tobin Ehlis225b59c2016-12-22 13:59:42 -0700323 if val_source.enum_count_dict[src_enum]['count'] > 1 and src_enum not in duplicate_exceptions:
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600324 multiple_uses = True
325 if src_enum not in val_db.db_implemented_enums:
326 imp_not_claimed.append(src_enum)
327 print(" Database file claims that %d checks (%s) are implemented in source." % (len(val_db.db_implemented_enums), "{0:.0f}%".format(float(len(val_db.db_implemented_enums))/db_enums * 100)))
Tobin Ehlis2bedc242017-01-12 13:45:55 -0700328 if len(val_db.db_unimplemented_implicit) > 0:
329 print(" Database file claims %d implicit checks (%s) that are not implemented." % (len(val_db.db_unimplemented_implicit), "{0:.0f}%".format(float(len(val_db.db_unimplemented_implicit))/db_enums * 100)))
330 total_checks = len(val_db.db_implemented_enums) + len(val_db.db_unimplemented_implicit)
331 print(" If all implicit checks are handled by parameter validation this is a total of %d (%s) checks covered." % (total_checks, "{0:.0f}%".format(float(total_checks)/db_enums * 100)))
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600332 if len(imp_not_found) == 0 and len(imp_not_claimed) == 0:
333 print(txt_color.green() + " All claimed Database implemented checks have been found in source, and no source checks aren't claimed in Database, GREAT!" + txt_color.endc())
334 else:
Tobin Ehlis20e32582016-12-05 14:50:03 -0700335 result = 1
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600336 print(txt_color.red() + " Uh oh, Database claimed implemented don't match Source :(" + txt_color.endc())
337 if len(imp_not_found) != 0:
Tobin Ehlis3f0b2772016-11-18 16:56:15 -0700338 print(txt_color.red() + " The following %d checks are claimed to be implemented in Database, but weren't found in source:" % (len(imp_not_found)) + txt_color.endc())
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600339 for not_imp_enum in imp_not_found:
340 print(txt_color.red() + " %s" % (not_imp_enum) + txt_color.endc())
341 if len(imp_not_claimed) != 0:
342 print(txt_color.red() + " The following checks are implemented in source, but not claimed to be in Database:" + txt_color.endc())
343 for imp_enum in imp_not_claimed:
344 print(txt_color.red() + " %s" % (imp_enum) + txt_color.endc())
345 if multiple_uses:
346 print(txt_color.yellow() + " Note that some checks are used multiple times. These may be good candidates for new valid usage spec language." + txt_color.endc())
347 print(txt_color.yellow() + " Here is a list of each check used multiple times with its number of uses:" + txt_color.endc())
348 for enum in val_source.enum_count_dict:
Tobin Ehlis225b59c2016-12-22 13:59:42 -0700349 if val_source.enum_count_dict[enum]['count'] > 1 and enum not in duplicate_exceptions:
Tobin Ehlis3d1f2bd2016-12-22 11:19:15 -0700350 print(txt_color.yellow() + " %s: %d uses in file,line:" % (enum, val_source.enum_count_dict[enum]['count']) + txt_color.endc())
351 for file_line in val_source.enum_count_dict[enum]['file_line']:
352 print(txt_color.yellow() + " \t%s" % (file_line) + txt_color.endc())
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600353 # Now check that tests claimed to be implemented are actual test names
354 bad_testnames = []
Tobin Ehlis9a68c982016-12-29 14:51:17 -0700355 tests_missing_enum = {} # Report tests that don't use validation error enum to check for error case
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600356 for enum in val_db.db_enum_to_tests:
357 for testname in val_db.db_enum_to_tests[enum]:
Tobin Ehlis9a68c982016-12-29 14:51:17 -0700358 if testname not in test_parser.test_to_errors:
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600359 bad_testnames.append(testname)
Tobin Ehlis9a68c982016-12-29 14:51:17 -0700360 else:
361 enum_found = False
362 for test_enum in test_parser.test_to_errors[testname]:
363 if test_enum == enum:
364 #print("Found test that correctly checks for enum: %s" % (enum))
365 enum_found = True
366 if not enum_found:
367 #print("Test %s is not using enum %s to check for error" % (testname, enum))
368 if testname not in tests_missing_enum:
369 tests_missing_enum[testname] = []
370 tests_missing_enum[testname].append(enum)
371 if tests_missing_enum:
372 print(txt_color.yellow() + " \nThe following tests do not use their reported enums to check for the validation error. You may want to update these to pass the expected enum to SetDesiredFailureMsg:" + txt_color.endc())
373 for testname in tests_missing_enum:
374 print(txt_color.yellow() + " Testname %s does not explicitly check for these ids:" % (testname) + txt_color.endc())
375 for enum in tests_missing_enum[testname]:
376 print(txt_color.yellow() + " %s" % (enum) + txt_color.endc())
377 # TODO : Go through all enums found in the test file and make sure they're correctly documented in the database file
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600378 print(" Database file claims that %d checks have tests written." % len(val_db.db_enum_to_tests))
379 if len(bad_testnames) == 0:
380 print(txt_color.green() + " All claimed tests have valid names. That's good!" + txt_color.endc())
381 else:
382 print(txt_color.red() + " The following testnames in Database appear to be invalid:")
Tobin Ehlis20e32582016-12-05 14:50:03 -0700383 result = 1
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600384 for bt in bad_testnames:
Tobin Ehlisb04c2c62016-11-21 15:51:45 -0700385 print(txt_color.red() + " %s" % (bt) + txt_color.endc())
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600386
Tobin Ehlis20e32582016-12-05 14:50:03 -0700387 return result
Tobin Ehlis35308dd2016-10-31 13:27:36 -0600388
389if __name__ == "__main__":
390 sys.exit(main())
391