/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_plugins.py

  • Committer: Alexander Belchenko
  • Date: 2007-04-30 19:52:44 UTC
  • mto: This revision was merged to the branch mainline in revision 2478.
  • Revision ID: bialix@ukr.net-20070430195244-pxc5r3x72ckh027e
Bugfix #110901: commit message template written with native line-endings; corresponding unit tests added

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# Copyright (C) 2005 Canonical Ltd
 
2
#
 
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.
 
7
#
 
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.
 
12
#
 
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
 
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
16
 
 
17
"""Tests for plugins"""
 
18
 
 
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.capture('help commands').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
 
179
        os.mkdir('plugin_test')
 
180
        f = open(pathjoin('plugin_test', 'myplug.py'), 'w')
 
181
        f.write(PLUGIN_TEXT)
 
182
        f.close()
 
183
 
 
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.capture('help myplug')
 
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
        try:
 
212
            bzrlib.plugin.load_from_zip(zip_name)
 
213
            self.assertTrue(plugin_name in dir(bzrlib.plugins),
 
214
                            'Plugin is not loaded')
 
215
        finally:
 
216
            # unregister plugin
 
217
            if getattr(bzrlib.plugins, plugin_name, None):
 
218
                delattr(bzrlib.plugins, plugin_name)
 
219
 
 
220
    def test_load_module(self):
 
221
        self.make_zipped_plugin('./test.zip', 'ziplug.py')
 
222
        self.check_plugin_load('./test.zip', 'ziplug')
 
223
 
 
224
    def test_load_package(self):
 
225
        self.make_zipped_plugin('./test.zip', 'ziplug/__init__.py')
 
226
        self.check_plugin_load('./test.zip', 'ziplug')
 
227
 
 
228
 
 
229
class TestSetPluginsPath(TestCase):
 
230
    
 
231
    def test_set_plugins_path(self):
 
232
        """set_plugins_path should set the module __path__ correctly."""
 
233
        old_path = bzrlib.plugins.__path__
 
234
        try:
 
235
            bzrlib.plugins.__path__ = []
 
236
            expected_path = bzrlib.plugin.set_plugins_path()
 
237
            self.assertEqual(expected_path, bzrlib.plugins.__path__)
 
238
        finally:
 
239
            bzrlib.plugins.__path__ = old_path
 
240
 
 
241
 
 
242
class TestHelpIndex(tests.TestCase):
 
243
    """Tests for the PluginsHelpIndex class."""
 
244
 
 
245
    def test_default_constructable(self):
 
246
        index = plugin.PluginsHelpIndex()
 
247
 
 
248
    def test_get_topics_None(self):
 
249
        """Searching for None returns an empty list."""
 
250
        index = plugin.PluginsHelpIndex()
 
251
        self.assertEqual([], index.get_topics(None))
 
252
 
 
253
    def test_get_topics_launchpad(self):
 
254
        """Searching for 'launchpad' returns the launchpad plugin docstring."""
 
255
        index = plugin.PluginsHelpIndex()
 
256
        self.assertFalse(sys.modules.has_key('bzrlib.plugins.get_topics'))
 
257
        demo_module = FakeModule('', 'bzrlib.plugins.get_topics')
 
258
        sys.modules['bzrlib.plugins.get_topics'] = demo_module
 
259
        try:
 
260
            topics = index.get_topics('get_topics')
 
261
            self.assertEqual(1, len(topics))
 
262
            self.assertIsInstance(topics[0], plugin.ModuleHelpTopic)
 
263
            self.assertEqual(demo_module, topics[0].module)
 
264
        finally:
 
265
            del sys.modules['bzrlib.plugins.get_topics']
 
266
 
 
267
    def test_get_topics_no_topic(self):
 
268
        """Searching for something that is not a plugin returns []."""
 
269
        # test this by using a name that cannot be a plugin - its not
 
270
        # a valid python identifier.
 
271
        index = plugin.PluginsHelpIndex()
 
272
        self.assertEqual([], index.get_topics('nothing by this name'))
 
