14
13
# You should have received a copy of the GNU General Public License
15
14
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18
17
"""Tests for the formatting and construction of errors."""
20
import bzrlib.bzrdir as bzrdir
21
import bzrlib.errors as errors
22
from bzrlib.tests import TestCaseWithTransport
30
from bzrlib.tests import (
32
TestCaseWithTransport,
25
37
class TestErrors(TestCaseWithTransport):
39
def test_no_arg_named_message(self):
40
"""Ensure the __init__ and _fmt in errors do not have "message" arg.
42
This test fails if __init__ or _fmt in errors has an argument
43
named "message" as this can cause errors in some Python versions.
44
Python 2.5 uses a slot for StandardError.message.
47
fmt_pattern = re.compile("%\(message\)[sir]")
48
subclasses_present = getattr(errors.BzrError, '__subclasses__', None)
49
if not subclasses_present:
50
raise TestSkipped('__subclasses__ attribute required for classes. '
51
'Requires Python 2.5 or later.')
52
for c in errors.BzrError.__subclasses__():
53
init = getattr(c, '__init__', None)
54
fmt = getattr(c, '_fmt', None)
56
args = inspect.getargspec(init)[0]
57
self.assertFalse('message' in args,
58
('Argument name "message" not allowed for '
59
'"errors.%s.__init__"' % c.__name__))
60
if fmt and fmt_pattern.search(fmt):
61
self.assertFalse(True, ('"message" not allowed in '
62
'"errors.%s._fmt"' % c.__name__))
64
def test_bad_filename_encoding(self):
65
error = errors.BadFilenameEncoding('bad/filen\xe5me', 'UTF-8')
67
"Filename 'bad/filen\\xe5me' is not valid in your current"
68
" filesystem encoding UTF-8",
71
def test_corrupt_dirstate(self):
72
error = errors.CorruptDirstate('path/to/dirstate', 'the reason why')
74
"Inconsistency in dirstate file path/to/dirstate.\n"
75
"Error: the reason why",
78
def test_dirstate_corrupt(self):
79
error = errors.DirstateCorrupt('.bzr/checkout/dirstate',
80
'trailing garbage: "x"')
81
self.assertEqualDiff("The dirstate file (.bzr/checkout/dirstate)"
82
" appears to be corrupt: trailing garbage: \"x\"",
85
def test_disabled_method(self):
86
error = errors.DisabledMethod("class name")
88
"The smart server method 'class name' is disabled.", str(error))
90
def test_duplicate_file_id(self):
91
error = errors.DuplicateFileId('a_file_id', 'foo')
92
self.assertEqualDiff('File id {a_file_id} already exists in inventory'
93
' as foo', str(error))
95
def test_duplicate_help_prefix(self):
96
error = errors.DuplicateHelpPrefix('foo')
97
self.assertEqualDiff('The prefix foo is in the help search path twice.',
100
def test_ghost_revisions_have_no_revno(self):
101
error = errors.GhostRevisionsHaveNoRevno('target', 'ghost_rev')
102
self.assertEqualDiff("Could not determine revno for {target} because"
103
" its ancestry shows a ghost at {ghost_rev}",
106
def test_incompatibleAPI(self):
107
error = errors.IncompatibleAPI("module", (1, 2, 3), (4, 5, 6), (7, 8, 9))
108
self.assertEqualDiff(
109
'The API for "module" is not compatible with "(1, 2, 3)". '
110
'It supports versions "(4, 5, 6)" to "(7, 8, 9)".',
113
def test_inconsistent_delta(self):
114
error = errors.InconsistentDelta('path', 'file-id', 'reason for foo')
115
self.assertEqualDiff(
116
"An inconsistent delta was supplied involving 'path', 'file-id'\n"
117
"reason: reason for foo",
120
def test_inconsistent_delta_delta(self):
121
error = errors.InconsistentDeltaDelta([], 'reason')
122
self.assertEqualDiff(
123
"An inconsistent delta was supplied: []\nreason: reason",
126
def test_in_process_transport(self):
127
error = errors.InProcessTransport('fpp')
128
self.assertEqualDiff(
129
"The transport 'fpp' is only accessible within this process.",
132
def test_invalid_http_range(self):
133
error = errors.InvalidHttpRange('path',
134
'Content-Range: potatoes 0-00/o0oo0',
136
self.assertEquals("Invalid http range"
137
" 'Content-Range: potatoes 0-00/o0oo0'"
138
" for path: bad range",
141
def test_invalid_range(self):
142
error = errors.InvalidRange('path', 12, 'bad range')
143
self.assertEquals("Invalid range access in path at 12: bad range",
146
def test_inventory_modified(self):
147
error = errors.InventoryModified("a tree to be repred")
148
self.assertEqualDiff("The current inventory for the tree 'a tree to "
149
"be repred' has been modified, so a clean inventory cannot be "
150
"read without data loss.",
153
def test_jail_break(self):
154
error = errors.JailBreak("some url")
155
self.assertEqualDiff("An attempt to access a url outside the server"
156
" jail was made: 'some url'.",
159
def test_lock_active(self):
160
error = errors.LockActive("lock description")
161
self.assertEqualDiff("The lock for 'lock description' is in use and "
165
def test_lock_corrupt(self):
166
error = errors.LockCorrupt("corruption info")
167
self.assertEqualDiff("Lock is apparently held, but corrupted: "
169
"Use 'bzr break-lock' to clear it",
172
def test_knit_data_stream_incompatible(self):
173
error = errors.KnitDataStreamIncompatible(
174
'stream format', 'target format')
175
self.assertEqual('Cannot insert knit data stream of format '
176
'"stream format" into knit of format '
177
'"target format".', str(error))
179
def test_knit_data_stream_unknown(self):
180
error = errors.KnitDataStreamUnknown(
182
self.assertEqual('Cannot parse knit data stream of format '
183
'"stream format".', str(error))
185
def test_knit_header_error(self):
186
error = errors.KnitHeaderError('line foo\n', 'path/to/file')
187
self.assertEqual("Knit header error: 'line foo\\n' unexpected"
188
" for file \"path/to/file\".", str(error))
190
def test_knit_index_unknown_method(self):
191
error = errors.KnitIndexUnknownMethod('http://host/foo.kndx',
193
self.assertEqual("Knit index http://host/foo.kndx does not have a"
194
" known method in options: ['bad', 'no-eol']",
197
def test_medium_not_connected(self):
198
error = errors.MediumNotConnected("a medium")
199
self.assertEqualDiff(
200
"The medium 'a medium' is not connected.", str(error))
202
def test_no_public_branch(self):
203
b = self.make_branch('.')
204
error = errors.NoPublicBranch(b)
205
url = urlutils.unescape_for_display(b.base, 'ascii')
206
self.assertEqualDiff(
207
'There is no public branch set for "%s".' % url, str(error))
27
209
def test_no_repo(self):
28
dir = bzrdir.BzrDir.create(self.get_url())
210
dir = controldir.ControlDir.create(self.get_url())
29
211
error = errors.NoRepositoryPresent(dir)
30
self.assertNotEqual(-1, str(error).find(repr(dir.transport.clone('..').base)))
31
self.assertEqual(-1, str(error).find(repr(dir.transport.base)))
212
self.assertNotEqual(-1, str(error).find((dir.transport.clone('..').base)))
213
self.assertEqual(-1, str(error).find((dir.transport.base)))
215
def test_no_smart_medium(self):
216
error = errors.NoSmartMedium("a transport")
217
self.assertEqualDiff("The transport 'a transport' cannot tunnel the "
221
def test_no_help_topic(self):
222
error = errors.NoHelpTopic("topic")
223
self.assertEqualDiff("No help could be found for 'topic'. "
224
"Please use 'bzr help topics' to obtain a list of topics.",
227
def test_no_such_id(self):
228
error = errors.NoSuchId("atree", "anid")
229
self.assertEqualDiff("The file id \"anid\" is not present in the tree "
233
def test_no_such_revision_in_tree(self):
234
error = errors.NoSuchRevisionInTree("atree", "anid")
235
self.assertEqualDiff("The revision id {anid} is not present in the"
236
" tree atree.", str(error))
237
self.assertIsInstance(error, errors.NoSuchRevision)
239
def test_not_stacked(self):
240
error = errors.NotStacked('a branch')
241
self.assertEqualDiff("The branch 'a branch' is not stacked.",
244
def test_not_write_locked(self):
245
error = errors.NotWriteLocked('a thing to repr')
246
self.assertEqualDiff("'a thing to repr' is not write locked but needs "
250
def test_lock_failed(self):
251
error = errors.LockFailed('http://canonical.com/', 'readonly transport')
252
self.assertEqualDiff("Cannot lock http://canonical.com/: readonly transport",
254
self.assertFalse(error.internal_error)
256
def test_too_many_concurrent_requests(self):
257
error = errors.TooManyConcurrentRequests("a medium")
258
self.assertEqualDiff("The medium 'a medium' has reached its concurrent "
259
"request limit. Be sure to finish_writing and finish_reading on "
260
"the currently open request.",
263
def test_unavailable_representation(self):
264
error = errors.UnavailableRepresentation(('key',), "mpdiff", "fulltext")
265
self.assertEqualDiff("The encoding 'mpdiff' is not available for key "
266
"('key',) which is encoded as 'fulltext'.",
269
def test_unknown_hook(self):
270
error = errors.UnknownHook("branch", "foo")
271
self.assertEqualDiff("The branch hook 'foo' is unknown in this version"
274
error = errors.UnknownHook("tree", "bar")
275
self.assertEqualDiff("The tree hook 'bar' is unknown in this version"
279
def test_unstackable_branch_format(self):
282
error = errors.UnstackableBranchFormat(format, url)
283
self.assertEqualDiff(
284
"The branch '/foo'(foo) is not a stackable format. "
285
"You will need to upgrade the branch to permit branch stacking.",
288
def test_unstackable_location(self):
289
error = errors.UnstackableLocationError('foo', 'bar')
290
self.assertEqualDiff("The branch 'foo' cannot be stacked on 'bar'.",
293
def test_unstackable_repository_format(self):
296
error = errors.UnstackableRepositoryFormat(format, url)
297
self.assertEqualDiff(
298
"The repository '/foo'(foo) is not a stackable format. "
299
"You will need to upgrade the repository to permit branch stacking.",
33
302
def test_up_to_date(self):
34
error = errors.UpToDateFormat(bzrdir.BzrDirFormat4())
35
self.assertEqualDiff("The branch format Bazaar-NG branch, "
36
"format 0.0.4 is already at the most "
303
error = errors.UpToDateFormat("someformat")
304
self.assertEqualDiff(
305
"The branch format someformat is already at the most "
306
"recent format.", str(error))
40
308
def test_corrupt_repository(self):
41
309
repo = self.make_repository('.')
44
312
"Please run bzr reconcile on this repository." %
45
313
repo.bzrdir.root_transport.base,
316
def test_read_error(self):
317
# a unicode path to check that %r is being used.
319
error = errors.ReadError(path)
320
self.assertEqualDiff("Error reading from u'a path'.", str(error))
322
def test_bad_index_format_signature(self):
323
error = errors.BadIndexFormatSignature("foo", "bar")
324
self.assertEqual("foo is not an index of type bar.",
327
def test_bad_index_data(self):
328
error = errors.BadIndexData("foo")
329
self.assertEqual("Error in data for index foo.",
332
def test_bad_index_duplicate_key(self):
333
error = errors.BadIndexDuplicateKey("foo", "bar")
334
self.assertEqual("The key 'foo' is already in index 'bar'.",
337
def test_bad_index_key(self):
338
error = errors.BadIndexKey("foo")
339
self.assertEqual("The key 'foo' is not a valid key.",
342
def test_bad_index_options(self):
343
error = errors.BadIndexOptions("foo")
344
self.assertEqual("Could not parse options for index foo.",
347
def test_bad_index_value(self):
348
error = errors.BadIndexValue("foo")
349
self.assertEqual("The value 'foo' is not a valid value.",
352
def test_bzrerror_from_literal_string(self):
353
# Some code constructs BzrError from a literal string, in which case
354
# no further formatting is done. (I'm not sure raising the base class
355
# is a great idea, but if the exception is not intended to be caught
356
# perhaps no more is needed.)
358
raise errors.BzrError('this is my errors; %d is not expanded')
359
except errors.BzrError, e:
360
self.assertEqual('this is my errors; %d is not expanded', str(e))
362
def test_reading_completed(self):
363
error = errors.ReadingCompleted("a request")
364
self.assertEqualDiff("The MediumRequest 'a request' has already had "
365
"finish_reading called upon it - the request has been completed and"
366
" no more data may be read.",
369
def test_writing_completed(self):
370
error = errors.WritingCompleted("a request")
371
self.assertEqualDiff("The MediumRequest 'a request' has already had "
372
"finish_writing called upon it - accept bytes may not be called "
376
def test_writing_not_completed(self):
377
error = errors.WritingNotComplete("a request")
378
self.assertEqualDiff("The MediumRequest 'a request' has not has "
379
"finish_writing called upon it - until the write phase is complete"
380
" no data may be read.",
383
def test_transport_not_possible(self):
384
error = errors.TransportNotPossible('readonly', 'original error')
385
self.assertEqualDiff('Transport operation not possible:'
386
' readonly original error', str(error))
388
def assertSocketConnectionError(self, expected, *args, **kwargs):
389
"""Check the formatting of a SocketConnectionError exception"""
390
e = errors.SocketConnectionError(*args, **kwargs)
391
self.assertEqual(expected, str(e))
393
def test_socket_connection_error(self):
394
"""Test the formatting of SocketConnectionError"""
396
# There should be a default msg about failing to connect
397
# we only require a host name.
398
self.assertSocketConnectionError(
399
'Failed to connect to ahost',
402
# If port is None, we don't put :None
403
self.assertSocketConnectionError(
404
'Failed to connect to ahost',
406
# But if port is supplied we include it
407
self.assertSocketConnectionError(
408
'Failed to connect to ahost:22',
411
# We can also supply extra information about the error
412
# with or without a port
413
self.assertSocketConnectionError(
414
'Failed to connect to ahost:22; bogus error',
415
'ahost', port=22, orig_error='bogus error')
416
self.assertSocketConnectionError(
417
'Failed to connect to ahost; bogus error',
418
'ahost', orig_error='bogus error')
419
# An exception object can be passed rather than a string
420
orig_error = ValueError('bad value')
421
self.assertSocketConnectionError(
422
'Failed to connect to ahost; %s' % (str(orig_error),),
423
host='ahost', orig_error=orig_error)
425
# And we can supply a custom failure message
426
self.assertSocketConnectionError(
427
'Unable to connect to ssh host ahost:444; my_error',
428
host='ahost', port=444, msg='Unable to connect to ssh host',
429
orig_error='my_error')
431
def test_target_not_branch(self):
432
"""Test the formatting of TargetNotBranch."""
433
error = errors.TargetNotBranch('foo')
435
"Your branch does not have all of the revisions required in "
436
"order to merge this merge directive and the target "
437
"location specified in the merge directive is not a branch: "
440
def test_malformed_bug_identifier(self):
441
"""Test the formatting of MalformedBugIdentifier."""
442
error = errors.MalformedBugIdentifier('bogus', 'reason for bogosity')
444
'Did not understand bug identifier bogus: reason for bogosity. '
445
'See "bzr help bugs" for more information on this feature.',
448
def test_unknown_bug_tracker_abbreviation(self):
449
"""Test the formatting of UnknownBugTrackerAbbreviation."""
450
branch = self.make_branch('some_branch')
451
error = errors.UnknownBugTrackerAbbreviation('xxx', branch)
453
"Cannot find registered bug tracker called xxx on %s" % branch,
456
def test_unexpected_smart_server_response(self):
457
e = errors.UnexpectedSmartServerResponse(('not yes',))
459
"Could not understand response from smart server: ('not yes',)",
462
def test_unknown_container_format(self):
463
"""Test the formatting of UnknownContainerFormatError."""
464
e = errors.UnknownContainerFormatError('bad format string')
466
"Unrecognised container format: 'bad format string'",
469
def test_unexpected_end_of_container(self):
470
"""Test the formatting of UnexpectedEndOfContainerError."""
471
e = errors.UnexpectedEndOfContainerError()
473
"Unexpected end of container stream", str(e))
475
def test_unknown_record_type(self):
476
"""Test the formatting of UnknownRecordTypeError."""
477
e = errors.UnknownRecordTypeError("X")
479
"Unknown record type: 'X'",
482
def test_invalid_record(self):
483
"""Test the formatting of InvalidRecordError."""
484
e = errors.InvalidRecordError("xxx")
486
"Invalid record: xxx",
489
def test_container_has_excess_data(self):
490
"""Test the formatting of ContainerHasExcessDataError."""
491
e = errors.ContainerHasExcessDataError("excess bytes")
493
"Container has data after end marker: 'excess bytes'",
496
def test_duplicate_record_name_error(self):
497
"""Test the formatting of DuplicateRecordNameError."""
498
e = errors.DuplicateRecordNameError(u"n\xe5me".encode('utf-8'))
500
"Container has multiple records with the same name: n\xc3\xa5me",
503
def test_check_error(self):
504
# This has a member called 'message', which is problematic in
505
# python2.5 because that is a slot on the base Exception class
506
e = errors.BzrCheckError('example check failure')
508
"Internal check failed: example check failure",
510
self.assertTrue(e.internal_error)
512
def test_repository_data_stream_error(self):
513
"""Test the formatting of RepositoryDataStreamError."""
514
e = errors.RepositoryDataStreamError(u"my reason")
516
"Corrupt or incompatible data stream: my reason", str(e))
518
def test_immortal_pending_deletion_message(self):
519
err = errors.ImmortalPendingDeletion('foo')
521
"Unable to delete transform temporary directory foo. "
522
"Please examine foo to see if it contains any files "
523
"you wish to keep, and delete it when you are done.",
526
def test_unable_create_symlink(self):
527
err = errors.UnableCreateSymlink()
529
"Unable to create symlink on this platform",
531
err = errors.UnableCreateSymlink(path=u'foo')
533
"Unable to create symlink 'foo' on this platform",
535
err = errors.UnableCreateSymlink(path=u'\xb5')
537
"Unable to create symlink u'\\xb5' on this platform",
540
def test_invalid_url_join(self):
541
"""Test the formatting of InvalidURLJoin."""
542
e = errors.InvalidURLJoin('Reason', 'base path', ('args',))
544
"Invalid URL join request: Reason: 'base path' + ('args',)",
547
def test_incorrect_url(self):
548
err = errors.InvalidBugTrackerURL('foo', 'http://bug.com/')
550
("The URL for bug tracker \"foo\" doesn't contain {id}: "
554
def test_unable_encode_path(self):
555
err = errors.UnableEncodePath('foo', 'executable')
556
self.assertEquals("Unable to encode executable path 'foo' in "
557
"user encoding " + osutils.get_user_encoding(),
560
def test_unknown_format(self):
561
err = errors.UnknownFormatError('bar', kind='foo')
562
self.assertEquals("Unknown foo format: 'bar'", str(err))
564
def test_unknown_rules(self):
565
err = errors.UnknownRules(['foo', 'bar'])
566
self.assertEquals("Unknown rules detected: foo, bar.", str(err))
568
def test_tip_change_rejected(self):
569
err = errors.TipChangeRejected(u'Unicode message\N{INTERROBANG}')
571
u'Tip change rejected: Unicode message\N{INTERROBANG}',
574
'Tip change rejected: Unicode message\xe2\x80\xbd',
577
def test_error_from_smart_server(self):
578
error_tuple = ('error', 'tuple')
579
err = errors.ErrorFromSmartServer(error_tuple)
581
"Error received from smart server: ('error', 'tuple')", str(err))
583
def test_untranslateable_error_from_smart_server(self):
584
error_tuple = ('error', 'tuple')
585
orig_err = errors.ErrorFromSmartServer(error_tuple)
586
err = errors.UnknownErrorFromSmartServer(orig_err)
588
"Server sent an unexpected error: ('error', 'tuple')", str(err))
590
def test_smart_message_handler_error(self):
591
# Make an exc_info tuple.
593
raise Exception("example error")
595
err = errors.SmartMessageHandlerError(sys.exc_info())
596
# GZ 2010-11-08: Should not store exc_info in exception instances.
598
self.assertStartsWith(
599
str(err), "The message handler raised an exception:\n")
600
self.assertEndsWith(str(err), "Exception: example error\n")
604
def test_must_have_working_tree(self):
605
err = errors.MustHaveWorkingTree('foo', 'bar')
606
self.assertEqual(str(err), "Branching 'bar'(foo) must create a"
609
def test_no_such_view(self):
610
err = errors.NoSuchView('foo')
611
self.assertEquals("No such view: foo.", str(err))
613
def test_views_not_supported(self):
614
err = errors.ViewsNotSupported('atree')
616
self.assertStartsWith(err_str, "Views are not supported by ")
617
self.assertEndsWith(err_str, "; use 'bzr upgrade' to change your "
618
"tree to a later format.")
620
def test_file_outside_view(self):
621
err = errors.FileOutsideView('baz', ['foo', 'bar'])
622
self.assertEquals('Specified file "baz" is outside the current view: '
623
'foo, bar', str(err))
625
def test_invalid_shelf_id(self):
627
err = errors.InvalidShelfId(invalid_id)
628
self.assertEqual('"foo" is not a valid shelf id, '
629
'try a number instead.', str(err))
631
def test_unresumable_write_group(self):
633
wg_tokens = ['token']
635
err = errors.UnresumableWriteGroup(repo, wg_tokens, reason)
637
"Repository dummy repo cannot resume write group "
638
"['token']: a reason", str(err))
640
def test_unsuspendable_write_group(self):
642
err = errors.UnsuspendableWriteGroup(repo)
644
'Repository dummy repo cannot suspend a write group.', str(err))
646
def test_not_branch_no_args(self):
647
err = errors.NotBranchError('path')
648
self.assertEqual('Not a branch: "path".', str(err))
650
def test_not_branch_bzrdir_with_repo(self):
651
bzrdir = self.make_repository('repo').bzrdir
652
err = errors.NotBranchError('path', bzrdir=bzrdir)
654
'Not a branch: "path": location is a repository.', str(err))
656
def test_not_branch_bzrdir_without_repo(self):
657
bzrdir = self.make_bzrdir('bzrdir')
658
err = errors.NotBranchError('path', bzrdir=bzrdir)
659
self.assertEqual('Not a branch: "path".', str(err))
661
def test_not_branch_bzrdir_with_recursive_not_branch_error(self):
662
class FakeBzrDir(object):
663
def open_repository(self):
664
# str() on the NotBranchError will trigger a call to this,
665
# which in turn will another, identical NotBranchError.
666
raise errors.NotBranchError('path', bzrdir=FakeBzrDir())
667
err = errors.NotBranchError('path', bzrdir=FakeBzrDir())
668
self.assertEqual('Not a branch: "path".', str(err))
670
def test_not_branch_laziness(self):
671
real_bzrdir = self.make_bzrdir('path')
672
class FakeBzrDir(object):
675
def open_repository(self):
676
self.calls.append('open_repository')
677
raise errors.NoRepositoryPresent(real_bzrdir)
678
fake_bzrdir = FakeBzrDir()
679
err = errors.NotBranchError('path', bzrdir=fake_bzrdir)
680
self.assertEqual([], fake_bzrdir.calls)
682
self.assertEqual(['open_repository'], fake_bzrdir.calls)
683
# Stringifying twice doesn't try to open a repository twice.
685
self.assertEqual(['open_repository'], fake_bzrdir.calls)
687
def test_invalid_pattern(self):
688
error = errors.InvalidPattern('Bad pattern msg.')
689
self.assertEqualDiff("Invalid pattern(s) found. Bad pattern msg.",
692
def test_recursive_bind(self):
693
error = errors.RecursiveBind('foo_bar_branch')
694
msg = ('Branch "foo_bar_branch" appears to be bound to itself. '
695
'Please use `bzr unbind` to fix.')
696
self.assertEqualDiff(msg, str(error))
698
def test_retry_with_new_packs(self):
699
fake_exc_info = ('{exc type}', '{exc value}', '{exc traceback}')
700
error = errors.RetryWithNewPacks(
701
'{context}', reload_occurred=False, exc_info=fake_exc_info)
703
'Pack files have changed, reload and retry. context: '
704
'{context} {exc value}', str(error))
707
class PassThroughError(errors.BzrError):
709
_fmt = """Pass through %(foo)s and %(bar)s"""
711
def __init__(self, foo, bar):
712
errors.BzrError.__init__(self, foo=foo, bar=bar)
715
class ErrorWithBadFormat(errors.BzrError):
717
_fmt = """One format specifier: %(thing)s"""
720
class ErrorWithNoFormat(errors.BzrError):
721
__doc__ = """This class has a docstring but no format string."""
724
class TestErrorFormatting(TestCase):
726
def test_always_str(self):
727
e = PassThroughError(u'\xb5', 'bar')
728
self.assertIsInstance(e.__str__(), str)
729
# In Python str(foo) *must* return a real byte string
730
# not a Unicode string. The following line would raise a
731
# Unicode error, because it tries to call str() on the string
732
# returned from e.__str__(), and it has non ascii characters
734
self.assertEqual('Pass through \xc2\xb5 and bar', s)
736
def test_missing_format_string(self):
737
e = ErrorWithNoFormat(param='randomvalue')
738
self.assertStartsWith(str(e),
739
"Unprintable exception ErrorWithNoFormat")
741
def test_mismatched_format_args(self):
742
# Even though ErrorWithBadFormat's format string does not match the
743
# arguments we constructing it with, we can still stringify an instance
744
# of this exception. The resulting string will say its unprintable.
745
e = ErrorWithBadFormat(not_thing='x')
746
self.assertStartsWith(
747
str(e), 'Unprintable exception ErrorWithBadFormat')
749
def test_cannot_bind_address(self):
750
# see <https://bugs.launchpad.net/bzr/+bug/286871>
751
e = errors.CannotBindAddress('example.com', 22,
752
socket.error(13, 'Permission denied'))
753
self.assertContainsRe(str(e),
754
r'Cannot bind address "example\.com:22":.*Permission denied')
756
def test_file_timestamp_unavailable(self):
757
e = errors.FileTimestampUnavailable("/path/foo")
758
self.assertEquals("The filestamp for /path/foo is not available.",
761
def test_transform_rename_failed(self):
762
e = errors.TransformRenameFailed(u"from", u"to", "readonly file", 2)
764
u"Failed to rename from to to: readonly file",