~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/selftest/plugins.py

- increment version

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2007 Canonical Ltd
2
 
#
 
1
# Copyright (C) 2005 by Canonical Ltd
 
2
 
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
5
5
# the Free Software Foundation; either version 2 of the License, or
6
6
# (at your option) any later version.
7
 
#
 
7
 
8
8
# This program is distributed in the hope that it will be useful,
9
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
11
# GNU General Public License for more details.
12
 
#
 
12
 
13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
16
 
 
17
 
17
18
"""Tests for plugins"""
18
19
 
19
 
# XXX: There are no plugin tests at the moment because the plugin module
20
 
# affects the global state of the process.  See bzrlib/plugins.py for more
21
 
# comments.
22
 
 
23
 
import os
24
 
from StringIO import StringIO
25
 
import sys
26
 
import zipfile
27
 
 
28
 
from bzrlib import plugin, tests
29
 
import bzrlib.plugin
30
 
import bzrlib.plugins
31
 
import bzrlib.commands
32
 
import bzrlib.help
33
 
from bzrlib.tests import TestCase, TestCaseInTempDir
34
 
from bzrlib.osutils import pathjoin, abspath
35
 
 
36
 
 
37
 
PLUGIN_TEXT = """\
38
 
import bzrlib.commands
39
 
class cmd_myplug(bzrlib.commands.Command):
40
 
    '''Just a simple test plugin.'''
41
 
    aliases = ['mplg']
42
 
    def run(self):
43
 
        print 'Hello from my plugin'
44
 
"""
45
 
 
46
 
# TODO: Write a test for plugin decoration of commands.
47
 
 
48
 
class TestLoadingPlugins(TestCaseInTempDir):
49
 
 
50
 
    activeattributes = {}
51
 
 
52
 
    def test_plugins_with_the_same_name_are_not_loaded(self):
53
 
        # This test tests that having two plugins in different directories does
54
 
        # not result in both being loaded when they have the same name.  get a
55
 
        # file name we can use which is also a valid attribute for accessing in
56
 
        # activeattributes. - we cannot give import parameters.
57
 
        tempattribute = "0"
58
 
        self.failIf(tempattribute in self.activeattributes)
59
 
        # set a place for the plugins to record their loading, and at the same
60
 
        # time validate that the location the plugins should record to is
61
 
        # valid and correct.
62
 
        bzrlib.tests.test_plugins.TestLoadingPlugins.activeattributes \
63
 
            [tempattribute] = []
64
 
        self.failUnless(tempattribute in self.activeattributes)
65
 
        # create two plugin directories
66
 
        os.mkdir('first')
67
 
        os.mkdir('second')
68
 
        # write a plugin that will record when its loaded in the 
69
 
        # tempattribute list.
70
 
        template = ("from bzrlib.tests.test_plugins import TestLoadingPlugins\n"
71
 
                    "TestLoadingPlugins.activeattributes[%r].append('%s')\n")
72
 
        print >> file(os.path.join('first', 'plugin.py'), 'w'), template % (tempattribute, 'first')
73
 
        print >> file(os.path.join('second', 'plugin.py'), 'w'), template % (tempattribute, 'second')
74
 
        try:
75
 
            bzrlib.plugin.load_from_path(['first', 'second'])
76
 
            self.assertEqual(['first'], self.activeattributes[tempattribute])
77
 
        finally:
78
 
            # remove the plugin 'plugin'
79
 
            del self.activeattributes[tempattribute]
80
 
            if getattr(bzrlib.plugins, 'plugin', None):
81
 
                del bzrlib.plugins.plugin
82
 
        self.failIf(getattr(bzrlib.plugins, 'plugin', None))
83
 
 
84
 
    def test_plugins_from_different_dirs_can_demand_load(self):
85
 
        # This test tests that having two plugins in different
86
 
        # directories with different names allows them both to be loaded, when
87
 
        # we do a direct import statement.
88
 
        # Determine a file name we can use which is also a valid attribute
89
 
        # for accessing in activeattributes. - we cannot give import parameters.
90
 
        tempattribute = "different-dirs"
91
 
        self.failIf(tempattribute in self.activeattributes)
