~bzr-pqm/bzr/bzr.dev

2052.3.2 by John Arbash Meinel
Change Copyright .. by Canonical to Copyright ... Canonical
1
# Copyright (C) 2004, 2005 Canonical Ltd
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
2
#
1185.12.59 by Aaron Bentley
Added command-quieting test
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
7
#
1185.12.59 by Aaron Bentley
Added command-quieting test
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
12
#
1185.12.59 by Aaron Bentley
Added command-quieting test
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
4183.7.1 by Sabin Iacob
update FSF mailing address
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
1185.12.59 by Aaron Bentley
Added command-quieting test
16
2155.2.3 by Marius Kruger
* commands.py
17
from cStringIO import StringIO
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
18
import errno
3785.1.4 by Aaron Bentley
Enable lazy-loading of commands
19
import sys
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
20
21
from bzrlib import (
4000.1.1 by Robert Collins
Add a new hook Commands['extend_command'] for plugins that want to alter commands without overriding the entire command.
22
    builtins,
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
23
    commands,
2155.2.1 by Marius Kruger
* Get command aliases to respect quoted arguments.
24
    config,
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
25
    errors,
4251.1.1 by Aaron Bentley
Support hidden options.
26
    option,
2425.2.1 by Robert Collins
Command objects can now declare related help topics by having _see_also
27
    tests,
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
28
    )
29
from bzrlib.commands import display_command
2425.2.1 by Robert Collins
Command objects can now declare related help topics by having _see_also
30
from bzrlib.tests import TestSkipped
31
32
33
class TestCommands(tests.TestCase):
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
34
1185.12.59 by Aaron Bentley
Added command-quieting test
35
    def test_display_command(self):
1185.33.18 by Martin Pool
[patch] handle bad IOError subclass raised by urlopen
36
        """EPIPE message is selectively suppressed"""
1185.12.59 by Aaron Bentley
Added command-quieting test
37
        def pipe_thrower():
38
            raise IOError(errno.EPIPE, "Bogus pipe error")
39
        self.assertRaises(IOError, pipe_thrower)
40
        @display_command
41
        def non_thrower():
42
            pipe_thrower()
43
        non_thrower()
44
        @display_command
45
        def other_thrower():
46
            raise IOError(errno.ESPIPE, "Bogus pipe error")
47
        self.assertRaises(IOError, other_thrower)
48
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
49
    def test_unicode_command(self):
1948.1.8 by John Arbash Meinel
Don't raise UnicodeCommand on request, instead just let it fall out when we get to NoSuchCommand
50
        # This error is thrown when we can't find the command in the
51
        # list of available commands
52
        self.assertRaises(errors.BzrCommandError,
1948.1.1 by John Arbash Meinel
Raise a reasonable error when a command is non-ascii
53
                          commands.run_bzr, [u'cmd\xb5'])
1948.1.8 by John Arbash Meinel
Don't raise UnicodeCommand on request, instead just let it fall out when we get to NoSuchCommand
54
55
    def test_unicode_option(self):
56
        # This error is actually thrown by optparse, when it
57
        # can't find the given option
1913.2.4 by Martin Pool
python2.5 apparently has trouble with unicode options
58
        import optparse
59
        if optparse.__version__ == "1.5.3":
60
            raise TestSkipped("optparse 1.5.3 can't handle unicode options")
1948.1.8 by John Arbash Meinel
Don't raise UnicodeCommand on request, instead just let it fall out when we get to NoSuchCommand
61
        self.assertRaises(errors.BzrCommandError,
62
                          commands.run_bzr, ['log', u'--option\xb5'])
63
4251.1.1 by Aaron Bentley
Support hidden options.
64
    @staticmethod
65
    def get_command(options):
4251.1.7 by Aaron Bentley
Use generated subclass instead of patching Command instance
66
        class cmd_foo(commands.Command):
67
            'Bar'
68
69
            takes_options = options
70
71
        return cmd_foo()
4251.1.1 by Aaron Bentley
Support hidden options.
72
73
    def test_help_hidden(self):
74
        c = self.get_command([option.Option('foo', hidden=True)])
75
        self.assertNotContainsRe(c.get_help_text(), '--foo')
76
77
    def test_help_not_hidden(self):
