diff options
author | Richard Purdie <richard.purdie@linuxfoundation.org> | 2011-02-09 13:01:23 +0000 |
---|---|---|
committer | Richard Purdie <rpurdie@linux.intel.com> | 2011-02-09 22:46:30 +0000 |
commit | 58ae42cbc386a8cd59f12527d3e47bea82d9b36f (patch) | |
tree | 7df0d8bf79ebe53ab7a8bce8a0596272d9264f41 | |
parent | bb3db5106d4722ac177fcd5b2ae3d4e45d12be47 (diff) | |
download | poky-58ae42cbc386a8cd59f12527d3e47bea82d9b36f.tar.gz |
lib.oe.path: Update copytree function to call shell commands since its twice as fast
As an added bonus, hardlinks between files in the tree will be preserved too.
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
-rw-r--r-- | meta/lib/oe/path.py | 44 |
1 files changed, 12 insertions, 32 deletions
diff --git a/meta/lib/oe/path.py b/meta/lib/oe/path.py index f42faeab26..da7811fac4 100644 --- a/meta/lib/oe/path.py +++ b/meta/lib/oe/path.py | |||
@@ -46,43 +46,23 @@ def format_display(path, metadata): | |||
46 | return rel | 46 | return rel |
47 | 47 | ||
48 | 48 | ||
49 | class Error(EnvironmentError): | 49 | class CopyFailed(Exception): |
50 | pass | 50 | pass |
51 | 51 | ||
52 | # Based on shutil.copytree but with features removed and | ||
53 | # No fatal error is dst already exists | ||
54 | # Handle symlinks that already exist | ||
55 | def copytree(src, dst): | 52 | def copytree(src, dst): |
56 | names = os.listdir(src) | 53 | # We could use something like shutil.copytree here but it turns out to |
54 | # to be slow. It takes twice as long copying to an empty directory. | ||
55 | # If dst already has contents performance can be 15 time slower | ||
56 | # This way we also preserve hardlinks between files in the tree. | ||
57 | 57 | ||
58 | bb.mkdirhier(dst) | 58 | import subprocess |
59 | 59 | ||
60 | errors = [] | 60 | bb.mkdirhier(dst) |
61 | for name in names: | 61 | cmd = 'tar -cf - -C %s -ps . | tar -xf - -C %s' % (src, dst) |
62 | srcname = os.path.join(src, name) | 62 | ret = subprocess.call(cmd, shell=True) |
63 | dstname = os.path.join(dst, name) | 63 | if ret != 0: |
64 | try: | 64 | raise CopyFailed("Command %s failed with return value %s" % (cmd, ret)) |
65 | if os.path.islink(srcname): | 65 | return |
66 | linkto = os.readlink(srcname) | ||
67 | if os.path.lexists(dstname): | ||
68 | os.unlink(dstname) | ||
69 | os.symlink(linkto, dstname) | ||
70 | elif os.path.isdir(srcname): | ||
71 | copytree(srcname, dstname) | ||
72 | else: | ||
73 | bb.utils.copyfile(srcname, dstname) | ||
74 | except (IOError, os.error), why: | ||
75 | errors.append((srcname, dstname, str(why))) | ||
76 | # catch the Error from the recursive copytree so that we can | ||
77 | # continue with other files | ||
78 | except Error, err: | ||
79 | errors.extend(err.args[0]) | ||
80 | try: | ||
81 | shutil.copystat(src, dst) | ||
82 | except OSError, why: | ||
83 | errors.extend((src, dst, str(why))) | ||
84 | if errors: | ||
85 | raise Error, errors | ||
86 | 66 | ||
87 | def remove(path): | 67 | def remove(path): |
88 | """Equivalent to rm -f or rm -rf""" | 68 | """Equivalent to rm -f or rm -rf""" |