92
 
        # set a place for the plugins to record their loading, and at the same
93
 
        # time validate that the location the plugins should record to is
94
 
        # valid and correct.
95
 
        bzrlib.tests.test_plugins.TestLoadingPlugins.activeattributes \
96
 
            [tempattribute] = []
97
 
        self.failUnless(tempattribute in self.activeattributes)
98
 
        # create two plugin directories
99
 
        os.mkdir('first')
100
 
        os.mkdir('second')
101
 
        # write plugins that will record when they are loaded in the 
102
 
        # tempattribute list.
103
 
        template = ("from bzrlib.tests.test_plugins import TestLoadingPlugins\n"
104
 
                    "TestLoadingPlugins.activeattributes[%r].append('%s')\n")
105
 
        print >> file(os.path.join('first', 'pluginone.py'), 'w'), template % (tempattribute, 'first')
106
 
        print >> file(os.path.join('second', 'plugintwo.py'), 'w'), template % (tempattribute, 'second')
107
 
        oldpath = bzrlib.plugins.__path__
108
 
        try:
109
 
            bzrlib.plugins.__path__ = ['first', 'second']
110
 
            exec "import bzrlib.plugins.pluginone"
111
 
            self.assertEqual(['first'], self.activeattributes[tempattribute])
112
 
            exec "import bzrlib.plugins.plugintwo"
113
 
            self.assertEqual(['first', 'second'],
114
 
                self.activeattributes[tempattribute])
115
 
        finally:
116
 
            # remove the plugin 'plugin'
117
 
            del self.activeattributes[tempattribute]
118
 
            if getattr(bzrlib.plugins, 'plugin', None):
119
 
                del bzrlib.plugins.plugin
120
 
        self.failIf(getattr(bzrlib.plugins, 'plugin', None))
121
 
 
122
 
 
123
 
class TestAllPlugins(TestCaseInTempDir):
124
 
 
125
 
    def test_plugin_appears_in_all_plugins(self):
126
 
        # This test tests a new plugin appears in bzrlib.plugin.all_plugins().
127
 
        # check the plugin is not loaded already
128
 
        self.failIf(getattr(bzrlib.plugins, 'plugin', None))
129
 
        # write a plugin that _cannot_ fail to load.
130
 
        print >> file('plugin.py', 'w'), ""
131
 
        try:
132
 
            bzrlib.plugin.load_from_path(['.'])
133
 
            self.failUnless('plugin' in bzrlib.plugin.all_plugins())
134
 
            self.failUnless(getattr(bzrlib.plugins, 'plugin', None))
135
 
            self.assertEqual(bzrlib.plugin.all_plugins()['plugin'],
136
 
                             bzrlib.plugins.plugin)
137
 
        finally:
138
 
            # remove the plugin 'plugin'
139
 
            if 'bzrlib.plugins.plugin' in sys.modules:
140
 
                del sys.modules['bzrlib.plugins.plugin']
141
 
            if getattr(bzrlib.plugins, 'plugin', None):
142
 
                del bzrlib.plugins.plugin
143
 
        self.failIf(getattr(bzrlib.plugins, 'plugin', None))
144
 
 
145
 
 
146
 
class TestPluginHelp(TestCaseInTempDir):
147
 
 
148
 
    def split_help_commands(self):
149
 
        help = {}
150
 
        current = None
151
 
        for line in self.run_bzr('help commands')[0].splitlines():
152
 
            if not line.startswith(' '):
153
 
                current = line.split()[0]
154
 
            help[current] = help.get(current, '') + line
155
 
 
156
 
        return help
157
 
 
158
 
    def test_plugin_help_builtins_unaffected(self):
159
 
        # Check we don't get false positives
160
 
        help_commands = self.split_help_commands()
161
 
        for cmd_name in bzrlib.commands.builtin_command_names():
162
 
            if cmd_name in bzrlib.commands.plugin_command_names():
163
 
                continue
164
 
            try:
165
 
                help = bzrlib.commands.get_cmd_object(cmd_name).get_help_text()
166
 
            except NotImplementedError:
167
 
                # some commands have no help
168
 
                pass
169
 
            else:
170
 
                self.assertNotContainsRe(help, 'From plugin "[^"]*"')
171
 
 
172
 
            if cmd_name in help_commands.keys():
173
 
                # some commands are hidden
174
 
                help = help_commands[cmd_name]
175
 
                self.assertNotContainsRe(help, 'From plugin "[^"]*"')
176
 
 
177
 
    def test_plugin_help_shows_plugin(self):
178
 
        # Create a test plugin
 
20
 
 
21
 
 
22
# **************************************************
 
23
# NOT RUN YET
 
24
# **************************************************
 
25
 
 
26
 
 
27
 
 
28
 
 
29
 
 
30
 
 
31
 
 
32
 
 
33
from bzrlib.selftest import TestCaseInTempDir
 
34
 
 
35
 
 
36
class PluginTest(TestCaseInTempDir):
 
37
    """Create an external plugin and test loading."""
 
38
    def test_plugin_loading(self):
 
39
        import os
 
40
        
 
41
        orig_help = self.backtick('bzr help commands') # No plugins yet
179
42
        os.mkdir('plugin_test')
180
 
        f = open(pathjoin('plugin_test', 'myplug.py'), 'w')
 
43
        f = open(os.path.join('plugin_test', 'myplug.py'), 'wt')
181
44
        f.write(PLUGIN_TEXT)
182
45
        f.close()
183
46
 
184
 
        try:
185
 
            # Check its help
186
 
            bzrlib.plugin.load_from_path(['plugin_test'])
187
 
            bzrlib.commands.register_command( bzrlib.plugins.myplug.cmd_myplug)
188
 
            help = self.run_bzr('help myplug')[0]
189
 
            self.assertContainsRe(help, 'From plugin "myplug"')
190
 
            help = self.split_help_commands()['myplug']
191
 
            self.assertContainsRe(help, '\[myplug\]')
192
 
        finally:
193
 
            # unregister command
194
 
            if bzrlib.commands.plugin_cmds.get('myplug', None):
195
 
                del bzrlib.commands.plugin_cmds['myplug']
196
 
            # remove the plugin 'myplug'
197
 
            if getattr(bzrlib.plugins, 'myplug', None):
198
 
                delattr(bzrlib.plugins, 'myplug')
199
 
 
200
 
 
201
 
class TestPluginFromZip(TestCaseInTempDir):
202
 
 
203
 
    def make_zipped_plugin(self, zip_name, filename):
204
 
        z = zipfile.ZipFile(zip_name, 'w')
205
 
        z.writestr(filename, PLUGIN_TEXT)
206
 
        z.close()
207
 
 
208
 
    def check_plugin_load(self, zip_name, plugin_name):
209
 
        self.assertFalse(plugin_name in dir(bzrlib.plugins),
210
 
                         'Plugin already loaded')
211
 
        old_path = bzrlib.plugins.__path__
212
 
        try:
213
 
            # this is normally done by load_plugins -> set_plugins_path
214
 
            bzrlib.plugins.__path__ = [zip_name]
215
 
            bzrlib.plugin.load_from_zip(zip_name)
216
 
            self.assertTrue(plugin_name in dir(bzrlib.plugins),
217
 
                            'Plugin is not loaded')
218
 
        finally:
219
 
            # unregister plugin
220
 
            if getattr(bzrlib.plugins, plugin_name, None):
221
 
                delattr(bzrlib.plugins, plugin_name)
222
 
                del sys.modules['bzrlib.plugins.' + plugin_name]
223
 
            bzrlib.plugins.__path__ = old_path
224
 
 
225
 
    def test_load_module(self):
226
 
        self.make_zipped_plugin('./test.zip', 'ziplug.py')
227
 
        self.check_plugin_load('./test.zip', 'ziplug')
228
 
 
229
 
    def test_load_package(self):
230
 
        self.make_zipped_plugin('./test.zip', 'ziplug/__init__.py')
231
 
        self.check_plugin_load('./test.zip', 'ziplug')
232
 
 
233
 
 
234
 
class TestSetPluginsPath(TestCase):
235
 
    
236
 
    def test_set_plugins_path(self):