78
        c = self.get_command([option.Option('foo', hidden=False)])
79
        self.assertContainsRe(c.get_help_text(), '--foo')
80
2155.2.3 by Marius Kruger
* commands.py
81
2425.2.1 by Robert Collins
Command objects can now declare related help topics by having _see_also
82
class TestGetAlias(tests.TestCase):
2155.2.3 by Marius Kruger
* commands.py
83
84
    def _get_config(self, config_text):
2155.2.1 by Marius Kruger
* Get command aliases to respect quoted arguments.
85
        my_config = config.GlobalConfig()
86
        config_file = StringIO(config_text.encode('utf-8'))
87
        my_config._parser = my_config._get_parser(file=config_file)
88
        return my_config
89
90
    def test_simple(self):
2155.2.3 by Marius Kruger
* commands.py
91
        my_config = self._get_config("[ALIASES]\n"
2155.2.1 by Marius Kruger
* Get command aliases to respect quoted arguments.
92
            "diff=diff -r -2..-1\n")
93
        self.assertEqual([u'diff', u'-r', u'-2..-1'],
94
            commands.get_alias("diff", config=my_config))
95
96
    def test_single_quotes(self):
2155.2.3 by Marius Kruger
* commands.py
97
        my_config = self._get_config("[ALIASES]\n"
2155.2.1 by Marius Kruger
* Get command aliases to respect quoted arguments.
98
            "diff=diff -r -2..-1 --diff-options "
99
            "'--strip-trailing-cr -wp'\n")
100
        self.assertEqual([u'diff', u'-r', u'-2..-1', u'--diff-options',
101
                          u'--strip-trailing-cr -wp'],
102
                          commands.get_alias("diff", config=my_config))
103
104
    def test_double_quotes(self):
2155.2.3 by Marius Kruger
* commands.py
105
        my_config = self._get_config("[ALIASES]\n"
2155.2.1 by Marius Kruger
* Get command aliases to respect quoted arguments.
106
            "diff=diff -r -2..-1 --diff-options "
107
            "\"--strip-trailing-cr -wp\"\n")
108
        self.assertEqual([u'diff', u'-r', u'-2..-1', u'--diff-options',
109
                          u'--strip-trailing-cr -wp'],
110
                          commands.get_alias("diff", config=my_config))
111
112
    def test_unicode(self):
2155.2.3 by Marius Kruger
* commands.py
113
        my_config = self._get_config("[ALIASES]\n"
2155.2.1 by Marius Kruger
* Get command aliases to respect quoted arguments.
114
            u"iam=whoami 'Erik B\u00e5gfors <erik@bagfors.nu>'\n")
115
        self.assertEqual([u'whoami', u'Erik B\u00e5gfors <erik@bagfors.nu>'],
116
                          commands.get_alias("iam", config=my_config))
2425.2.1 by Robert Collins
Command objects can now declare related help topics by having _see_also
117
118
119
class TestSeeAlso(tests.TestCase):
120
    """Tests for the see also functional of Command."""
121
122
    def test_default_subclass_no_see_also(self):
123
        class ACommand(commands.Command):
124
            """A sample command."""
125
        command = ACommand()
126
        self.assertEqual([], command.get_see_also())
127
128
    def test__see_also(self):
129
        """When _see_also is defined, it sets the result of get_see_also()."""
130
        class ACommand(commands.Command):
131
            _see_also = ['bar', 'foo']
132
        command = ACommand()
133
        self.assertEqual(['bar', 'foo'], command.get_see_also())
134
135
    def test_deduplication(self):
136
        """Duplicates in _see_also are stripped out."""
137
        class ACommand(commands.Command):
138
            _see_also = ['foo', 'foo']
139
        command = ACommand()
140
        self.assertEqual(['foo'], command.get_see_also())
141
142
    def test_sorted(self):
143
        """_see_also is sorted by get_see_also."""
144
        class ACommand(commands.Command):
145
            _see_also = ['foo', 'bar']
146
        command = ACommand()
147
        self.assertEqual(['bar', 'foo'], command.get_see_also())
2432.1.21 by Robert Collins
Teach Command.get_help_text to show additional help cross references when supplied.
148
149
    def test_additional_terms(self):
150
        """Additional terms can be supplied and are deduped and sorted."""
