• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1#!/usr/bin/env python
2# -*- coding: utf-8 -*-
3# Copyright 2014 The Chromium Authors. All rights reserved.
4# Use of this source code is governed by a BSD-style license that can be
5# found in the LICENSE file.
6"""Copies files to a directory."""
7
8import filecmp
9import itertools
10import optparse
11import os
12import shutil
13import sys
14
15from util import build_utils
16
17
18def copy_tree(src: str,
19              dest: str,
20              follow_all_symlinks=False,
21              follow_outside_symlinks=False):
22    """copy src/* to dest/
23  I. If follow_outside_symlinks is true,
24     1. If src item is a symlink, and points to some item inside src,
25        then copy the symlink to dest.
26     2. If src item points to items outside src, then follow links to copy
27        the original file to dest.
28     3. Else copy src item to dest.
29  II. If follow_all_symlinks is true,
30     1. If src item is a symlink, then follow links to copy the original file
31        to dest.
32     2. Else copy src item to dest.
33  follow_outside_symlinks is true when follow_all_symlinks is true.
34  """
35    with os.scandir(src) as itr:
36        items = list(itr)
37    return _do_copy_tree(items,
38                         src,
39                         dest,
40                         follow_all_symlinks=follow_all_symlinks,
41                         follow_outside_symlinks=follow_outside_symlinks)
42
43
44def _do_copy_tree(items: list,
45                  src: str,
46                  dest: str,
47                  follow_all_symlinks: bool=False,
48                  follow_outside_symlinks: bool=False) -> str:
49    os.makedirs(dest, exist_ok=True)
50    for item in items:
51        srcname = os.path.join(src, item.name)
52        destname = os.path.join(dest, item.name)
53        is_symlink = item.is_symlink()
54        if is_symlink:
55            org_linkto = os.readlink(srcname)
56            linkto = org_linkto
57            if not os.path.isabs(org_linkto):
58                linkto = os.path.join(os.path.dirname(item), org_linkto)
59
60            if not os.path.exists(linkto):
61                os.symlink(org_linkto, destname)
62                shutil.copymode(srcname, destname, follow_symlinks=False)
63                continue
64
65            if follow_all_symlinks:
66                if item.is_dir():
67                    copy_tree(item,
68                              destname,
69                              follow_all_symlinks=follow_all_symlinks,
70                              follow_outside_symlinks=follow_outside_symlinks)
71                else:
72                    shutil.copy(item, destname)
73                    shutil.copymode(item, destname)
74
75            elif follow_outside_symlinks:
76                if os.path.abspath(src) in os.path.abspath(
77                        linkto) and not os.path.isabs(org_linkto):
78                    os.symlink(org_linkto, destname)
79                    shutil.copymode(srcname, destname, follow_symlinks=False)
80                else:
81                    if item.is_dir():
82                        copy_tree(
83                            item,
84                            destname,
85                            follow_all_symlinks=follow_all_symlinks,
86                            follow_outside_symlinks=follow_outside_symlinks)
87                    else:
88                        shutil.copy(item, destname)
89                        shutil.copymode(item, destname)
90            else:
91                os.symlink(org_linkto, destname)
92                shutil.copymode(srcname, destname, follow_symlinks=False)
93        elif item.is_dir():
94            copy_tree(item,
95                      destname,
96                      follow_all_symlinks=follow_all_symlinks,
97                      follow_outside_symlinks=follow_outside_symlinks)
98        else:
99            shutil.copy(item, destname)
100            shutil.copymode(item, destname)
101    shutil.copystat(src, dest)
102    return dest
103
104
105def copy_file(f: str,
106              dest: str,
107              deps: list,
108              follow_all_symlinks: bool=False,
109              follow_outside_symlinks: bool=False):
110    """Copy file or directory and update deps."""
111    if os.path.isdir(f):
112        copy_tree(f,
113                  os.path.join(dest, os.path.basename(f)),
114                  follow_all_symlinks=follow_all_symlinks,
115                  follow_outside_symlinks=follow_outside_symlinks)
116        deps.extend(build_utils.get_all_files(f))
117    else:
118        if os.path.isfile(os.path.join(dest, os.path.basename(f))):
119            dest = os.path.join(dest, os.path.basename(f))
120
121        deps.append(f)
122
123        if os.path.isfile(dest):
124            if filecmp.cmp(dest, f, shallow=False):
125                return
126            # The shutil.copy() below would fail if the file does not
127            # have write permissions. Deleting the file
128            # has similar costs to modifying the permissions.
129            os.unlink(dest)
130
131        shutil.copy(f, dest)
132
133
134def do_copy(options, deps: list):
135    """Copy files or directories given in options.files and update deps."""
136    files = list(
137        itertools.chain.from_iterable(
138            build_utils.parse_gn_list(f) for f in options.files))
139
140    for f in files:
141        if not options.ignore_stale:
142            if os.path.isdir(f) and not options.clear:
143                print('To avoid stale files you must use --clear when copying '
144                      'directories')
145                sys.exit(-1)
146        copy_file(f,
147                  options.dest,
148                  deps,
149                  follow_all_symlinks=options.follow_all_symlinks,
150                  follow_outside_symlinks=options.follow_outside_symlinks)
151
152
153def do_renaming(options, deps: list):
154    """Copy and rename files given in options.renaming_sources
155    and update deps.
156    """
157    src_files = list(
158        itertools.chain.from_iterable(
159            build_utils.parse_gn_list(f) for f in options.renaming_sources))
160
161    dest_files = list(
162        itertools.chain.from_iterable(
163            build_utils.parse_gn_list(f)
164            for f in options.renaming_destinations))
165
166    if (len(src_files) != len(dest_files)):
167        print('Renaming source and destination files not match.')
168        sys.exit(-1)
169
170    for src, dest in zip(src_files, dest_files):
171        if os.path.isdir(src):
172            print('renaming directory is not supported.')
173            sys.exit(-1)
174        else:
175            copy_file(src, os.path.join(options.dest, dest), deps)
176
177
178def main(args):
179    args = build_utils.expand_file_args(args)
180
181    parser = optparse.OptionParser()
182    build_utils.add_depfile_option(parser)
183
184    parser.add_option('--dest', help='Directory to copy files to.')
185    parser.add_option('--files',
186                      action='append',
187                      help='List of files to copy.')
188    parser.add_option(
189        '--clear',
190        action='store_true',
191        help='If set, the destination directory will be deleted '
192        'before copying files to it. This is highly recommended to '
193        'ensure that no stale files are left in the directory.')
194    parser.add_option('--ignore-stale',
195                      action='store_true',
196                      help='Do copy even there may be stale files. '
197                      'If set, overrule options.clear')
198    parser.add_option('--stamp', help='Path to touch on success.')
199    parser.add_option('--follow-all-symlinks',
200                      action='store_true',
201                      help='whether to follow symlinks')
202    parser.add_option(
203        '--follow-outside-symlinks',
204        action='store_true',
205        help='whether to follow symlinks which points to targets outside of'
206        'source directory')
207    parser.add_option('--renaming-sources',
208                      action='append',
209                      help='List of files need to be renamed while being '
210                      'copied to dest directory')
211    parser.add_option('--renaming-destinations',
212                      action='append',
213                      help='List of destination file name without path, the '
214                      'number of elements must match rename-sources.')
215
216    options, _ = parser.parse_args(args)
217    if options.follow_all_symlinks:
218        options.follow_outside_symlinks = True
219
220    if options.clear and not options.ignore_stale:
221        build_utils.delete_directory(options.dest)
222        build_utils.make_directory(options.dest)
223
224    deps = []
225
226    if options.files:
227        do_copy(options, deps)
228
229    if options.renaming_sources:
230        do_renaming(options, deps)
231
232    if options.depfile:
233        build_utils.write_depfile(options.depfile,
234                                  options.stamp,
235                                  deps,
236                                  add_pydeps=False)
237
238    if options.stamp:
239        build_utils.touch(options.stamp)
240
241
242if __name__ == '__main__':
243    sys.exit(main(sys.argv[1:]))
244