237
 
        """set_plugins_path should set the module __path__ correctly."""
238
 
        old_path = bzrlib.plugins.__path__
239
 
        try:
240
 
            bzrlib.plugins.__path__ = []
241
 
            expected_path = bzrlib.plugin.set_plugins_path()
242
 
            self.assertEqual(expected_path, bzrlib.plugins.__path__)
243
 
        finally:
244
 
            bzrlib.plugins.__path__ = old_path
245
 
 
246
 
 
247
 
class TestHelpIndex(tests.TestCase):
248
 
    """Tests for the PluginsHelpIndex class."""
249
 
 
250
 
    def test_default_constructable(self):
251
 
        index = plugin.PluginsHelpIndex()
252
 
 
253
 
    def test_get_topics_None(self):
254
 
        """Searching for None returns an empty list."""
255
 
        index = plugin.PluginsHelpIndex()
256
 
        self.assertEqual([], index.get_topics(None))
257
 
 
258
 
    def test_get_topics_for_plugin(self):
259
 
        """Searching for plugin name gets its docstring."""
260
 
        index = plugin.PluginsHelpIndex()
261
 
        # make a new plugin here for this test, even if we're run with
262
 
        # --no-plugins
263
 
        self.assertFalse(sys.modules.has_key('bzrlib.plugins.demo_module'))
264
 
        demo_module = FakeModule('', 'bzrlib.plugins.demo_module')
265
 
        sys.modules['bzrlib.plugins.demo_module'] = demo_module
266
 
        try:
267
 
            topics = index.get_topics('demo_module')
268
 
            self.assertEqual(1, len(topics))
269
 
            self.assertIsInstance(topics[0], plugin.ModuleHelpTopic)
270
 
            self.assertEqual(demo_module, topics[0].module)
271
 
        finally:
272
 
            del sys.modules['bzrlib.plugins.demo_module']
273
 
 
274
 
    def test_get_topics_no_topic(self):
275
 
        """Searching for something that is not a plugin returns []."""
276
 
        # test this by using a name that cannot be a plugin - its not
277
 
        # a valid python identifier.
278
 
        index = plugin.PluginsHelpIndex()
279
 
        self.assertEqual([], index.get_topics('nothing by this name'))
280
 
 
281
 
    def test_prefix(self):
282
 
        """PluginsHelpIndex has a prefix of 'plugins/'."""
283
 
        index = plugin.PluginsHelpIndex()
284
 
        self.assertEqual('plugins/', index.prefix)
285
 
 
286
 
    def test_get_plugin_topic_with_prefix(self):
287
 
        """Searching for plugins/demo_module returns help."""
288
 
        index = plugin.PluginsHelpIndex()
289
 
        self.assertFalse(sys.modules.has_key('bzrlib.plugins.demo_module'))
290
 
        demo_module = FakeModule('', 'bzrlib.plugins.demo_module')
291
 
        sys.modules['bzrlib.plugins.demo_module'] = demo_module
292
 
        try:
293
 
            topics = index.get_topics('plugins/demo_module')
294
 
            self.assertEqual(1, len(topics))
295
 
            self.assertIsInstance(topics[0], plugin.ModuleHelpTopic)
296
 
            self.assertEqual(demo_module, topics[0].module)
297
 
        finally:
298
 
            del sys.modules['bzrlib.plugins.demo_module']
299
 
 
300
 
 
301
 
class FakeModule(object):
302
 
    """A fake module to test with."""
303
 
 
304
 
    def __init__(self, doc, name):
305
 
        self.__doc__ = doc
306
 
        self.__name__ = name
307
 
 
308
 
 
309
 
class TestModuleHelpTopic(tests.TestCase):
310
 
    """Tests for the ModuleHelpTopic class."""
311
 
 
312
 
    def test_contruct(self):
313
 
        """Construction takes the module to document."""
314
 
        mod = FakeModule('foo', 'foo')
315
 
        topic = plugin.ModuleHelpTopic(mod)
316
 
        self.assertEqual(mod, topic.module)
317
 
 
318
 
    def test_get_help_text_None(self):
319
 
        """A ModuleHelpTopic returns the docstring for get_help_text."""