151
        class ACommand(commands.Command):
152
            _see_also = ['foo', 'bar']
153
        command = ACommand()
154
        self.assertEqual(['bar', 'foo', 'gam'],
155
            command.get_see_also(['gam', 'bar', 'gam']))
156
3785.1.4 by Aaron Bentley
Enable lazy-loading of commands
157
158
class TestRegisterLazy(tests.TestCase):
159
160
    def setUp(self):
4153.1.2 by Andrew Bennetts
Add missing TestCase.setUp upcalls.
161
        tests.TestCase.setUp(self)
3785.1.4 by Aaron Bentley
Enable lazy-loading of commands
162
        import bzrlib.tests.fake_command
163
        del sys.modules['bzrlib.tests.fake_command']
164
        global lazy_command_imported
165
        lazy_command_imported = False
166
167
    @staticmethod
168
    def remove_fake():
169
        commands.plugin_cmds.remove('fake')
170
171
    def assertIsFakeCommand(self, cmd_obj):
172
        from bzrlib.tests.fake_command import cmd_fake
173
        self.assertIsInstance(cmd_obj, cmd_fake)
174
175
    def test_register_lazy(self):
176
        """Ensure lazy registration works"""
177
        commands.plugin_cmds.register_lazy('cmd_fake', [],
178
                                           'bzrlib.tests.fake_command')
179
        self.addCleanup(self.remove_fake)
180
        self.assertFalse(lazy_command_imported)
181
        fake_instance = commands.get_cmd_object('fake')
182
        self.assertTrue(lazy_command_imported)
183
        self.assertIsFakeCommand(fake_instance)
184
185
    def test_get_unrelated_does_not_import(self):
186
        commands.plugin_cmds.register_lazy('cmd_fake', [],
187
                                           'bzrlib.tests.fake_command')
188
        self.addCleanup(self.remove_fake)
189
        commands.get_cmd_object('status')
190
        self.assertFalse(lazy_command_imported)
191
192
    def test_aliases(self):
193
        commands.plugin_cmds.register_lazy('cmd_fake', ['fake_alias'],
194
                                           'bzrlib.tests.fake_command')
195
        self.addCleanup(self.remove_fake)
196
        fake_instance = commands.get_cmd_object('fake_alias')
197
        self.assertIsFakeCommand(fake_instance)
4000.1.1 by Robert Collins
Add a new hook Commands['extend_command'] for plugins that want to alter commands without overriding the entire command.
198
199
200
class TestExtendCommandHook(tests.TestCase):
201
202
    def test_fires_on_get_cmd_object(self):
203
        # The extend_command(cmd) hook fires when commands are delivered to the
204
        # ui, not simply at registration (because lazy registered plugin
205
        # commands are registered).
206
        # when they are simply created.
207
        hook_calls = []
208
        commands.Command.hooks.install_named_hook(
209
            "extend_command", hook_calls.append, None)
210
        # create a command, should not fire
211
        class ACommand(commands.Command):
212
            """A sample command."""
213
        cmd = ACommand()
214
        self.assertEqual([], hook_calls)
215
        # -- as a builtin
216
        # register the command class, should not fire
217
        try:
218
            builtins.cmd_test_extend_command_hook = ACommand
219
            self.assertEqual([], hook_calls)
220
            # and ask for the object, should fire
221
            cmd = commands.get_cmd_object('test-extend-command-hook')
222
            # For resilience - to ensure all code paths hit it - we
223
            # fire on everything returned in the 'cmd_dict', which is currently
224
            # all known commands, so assert that cmd is in hook_calls
225
            self.assertSubset([cmd], hook_calls)
226
            del hook_calls[:]
227
        finally:
228
            del builtins.cmd_test_extend_command_hook
229
        # -- as a plugin lazy registration
230
        try:
231
            # register the command class, should not fire
232
            commands.plugin_cmds.register_lazy('cmd_fake', [],
233
                                               'bzrlib.tests.fake_command')
234
            self.assertEqual([], hook_calls)
235
            # and ask for the object, should fire
236
            cmd = commands.get_cmd_object('fake')
237
            self.assertEqual([cmd], hook_calls)
238
        finally:
239
            commands.plugin_cmds.remove('fake')