2
from StringIO import StringIO
3
from shutil import rmtree, copy2, copytree
6
from unittest import makeSuite
10
revision as _mod_revision,
13
from bzrlib.bzrdir import BzrDir
14
from bzrlib.export import export
15
from bzrlib.plugins.bzrtools import errors
16
from bzrlib.plugins.bzrtools.upstream_import import (
26
from bzrlib.tests import (
28
TestCaseWithTransport,
31
from bzrlib.tests.features import UnicodeFilenameFeature
32
except ImportError: # bzr < 2.5
33
from bzrlib.tests import UnicodeFilenameFeature
36
def import_tar_broken(tree, tar_input):
38
Import a tarfile with names that that end in //, e.g. Feisty Python 2.5
40
tar_file = tarfile.open('lala', 'r', tar_input)
41
for member in tar_file.members:
42
if member.name.endswith('/'):
44
import_archive(tree, tar_file)
47
class DirFileWriter(object):
49
def __init__(self, fileobj, mode):
50
# We may be asked to 'append'. If so, fileobj already has a path.
51
# So we copy the existing tree, and overwrite afterward.
53
existing = fileobj.read()
55
path = tempfile.mkdtemp(dir=os.getcwd())
57
# copytree requires the directory not to exist
59
copytree(existing, path)
64
target_path = os.path.join(self.root, path)
65
parent = osutils.dirname(target_path)
66
if not os.path.exists(parent):
68
kind = osutils.file_kind(path)
70
copy2(path, target_path)
71
if kind == 'directory':
78
class TestImport(TestCaseInTempDir):
80
def make_tar(self, mode='w'):
82
return tarfile.open('project-0.1.tar', mode, fileobj)
83
return self.make_archive(maker)
85
def make_archive(self, maker, subdir=True):
87
archive_file = maker(result)
89
os.mkdir('project-0.1')
92
archive_file.add('project-0.1')
95
os.chdir('project-0.1')
96
os.mkdir(prefix + 'junk')
97
archive_file.add(prefix + 'junk')
99
f = file(prefix + 'README', 'wb')
102
archive_file.add(prefix + 'README')
104
f = file(prefix + 'FEEDME', 'wb')
107
archive_file.add(prefix + 'FEEDME')
113
rmtree('project-0.1')
117
def make_archive2(self, builder, subdir):
119
archive_file = builder(result)
120
os.mkdir('project-0.2')
123
prefix='project-0.2/'
124
archive_file.add('project-0.2')
127
os.chdir('project-0.2')
129
os.mkdir(prefix + 'junk')
130
archive_file.add(prefix + 'junk')
132
f = file(prefix + 'README', 'wb')
135
archive_file.add(prefix + 'README')
137
f = file(prefix + 'README', 'wb')
140
# Add a second entry for README with different contents.
141
archive_file.add(prefix + 'README')
150
def make_messed_tar(self):
152
tar_file = tarfile.open('project-0.1.tar', 'w', result)
153
os.mkdir('project-0.1')
154
tar_file.add('project-0.1')
156
os.mkdir('project-0.2')
157
tar_file.add('project-0.2')
159
f = file('project-0.1/README', 'wb')
162
tar_file.add('project-0.1/README')
164
rmtree('project-0.1')
170
return ZipFileWrapper(fileobj, 'w')
171
return self.make_archive(maker)
173
def make_tar_with_bzrdir(self):
175
tar_file = tarfile.open('tar-with-bzrdir.tar', 'w', result)
176
os.mkdir('toplevel-dir')
177
tar_file.add('toplevel-dir')
178
os.mkdir('toplevel-dir/.bzr')
179
tar_file.add('toplevel-dir/.bzr')
181
rmtree('toplevel-dir')
185
def test_top_path(self):
186
self.assertEqual(top_path('ab/b/c'), 'ab')
187
self.assertEqual(top_path('etc'), 'etc')
188
self.assertEqual(top_path('project-0.1'), 'project-0.1')
190
def test_common_directory(self):
191
self.assertEqual(common_directory(['ab/c/d', 'ab/c/e']), 'ab')
192
self.assertIs(common_directory(['ab/c/d', 'ac/c/e']), None)
193
self.assertEqual('FEEDME', common_directory(['FEEDME']))
195
def test_untar(self):
196
def builder(fileobj, mode='w'):
197
return tarfile.open('project-0.1.tar', mode, fileobj)
198
self.archive_test(builder, import_tar)
200
def test_broken_tar(self):
201
def builder(fileobj, mode='w'):
202
return tarfile.open('project-0.1.tar', mode, fileobj)
203
self.archive_test(builder, import_tar_broken, subdir=True)
205
def test_unzip(self):
206
def builder(fileobj, mode='w'):
207
return ZipFileWrapper(fileobj, mode)
208
self.archive_test(builder, import_zip)
210
def test_copydir_nosub(self):
211
def builder(fileobj, mode='w'):
212
return DirFileWriter(fileobj, mode)
213
# It would be bogus to test with the result in a subdirectory,
214
# because for directories, the input root is always the output root.
215
self.archive_test(builder, import_dir)
217
def archive_test(self, builder, importer, subdir=False):
218
archive_file = self.make_archive(builder, subdir)
219
tree = BzrDir.create_standalone_workingtree('tree')
222
importer(tree, archive_file)
223
self.assertTrue(tree.path2id('README') is not None)
224
self.assertTrue(tree.path2id('FEEDME') is not None)
225
self.assertTrue(os.path.isfile(tree.abspath('README')))
226
self.assertEqual(tree.stored_kind(tree.path2id('README')),
228
self.assertEqual(tree.stored_kind(tree.path2id('FEEDME')),
230
f = file(tree.abspath('junk/food'), 'wb')
231
f.write('I like food\n')
234
archive_file = self.make_archive2(builder, subdir)
235
importer(tree, archive_file)
236
self.assertTrue(tree.path2id('README') is not None)
237
# Ensure the second version of the file is used.
238
self.assertEqual(tree.get_file_text(tree.path2id('README')),
240
self.assertTrue(not os.path.exists(tree.abspath('FEEDME')))
245
def test_untar2(self):
246
tar_file = self.make_messed_tar()
247
tree = BzrDir.create_standalone_workingtree('tree')
248
import_tar(tree, tar_file)
249
self.assertTrue(tree.path2id('project-0.1/README') is not None)
251
def test_untar_gzip(self):
252
tar_file = self.make_tar(mode='w:gz')
253
tree = BzrDir.create_standalone_workingtree('tree')
254
import_tar(tree, tar_file)
255
self.assertTrue(tree.path2id('README') is not None)
257
def test_no_crash_with_bzrdir(self):
258
tar_file = self.make_tar_with_bzrdir()
259
tree = BzrDir.create_standalone_workingtree('tree')
260
import_tar(tree, tar_file)
261
# So long as it did not crash, that should be ok
263
def test_get_archive_type(self):
264
self.assertEqual(('tar', None), get_archive_type('foo.tar'))
265
self.assertEqual(('zip', None), get_archive_type('foo.zip'))
266
self.assertRaises(errors.NotArchiveType, get_archive_type, 'foo.gif')
267
self.assertEqual(('tar', 'gz'), get_archive_type('foo.tar.gz'))
268
self.assertRaises(errors.NotArchiveType, get_archive_type,
270
self.assertEqual(('tar', 'gz'), get_archive_type('foo.tgz'))
271
self.assertEqual(('tar', 'lzma'), get_archive_type('foo.tar.lzma'))
272
self.assertEqual(('tar', 'lzma'), get_archive_type('foo.tar.xz'))
273
self.assertEqual(('tar', 'bz2'), get_archive_type('foo.tar.bz2'))
276
class TestWithStuff(TestCaseWithTransport):
278
def transform_to_tar(self, tt):
280
export(tt.get_preview_tree(), root='', fileobj=stream, format='tar',
284
def get_empty_tt(self):
285
b = self.make_repository('foo')
286
null_tree = b.revision_tree(_mod_revision.NULL_REVISION)
287
tt = transform.TransformPreview(null_tree)
288
root = tt.new_directory('', transform.ROOT_PARENT, 'tree-root')
290
self.addCleanup(tt.finalize)
293
def test_nonascii_paths(self):
294
self.requireFeature(UnicodeFilenameFeature)
295
tt = self.get_empty_tt()
296
encoded_file = tt.new_file(
297
u'\u1234file', tt.root, 'contents', 'new-file')
298
encoded_file = tt.new_file(
299
'other', tt.root, 'contents', 'other-file')
300
tarfile = self.transform_to_tar(tt)
302
tree = self.make_branch_and_tree('bar')
303
import_tar(tree, tarfile)
304
self.assertPathExists(u'bar/\u1234file')
308
return makeSuite(TestImport)