320
 
        mod = FakeModule(None, 'demo')
321
 
        topic = plugin.ModuleHelpTopic(mod)
322
 
        self.assertEqual("Plugin 'demo' has no docstring.\n",
323
 
            topic.get_help_text())
324
 
 
325
 
    def test_get_help_text_no_carriage_return(self):
326
 
        """ModuleHelpTopic.get_help_text adds a \n if needed."""
327
 
        mod = FakeModule('one line of help', 'demo')
328
 
        topic = plugin.ModuleHelpTopic(mod)
329
 
        self.assertEqual("one line of help\n",
330
 
            topic.get_help_text())
331
 
 
332
 
    def test_get_help_text_carriage_return(self):
333
 
        """ModuleHelpTopic.get_help_text adds a \n if needed."""
334
 
        mod = FakeModule('two lines of help\nand more\n', 'demo')
335
 
        topic = plugin.ModuleHelpTopic(mod)
336
 
        self.assertEqual("two lines of help\nand more\n",
337
 
            topic.get_help_text())
338
 
 
339
 
    def test_get_help_text_with_additional_see_also(self):
340
 
        mod = FakeModule('two lines of help\nand more', 'demo')
341
 
        topic = plugin.ModuleHelpTopic(mod)
342
 
        self.assertEqual("two lines of help\nand more\nSee also: bar, foo\n",
343
 
            topic.get_help_text(['foo', 'bar']))
344
 
 
345
 
    def test_get_help_topic(self):
346
 
        """The help topic for a plugin is its module name."""
347
 
        mod = FakeModule('two lines of help\nand more', 'bzrlib.plugins.demo')
348
 
        topic = plugin.ModuleHelpTopic(mod)
349
 
        self.assertEqual('demo', topic.get_help_topic())
350
 
        mod = FakeModule('two lines of help\nand more', 'bzrlib.plugins.foo_bar')
351
 
        topic = plugin.ModuleHelpTopic(mod)
352
 
        self.assertEqual('foo_bar', topic.get_help_topic())
 
47
        newhelp = backtick('bzr help commands')
 
48
        assert newhelp.startswith('You have been overridden\n')
 
49
        # We added a line, but the rest should work
 
50
        assert newhelp[25:] == help
 
51
 
 
52
        assert backtick('bzr commit -m test') == "I'm sorry dave, you can't do that\n"
 
53
 
 
54
        shutil.rmtree('plugin_test')
 
55
 
 
56
 
 
57
 
 
58
 
 
59
#         PLUGIN_TEXT = \
 
60
#         """import bzrlib, bzrlib.commands
 
61
#         class cmd_myplug(bzrlib.commands.Command):
 
62
#             '''Just a simple test plugin.'''
 
63
#             aliases = ['mplg']
 
64
#             def run(self):
 
65
#                 print 'Hello from my plugin'
 
66
#         """
 
67
#         f.close()
 
68
 
 
69
#         os.environ['BZRPLUGINPATH'] = os.path.abspath('plugin_test')
 
70
#         help = backtick('bzr help commands')
 
71
#         assert help.find('myplug') != -1
 
72
#         assert help.find('Just a simple test plugin.') != -1
 
73
 
 
74
 
 
75
#         assert backtick('bzr myplug') == 'Hello from my plugin\n'
 
76
#         assert backtick('bzr mplg') == 'Hello from my plugin\n'
 
77
 
 
78
#         f = open(os.path.join('plugin_test', 'override.py'), 'wb')
 
79
#         f.write("""import bzrlib, bzrlib.commands
 
80
#     class cmd_commit(bzrlib.commands.cmd_commit):
 
81
#         '''Commit changes into a new revision.'''
 
82
#         def run(self, *args, **kwargs):
 
83
#             print "I'm sorry dave, you can't do that"
 
84
 
 
85
#     class cmd_help(bzrlib.commands.cmd_help):
 
86
#         '''Show help on a command or other topic.'''
 
87
#         def run(self, *args, **kwargs):
 
88
#             print "You have been overridden"
 
89
#             bzrlib.commands.cmd_help.run(self, *args, **kwargs)
 
90
 
 
91
#         """