~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_plugins.py

  • Committer: Ian Clatworthy
  • Date: 2007-08-14 03:59:22 UTC
  • mto: (2733.1.1 ianc-integration)
  • mto: This revision was merged to the branch mainline in revision 2734.
  • Revision ID: ian.clatworthy@internode.on.net-20070814035922-siavg542cwvkf4r5
Fix pretty doc generation so works for all html docs

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005 by Canonical Ltd
 
1
# Copyright (C) 2005, 2007 Canonical Ltd
2
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
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
 
 
18
17
"""Tests for plugins"""
19
18
 
20
19
# XXX: There are no plugin tests at the moment because the plugin module
23
22
 
24
23
import os
25
24
from StringIO import StringIO
 
25
import sys
 
26
import zipfile
26
27
 
 
28
from bzrlib import plugin, tests
27
29
import bzrlib.plugin
28
30
import bzrlib.plugins
29
31
import bzrlib.commands
30
32
import bzrlib.help
31
 
from bzrlib.tests import TestCaseInTempDir
 
33
from bzrlib.tests import TestCase, TestCaseInTempDir
32
34
from bzrlib.osutils import pathjoin, abspath
33
35
 
34
 
class PluginTest(TestCaseInTempDir):
35
 
    """Create an external plugin and test loading."""
36
 
#    def test_plugin_loading(self):
37
 
#        orig_help = self.run_bzr_captured('bzr help commands')[0]
38
 
#        os.mkdir('plugin_test')
39
 
#        f = open(pathjoin('plugin_test', 'myplug.py'), 'wt')
40
 
#        f.write(PLUGIN_TEXT)
41
 
#        f.close()
42
 
#        newhelp = self.run_bzr_captured('bzr help commands')[0]
43
 
#        assert newhelp.startswith('You have been overridden\n')
44
 
#        # We added a line, but the rest should work
45
 
#        assert newhelp[25:] == help
46
 
#
47
 
#        assert backtick('bzr commit -m test') == "I'm sorry dave, you can't do that\n"
48
 
#
49
 
#        shutil.rmtree('plugin_test')
50
 
#
51
 
 
52
 
#         os.environ['BZRPLUGINPATH'] = abspath('plugin_test')
53
 
#         help = backtick('bzr help commands')
54
 
#         assert help.find('myplug') != -1
55
 
#         assert help.find('Just a simple test plugin.') != -1
56
 
 
57
 
 
58
 
#         assert backtick('bzr myplug') == 'Hello from my plugin\n'
59
 
#         assert backtick('bzr mplg') == 'Hello from my plugin\n'
60
 
 
61
 
#         f = open(pathjoin('plugin_test', 'override.py'), 'wb')
62
 
#         f.write("""import bzrlib, bzrlib.commands
63
 
#     class cmd_commit(bzrlib.commands.cmd_commit):
64
 
#         '''Commit changes into a new revision.'''
65
 
#         def run(self, *args, **kwargs):
66
 
#             print "I'm sorry dave, you can't do that"
67
 
 
68
 
#     class cmd_help(bzrlib.commands.cmd_help):
69
 
#         '''Show help on a command or other topic.'''
70
 
#         def run(self, *args, **kwargs):
71
 
#             print "You have been overridden"
72
 
#             bzrlib.commands.cmd_help.run(self, *args, **kwargs)
73
 
 
74
 
#         """
75
36
 
