183
172
def test_get(self):
184
173
t = self.get_transport()
186
files = ['a', 'b', 'e', 'g']
187
contents = ['contents of a\n',
192
self.build_tree(files, transport=t, line_endings='binary')
193
self.check_transport_contents('contents of a\n', t, 'a')
194
content_f = t.get_multi(files)
195
# Use itertools.izip() instead of use zip() or map(), since they fully
196
# evaluate their inputs, the transport requests should be issued and
197
# handled sequentially (we don't want to force transport to buffer).
198
for content, f in itertools.izip(contents, content_f):
199
self.assertEqual(content, f.read())
201
content_f = t.get_multi(iter(files))
202
# Use itertools.izip() for the same reason
203
for content, f in itertools.izip(contents, content_f):
204
self.assertEqual(content, f.read())
176
content = b'contents of a\n'
177
self.build_tree(['a'], transport=t, line_endings='binary')
178
self.check_transport_contents(b'contents of a\n', t, 'a')
180
self.assertEqual(content, f.read())
206
182
def test_get_unknown_file(self):
207
183
t = self.get_transport()
208
184
files = ['a', 'b']
209
contents = ['contents of a\n',
185
contents = [b'contents of a\n',
212
188
self.build_tree(files, transport=t, line_endings='binary')
213
189
self.assertRaises(NoSuchFile, t.get, 'c')
214
self.assertListRaises(NoSuchFile, t.get_multi, ['a', 'b', 'c'])
215
self.assertListRaises(NoSuchFile, t.get_multi, iter(['a', 'b', 'c']))
191
def iterate_and_close(func, *args):
192
for f in func(*args):
193
# We call f.read() here because things like paramiko actually
194
# spawn a thread to prefetch the content, which we want to
195
# consume before we close the handle.
217
199
def test_get_directory_read_gives_ReadError(self):
218
200
"""consistent errors for read() on a file returned by get()."""
238
220
t = self.get_transport()
240
222
files = ['a', 'b', 'e', 'g']
241
contents = ['contents of a\n',
223
contents = [b'contents of a\n',
246
228
self.build_tree(files, transport=t, line_endings='binary')
247
self.check_transport_contents('contents of a\n', t, 'a')
229
self.check_transport_contents(b'contents of a\n', t, 'a')
249
231
for content, fname in zip(contents, files):
250
232
self.assertEqual(content, t.get_bytes(fname))
252
234
def test_get_bytes_unknown_file(self):
253
235
t = self.get_transport()
255
236
self.assertRaises(NoSuchFile, t.get_bytes, 'c')
257
238
def test_get_with_open_write_stream_sees_all_content(self):
258
239
t = self.get_transport()
259
240
if t.is_readonly():
261
handle = t.open_write_stream('foo')
264
self.assertEqual('b', t.get('foo').read())
242
with t.open_write_stream('foo') as handle:
244
self.assertEqual(b'b', t.get_bytes('foo'))
268
246
def test_get_bytes_with_open_write_stream_sees_all_content(self):
269
247
t = self.get_transport()
270
248
if t.is_readonly():
272
handle = t.open_write_stream('foo')
275
self.assertEqual('b', t.get_bytes('foo'))
276
self.assertEqual('b', t.get('foo').read())
250
with t.open_write_stream('foo') as handle:
252
self.assertEqual(b'b', t.get_bytes('foo'))
253
with t.get('foo') as f:
254
self.assertEqual(b'b', f.read())
280
256
def test_put_bytes(self):
281
257
t = self.get_transport()
283
259
if t.is_readonly():
284
260
self.assertRaises(TransportNotPossible,
285
t.put_bytes, 'a', 'some text for a\n')
261
t.put_bytes, 'a', b'some text for a\n')
288
t.put_bytes('a', 'some text for a\n')
289
self.failUnless(t.has('a'))
290
self.check_transport_contents('some text for a\n', t, 'a')
264
t.put_bytes('a', b'some text for a\n')
265
self.assertTrue(t.has('a'))
266
self.check_transport_contents(b'some text for a\n', t, 'a')
292
268
# The contents should be overwritten
293
t.put_bytes('a', 'new text for a\n')
294
self.check_transport_contents('new text for a\n', t, 'a')
269
t.put_bytes('a', b'new text for a\n')
270
self.check_transport_contents(b'new text for a\n', t, 'a')
296
272
self.assertRaises(NoSuchFile,
297
t.put_bytes, 'path/doesnt/exist/c', 'contents')
273
t.put_bytes, 'path/doesnt/exist/c', b'contents')
299
275
def test_put_bytes_non_atomic(self):
300
276
t = self.get_transport()
302
278
if t.is_readonly():
303
279
self.assertRaises(TransportNotPossible,
304
t.put_bytes_non_atomic, 'a', 'some text for a\n')
280
t.put_bytes_non_atomic, 'a', b'some text for a\n')
307
self.failIf(t.has('a'))
308
t.put_bytes_non_atomic('a', 'some text for a\n')
309
self.failUnless(t.has('a'))
310
self.check_transport_contents('some text for a\n', t, 'a')
283
self.assertFalse(t.has('a'))
284
t.put_bytes_non_atomic('a', b'some text for a\n')
285
self.assertTrue(t.has('a'))
286
self.check_transport_contents(b'some text for a\n', t, 'a')
311
287
# Put also replaces contents
312
t.put_bytes_non_atomic('a', 'new\ncontents for\na\n')
313
self.check_transport_contents('new\ncontents for\na\n', t, 'a')
288
t.put_bytes_non_atomic('a', b'new\ncontents for\na\n')
289
self.check_transport_contents(b'new\ncontents for\na\n', t, 'a')
315
291
# Make sure we can create another file
316
t.put_bytes_non_atomic('d', 'contents for\nd\n')
292
t.put_bytes_non_atomic('d', b'contents for\nd\n')
317
293
# And overwrite 'a' with empty contents
318
t.put_bytes_non_atomic('a', '')
319
self.check_transport_contents('contents for\nd\n', t, 'd')
320
self.check_transport_contents('', t, 'a')
294
t.put_bytes_non_atomic('a', b'')
295
self.check_transport_contents(b'contents for\nd\n', t, 'd')
296
self.check_transport_contents(b'', t, 'a')
322
298
self.assertRaises(NoSuchFile, t.put_bytes_non_atomic, 'no/such/path',
324
300
# Now test the create_parent flag
325
301
self.assertRaises(NoSuchFile, t.put_bytes_non_atomic, 'dir/a',
327
self.failIf(t.has('dir/a'))
328
t.put_bytes_non_atomic('dir/a', 'contents for dir/a\n',
303
self.assertFalse(t.has('dir/a'))
304
t.put_bytes_non_atomic('dir/a', b'contents for dir/a\n',
329
305
create_parent_dir=True)
330
self.check_transport_contents('contents for dir/a\n', t, 'dir/a')
306
self.check_transport_contents(b'contents for dir/a\n', t, 'dir/a')
332
308
# But we still get NoSuchFile if we can't make the parent dir
333
309
self.assertRaises(NoSuchFile, t.put_bytes_non_atomic, 'not/there/a',
335
create_parent_dir=True)
311
create_parent_dir=True)
337
313
def test_put_bytes_permissions(self):
338
314
t = self.get_transport()
342
318
if not t._can_roundtrip_unix_modebits():
343
319
# Can't roundtrip, so no need to run this test
345
t.put_bytes('mode644', 'test text\n', mode=0644)
346
self.assertTransportMode(t, 'mode644', 0644)
347
t.put_bytes('mode666', 'test text\n', mode=0666)
348
self.assertTransportMode(t, 'mode666', 0666)
349
t.put_bytes('mode600', 'test text\n', mode=0600)
350
self.assertTransportMode(t, 'mode600', 0600)
321
t.put_bytes('mode644', b'test text\n', mode=0o644)
322
self.assertTransportMode(t, 'mode644', 0o644)
323
t.put_bytes('mode666', b'test text\n', mode=0o666)
324
self.assertTransportMode(t, 'mode666', 0o666)
325
t.put_bytes('mode600', b'test text\n', mode=0o600)
326
self.assertTransportMode(t, 'mode600', 0o600)
351
327
# Yes, you can put_bytes a file such that it becomes readonly
352
t.put_bytes('mode400', 'test text\n', mode=0400)
353
self.assertTransportMode(t, 'mode400', 0400)
328
t.put_bytes('mode400', b'test text\n', mode=0o400)
329
self.assertTransportMode(t, 'mode400', 0o400)
355
331
# The default permissions should be based on the current umask
356
332
umask = osutils.get_umask()
357
t.put_bytes('nomode', 'test text\n', mode=None)
358
self.assertTransportMode(t, 'nomode', 0666 & ~umask)
333
t.put_bytes('nomode', b'test text\n', mode=None)
334
self.assertTransportMode(t, 'nomode', 0o666 & ~umask)
360
336
def test_put_bytes_non_atomic_permissions(self):
361
337
t = self.get_transport()
365
341
if not t._can_roundtrip_unix_modebits():
366
342
# Can't roundtrip, so no need to run this test
368
t.put_bytes_non_atomic('mode644', 'test text\n', mode=0644)
369
self.assertTransportMode(t, 'mode644', 0644)
370
t.put_bytes_non_atomic('mode666', 'test text\n', mode=0666)
371
self.assertTransportMode(t, 'mode666', 0666)
372
t.put_bytes_non_atomic('mode600', 'test text\n', mode=0600)
373
self.assertTransportMode(t, 'mode600', 0600)
374
t.put_bytes_non_atomic('mode400', 'test text\n', mode=0400)
375
self.assertTransportMode(t, 'mode400', 0400)
344
t.put_bytes_non_atomic('mode644', b'test text\n', mode=0o644)
345
self.assertTransportMode(t, 'mode644', 0o644)
346
t.put_bytes_non_atomic('mode666', b'test text\n', mode=0o666)
347
self.assertTransportMode(t, 'mode666', 0o666)
348
t.put_bytes_non_atomic('mode600', b'test text\n', mode=0o600)
349
self.assertTransportMode(t, 'mode600', 0o600)
350
t.put_bytes_non_atomic('mode400', b'test text\n', mode=0o400)
351
self.assertTransportMode(t, 'mode400', 0o400)
377
353
# The default permissions should be based on the current umask
378
354
umask = osutils.get_umask()
379
t.put_bytes_non_atomic('nomode', 'test text\n', mode=None)
380
self.assertTransportMode(t, 'nomode', 0666 & ~umask)
355
t.put_bytes_non_atomic('nomode', b'test text\n', mode=None)
356
self.assertTransportMode(t, 'nomode', 0o666 & ~umask)
382
358
# We should also be able to set the mode for a parent directory
383
359
# when it is created
384
t.put_bytes_non_atomic('dir700/mode664', 'test text\n', mode=0664,
385
dir_mode=0700, create_parent_dir=True)
386
self.assertTransportMode(t, 'dir700', 0700)
387
t.put_bytes_non_atomic('dir770/mode664', 'test text\n', mode=0664,
388
dir_mode=0770, create_parent_dir=True)
389
self.assertTransportMode(t, 'dir770', 0770)
390
t.put_bytes_non_atomic('dir777/mode664', 'test text\n', mode=0664,
391
dir_mode=0777, create_parent_dir=True)
392
self.assertTransportMode(t, 'dir777', 0777)
360
t.put_bytes_non_atomic('dir700/mode664', b'test text\n', mode=0o664,
361
dir_mode=0o700, create_parent_dir=True)
362
self.assertTransportMode(t, 'dir700', 0o700)
363
t.put_bytes_non_atomic('dir770/mode664', b'test text\n', mode=0o664,
364
dir_mode=0o770, create_parent_dir=True)
365
self.assertTransportMode(t, 'dir770', 0o770)
366
t.put_bytes_non_atomic('dir777/mode664', b'test text\n', mode=0o664,
367
dir_mode=0o777, create_parent_dir=True)
368
self.assertTransportMode(t, 'dir777', 0o777)
394
370
def test_put_file(self):
395
371
t = self.get_transport()
397
373
if t.is_readonly():
398
374
self.assertRaises(TransportNotPossible,
399
t.put_file, 'a', StringIO('some text for a\n'))
375
t.put_file, 'a', BytesIO(b'some text for a\n'))
402
result = t.put_file('a', StringIO('some text for a\n'))
378
result = t.put_file('a', BytesIO(b'some text for a\n'))
403
379
# put_file returns the length of the data written
404
380
self.assertEqual(16, result)
405
self.failUnless(t.has('a'))
406
self.check_transport_contents('some text for a\n', t, 'a')
381
self.assertTrue(t.has('a'))
382
self.check_transport_contents(b'some text for a\n', t, 'a')
407
383
# Put also replaces contents
408
result = t.put_file('a', StringIO('new\ncontents for\na\n'))
384
result = t.put_file('a', BytesIO(b'new\ncontents for\na\n'))
409
385
self.assertEqual(19, result)
410
self.check_transport_contents('new\ncontents for\na\n', t, 'a')
386
self.check_transport_contents(b'new\ncontents for\na\n', t, 'a')
411
387
self.assertRaises(NoSuchFile,
412
388
t.put_file, 'path/doesnt/exist/c',
413
StringIO('contents'))
389
BytesIO(b'contents'))
415
391
def test_put_file_non_atomic(self):
416
392
t = self.get_transport()
418
394
if t.is_readonly():
419
395
self.assertRaises(TransportNotPossible,
420
t.put_file_non_atomic, 'a', StringIO('some text for a\n'))
396
t.put_file_non_atomic, 'a', BytesIO(b'some text for a\n'))
423
self.failIf(t.has('a'))
424
t.put_file_non_atomic('a', StringIO('some text for a\n'))
425
self.failUnless(t.has('a'))
426
self.check_transport_contents('some text for a\n', t, 'a')
399
self.assertFalse(t.has('a'))
400
t.put_file_non_atomic('a', BytesIO(b'some text for a\n'))
401
self.assertTrue(t.has('a'))
402
self.check_transport_contents(b'some text for a\n', t, 'a')
427
403
# Put also replaces contents
428
t.put_file_non_atomic('a', StringIO('new\ncontents for\na\n'))
429
self.check_transport_contents('new\ncontents for\na\n', t, 'a')
404
t.put_file_non_atomic('a', BytesIO(b'new\ncontents for\na\n'))
405
self.check_transport_contents(b'new\ncontents for\na\n', t, 'a')
431
407
# Make sure we can create another file
432
t.put_file_non_atomic('d', StringIO('contents for\nd\n'))
408
t.put_file_non_atomic('d', BytesIO(b'contents for\nd\n'))
433
409
# And overwrite 'a' with empty contents
434
t.put_file_non_atomic('a', StringIO(''))
435
self.check_transport_contents('contents for\nd\n', t, 'd')
436
self.check_transport_contents('', t, 'a')
410
t.put_file_non_atomic('a', BytesIO(b''))
411
self.check_transport_contents(b'contents for\nd\n', t, 'd')
412
self.check_transport_contents(b'', t, 'a')
438
414
self.assertRaises(NoSuchFile, t.put_file_non_atomic, 'no/such/path',
439
StringIO('contents\n'))
415
BytesIO(b'contents\n'))
440
416
# Now test the create_parent flag
441
417
self.assertRaises(NoSuchFile, t.put_file_non_atomic, 'dir/a',
442
StringIO('contents\n'))
443
self.failIf(t.has('dir/a'))
444
t.put_file_non_atomic('dir/a', StringIO('contents for dir/a\n'),
418
BytesIO(b'contents\n'))
419
self.assertFalse(t.has('dir/a'))
420
t.put_file_non_atomic('dir/a', BytesIO(b'contents for dir/a\n'),
445
421
create_parent_dir=True)
446
self.check_transport_contents('contents for dir/a\n', t, 'dir/a')
422
self.check_transport_contents(b'contents for dir/a\n', t, 'dir/a')
448
424
# But we still get NoSuchFile if we can't make the parent dir
449
425
self.assertRaises(NoSuchFile, t.put_file_non_atomic, 'not/there/a',
450
StringIO('contents\n'),
451
create_parent_dir=True)
426
BytesIO(b'contents\n'),
427
create_parent_dir=True)
453
429
def test_put_file_permissions(self):
459
435
if not t._can_roundtrip_unix_modebits():
460
436
# Can't roundtrip, so no need to run this test
462
t.put_file('mode644', StringIO('test text\n'), mode=0644)
463
self.assertTransportMode(t, 'mode644', 0644)
464
t.put_file('mode666', StringIO('test text\n'), mode=0666)
465
self.assertTransportMode(t, 'mode666', 0666)
466
t.put_file('mode600', StringIO('test text\n'), mode=0600)
467
self.assertTransportMode(t, 'mode600', 0600)
438
t.put_file('mode644', BytesIO(b'test text\n'), mode=0o644)
439
self.assertTransportMode(t, 'mode644', 0o644)
440
t.put_file('mode666', BytesIO(b'test text\n'), mode=0o666)
441
self.assertTransportMode(t, 'mode666', 0o666)
442
t.put_file('mode600', BytesIO(b'test text\n'), mode=0o600)
443
self.assertTransportMode(t, 'mode600', 0o600)
468
444
# Yes, you can put a file such that it becomes readonly
469
t.put_file('mode400', StringIO('test text\n'), mode=0400)
470
self.assertTransportMode(t, 'mode400', 0400)
445
t.put_file('mode400', BytesIO(b'test text\n'), mode=0o400)
446
self.assertTransportMode(t, 'mode400', 0o400)
471
447
# The default permissions should be based on the current umask
472
448
umask = osutils.get_umask()
473
t.put_file('nomode', StringIO('test text\n'), mode=None)
474
self.assertTransportMode(t, 'nomode', 0666 & ~umask)
449
t.put_file('nomode', BytesIO(b'test text\n'), mode=None)
450
self.assertTransportMode(t, 'nomode', 0o666 & ~umask)
476
452
def test_put_file_non_atomic_permissions(self):
477
453
t = self.get_transport()
481
457
if not t._can_roundtrip_unix_modebits():
482
458
# Can't roundtrip, so no need to run this test
484
t.put_file_non_atomic('mode644', StringIO('test text\n'), mode=0644)
485
self.assertTransportMode(t, 'mode644', 0644)
486
t.put_file_non_atomic('mode666', StringIO('test text\n'), mode=0666)
487
self.assertTransportMode(t, 'mode666', 0666)
488
t.put_file_non_atomic('mode600', StringIO('test text\n'), mode=0600)
489
self.assertTransportMode(t, 'mode600', 0600)
460
t.put_file_non_atomic('mode644', BytesIO(b'test text\n'), mode=0o644)
461
self.assertTransportMode(t, 'mode644', 0o644)
462
t.put_file_non_atomic('mode666', BytesIO(b'test text\n'), mode=0o666)
463
self.assertTransportMode(t, 'mode666', 0o666)
464
t.put_file_non_atomic('mode600', BytesIO(b'test text\n'), mode=0o600)
465
self.assertTransportMode(t, 'mode600', 0o600)
490
466
# Yes, you can put_file_non_atomic a file such that it becomes readonly
491
t.put_file_non_atomic('mode400', StringIO('test text\n'), mode=0400)
492
self.assertTransportMode(t, 'mode400', 0400)
467
t.put_file_non_atomic('mode400', BytesIO(b'test text\n'), mode=0o400)
468
self.assertTransportMode(t, 'mode400', 0o400)
494
470
# The default permissions should be based on the current umask
495
471
umask = osutils.get_umask()
496
t.put_file_non_atomic('nomode', StringIO('test text\n'), mode=None)
497
self.assertTransportMode(t, 'nomode', 0666 & ~umask)
472
t.put_file_non_atomic('nomode', BytesIO(b'test text\n'), mode=None)
473
self.assertTransportMode(t, 'nomode', 0o666 & ~umask)
499
475
# We should also be able to set the mode for a parent directory
500
476
# when it is created
502
t.put_file_non_atomic('dir700/mode664', sio, mode=0664,
503
dir_mode=0700, create_parent_dir=True)
504
self.assertTransportMode(t, 'dir700', 0700)
505
t.put_file_non_atomic('dir770/mode664', sio, mode=0664,
506
dir_mode=0770, create_parent_dir=True)
507
self.assertTransportMode(t, 'dir770', 0770)
508
t.put_file_non_atomic('dir777/mode664', sio, mode=0664,
509
dir_mode=0777, create_parent_dir=True)
510
self.assertTransportMode(t, 'dir777', 0777)
478
t.put_file_non_atomic('dir700/mode664', sio, mode=0o664,
479
dir_mode=0o700, create_parent_dir=True)
480
self.assertTransportMode(t, 'dir700', 0o700)
481
t.put_file_non_atomic('dir770/mode664', sio, mode=0o664,
482
dir_mode=0o770, create_parent_dir=True)
483
self.assertTransportMode(t, 'dir770', 0o770)
484
t.put_file_non_atomic('dir777/mode664', sio, mode=0o664,
485
dir_mode=0o777, create_parent_dir=True)
486
self.assertTransportMode(t, 'dir777', 0o777)
512
488
def test_put_bytes_unicode(self):
513
# Expect put_bytes to raise AssertionError or UnicodeEncodeError if
514
# given unicode "bytes". UnicodeEncodeError doesn't really make sense
515
# (we don't want to encode unicode here at all, callers should be
516
# strictly passing bytes to put_bytes), but we allow it for backwards
517
# compatibility. At some point we should use a specific exception.
518
# See https://bugs.launchpad.net/bzr/+bug/106898.
519
489
t = self.get_transport()
520
490
if t.is_readonly():
522
492
unicode_string = u'\u1234'
524
(AssertionError, UnicodeEncodeError),
525
t.put_bytes, 'foo', unicode_string)
527
def test_put_file_unicode(self):
528
# Like put_bytes, except with a StringIO.StringIO of a unicode string.
529
# This situation can happen (and has) if code is careless about the type
530
# of "string" they initialise/write to a StringIO with. We cannot use
531
# cStringIO, because it never returns unicode from read.
532
# Like put_bytes, UnicodeEncodeError isn't quite the right exception to
533
# raise, but we raise it for hysterical raisins.
534
t = self.get_transport()
537
unicode_file = pyStringIO(u'\u1234')
538
self.assertRaises(UnicodeEncodeError, t.put_file, 'foo', unicode_file)
493
self.assertRaises(TypeError, t.put_bytes, 'foo', unicode_string)
540
495
def test_mkdir(self):
541
496
t = self.get_transport()
591
541
# no sense testing on this transport
593
543
# Test mkdir with a mode
594
t.mkdir('dmode755', mode=0755)
595
self.assertTransportMode(t, 'dmode755', 0755)
596
t.mkdir('dmode555', mode=0555)
597
self.assertTransportMode(t, 'dmode555', 0555)
598
t.mkdir('dmode777', mode=0777)
599
self.assertTransportMode(t, 'dmode777', 0777)
600
t.mkdir('dmode700', mode=0700)
601
self.assertTransportMode(t, 'dmode700', 0700)
602
t.mkdir_multi(['mdmode755'], mode=0755)
603
self.assertTransportMode(t, 'mdmode755', 0755)
544
t.mkdir('dmode755', mode=0o755)
545
self.assertTransportMode(t, 'dmode755', 0o755)
546
t.mkdir('dmode555', mode=0o555)
547
self.assertTransportMode(t, 'dmode555', 0o555)
548
t.mkdir('dmode777', mode=0o777)
549
self.assertTransportMode(t, 'dmode777', 0o777)
550
t.mkdir('dmode700', mode=0o700)
551
self.assertTransportMode(t, 'dmode700', 0o700)
552
t.mkdir('mdmode755', mode=0o755)
553
self.assertTransportMode(t, 'mdmode755', 0o755)
605
555
# Default mode should be based on umask
606
556
umask = osutils.get_umask()
607
557
t.mkdir('dnomode', mode=None)
608
self.assertTransportMode(t, 'dnomode', 0777 & ~umask)
558
self.assertTransportMode(t, 'dnomode', 0o777 & ~umask)
610
560
def test_opening_a_file_stream_creates_file(self):
611
561
t = self.get_transport()
614
564
handle = t.open_write_stream('foo')
616
self.assertEqual('', t.get_bytes('foo'))
566
self.assertEqual(b'', t.get_bytes('foo'))
620
570
def test_opening_a_file_stream_can_set_mode(self):
621
571
t = self.get_transport()
622
572
if t.is_readonly():
573
self.assertRaises((TransportNotPossible, NotImplementedError),
574
t.open_write_stream, 'foo')
624
576
if not t._can_roundtrip_unix_modebits():
625
577
# Can't roundtrip, so no need to run this test
627
580
def check_mode(name, mode, expected):
628
581
handle = t.open_write_stream(name, mode=mode)
630
583
self.assertTransportMode(t, name, expected)
631
check_mode('mode644', 0644, 0644)
632
check_mode('mode666', 0666, 0666)
633
check_mode('mode600', 0600, 0600)
584
check_mode('mode644', 0o644, 0o644)
585
check_mode('mode666', 0o666, 0o666)
586
check_mode('mode600', 0o600, 0o600)
634
587
# The default permissions should be based on the current umask
635
check_mode('nomode', None, 0666 & ~osutils.get_umask())
588
check_mode('nomode', None, 0o666 & ~osutils.get_umask())
637
590
def test_copy_to(self):
638
591
# FIXME: test: same server to same server (partly done)
700
654
if t.is_readonly():
701
655
self.assertRaises(TransportNotPossible,
702
t.append_file, 'a', 'add\nsome\nmore\ncontents\n')
656
t.append_file, 'a', 'add\nsome\nmore\ncontents\n')
704
t.put_bytes('a', 'diff\ncontents for\na\n')
705
t.put_bytes('b', 'contents\nfor b\n')
658
t.put_bytes('a', b'diff\ncontents for\na\n')
659
t.put_bytes('b', b'contents\nfor b\n')
707
661
self.assertEqual(20,
708
t.append_file('a', StringIO('add\nsome\nmore\ncontents\n')))
662
t.append_file('a', BytesIO(b'add\nsome\nmore\ncontents\n')))
710
664
self.check_transport_contents(
711
'diff\ncontents for\na\nadd\nsome\nmore\ncontents\n',
665
b'diff\ncontents for\na\nadd\nsome\nmore\ncontents\n',
714
668
# a file with no parent should fail..
715
669
self.assertRaises(NoSuchFile,
716
t.append_file, 'missing/path', StringIO('content'))
670
t.append_file, 'missing/path', BytesIO(b'content'))
718
672
# And we can create new files, too
719
673
self.assertEqual(0,
720
t.append_file('c', StringIO('some text\nfor a missing file\n')))
721
self.check_transport_contents('some text\nfor a missing file\n',
674
t.append_file('c', BytesIO(b'some text\nfor a missing file\n')))
675
self.check_transport_contents(b'some text\nfor a missing file\n',
724
678
def test_append_bytes(self):
727
681
if t.is_readonly():
728
682
self.assertRaises(TransportNotPossible,
729
t.append_bytes, 'a', 'add\nsome\nmore\ncontents\n')
683
t.append_bytes, 'a', b'add\nsome\nmore\ncontents\n')
732
self.assertEqual(0, t.append_bytes('a', 'diff\ncontents for\na\n'))
733
self.assertEqual(0, t.append_bytes('b', 'contents\nfor b\n'))
686
self.assertEqual(0, t.append_bytes('a', b'diff\ncontents for\na\n'))
687
self.assertEqual(0, t.append_bytes('b', b'contents\nfor b\n'))
735
689
self.assertEqual(20,
736
t.append_bytes('a', 'add\nsome\nmore\ncontents\n'))
690
t.append_bytes('a', b'add\nsome\nmore\ncontents\n'))
738
692
self.check_transport_contents(
739
'diff\ncontents for\na\nadd\nsome\nmore\ncontents\n',
693
b'diff\ncontents for\na\nadd\nsome\nmore\ncontents\n',
742
696
# a file with no parent should fail..
743
697
self.assertRaises(NoSuchFile,
744
t.append_bytes, 'missing/path', 'content')
746
def test_append_multi(self):
747
t = self.get_transport()
751
t.put_bytes('a', 'diff\ncontents for\na\n'
752
'add\nsome\nmore\ncontents\n')
753
t.put_bytes('b', 'contents\nfor b\n')
755
self.assertEqual((43, 15),
756
t.append_multi([('a', StringIO('and\nthen\nsome\nmore\n')),
757
('b', StringIO('some\nmore\nfor\nb\n'))]))
759
self.check_transport_contents(
760
'diff\ncontents for\na\n'
761
'add\nsome\nmore\ncontents\n'
762
'and\nthen\nsome\nmore\n',
764
self.check_transport_contents(
766
'some\nmore\nfor\nb\n',
769
self.assertEqual((62, 31),
770
t.append_multi(iter([('a', StringIO('a little bit more\n')),
771
('b', StringIO('from an iterator\n'))])))
772
self.check_transport_contents(
773
'diff\ncontents for\na\n'
774
'add\nsome\nmore\ncontents\n'
775
'and\nthen\nsome\nmore\n'
776
'a little bit more\n',
778
self.check_transport_contents(
780
'some\nmore\nfor\nb\n'
781
'from an iterator\n',
784
self.assertEqual((80, 0),
785
t.append_multi([('a', StringIO('some text in a\n')),
786
('d', StringIO('missing file r\n'))]))
788
self.check_transport_contents(
789
'diff\ncontents for\na\n'
790
'add\nsome\nmore\ncontents\n'
791
'and\nthen\nsome\nmore\n'
792
'a little bit more\n'
795
self.check_transport_contents('missing file r\n', t, 'd')
698
t.append_bytes, 'missing/path', b'content')
797
700
def test_append_file_mode(self):
798
701
"""Check that append accepts a mode parameter"""
822
725
self.assertRaises(TransportNotPossible, t.delete, 'missing')
825
t.put_bytes('a', 'a little bit of text\n')
826
self.failUnless(t.has('a'))
728
t.put_bytes('a', b'a little bit of text\n')
729
self.assertTrue(t.has('a'))
828
self.failIf(t.has('a'))
731
self.assertFalse(t.has('a'))
830
733
self.assertRaises(NoSuchFile, t.delete, 'a')
832
t.put_bytes('a', 'a text\n')
833
t.put_bytes('b', 'b text\n')
834
t.put_bytes('c', 'c text\n')
735
t.put_bytes('a', b'a text\n')
736
t.put_bytes('b', b'b text\n')
737
t.put_bytes('c', b'c text\n')
835
738
self.assertEqual([True, True, True],
836
list(t.has_multi(['a', 'b', 'c'])))
837
t.delete_multi(['a', 'c'])
739
[t.has(n) for n in ['a', 'b', 'c']])
838
742
self.assertEqual([False, True, False],
839
list(t.has_multi(['a', 'b', 'c'])))
840
self.failIf(t.has('a'))
841
self.failUnless(t.has('b'))
842
self.failIf(t.has('c'))
844
self.assertRaises(NoSuchFile,
845
t.delete_multi, ['a', 'b', 'c'])
847
self.assertRaises(NoSuchFile,
848
t.delete_multi, iter(['a', 'b', 'c']))
850
t.put_bytes('a', 'another a text\n')
851
t.put_bytes('c', 'another c text\n')
852
t.delete_multi(iter(['a', 'b', 'c']))
743
[t.has(n) for n in ['a', 'b', 'c']])
744
self.assertFalse(t.has('a'))
745
self.assertTrue(t.has('b'))
746
self.assertFalse(t.has('c'))
748
for name in ['a', 'c', 'd']:
749
self.assertRaises(NoSuchFile, t.delete, name)
854
751
# We should have deleted everything
855
752
# SftpServer creates control files in the
991
892
# creates control files in the working directory
992
893
# perhaps all of this could be done in a subdirectory
994
t.put_bytes('a', 'a first file\n')
995
self.assertEquals([True, False], list(t.has_multi(['a', 'b'])))
895
t.put_bytes('a', b'a first file\n')
896
self.assertEqual([True, False], [t.has(n) for n in ['a', 'b']])
998
self.failUnless(t.has('b'))
999
self.failIf(t.has('a'))
899
self.assertTrue(t.has('b'))
900
self.assertFalse(t.has('a'))
1001
self.check_transport_contents('a first file\n', t, 'b')
1002
self.assertEquals([False, True], list(t.has_multi(['a', 'b'])))
902
self.check_transport_contents(b'a first file\n', t, 'b')
903
self.assertEqual([False, True], [t.has(n) for n in ['a', 'b']])
1004
905
# Overwrite a file
1005
t.put_bytes('c', 'c this file\n')
906
t.put_bytes('c', b'c this file\n')
1006
907
t.move('c', 'b')
1007
self.failIf(t.has('c'))
1008
self.check_transport_contents('c this file\n', t, 'b')
908
self.assertFalse(t.has('c'))
909
self.check_transport_contents(b'c this file\n', t, 'b')
1010
911
# TODO: Try to write a test for atomicity
1011
912
# TODO: Test moving into a non-existent subdirectory
1012
# TODO: Test Transport.move_multi
1014
914
def test_copy(self):
1015
915
t = self.get_transport()
1294
1203
self.build_tree(['a', 'b/', 'b/c'], transport=t1)
1296
self.failUnless(t1.has('a'))
1297
self.failUnless(t1.has('b/c'))
1298
self.failIf(t1.has('c'))
1205
self.assertTrue(t1.has('a'))
1206
self.assertTrue(t1.has('b/c'))
1207
self.assertFalse(t1.has('c'))
1300
1209
t2 = t1.clone('b')
1301
1210
self.assertEqual(t1.base + 'b/', t2.base)
1303
self.failUnless(t2.has('c'))
1304
self.failIf(t2.has('a'))
1212
self.assertTrue(t2.has('c'))
1213
self.assertFalse(t2.has('a'))
1306
1215
t3 = t2.clone('..')
1307
self.failUnless(t3.has('a'))
1308
self.failIf(t3.has('c'))
1216
self.assertTrue(t3.has('a'))
1217
self.assertFalse(t3.has('c'))
1310
self.failIf(t1.has('b/d'))
1311
self.failIf(t2.has('d'))
1312
self.failIf(t3.has('b/d'))
1219
self.assertFalse(t1.has('b/d'))
1220
self.assertFalse(t2.has('d'))
1221
self.assertFalse(t3.has('b/d'))
1314
1223
if t1.is_readonly():
1315
self.build_tree_contents([('b/d', 'newfile\n')])
1224
self.build_tree_contents([('b/d', b'newfile\n')])
1317
t2.put_bytes('d', 'newfile\n')
1226
t2.put_bytes('d', b'newfile\n')
1319
self.failUnless(t1.has('b/d'))
1320
self.failUnless(t2.has('d'))
1321
self.failUnless(t3.has('b/d'))
1228
self.assertTrue(t1.has('b/d'))
1229
self.assertTrue(t2.has('d'))
1230
self.assertTrue(t3.has('b/d'))
1323
1232
def test_clone_to_root(self):
1324
1233
orig_transport = self.get_transport()
1535
1445
# '\xe5' and '\xe4' actually map to the same file
1536
1446
# adding a suffix kicks in the 'preserving but insensitive'
1537
1447
# route, and maintains the right files
1538
files = [u'\xe5.1', # a w/ circle iso-8859-1
1539
u'\xe4.2', # a w/ dots iso-8859-1
1540
u'\u017d', # Z with umlat iso-8859-2
1541
u'\u062c', # Arabic j
1542
u'\u0410', # Russian A
1543
u'\u65e5', # Kanji person
1448
files = [u'\xe5.1', # a w/ circle iso-8859-1
1449
u'\xe4.2', # a w/ dots iso-8859-1
1450
u'\u017d', # Z with umlat iso-8859-2
1451
u'\u062c', # Arabic j
1452
u'\u0410', # Russian A
1453
u'\u65e5', # Kanji person
1546
1456
no_unicode_support = getattr(self._server, 'no_unicode_support', False)
1547
1457
if no_unicode_support:
1548
raise tests.KnownFailure("test server cannot handle unicode paths")
1458
self.knownFailure("test server cannot handle unicode paths")
1551
1461
self.build_tree(files, transport=t, line_endings='binary')
1552
1462
except UnicodeError:
1553
raise TestSkipped("cannot handle unicode paths in current encoding")
1464
"cannot handle unicode paths in current encoding")
1555
1466
# A plain unicode string is not a valid url
1556
1467
for fname in files:
1557
self.assertRaises(InvalidURL, t.get, fname)
1468
self.assertRaises(urlutils.InvalidURL, t.get, fname)
1559
1470
for fname in files:
1560
1471
fname_utf8 = fname.encode('utf-8')
1561
contents = 'contents of %s\n' % (fname_utf8,)
1472
contents = b'contents of %s\n' % (fname_utf8,)
1562
1473
self.check_transport_contents(contents, t, urlutils.escape(fname))
1564
1475
def test_connect_twice_is_same_content(self):
1616
1528
def test_readv(self):
1617
1529
transport = self.get_transport()
1618
1530
if transport.is_readonly():
1619
file('a', 'w').write('0123456789')
1531
with open('a', 'w') as f:
1532
f.write('0123456789')
1621
transport.put_bytes('a', '0123456789')
1534
transport.put_bytes('a', b'0123456789')
1623
1536
d = list(transport.readv('a', ((0, 1),)))
1624
self.assertEqual(d[0], (0, '0'))
1537
self.assertEqual(d[0], (0, b'0'))
1626
1539
d = list(transport.readv('a', ((0, 1), (1, 1), (3, 2), (9, 1))))
1627
self.assertEqual(d[0], (0, '0'))
1628
self.assertEqual(d[1], (1, '1'))
1629
self.assertEqual(d[2], (3, '34'))
1630
self.assertEqual(d[3], (9, '9'))
1540
self.assertEqual(d[0], (0, b'0'))
1541
self.assertEqual(d[1], (1, b'1'))
1542
self.assertEqual(d[2], (3, b'34'))
1543
self.assertEqual(d[3], (9, b'9'))
1632
1545
def test_readv_out_of_order(self):
1633
1546
transport = self.get_transport()
1634
1547
if transport.is_readonly():
1635
file('a', 'w').write('0123456789')
1548
with open('a', 'w') as f:
1549
f.write('0123456789')
1637
transport.put_bytes('a', '01234567890')
1551
transport.put_bytes('a', b'01234567890')
1639
1553
d = list(transport.readv('a', ((1, 1), (9, 1), (0, 1), (3, 2))))
1640
self.assertEqual(d[0], (1, '1'))
1641
self.assertEqual(d[1], (9, '9'))
1642
self.assertEqual(d[2], (0, '0'))
1643
self.assertEqual(d[3], (3, '34'))
1554
self.assertEqual(d[0], (1, b'1'))
1555
self.assertEqual(d[1], (9, b'9'))
1556
self.assertEqual(d[2], (0, b'0'))
1557
self.assertEqual(d[3], (3, b'34'))
1645
1559
def test_readv_with_adjust_for_latency(self):
1646
1560
transport = self.get_transport()
1710
1625
transport = self.get_transport()
1711
1626
# test from observed failure case.
1712
1627
if transport.is_readonly():
1713
file('a', 'w').write('a'*1024*1024)
1628
with open('a', 'w') as f:
1629
f.write('a' * 1024 * 1024)
1715
transport.put_bytes('a', 'a'*1024*1024)
1631
transport.put_bytes('a', b'a' * 1024 * 1024)
1716
1632
broken_vector = [(465219, 800), (225221, 800), (445548, 800),
1717
(225037, 800), (221357, 800), (437077, 800), (947670, 800),
1718
(465373, 800), (947422, 800)]
1719
results = list(transport.readv('a', broken_vector, True, 1024*1024))
1720
found_items = [False]*9
1633
(225037, 800), (221357, 800), (437077, 800), (947670, 800),
1634
(465373, 800), (947422, 800)]
1635
results = list(transport.readv('a', broken_vector, True, 1024 * 1024))
1636
found_items = [False] * 9
1721
1637
for pos, (start, length) in enumerate(broken_vector):
1722
1638
# check the range is covered by the result
1723
1639
for offset, data in results:
1724
1640
if offset <= start and start + length <= offset + len(data):
1725
1641
found_items[pos] = True
1726
self.assertEqual([True]*9, found_items)
1642
self.assertEqual([True] * 9, found_items)
1728
1644
def test_get_with_open_write_stream_sees_all_content(self):
1729
1645
t = self.get_transport()
1730
1646
if t.is_readonly():
1732
handle = t.open_write_stream('foo')
1735
self.assertEqual([(0, 'b'), (2, 'd')], list(t.readv('foo', ((0,1), (2,1)))))
1648
with t.open_write_stream('foo') as handle:
1649
handle.write(b'bcd')
1650
self.assertEqual([(0, b'b'), (2, b'd')], list(
1651
t.readv('foo', ((0, 1), (2, 1)))))
1739
1653
def test_get_smart_medium(self):
1740
1654
"""All transports must either give a smart medium, or know they can't.
1742
1656
transport = self.get_transport()
1744
1658
client_medium = transport.get_smart_medium()
1745
self.assertIsInstance(client_medium, medium.SmartClientMedium)
1746
1659
except errors.NoSmartMedium:
1747
1660
# as long as we got it we're fine
1663
from ..bzr.smart import medium
1664
self.assertIsInstance(client_medium, medium.SmartClientMedium)
1750
1666
def test_readv_short_read(self):
1751
1667
transport = self.get_transport()
1752
1668
if transport.is_readonly():
1753
file('a', 'w').write('0123456789')
1669
with open('a', 'w') as f:
1670
f.write('0123456789')
1755
transport.put_bytes('a', '01234567890')
1672
transport.put_bytes('a', b'01234567890')
1757
1674
# This is intentionally reading off the end of the file
1758
1675
# since we are sure that it cannot get there
1759
1676
self.assertListRaises((errors.ShortReadvError, errors.InvalidRange,
1760
1677
# Can be raised by paramiko
1761
1678
AssertionError),
1762
transport.readv, 'a', [(1,1), (8,10)])
1679
transport.readv, 'a', [(1, 1), (8, 10)])
1764
1681
# This is trying to seek past the end of the file, it should
1765
1682
# also raise a special error
1766
1683
self.assertListRaises((errors.ShortReadvError, errors.InvalidRange),
1767
transport.readv, 'a', [(12,2)])
1684
transport.readv, 'a', [(12, 2)])
1686
def test_no_segment_parameters(self):
1687
"""Segment parameters should be stripped and stored in
1688
transport.segment_parameters."""
1689
transport = self.get_transport("foo")
1690
self.assertEqual({}, transport.get_segment_parameters())
1692
def test_segment_parameters(self):
1693
"""Segment parameters should be stripped and stored in
1694
transport.get_segment_parameters()."""
1695
base_url = self._server.get_url()
1696
parameters = {"key1": "val1", "key2": "val2"}
1697
url = urlutils.join_segment_parameters(base_url, parameters)
1698
transport = _mod_transport.get_transport_from_url(url)
1699
self.assertEqual(parameters, transport.get_segment_parameters())
1701
def test_set_segment_parameters(self):
1702
"""Segment parameters can be set and show up in base."""
1703
transport = self.get_transport("foo")
1704
orig_base = transport.base
1705
transport.set_segment_parameter("arm", "board")
1706
self.assertEqual("%s,arm=board" % orig_base, transport.base)
1707
self.assertEqual({"arm": "board"}, transport.get_segment_parameters())
1708
transport.set_segment_parameter("arm", None)
1709
transport.set_segment_parameter("nonexistant", None)
1710
self.assertEqual({}, transport.get_segment_parameters())
1711
self.assertEqual(orig_base, transport.base)
1713
def test_stat_symlink(self):
1714
# if a transport points directly to a symlink (and supports symlinks
1715
# at all) you can tell this. helps with bug 32669.
1716
t = self.get_transport()
1718
t.symlink('target', 'link')
1719
except TransportNotPossible:
1720
raise TestSkipped("symlinks not supported")
1721
t2 = t.clone('link')
1723
self.assertTrue(stat.S_ISLNK(st.st_mode))
1725
def test_abspath_url_unquote_unreserved(self):
1726
"""URLs from abspath should have unreserved characters unquoted
1728
Need consistent quoting notably for tildes, see lp:842223 for more.
1730
t = self.get_transport()
1731
needlessly_escaped_dir = "%2D%2E%30%39%41%5A%5F%61%7A%7E/"
1732
self.assertEqual(t.base + "-.09AZ_az~",
1733
t.abspath(needlessly_escaped_dir))
1735
def test_clone_url_unquote_unreserved(self):
1736
"""Base URL of a cloned branch needs unreserved characters unquoted
1738
Cloned transports should be prefix comparable for things like the
1739
isolation checking of tests, see lp:842223 for more.
1741
t1 = self.get_transport()
1742
needlessly_escaped_dir = "%2D%2E%30%39%41%5A%5F%61%7A%7E/"
1743
self.build_tree([needlessly_escaped_dir], transport=t1)
1744
t2 = t1.clone(needlessly_escaped_dir)
1745
self.assertEqual(t1.base + "-.09AZ_az~/", t2.base)
1747
def test_hook_post_connection_one(self):
1748
"""Fire post_connect hook after a ConnectedTransport is first used"""
1750
Transport.hooks.install_named_hook("post_connect", log.append, None)
1751
t = self.get_transport()
1752
self.assertEqual([], log)
1753
t.has("non-existant")
1754
if isinstance(t, RemoteTransport):
1755
self.assertEqual([t.get_smart_medium()], log)
1756
elif isinstance(t, ConnectedTransport):
1757
self.assertEqual([t], log)
1759
self.assertEqual([], log)
1761
def test_hook_post_connection_multi(self):
1762
"""Fire post_connect hook once per unshared underlying connection"""
1764
Transport.hooks.install_named_hook("post_connect", log.append, None)
1765
t1 = self.get_transport()
1767
t3 = self.get_transport()
1768
self.assertEqual([], log)
1772
if isinstance(t1, RemoteTransport):
1773
self.assertEqual([t.get_smart_medium() for t in [t1, t3]], log)
1774
elif isinstance(t1, ConnectedTransport):
1775
self.assertEqual([t1, t3], log)
1777
self.assertEqual([], log)