xref: /openbmc/u-boot/tools/patman/func_test.py (revision 1f4e25780a827de9526b5f60b8a574b1e4f45b9c)
1# -*- coding: utf-8 -*-
2#
3# Copyright 2017 Google, Inc
4#
5# SPDX-License-Identifier:	GPL-2.0+
6#
7
8import contextlib
9import os
10import re
11import shutil
12import sys
13import tempfile
14import unittest
15
16import gitutil
17import patchstream
18import settings
19
20
21@contextlib.contextmanager
22def capture():
23    import sys
24    from cStringIO import StringIO
25    oldout,olderr = sys.stdout, sys.stderr
26    try:
27        out=[StringIO(), StringIO()]
28        sys.stdout,sys.stderr = out
29        yield out
30    finally:
31        sys.stdout,sys.stderr = oldout, olderr
32        out[0] = out[0].getvalue()
33        out[1] = out[1].getvalue()
34
35
36class TestFunctional(unittest.TestCase):
37    def setUp(self):
38        self.tmpdir = tempfile.mkdtemp(prefix='patman.')
39
40    def tearDown(self):
41        shutil.rmtree(self.tmpdir)
42
43    @staticmethod
44    def GetPath(fname):
45        return os.path.join(os.path.dirname(os.path.realpath(sys.argv[0])),
46                            'test', fname)
47
48    @classmethod
49    def GetText(self, fname):
50        return open(self.GetPath(fname)).read()
51
52    @classmethod
53    def GetPatchName(self, subject):
54        fname = re.sub('[ :]', '-', subject)
55        return fname.replace('--', '-')
56
57    def CreatePatchesForTest(self, series):
58        cover_fname = None
59        fname_list = []
60        for i, commit in enumerate(series.commits):
61            clean_subject = self.GetPatchName(commit.subject)
62            src_fname = '%04d-%s.patch' % (i + 1, clean_subject[:52])
63            fname = os.path.join(self.tmpdir, src_fname)
64            shutil.copy(self.GetPath(src_fname), fname)
65            fname_list.append(fname)
66        if series.get('cover'):
67            src_fname = '0000-cover-letter.patch'
68            cover_fname = os.path.join(self.tmpdir, src_fname)
69            fname = os.path.join(self.tmpdir, src_fname)
70            shutil.copy(self.GetPath(src_fname), fname)
71
72        return cover_fname, fname_list
73
74    def testBasic(self):
75        """Tests the basic flow of patman
76
77        This creates a series from some hard-coded patches build from a simple
78        tree with the following metadata in the top commit:
79
80            Series-to: u-boot
81            Series-prefix: RFC
82            Series-cc: Stefan Brüns <stefan.bruens@rwth-aachen.de>
83            Cover-letter-cc: Lord Mëlchett <clergy@palace.gov>
84            Series-version: 2
85            Series-changes: 4
86            - Some changes
87
88            Cover-letter:
89            test: A test patch series
90            This is a test of how the cover
91            leter
92            works
93            END
94
95        and this in the first commit:
96
97            Series-notes:
98            some notes
99            about some things
100            from the first commit
101            END
102
103            Commit-notes:
104            Some notes about
105            the first commit
106            END
107
108        with the following commands:
109
110           git log -n2 --reverse >/path/to/tools/patman/test/test01.txt
111           git format-patch --subject-prefix RFC --cover-letter HEAD~2
112           mv 00* /path/to/tools/patman/test
113
114        It checks these aspects:
115            - git log can be processed by patchstream
116            - emailing patches uses the correct command
117            - CC file has information on each commit
118            - cover letter has the expected text and subject
119            - each patch has the correct subject
120            - dry-run information prints out correctly
121            - unicode is handled correctly
122            - Series-to, Series-cc, Series-prefix, Cover-letter
123            - Cover-letter-cc, Series-version, Series-changes, Series-notes
124            - Commit-notes
125        """
126        process_tags = True
127        ignore_bad_tags = True
128        stefan = u'Stefan Brüns <stefan.bruens@rwth-aachen.de>'
129        rick = 'Richard III <richard@palace.gov>'
130        mel = u'Lord Mëlchett <clergy@palace.gov>'
131        ed = u'Lond Edmund Blackaddër <weasel@blackadder.org'
132        fred = 'Fred Bloggs <f.bloggs@napier.net>'
133        add_maintainers = [stefan, rick]
134        dry_run = True
135        in_reply_to = mel
136        count = 2
137        settings.alias = {
138                'fdt': ['simon'],
139                'u-boot': ['u-boot@lists.denx.de'],
140                'simon': [ed],
141                'fred': [fred],
142        }
143
144        text = self.GetText('test01.txt')
145        series = patchstream.GetMetaDataForTest(text)
146        cover_fname, args = self.CreatePatchesForTest(series)
147        with capture() as out:
148            patchstream.FixPatches(series, args)
149            if cover_fname and series.get('cover'):
150                patchstream.InsertCoverLetter(cover_fname, series, count)
151            series.DoChecks()
152            cc_file = series.MakeCcFile(process_tags, cover_fname,
153                                        not ignore_bad_tags, add_maintainers)
154            cmd = gitutil.EmailPatches(series, cover_fname, args,
155                    dry_run, not ignore_bad_tags, cc_file,
156                    in_reply_to=in_reply_to, thread=None)
157            series.ShowActions(args, cmd, process_tags)
158        cc_lines = open(cc_file).read().splitlines()
159        os.remove(cc_file)
160
161        lines = out[0].splitlines()
162        #print '\n'.join(lines)
163        self.assertEqual('Cleaned %s patches' % len(series.commits), lines[0])
164        self.assertEqual('Change log missing for v2', lines[1])
165        self.assertEqual('Change log missing for v3', lines[2])
166        self.assertEqual('Change log for unknown version v4', lines[3])
167        self.assertEqual("Alias 'pci' not found", lines[4])
168        self.assertIn('Dry run', lines[5])
169        self.assertIn('Send a total of %d patches' % count, lines[7])
170        line = 8
171        for i, commit in enumerate(series.commits):
172            self.assertEqual('   %s' % args[i], lines[line + 0])
173            line += 1
174            while 'Cc:' in lines[line]:
175                line += 1
176        self.assertEqual('To:	  u-boot@lists.denx.de', lines[line])
177        self.assertEqual('Cc:	  %s' % stefan.encode('utf-8'), lines[line + 1])
178        self.assertEqual('Version:  3', lines[line + 2])
179        self.assertEqual('Prefix:\t  RFC', lines[line + 3])
180        self.assertEqual('Cover: 4 lines', lines[line + 4])
181        line += 5
182        self.assertEqual('      Cc:  %s' % mel.encode('utf-8'), lines[line + 0])
183        self.assertEqual('      Cc:  %s' % rick, lines[line + 1])
184        self.assertEqual('      Cc:  %s' % fred, lines[line + 2])
185        self.assertEqual('      Cc:  %s' % ed.encode('utf-8'), lines[line + 3])
186        expected = ('Git command: git send-email --annotate '
187                    '--in-reply-to="%s" --to "u-boot@lists.denx.de" '
188                    '--cc "%s" --cc-cmd "%s --cc-cmd %s" %s %s'
189                    % (in_reply_to, stefan, sys.argv[0], cc_file, cover_fname,
190                       ' '.join(args))).encode('utf-8')
191        line += 4
192        self.assertEqual(expected, lines[line])
193
194        self.assertEqual(('%s %s, %s' % (args[0], rick, stefan))
195                         .encode('utf-8'), cc_lines[0])
196        self.assertEqual(('%s %s, %s, %s, %s' % (args[1], fred, rick, stefan,
197                                            ed)).encode('utf-8'), cc_lines[1])
198
199        expected = '''
200This is a test of how the cover
201leter
202works
203
204some notes
205about some things
206from the first commit
207
208Changes in v4:
209- Some changes
210
211Simon Glass (2):
212  pci: Correct cast for sandbox
213  fdt: Correct cast for sandbox in fdtdec_setup_memory_size()
214
215 cmd/pci.c                   | 3 ++-
216 fs/fat/fat.c                | 1 +
217 lib/efi_loader/efi_memory.c | 1 +
218 lib/fdtdec.c                | 3 ++-
219 4 files changed, 6 insertions(+), 2 deletions(-)
220
221--\x20
2222.7.4
223
224'''
225        lines = open(cover_fname).read().splitlines()
226        #print '\n'.join(lines)
227        self.assertEqual(
228                'Subject: [RFC PATCH v3 0/2] test: A test patch series',
229                lines[3])
230        self.assertEqual(expected.splitlines(), lines[7:])
231
232        for i, fname in enumerate(args):
233            lines = open(fname).read().splitlines()
234            #print '\n'.join(lines)
235            subject = [line for line in lines if line.startswith('Subject')]
236            self.assertEqual('Subject: [RFC %d/%d]' % (i + 1, count),
237                             subject[0][:18])
238            if i == 0:
239                # Check that we got our commit notes
240                self.assertEqual('---', lines[17])
241                self.assertEqual('Some notes about', lines[18])
242                self.assertEqual('the first commit', lines[19])
243