| #!/usr/bin/env python3 |
| # Copyright 2013 The Chromium Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| import fnmatch |
| import os |
| import sys |
| import tempfile |
| import unittest |
| import zipfile |
| |
| sys.path.insert( |
| 0, os.path.abspath(os.path.join(os.path.dirname(__file__), os.pardir))) |
| from util import md5_check |
| |
| |
| def _WriteZipFile(path, entries): |
| with zipfile.ZipFile(path, 'w') as zip_file: |
| for subpath, data in entries: |
| zip_file.writestr(subpath, data) |
| |
| |
| class TestMd5Check(unittest.TestCase): |
| def setUp(self): |
| self.called = False |
| self.changes = None |
| |
| def testCallAndRecordIfStale(self): |
| input_strings = ['string1', 'string2'] |
| input_file1 = tempfile.NamedTemporaryFile(suffix='.txt') |
| input_file2 = tempfile.NamedTemporaryFile(suffix='.zip') |
| file1_contents = b'input file 1' |
| input_file1.write(file1_contents) |
| input_file1.flush() |
| # Test out empty zip file to start. |
| _WriteZipFile(input_file2.name, []) |
| input_files = [input_file1.name, input_file2.name] |
| zip_paths = [input_file2.name] |
| |
| record_path = tempfile.NamedTemporaryFile(suffix='.stamp') |
| |
| def CheckCallAndRecord(should_call, |
| message, |
| force=False, |
| outputs_specified=False, |
| outputs_missing=False, |
| expected_changes=None, |
| added_or_modified_only=None, |
| track_subentries=False, |
| output_newer_than_record=False): |
| output_paths = None |
| if outputs_specified: |
| output_file1 = tempfile.NamedTemporaryFile() |
| if outputs_missing: |
| output_file1.close() # Gets deleted on close(). |
| output_paths = [output_file1.name] |
| if output_newer_than_record: |
| output_mtime = os.path.getmtime(output_file1.name) |
| os.utime(record_path.name, (output_mtime - 1, output_mtime - 1)) |
| else: |
| # touch the record file so it doesn't look like it's older that |
| # the output we've just created |
| os.utime(record_path.name, None) |
| |
| self.called = False |
| self.changes = None |
| if expected_changes or added_or_modified_only is not None: |
| def MarkCalled(changes): |
| self.called = True |
| self.changes = changes |
| else: |
| def MarkCalled(): |
| self.called = True |
| |
| md5_check.CallAndRecordIfStale( |
| MarkCalled, |
| record_path=record_path.name, |
| input_paths=input_files, |
| input_strings=input_strings, |
| output_paths=output_paths, |
| force=force, |
| pass_changes=(expected_changes or added_or_modified_only) is not None, |
| track_subpaths_allowlist=zip_paths if track_subentries else None) |
| self.assertEqual(should_call, self.called, message) |
| if expected_changes: |
| description = self.changes.DescribeDifference() |
| self.assertTrue(fnmatch.fnmatch(description, expected_changes), |
| 'Expected %s to match %s' % ( |
| repr(description), repr(expected_changes))) |
| if should_call and added_or_modified_only is not None: |
| self.assertEqual(added_or_modified_only, |
| self.changes.AddedOrModifiedOnly()) |
| |
| CheckCallAndRecord(True, 'should call when record doesn\'t exist', |
| expected_changes='Previous stamp file not found.', |
| added_or_modified_only=False) |
| CheckCallAndRecord(False, 'should not call when nothing changed') |
| input_files = input_files[::-1] |
| CheckCallAndRecord(False, 'reordering of inputs shouldn\'t trigger call') |
| |
| CheckCallAndRecord(False, 'should not call when nothing changed #2', |
| outputs_specified=True, outputs_missing=False) |
| CheckCallAndRecord(True, 'should call when output missing', |
| outputs_specified=True, outputs_missing=True, |
| expected_changes='Outputs do not exist:*', |
| added_or_modified_only=False) |
| CheckCallAndRecord(True, |
| 'should call when output is newer than record', |
| expected_changes='Outputs newer than stamp file:*', |
| outputs_specified=True, |
| outputs_missing=False, |
| added_or_modified_only=False, |
| output_newer_than_record=True) |
| CheckCallAndRecord(True, force=True, message='should call when forced', |
| expected_changes='force=True', |
| added_or_modified_only=False) |
| |
| input_file1.write(b'some more input') |
| input_file1.flush() |
| CheckCallAndRecord(True, 'changed input file should trigger call', |
| expected_changes='*Modified: %s' % input_file1.name, |
| added_or_modified_only=True) |
| |
| input_files = input_files[:1] |
| CheckCallAndRecord(True, 'removing file should trigger call', |
| expected_changes='*Removed: %s' % input_file1.name, |
| added_or_modified_only=False) |
| |
| input_files.append(input_file1.name) |
| CheckCallAndRecord(True, 'added input file should trigger call', |
| expected_changes='*Added: %s' % input_file1.name, |
| added_or_modified_only=True) |
| |
| input_strings[0] = input_strings[0] + ' a bit longer' |
| CheckCallAndRecord(True, 'changed input string should trigger call', |
| expected_changes='*Input strings changed*', |
| added_or_modified_only=False) |
| |
| input_strings = input_strings[::-1] |
| CheckCallAndRecord(True, 'reordering of string inputs should trigger call', |
| expected_changes='*Input strings changed*') |
| |
| input_strings = input_strings[:1] |
| CheckCallAndRecord(True, 'removing a string should trigger call') |
| |
| input_strings.append('a brand new string') |
| CheckCallAndRecord( |
| True, |
| 'added input string should trigger call', |
| added_or_modified_only=False) |
| |
| _WriteZipFile(input_file2.name, [('path/1.txt', '1')]) |
| CheckCallAndRecord( |
| True, |
| 'added subpath should trigger call', |
| expected_changes='*Modified: %s*Subpath added: %s' % (input_file2.name, |
| 'path/1.txt'), |
| added_or_modified_only=True, |
| track_subentries=True) |
| _WriteZipFile(input_file2.name, [('path/1.txt', '2')]) |
| CheckCallAndRecord( |
| True, |
| 'changed subpath should trigger call', |
| expected_changes='*Modified: %s*Subpath modified: %s' % |
| (input_file2.name, 'path/1.txt'), |
| added_or_modified_only=True, |
| track_subentries=True) |
| |
| _WriteZipFile(input_file2.name, []) |
| CheckCallAndRecord(True, 'removed subpath should trigger call', |
| expected_changes='*Modified: %s*Subpath removed: %s' % ( |
| input_file2.name, 'path/1.txt'), |
| added_or_modified_only=False) |
| |
| |
| if __name__ == '__main__': |
| unittest.main() |