| #!/usr/bin/env python3 |
| # |
| # This file is part of GCC. |
| # |
| # GCC is free software; you can redistribute it and/or modify it under |
| # the terms of the GNU General Public License as published by the Free |
| # Software Foundation; either version 3, or (at your option) any later |
| # version. |
| # |
| # GCC is distributed in the hope that it will be useful, but WITHOUT ANY |
| # WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
| # for more details. |
| # |
| # You should have received a copy of the GNU General Public License |
| # along with GCC; see the file COPYING3. If not see |
| # <http://www.gnu.org/licenses/>. */ |
| |
| import os |
| import tempfile |
| import unittest |
| |
| from git_commit import GitCommit |
| |
| from git_email import GitEmail |
| |
| import unidiff |
| |
| script_path = os.path.dirname(os.path.realpath(__file__)) |
| |
| unidiff_supports_renaming = hasattr(unidiff.PatchedFile(), 'is_rename') |
| |
| |
| NAME_STATUS1 = """ |
| M gcc/ada/impunit.adb' |
| R097 gcc/ada/libgnat/s-atopar.adb gcc/ada/libgnat/s-aoinar.adb |
| """ |
| |
| |
| class TestGccChangelog(unittest.TestCase): |
| def setUp(self): |
| self.patches = {} |
| self.temps = [] |
| |
| filename = None |
| patch_lines = [] |
| with open(os.path.join(script_path, 'test_patches.txt')) as f: |
| lines = f.read() |
| for line in lines.split('\n'): |
| if line.startswith('==='): |
| if patch_lines: |
| self.patches[filename] = patch_lines |
| filename = line.split(' ')[1] |
| patch_lines = [] |
| else: |
| patch_lines.append(line) |
| if patch_lines: |
| self.patches[filename] = patch_lines |
| |
| def tearDown(self): |
| for t in self.temps: |
| assert t.endswith('.patch') |
| os.remove(t) |
| |
| def get_git_email(self, filename): |
| with tempfile.NamedTemporaryFile(mode='w+', suffix='.patch', |
| delete=False) as f: |
| f.write('\n'.join(self.patches[filename])) |
| self.temps.append(f.name) |
| return GitEmail(f.name) |
| |
| def from_patch_glob(self, name): |
| files = [f for f in self.patches.keys() if f.startswith(name)] |
| assert len(files) == 1 |
| return self.get_git_email(files[0]) |
| |
| def test_simple_patch_format(self): |
| email = self.get_git_email('0577-aarch64-Add-an-and.patch') |
| assert not email.errors |
| assert len(email.changelog_entries) == 2 |
| entry = email.changelog_entries[0] |
| assert (entry.author_lines == |
| [('Richard Sandiford <richard.sandiford@arm.com>', |
| '2020-02-06')]) |
| assert len(entry.authors) == 1 |
| assert (entry.authors[0] |
| == 'Richard Sandiford <richard.sandiford@arm.com>') |
| assert entry.folder == 'gcc' |
| assert entry.prs == ['PR target/87763'] |
| assert len(entry.files) == 3 |
| assert entry.files[0] == 'config/aarch64/aarch64-protos.h' |
| |
| def test_daily_bump(self): |
| email = self.get_git_email('0085-Daily-bump.patch') |
| assert not email.errors |
| assert not email.changelog_entries |
| |
| def test_deduce_changelog_entries(self): |
| email = self.from_patch_glob('0040') |
| assert len(email.changelog_entries) == 2 |
| assert email.changelog_entries[0].folder == 'gcc/cp' |
| assert email.changelog_entries[0].prs == ['PR c++/90916'] |
| assert email.changelog_entries[0].files == ['pt.c'] |
| # this one is added automatically |
| assert email.changelog_entries[1].folder == 'gcc/testsuite' |
| |
| def test_only_changelog_updated(self): |
| email = self.from_patch_glob('0129') |
| assert not email.errors |
| assert not email.changelog_entries |
| |
| def test_wrong_mentioned_filename(self): |
| email = self.from_patch_glob('0096') |
| assert email.errors |
| err = email.errors[0] |
| assert err.message == 'unchanged file mentioned in a ChangeLog (did ' \ |
| 'you mean "gcc/testsuite/gcc.target/aarch64/' \ |
| 'advsimd-intrinsics/vdot-3-1.c"?)' |
| assert err.line == 'gcc/testsuite/gcc.target/aarch64/' \ |
| 'advsimd-intrinsics/vdot-compile-3-1.c' |
| |
| def test_missing_tab(self): |
| email = self.from_patch_glob('0031') |
| assert len(email.errors) == 2 |
| err = email.errors[0] |
| assert err.message == 'line should start with a tab' |
| assert err.line == ' * cfgloopanal.c (average_num_loop_insns): ' \ |
| 'Free bbs when early' |
| |
| def test_leading_changelog_format(self): |
| email = self.from_patch_glob('0184') |
| assert len(email.errors) == 4 |
| assert email.errors[0].line == 'gcc/c-family/c-cppbuiltins.c' |
| assert email.errors[2].line == 'gcc/c-family/c-cppbuiltin.c' |
| |
| def test_cannot_deduce_no_blank_line(self): |
| email = self.from_patch_glob('0334') |
| assert len(email.errors) == 1 |
| assert len(email.changelog_entries) == 1 |
| assert email.changelog_entries[0].folder is None |
| |
| def test_author_lines(self): |
| email = self.from_patch_glob('0814') |
| assert not email.errors |
| assert (email.changelog_entries[0].author_lines == |
| [('Martin Jambor <mjambor@suse.cz>', '2020-02-19')]) |
| |
| def test_multiple_authors_and_prs(self): |
| email = self.from_patch_glob('0735') |
| assert len(email.changelog_entries) == 1 |
| entry = email.changelog_entries[0] |
| assert len(entry.author_lines) == 2 |
| assert len(entry.authors) == 2 |
| assert (entry.author_lines[1] == |
| ('Bernd Edlinger <bernd.edlinger@hotmail.de>', None)) |
| |
| def test_multiple_prs(self): |
| email = self.from_patch_glob('1699') |
| assert len(email.changelog_entries) == 2 |
| assert len(email.changelog_entries[0].prs) == 2 |
| |
| def test_missing_PR_component(self): |
| email = self.from_patch_glob('0735') |
| assert len(email.errors) == 1 |
| assert email.errors[0].message == 'missing PR component' |
| |
| def test_invalid_PR_component(self): |
| email = self.from_patch_glob('0198') |
| assert len(email.errors) == 1 |
| assert email.errors[0].message == 'invalid PR component' |
| |
| def test_additional_author_list(self): |
| email = self.from_patch_glob('0342') |
| msg = 'additional author must be indented ' \ |
| 'with one tab and four spaces' |
| assert email.errors[1].message == msg |
| |
| def test_trailing_whitespaces(self): |
| email = self.get_git_email('trailing-whitespaces.patch') |
| assert len(email.errors) == 3 |
| |
| def test_space_after_asterisk(self): |
| email = self.from_patch_glob('1999') |
| assert len(email.errors) == 1 |
| assert email.errors[0].message == 'one space should follow asterisk' |
| |
| def test_long_lines(self): |
| email = self.get_git_email('long-lines.patch') |
| assert len(email.errors) == 1 |
| assert email.errors[0].message == 'line exceeds 100 character limit' |
| |
| def test_new_files(self): |
| email = self.from_patch_glob('0030') |
| assert not email.errors |
| |
| def test_wrong_changelog_location(self): |
| email = self.from_patch_glob('0043') |
| assert len(email.errors) == 2 |
| assert (email.errors[0].message == |
| 'wrong ChangeLog location "gcc", should be "gcc/testsuite"') |
| |
| def test_single_author_name(self): |
| email = self.from_patch_glob('1975') |
| assert len(email.changelog_entries) == 2 |
| assert len(email.changelog_entries[0].author_lines) == 1 |
| assert len(email.changelog_entries[1].author_lines) == 1 |
| |
| def test_bad_first_line(self): |
| email = self.from_patch_glob('0413') |
| assert len(email.errors) == 1 |
| |
| def test_co_authored_by(self): |
| email = self.from_patch_glob('1850') |
| assert email.co_authors == ['Jakub Jelinek <jakub@redhat.com>'] |
| output_entries = list(email.to_changelog_entries()) |
| assert len(output_entries) == 2 |
| ent0 = output_entries[0] |
| assert ent0[1].startswith('2020-04-16 Martin Liska ' |
| '<mliska@suse.cz>\n\t' |
| ' Jakub Jelinek <jakub@redhat.com>') |
| |
| def test_multiple_co_author_formats(self): |
| email = self.get_git_email('co-authored-by.patch') |
| assert len(email.co_authors) == 3 |
| assert email.co_authors[0] == 'Jakub Jelinek <jakub@redhat.com>' |
| assert email.co_authors[1] == 'John Miller <jm@example.com>' |
| assert email.co_authors[2] == 'John Miller2 <jm2@example.com>' |
| |
| def test_new_file_added_entry(self): |
| email = self.from_patch_glob('1957') |
| output_entries = list(email.to_changelog_entries()) |
| assert len(output_entries) == 2 |
| needle = ('\t* g++.dg/cpp2a/lambda-generic-variadic20.C' |
| ': New file.') |
| assert output_entries[1][1].endswith(needle) |
| assert email.changelog_entries[1].prs == ['PR c++/94546'] |
| |
| def test_global_pr_entry(self): |
| email = self.from_patch_glob('2004') |
| assert not email.errors |
| assert email.changelog_entries[0].prs == ['PR other/94629'] |
| |
| def test_unique_prs(self): |
| email = self.get_git_email('pr-check1.patch') |
| assert not email.errors |
| assert email.changelog_entries[0].prs == ['PR ipa/12345'] |
| assert email.changelog_entries[1].prs == [] |
| |
| def test_multiple_prs_not_added(self): |
| email = self.from_patch_glob('0002-Add-patch_are') |
| assert not email.errors |
| assert email.changelog_entries[0].prs == ['PR target/93492'] |
| assert email.changelog_entries[1].prs == ['PR target/12345'] |
| assert email.changelog_entries[2].prs == [] |
| assert email.changelog_entries[2].folder == 'gcc/testsuite' |
| |
| def test_strict_mode(self): |
| email = self.from_patch_glob('0001-Add-patch_are') |
| msg = 'ChangeLog, DATESTAMP, BASE-VER and DEV-PHASE updates should ' \ |
| 'be done separately from normal commits' |
| assert email.errors[0].message.startswith(msg) |
| |
| def test_strict_mode_normal_patch(self): |
| email = self.get_git_email('0001-Just-test-it.patch') |
| assert not email.errors |
| |
| def test_strict_mode_datestamp_only(self): |
| email = self.get_git_email('0002-Bump-date.patch') |
| assert not email.errors |
| |
| def test_wrong_changelog_entry(self): |
| email = self.from_patch_glob('0020-IPA-Avoid') |
| msg = 'first line should start with a tab, an asterisk and a space' |
| assert (email.errors[0].message == msg) |
| |
| def test_cherry_pick_format(self): |
| email = self.from_patch_glob('0001-c-Alias.patch') |
| assert not email.errors |
| |
| def test_signatures(self): |
| email = self.from_patch_glob('0001-RISC-V-Make-unique.patch') |
| assert not email.errors |
| assert len(email.changelog_entries) == 1 |
| |
| def test_duplicate_top_level_author(self): |
| email = self.from_patch_glob('0001-Fortran-ProcPtr-function.patch') |
| assert not email.errors |
| assert len(email.changelog_entries[0].author_lines) == 1 |
| |
| def test_dr_entry(self): |
| email = self.from_patch_glob('0001-c-C-20-DR-2237.patch') |
| assert email.changelog_entries[0].prs == ['DR 2237'] |
| |
| def test_changes_only_in_ignored_location(self): |
| email = self.from_patch_glob('0001-go-in-ignored-location.patch') |
| assert not email.errors |
| |
| def test_changelog_for_ignored_location(self): |
| email = self.from_patch_glob('0001-Update-merge.sh-to-reflect.patch') |
| assert (email.changelog_entries[0].lines[0] |
| == '\t* LOCAL_PATCHES: Use git hash instead of SVN id.') |
| |
| def test_multiline_file_list(self): |
| email = self.from_patch_glob( |
| '0001-Ada-Reuse-Is_Package_Or_Generic_Package-where-possib.patch') |
| assert (email.changelog_entries[0].files |
| == ['contracts.adb', 'einfo.adb', 'exp_ch9.adb', |
| 'sem_ch12.adb', 'sem_ch4.adb', 'sem_ch7.adb', |
| 'sem_ch8.adb', 'sem_elab.adb', 'sem_type.adb', |
| 'sem_util.adb']) |
| |
| @unittest.skipIf(not unidiff_supports_renaming, |
| 'Newer version of unidiff is needed (0.6.0+)') |
| def test_renamed_file(self): |
| email = self.from_patch_glob( |
| '0001-Ada-Add-support-for-XDR-streaming-in-the-default-run.patch') |
| assert not email.errors |
| |
| def test_duplicite_author_lines(self): |
| email = self.from_patch_glob('0001-Fortran-type-is-real-kind-1.patch') |
| assert (email.changelog_entries[0].author_lines[0][0] |
| == 'Steven G. Kargl <kargl@gcc.gnu.org>') |
| assert (email.changelog_entries[0].author_lines[1][0] |
| == 'Mark Eggleston <markeggleston@gcc.gnu.org>') |
| |
| def test_missing_change_description(self): |
| email = self.from_patch_glob('0001-Missing-change-description.patch') |
| assert len(email.errors) == 2 |
| assert email.errors[0].message == 'missing description of a change' |
| assert email.errors[1].message == 'missing description of a change' |
| |
| def test_libstdcxx_html_regenerated(self): |
| email = self.from_patch_glob('0001-Fix-text-of-hyperlink') |
| assert not email.errors |
| email = self.from_patch_glob('0002-libstdc-Fake-test-change-1.patch') |
| assert len(email.errors) == 1 |
| msg = "pattern doesn't match any changed files" |
| assert email.errors[0].message == msg |
| assert email.errors[0].line == 'libstdc++-v3/doc/html/' |
| email = self.from_patch_glob('0003-libstdc-Fake-test-change-2.patch') |
| assert len(email.errors) == 1 |
| msg = 'changed file not mentioned in a ChangeLog' |
| assert email.errors[0].message == msg |
| |
| def test_not_deduce(self): |
| email = self.from_patch_glob('0001-configure.patch') |
| assert not email.errors |
| assert len(email.changelog_entries) == 2 |
| |
| def test_parse_git_name_status(self): |
| modified_files = GitCommit.parse_git_name_status(NAME_STATUS1) |
| assert len(modified_files) == 3 |
| assert modified_files[1] == ('gcc/ada/libgnat/s-atopar.adb', 'D') |
| assert modified_files[2] == ('gcc/ada/libgnat/s-aoinar.adb', 'A') |
| |
| def test_backport(self): |
| email = self.from_patch_glob('0001-asan-fix-RTX-emission.patch') |
| assert not email.errors |
| expected_hash = '8cff672cb9a132d3d3158c2edfc9a64b55292b80' |
| assert email.cherry_pick_commit == expected_hash |
| assert len(email.changelog_entries) == 1 |
| entry = list(email.to_changelog_entries())[0][1] |
| assert entry.startswith('2020-06-11 Martin Liska <mliska@suse.cz>') |
| assert '\tBackported from master:' in entry |
| assert '\t2020-06-11 Martin Liska <mliska@suse.cz>' in entry |
| assert '\t\t Jakub Jelinek <jakub@redhat.com>' in entry |
| |
| def test_backport_double_cherry_pick(self): |
| email = self.from_patch_glob('double-cherry-pick.patch') |
| assert email.errors[0].message.startswith('multiple cherry pick lines') |
| |
| def test_square_and_lt_gt(self): |
| email = self.from_patch_glob('0001-Check-for-more-missing') |
| assert not email.errors |
| |
| def test_empty_parenthesis(self): |
| email = self.from_patch_glob('0001-tree-optimization-97633-fix') |
| assert len(email.errors) == 1 |
| assert email.errors[0].message == 'empty group "()" found' |
| |
| def test_emptry_entry_desc(self): |
| email = self.from_patch_glob('0001-c-Set-CALL_FROM_NEW_OR') |
| assert len(email.errors) == 1 |
| assert email.errors[0].message == 'missing description of a change' |
| |
| def test_emptry_entry_desc_2(self): |
| email = self.from_patch_glob('0001-lto-fix-LTO-debug') |
| assert not email.errors |
| assert len(email.changelog_entries) == 1 |
| |
| def test_wildcard_in_subdir(self): |
| email = self.from_patch_glob('0001-Wildcard-subdirs.patch') |
| assert len(email.changelog_entries) == 1 |
| err = email.errors[0] |
| assert err.message == "pattern doesn't match any changed files" |
| assert err.line == 'libstdc++-v3/testsuite/28_regex_not-existing/' |
| |
| def test_unicode_chars_in_filename(self): |
| email = self.from_patch_glob('0001-Add-horse.patch') |
| assert not email.errors |
| |
| def test_bad_unicode_chars_in_filename(self): |
| email = self.from_patch_glob('0001-Add-horse2.patch') |
| assert not email.errors |
| assert email.changelog_entries[0].files == ['koníček.txt'] |
| |
| def test_modification_of_old_changelog(self): |
| email = self.from_patch_glob('0001-fix-old-ChangeLog.patch') |
| assert not email.errors |
| |
| def test_multiline_parentheses(self): |
| email = self.from_patch_glob('0001-Add-macro.patch') |
| assert not email.errors |
| |
| def test_multiline_bad_parentheses(self): |
| email = self.from_patch_glob('0002-Wrong-macro-changelog.patch') |
| assert email.errors[0].message == 'bad parentheses wrapping' |
| |
| def test_changelog_removal(self): |
| email = self.from_patch_glob('0001-ChangeLog-removal.patch') |
| assert not email.errors |
| |
| def test_long_filenames(self): |
| email = self.from_patch_glob('0001-long-filenames') |
| assert not email.errors |
| |
| def test_multi_same_file(self): |
| email = self.from_patch_glob('0001-OpenMP-Fix-SIMT') |
| assert email.errors[0].message == 'same file specified multiple times' |
| |
| def test_pr_only_in_subject(self): |
| email = self.from_patch_glob('0001-rs6000-Support-doubleword') |
| assert (email.errors[0].message == |
| 'PR 100085 in subject but not in changelog') |
| |
| def test_wrong_pr_comp_in_subject(self): |
| email = self.from_patch_glob('pr-wrong-comp.patch') |
| assert email.errors[0].message == 'invalid PR component in subject' |
| |
| def test_copyright_years(self): |
| email = self.from_patch_glob('copyright-years.patch') |
| assert not email.errors |
| |
| def test_non_ascii_email(self): |
| email = self.from_patch_glob('non-ascii-email.patch') |
| assert (email.errors[0].message == |
| 'non-ASCII characters in git commit email address (jbglaw@ług-owl.de)') |