76
37
PLUGIN_TEXT = """\
77
38
import bzrlib.commands
84
45
 
85
46
# TODO: Write a test for plugin decoration of commands.
86
47
 
87
 
class TestOneNamedPluginOnly(TestCaseInTempDir):
 
48
class TestLoadingPlugins(TestCaseInTempDir):
88
49
 
89
50
    activeattributes = {}
90
51
 
91
52
    def test_plugins_with_the_same_name_are_not_loaded(self):
92
 
        # This test tests that having two plugins in different
93
 
        # directories does not result in both being loaded.
94
 
        # get a file name we can use which is also a valid attribute
95
 
        # for accessing in activeattributes. - we cannot give import parameters.
 
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.
96
57
        tempattribute = "0"
97
58
        self.failIf(tempattribute in self.activeattributes)
98
59
        # set a place for the plugins to record their loading, and at the same
99
60
        # time validate that the location the plugins should record to is
100
61
        # valid and correct.
101
 
        bzrlib.tests.test_plugins.TestOneNamedPluginOnly.activeattributes \
 
62
        bzrlib.tests.test_plugins.TestLoadingPlugins.activeattributes \
102
63
            [tempattribute] = []
103
64
        self.failUnless(tempattribute in self.activeattributes)
104
65
        # create two plugin directories
106
67
        os.mkdir('second')
107
68
        # write a plugin that will record when its loaded in the 
108
69
        # tempattribute list.
109
 
        template = ("from bzrlib.tests.test_plugins import TestOneNamedPluginOnly\n"
110
 
                    "TestOneNamedPluginOnly.activeattributes[%r].append('%s')\n")
 
70
        template = ("from bzrlib.tests.test_plugins import TestLoadingPlugins\n"
 
71
                    "TestLoadingPlugins.activeattributes[%r].append('%s')\n")
111
72
        print >> file(os.path.join('first', 'plugin.py'), 'w'), template % (tempattribute, 'first')
112
73
        print >> file(os.path.join('second', 'plugin.py'), 'w'), template % (tempattribute, 'second')
113
74
        try:
114
 
            bzrlib.plugin.load_from_dirs(['first', 'second'])
115
 
            self.assertEqual(['first'], self.activeattributes[tempattribute])
 
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])
116
115
        finally:
117
116
            # remove the plugin 'plugin'
118
117
            del self.activeattributes[tempattribute]
130
129
        # write a plugin that _cannot_ fail to load.
131
130
        print >> file('plugin.py', 'w'), ""
132
131
        try:
133
 
            bzrlib.plugin.load_from_dirs(['.'])
 
132
            bzrlib.plugin.load_from_path(['.'])
134
133
            self.failUnless('plugin' in bzrlib.plugin.all_plugins())
135
134
            self.failUnless(getattr(bzrlib.plugins, 'plugin', None))
136
135
            self.assertEqual(bzrlib.plugin.all_plugins()['plugin'],
137
136
                             bzrlib.plugins.plugin)
138
137
        finally:
139
138
            # remove the plugin 'plugin'
 
139
            if 'bzrlib.plugins.plugin' in sys.modules:
 
140
                del sys.modules['bzrlib.plugins.plugin']
140
141
            if getattr(bzrlib.plugins, 'plugin', None):
141
142
                del bzrlib.plugins.plugin
142
143
        self.failIf(getattr(bzrlib.plugins, 'plugin', None))
147
148
    def split_help_commands(self):
148
149
        help = {}
149
150
        current = None
150
 
        for line in self.capture('help commands').splitlines():
151
 
            if line.startswith('bzr '):
152
 
                current = line.split()[1]
 
151
        for line in self.run_bzr('help commands')[0].splitlines():
 
152
            if not line.startswith(' '):
 
153
                current = line.split()[0]
153
154
            help[current] = help.get(current, '') + line
154
155
 
155
156
        return help
160
161
        for cmd_name in bzrlib.commands.builtin_command_names():
161
162
            if cmd_name in bzrlib.commands.plugin_command_names():
162
163
                continue
163
 
            help = StringIO()
164
164
            try:
165
 
                bzrlib.help.help_on_command(cmd_name, help)
 
165
                help = bzrlib.commands.get_cmd_object(cmd_name).get_help_text()
166
166
            except NotImplementedError:
167
167
                # some commands have no help
168
168
                pass
169
169
            else:
170
 
                help.seek(0)
171
 
                self.assertNotContainsRe(help.read(), 'From plugin "[^"]*"')
 
170
                self.assertNotContainsRe(help, 'plugin "[^"]*"')
172
171
 
173
 
            if help in help_commands.keys():
 
172
            if cmd_name in help_commands.keys():
174
173
                # some commands are hidden
175
174
                help = help_commands[cmd_name]
176
 
                self.assertNotContainsRe(help, 'From plugin "[^"]*"')
 
175
                self.assertNotContainsRe(help, 'plugin "[^"]*"')
177
176
 
178
177
    def test_plugin_help_shows_plugin(self):
179
178
        # Create a test plugin
184
183
 
185
184
        try:
186
185
            # Check its help
187
 
            bzrlib.plugin.load_from_dirs(['plugin_test'])
 
186
            bzrlib.plugin.load_from_path(['plugin_test'])
188
187
            bzrlib.commands.register_command( bzrlib.plugins.myplug.cmd_myplug)
189
 
            help = self.capture('help myplug')
190
 
            self.assertContainsRe(help, 'From plugin "myplug"')
 
188
            help = self.run_bzr('help myplug')[0]
 
189
            self.assertContainsRe(help, 'plugin "myplug"')
191
190
            help = self.split_help_commands()['myplug']
192
 
            self.assertContainsRe(help, 'From plugin "myplug"')
193
 
        finally:
194
 
            # remove the plugin 'plugin'
195
 
            if getattr(bzrlib.plugins, 'plugin', None):
196
 
                del bzrlib.plugins.plugin
 
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())