~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_plugins.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2006-07-06 03:15:29 UTC
  • mfrom: (1711.2.78 jam-integration)
  • Revision ID: pqm@pqm.ubuntu.com-20060706031529-e189d8c3f42076be
(jam) allow plugins to include benchmarks

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
20
# XXX: There are no plugin tests at the moment because the plugin module
22
23
 
23
24
import os
24
25
from StringIO import StringIO
25
 
import sys
26
 
import zipfile
27
26
 
28
 
from bzrlib import plugin, tests
29
27
import bzrlib.plugin
30
28
import bzrlib.plugins
31
29
import bzrlib.commands
32
30
import bzrlib.help
33
 
from bzrlib.tests import TestCase, TestCaseInTempDir
 
31
from bzrlib.tests import TestCaseInTempDir
34
32
from bzrlib.osutils import pathjoin, abspath
35
33
 
 
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
#         """
36
75
 
37
76
PLUGIN_TEXT = """\
38
77
import bzrlib.commands
45
84
 
46
85
# TODO: Write a test for plugin decoration of commands.
47
86
 
48
 
class TestLoadingPlugins(TestCaseInTempDir):
 
87
class TestOneNamedPluginOnly(TestCaseInTempDir):
49
88
 
50
89
    activeattributes = {}
51
90
 
52
91
    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.
 
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.
57
96
        tempattribute = "0"
58
97
        self.failIf(tempattribute in self.activeattributes)
59
98
        # set a place for the plugins to record their loading, and at the same
60
99
        # time validate that the location the plugins should record to is
61
100
        # valid and correct.
62
 
        bzrlib.tests.test_plugins.TestLoadingPlugins.activeattributes \
 
101
        bzrlib.tests.test_plugins.TestOneNamedPluginOnly.activeattributes \
63
102
            [tempattribute] = []
64
103
        self.failUnless(tempattribute in self.activeattributes)
65
104
        # create two plugin directories
67
106
        os.mkdir('second')
68
107
        # write a plugin that will record when its loaded in the 
69
108
        # tempattribute list.
70
 
        template = ("from bzrlib.tests.test_plugins import TestLoadingPlugins\n"
71
 
                    "TestLoadingPlugins.activeattributes[%r].append('%s')\n")
 
109
        template = ("from bzrlib.tests.test_plugins import TestOneNamedPluginOnly\n"
 
110
                    "TestOneNamedPluginOnly.activeattributes[%r].append('%s')\n")
72
111
        print >> file(os.path.join('first', 'plugin.py'), 'w'), template % (tempattribute, 'first')
73
112
        print >> file(os.path.join('second', 'plugin.py'), 'w'), template % (tempattribute, 'second')
74
113
        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])
 
114
            bzrlib.plugin.load_from_dirs(['first', 'second'])
 
115
            self.assertEqual(['first'], self.activeattributes[tempattribute])
115
116
        finally:
116
117
            # remove the plugin 'plugin'
117
118
            del self.activeattributes[tempattribute]
129
130
        # write a plugin that _cannot_ fail to load.
130
131
        print >> file('plugin.py', 'w'), ""
131
132
        try:
132
 
            bzrlib.plugin.load_from_path(['.'])
 
133
            bzrlib.plugin.load_from_dirs(['.'])
133
134
            self.failUnless('plugin' in bzrlib.plugin.all_plugins())
134
135
            self.failUnless(getattr(bzrlib.plugins, 'plugin', None))
135
136
            self.assertEqual(bzrlib.plugin.all_plugins()['plugin'],
136
137
                             bzrlib.plugins.plugin)
137
138
        finally:
138
139
            # remove the plugin 'plugin'
139
 
            if 'bzrlib.plugins.plugin' in sys.modules:
140
 
                del sys.modules['bzrlib.plugins.plugin']
141
140
            if getattr(bzrlib.plugins, 'plugin', None):
142
141
                del bzrlib.plugins.plugin
143
142
        self.failIf(getattr(bzrlib.plugins, 'plugin', None))
148
147
    def split_help_commands(self):
149
148
        help = {}
150
149
        current = None
151
 
        for line in self.run_bzr('help commands')[0].splitlines():
152
 
            if not line.startswith(' '):
153
 
                current = line.split()[0]
 
150
        for line in self.capture('help commands').splitlines():
 
151
            if line.startswith('bzr '):
 
152
                current = line.split()[1]
154
153
            help[current] = help.get(current, '') + line
155
154
 
156
155
        return help
161
160
        for cmd_name in bzrlib.commands.builtin_command_names():
162
161
            if cmd_name in bzrlib.commands.plugin_command_names():
163
162
                continue
 
163
            help = StringIO()
164
164
            try:
165
 
                help = bzrlib.commands.get_cmd_object(cmd_name).get_help_text()
 
165
                bzrlib.help.help_on_command(cmd_name, help)
166
166
            except NotImplementedError:
167
167
                # some commands have no help
168
168
                pass
169
169
            else:
170
 
                self.assertNotContainsRe(help, 'From plugin "[^"]*"')
 
170
                help.seek(0)
 
171
                self.assertNotContainsRe(help.read(), 'From plugin "[^"]*"')
171
172
 
172
 
            if cmd_name in help_commands.keys():
 
173
            if help in help_commands.keys():
173
174
                # some commands are hidden
174
175
                help = help_commands[cmd_name]
175
176
                self.assertNotContainsRe(help, 'From plugin "[^"]*"')
183
184
 
184
185
        try:
185
186
            # Check its help
186
 
            bzrlib.plugin.load_from_path(['plugin_test'])
 
187
            bzrlib.plugin.load_from_dirs(['plugin_test'])
187
188
            bzrlib.commands.register_command( bzrlib.plugins.myplug.cmd_myplug)
188
 
            help = self.run_bzr('help myplug')[0]
 
189
            help = self.capture('help myplug')
189
190
            self.assertContainsRe(help, 'From plugin "myplug"')
190
191
            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())
 
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