#!/usr/bin/env python # Protocol Buffers - Google's data interchange format # Copyright 2008 Google Inc. All rights reserved. # # Use of this source code is governed by a BSD-style # license that can be found in the LICENSE file or at # https://developers.google.com/open-source/licenses/bsd """Script to update a failure list file to add/remove failures. When adding, will attempt to place them in their correct lexicographical position relative to other test names. This requires that the failure list is already sorted. This does not guarantee that the tests will appear neatly one after the other, as there may be comments in between. If the failure list is not sorted, sorted blocks may be produced, but the list as a whole will not. Lastly, be wary of lists that have tests stripped from OSS; if you catch that a test was added inside a stripped block, you may need to move it out. This is sort of like comm(1), except it recognizes comments and ignores them. """ import argparse parser = argparse.ArgumentParser( description='Adds/removes failures from the failure list.') parser.add_argument('filename', type=str, help='failure list file to update') parser.add_argument('--add', dest='add_list', action='append') parser.add_argument('--remove', dest='remove_list', action='append') DEFAULT_ALIGNMENT = 114 args = parser.parse_args() add_set = set() remove_set = set() # Adds test + failure message for add_file in (args.add_list or []): with open(add_file) as f: for line in f: add_set.add(line) # We only remove by name for remove_file in (args.remove_list or []): with open(remove_file) as f: for line in f: if line in add_set: raise Exception("Asked to both add and remove test: " + line) remove_set.add(line.split('#')[0].strip()) add_list = sorted(add_set, reverse=True) with open(args.filename) as in_file: existing_list = in_file.read() with open(args.filename, 'w') as f: split_lines = existing_list.splitlines(True) # Make sure that we have one newline for potential tests that have to be # added at the end. split_lines[-1] = split_lines[-1].rstrip() + '\n' for line in split_lines: test = line.split('#')[0].strip() # As long as the tests we are adding appear lexicographically before our # read test, put them first followed by our read test. while add_list and test > add_list[-1]: f.write(add_list.pop()) if test not in remove_set: f.write(line) # Any remaining tests are added at the end while add_list: f.write(add_list.pop()) # Update our read of the existing file with open(args.filename, 'r') as f: existing_list = f.read() # Actual alignment of failure messages to 'DEFAULT_ALIGNMENT' # If test name exceeds DEFAULT_ALIGNMENT, it cannot and will not be aligned. with open(args.filename, 'w') as f: for line in existing_list.splitlines(True): split = line.split('#', 1) test = split[0].strip() if len(split) > 1 and test: message = split[1].lstrip() spaces = ' ' * (DEFAULT_ALIGNMENT - len(test)) line = test + spaces + ' # ' + message f.write(line) else: # ignore blank lines/lines that do not have '#'/comments f.write(line)