273
 
 
274
    def test_prefix(self):
 
275
        """PluginsHelpIndex has a prefix of 'plugins/'."""
 
276
        index = plugin.PluginsHelpIndex()
 
277
        self.assertEqual('plugins/', index.prefix)
 
278
 
 
279
    def test_get_topic_with_prefix(self):
 
280
        """Searching for plugins/launchpad returns launchpad module help."""
 
281
        index = plugin.PluginsHelpIndex()
 
282
        self.assertFalse(sys.modules.has_key('bzrlib.plugins.get_topics'))
 
283
        demo_module = FakeModule('', 'bzrlib.plugins.get_topics')
 
284
        sys.modules['bzrlib.plugins.get_topics'] = demo_module
 
285
        try:
 
286
            topics = index.get_topics('plugins/get_topics')
 
287
            self.assertEqual(1, len(topics))
 
288
            self.assertIsInstance(topics[0], plugin.ModuleHelpTopic)
 
289
            self.assertEqual(demo_module, topics[0].module)
 
290
        finally:
 
291
            del sys.modules['bzrlib.plugins.get_topics']
 
292
 
 
293
 
 
294
class FakeModule(object):
 
295
    """A fake module to test with."""
 
296
 
 
297
    def __init__(self, doc, name):
 
298
        self.__doc__ = doc
 
299
        self.__name__ = name
 
300
 
 
301
 
 
302
class TestModuleHelpTopic(tests.TestCase):
 
303
    """Tests for the ModuleHelpTopic class."""
 
304
 
 
305
    def test_contruct(self):
 
306
        """Construction takes the module to document."""
 
307
        mod = FakeModule('foo', 'foo')
 
308
        topic = plugin.ModuleHelpTopic(mod)
 
309
        self.assertEqual(mod, topic.module)
 
310
 
 
311
    def test_get_help_text_None(self):
 
312
        """A ModuleHelpTopic returns the docstring for get_help_text."""
 
313
        mod = FakeModule(None, 'demo')
 
314
        topic = plugin.ModuleHelpTopic(mod)
 
315
        self.assertEqual("Plugin 'demo' has no docstring.\n",
 
316
            topic.get_help_text())
 
317
 
 
318
    def test_get_help_text_no_carriage_return(self):
 
319
        """ModuleHelpTopic.get_help_text adds a \n if needed."""
 
320
        mod = FakeModule('one line of help', 'demo')
 
321
        topic = plugin.ModuleHelpTopic(mod)
 
322
        self.assertEqual("one line of help\n",
 
323
            topic.get_help_text())
 
324
 
 
325
    def test_get_help_text_carriage_return(self):
 
326
        """ModuleHelpTopic.get_help_text adds a \n if needed."""
 
327
        mod = FakeModule('two lines of help\nand more\n', 'demo')
 
328
        topic = plugin.ModuleHelpTopic(mod)
 
329
        self.assertEqual("two lines of help\nand more\n",
 
330
            topic.get_help_text())
 
331
 
 
332
    def test_get_help_text_with_additional_see_also(self):
 
333
        mod = FakeModule('two lines of help\nand more', 'demo')
 
334
        topic = plugin.ModuleHelpTopic(mod)
 
335
        self.assertEqual("two lines of help\nand more\nSee also: bar, foo\n",
 
336
            topic.get_help_text(['foo', 'bar']))
 
337
 
 
338
    def test_get_help_topic(self):
 
339
        """The help topic for a plugin is its module name."""
 
340
        mod = FakeModule('two lines of help\nand more', 'bzrlib.plugins.demo')
 
341
        topic = plugin.ModuleHelpTopic(mod)
 
342
        self.assertEqual('demo', topic.get_help_topic())
 
343
        mod = FakeModule('two lines of help\nand more', 'bzrlib.plugins.foo_bar')
 
344
        topic = plugin.ModuleHelpTopic(mod)
 
345
        self.assertEqual('foo_bar', topic.get_help_topic())