blob: 087ae9dcc66eb3c494e8434f1b8aadce40c4662b [file] [log] [blame]
Greg Warda82122b2000-02-17 23:56:15 +00001"""distutils.command.sdist
2
3Implements the Distutils 'sdist' command (create a source distribution)."""
4
Greg Ward3ce77fd2000-03-02 01:49:45 +00005__revision__ = "$Id$"
Greg Warda82122b2000-02-17 23:56:15 +00006
Tarek Ziadé2b66da72009-12-21 01:22:46 +00007import os
8import string
Tarek Ziadé85d6fb52009-01-04 00:04:49 +00009import sys
Greg Warda82122b2000-02-17 23:56:15 +000010from glob import glob
Tarek Ziadécb768042009-05-16 16:37:06 +000011from warnings import warn
12
Greg Warda82122b2000-02-17 23:56:15 +000013from distutils.core import Command
Greg Wardab3a0f32000-08-05 01:31:54 +000014from distutils import dir_util, dep_util, file_util, archive_util
Greg Warda82122b2000-02-17 23:56:15 +000015from distutils.text_file import TextFile
Tarek Ziadé2b66da72009-12-21 01:22:46 +000016from distutils.errors import (DistutilsPlatformError, DistutilsOptionError,
17 DistutilsTemplateError)
Greg Ward4571ac12000-07-30 01:05:02 +000018from distutils.filelist import FileList
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +000019from distutils import log
Tarek Ziadéf68b5b82009-02-17 09:42:44 +000020from distutils.util import convert_path
Greg Warda82122b2000-02-17 23:56:15 +000021
Tarek Ziadé064a7682009-05-14 12:40:59 +000022def show_formats():
Greg Ward34593812000-06-24 01:23:37 +000023 """Print all possible values for the 'formats' option (used by
24 the "--help-formats" command-line option).
25 """
26 from distutils.fancy_getopt import FancyGetopt
27 from distutils.archive_util import ARCHIVE_FORMATS
Tarek Ziadé064a7682009-05-14 12:40:59 +000028 formats = []
Greg Ward34593812000-06-24 01:23:37 +000029 for format in ARCHIVE_FORMATS.keys():
30 formats.append(("formats=" + format, None,
31 ARCHIVE_FORMATS[format][2]))
32 formats.sort()
Tarek Ziadé064a7682009-05-14 12:40:59 +000033 FancyGetopt(formats).print_help(
Greg Ward34593812000-06-24 01:23:37 +000034 "List of available source distribution formats:")
35
Tarek Ziadé89539132009-05-14 14:56:14 +000036class sdist(Command):
Greg Warda82122b2000-02-17 23:56:15 +000037
38 description = "create a source distribution (tarball, zip file, etc.)"
39
Tarek Ziadécb768042009-05-16 16:37:06 +000040 def checking_metadata(self):
41 """Callable used for the check sub-command.
42
43 Placed here so user_options can view it"""
44 return self.metadata_check
45
Greg Wardbbeceea2000-02-18 00:25:39 +000046 user_options = [
47 ('template=', 't',
48 "name of manifest template file [default: MANIFEST.in]"),
49 ('manifest=', 'm',
50 "name of manifest file [default: MANIFEST]"),
51 ('use-defaults', None,
52 "include the default file set in the manifest "
53 "[default; disable with --no-defaults]"),
Greg Ward499822d2000-06-29 02:06:29 +000054 ('no-defaults', None,
55 "don't include the default file set"),
56 ('prune', None,
57 "specifically exclude files/directories that should not be "
58 "distributed (build tree, RCS/CVS dirs, etc.) "
59 "[default; disable with --no-prune]"),
60 ('no-prune', None,
61 "don't automatically exclude anything"),
Greg Ward839d5322000-04-26 01:14:33 +000062 ('manifest-only', 'o',
Greg Wardc3c8c6e2000-06-08 00:46:45 +000063 "just regenerate the manifest and then stop "
64 "(implies --force-manifest)"),
Greg Ward839d5322000-04-26 01:14:33 +000065 ('force-manifest', 'f',
Tarek Ziadé4fc2a002010-05-17 10:54:43 +000066 "forcibly regenerate the manifest and carry on as usual. "
67 "Deprecated: now the manifest is always regenerated."),
Greg Wardbbeceea2000-02-18 00:25:39 +000068 ('formats=', None,
Greg Ward2ff78872000-06-24 00:23:20 +000069 "formats for source distribution (comma-separated list)"),
Greg Wardaf64aed2000-09-25 01:51:01 +000070 ('keep-temp', 'k',
Greg Wardbbeceea2000-02-18 00:25:39 +000071 "keep the distribution tree around after creating " +
72 "archive file(s)"),
Greg Wardc0614102000-07-05 03:06:46 +000073 ('dist-dir=', 'd',
74 "directory to put the source distribution archive(s) in "
75 "[default: dist]"),
Tarek Ziadécb768042009-05-16 16:37:06 +000076 ('medata-check', None,
77 "Ensure that all required elements of meta-data "
78 "are supplied. Warn if any missing. [default]"),
Tarek Ziadé1b486712009-10-02 23:49:48 +000079 ('owner=', 'u',
80 "Owner name used when creating a tar file [default: current user]"),
81 ('group=', 'g',
82 "Group name used when creating a tar file [default: current group]"),
Greg Wardbbeceea2000-02-18 00:25:39 +000083 ]
Greg Wardf1fe1032000-06-08 00:14:18 +000084
Greg Ward99b032e2000-09-25 01:41:15 +000085 boolean_options = ['use-defaults', 'prune',
86 'manifest-only', 'force-manifest',
Tarek Ziadécb768042009-05-16 16:37:06 +000087 'keep-temp', 'metadata-check']
Greg Wardf1fe1032000-06-08 00:14:18 +000088
Greg Ward9d17a7a2000-06-07 03:00:06 +000089 help_options = [
90 ('help-formats', None,
Greg Ward2ff78872000-06-24 00:23:20 +000091 "list available distribution formats", show_formats),
Greg Wardfa9ff762000-10-14 04:06:40 +000092 ]
Greg Ward9d17a7a2000-06-07 03:00:06 +000093
Greg Ward499822d2000-06-29 02:06:29 +000094 negative_opt = {'no-defaults': 'use-defaults',
95 'no-prune': 'prune' }
Greg Warda82122b2000-02-17 23:56:15 +000096
Tarek Ziadé89539132009-05-14 14:56:14 +000097 default_format = {'posix': 'gztar',
98 'nt': 'zip' }
Greg Warda82122b2000-02-17 23:56:15 +000099
Tarek Ziadécb768042009-05-16 16:37:06 +0000100 sub_commands = [('check', checking_metadata)]
101
Tarek Ziadé89539132009-05-14 14:56:14 +0000102 def initialize_options(self):
Greg Warda82122b2000-02-17 23:56:15 +0000103 # 'template' and 'manifest' are, respectively, the names of
104 # the manifest template and manifest file.
105 self.template = None
106 self.manifest = None
107
108 # 'use_defaults': if true, we will include the default file set
109 # in the manifest
110 self.use_defaults = 1
Greg Ward499822d2000-06-29 02:06:29 +0000111 self.prune = 1
Greg Warda82122b2000-02-17 23:56:15 +0000112
113 self.manifest_only = 0
114 self.force_manifest = 0
115
116 self.formats = None
Greg Wardaf64aed2000-09-25 01:51:01 +0000117 self.keep_temp = 0
Greg Wardc0614102000-07-05 03:06:46 +0000118 self.dist_dir = None
Greg Warda82122b2000-02-17 23:56:15 +0000119
Greg Wardd87eb732000-06-01 01:10:56 +0000120 self.archive_files = None
Tarek Ziadécb768042009-05-16 16:37:06 +0000121 self.metadata_check = 1
Tarek Ziadé1b486712009-10-02 23:49:48 +0000122 self.owner = None
123 self.group = None
Greg Wardd87eb732000-06-01 01:10:56 +0000124
Tarek Ziadé89539132009-05-14 14:56:14 +0000125 def finalize_options(self):
Greg Warda82122b2000-02-17 23:56:15 +0000126 if self.manifest is None:
127 self.manifest = "MANIFEST"
128 if self.template is None:
129 self.template = "MANIFEST.in"
130
Greg Ward62d5a572000-06-04 15:12:51 +0000131 self.ensure_string_list('formats')
Greg Warda82122b2000-02-17 23:56:15 +0000132 if self.formats is None:
133 try:
134 self.formats = [self.default_format[os.name]]
135 except KeyError:
136 raise DistutilsPlatformError, \
Greg Ward578c10d2000-03-31 02:50:04 +0000137 "don't know how to create source distributions " + \
138 "on platform %s" % os.name
Greg Warda82122b2000-02-17 23:56:15 +0000139
Greg Wardcb1f4c42000-09-30 18:27:54 +0000140 bad_format = archive_util.check_archive_formats(self.formats)
Greg Ward6a9a5452000-04-22 03:11:55 +0000141 if bad_format:
142 raise DistutilsOptionError, \
143 "unknown archive format '%s'" % bad_format
144
Greg Wardc0614102000-07-05 03:06:46 +0000145 if self.dist_dir is None:
146 self.dist_dir = "dist"
147
Tarek Ziadé89539132009-05-14 14:56:14 +0000148 def run(self):
Greg Ward23266fe2000-07-30 01:30:31 +0000149 # 'filelist' contains the list of files that will make up the
150 # manifest
151 self.filelist = FileList()
Fred Drake21d45352001-12-06 21:01:19 +0000152
Tarek Ziadécb768042009-05-16 16:37:06 +0000153 # Run sub commands
154 for cmd_name in self.get_sub_commands():
155 self.run_command(cmd_name)
Greg Warda82122b2000-02-17 23:56:15 +0000156
157 # Do whatever it takes to get the list of files to process
158 # (process the manifest template, read an existing manifest,
Greg Ward23266fe2000-07-30 01:30:31 +0000159 # whatever). File list is accumulated in 'self.filelist'.
Greg Wardcb1f4c42000-09-30 18:27:54 +0000160 self.get_file_list()
Greg Warda82122b2000-02-17 23:56:15 +0000161
162 # If user just wanted us to regenerate the manifest, stop now.
163 if self.manifest_only:
164 return
165
166 # Otherwise, go ahead and create the source distribution tarball,
167 # or zipfile, or whatever.
Greg Wardcb1f4c42000-09-30 18:27:54 +0000168 self.make_distribution()
Greg Warda82122b2000-02-17 23:56:15 +0000169
Tarek Ziadé89539132009-05-14 14:56:14 +0000170 def check_metadata(self):
Tarek Ziadécb768042009-05-16 16:37:06 +0000171 """Deprecated API."""
172 warn("distutils.command.sdist.check_metadata is deprecated, \
173 use the check command instead", PendingDeprecationWarning)
174 check = self.distribution.get_command_obj('check')
175 check.ensure_finalized()
176 check.run()
Greg Warda82122b2000-02-17 23:56:15 +0000177
Tarek Ziadé89539132009-05-14 14:56:14 +0000178 def get_file_list(self):
Greg Warda82122b2000-02-17 23:56:15 +0000179 """Figure out the list of files to include in the source
Greg Ward23266fe2000-07-30 01:30:31 +0000180 distribution, and put it in 'self.filelist'. This might involve
Greg Warde0c8c2f2000-06-08 00:24:01 +0000181 reading the manifest template (and writing the manifest), or just
182 reading the manifest, or just using the default file set -- it all
Tarek Ziadé422545f2010-05-17 10:06:20 +0000183 depends on the user's options.
Greg Warde0c8c2f2000-06-08 00:24:01 +0000184 """
Tarek Ziadé422545f2010-05-17 10:06:20 +0000185 # new behavior:
186 # the file list is recalculated everytime because
187 # even if MANIFEST.in or setup.py are not changed
188 # the user might have added some files in the tree that
189 # need to be included.
190 #
191 # This makes --force the default and only behavior.
Greg Ward23266fe2000-07-30 01:30:31 +0000192 template_exists = os.path.isfile(self.template)
Tarek Ziadé422545f2010-05-17 10:06:20 +0000193 if not template_exists:
194 self.warn(("manifest template '%s' does not exist " +
195 "(using default file list)") %
196 self.template)
197 self.filelist.findall()
198
199 if self.use_defaults:
200 self.add_defaults()
201
Greg Warda82122b2000-02-17 23:56:15 +0000202 if template_exists:
Tarek Ziadé422545f2010-05-17 10:06:20 +0000203 self.read_template()
Greg Warda82122b2000-02-17 23:56:15 +0000204
Tarek Ziadé422545f2010-05-17 10:06:20 +0000205 if self.prune:
206 self.prune_file_list()
Greg Wardb2db0eb2000-06-21 03:29:57 +0000207
Tarek Ziadé422545f2010-05-17 10:06:20 +0000208 self.filelist.sort()
209 self.filelist.remove_duplicates()
210 self.write_manifest()
Greg Warda82122b2000-02-17 23:56:15 +0000211
Tarek Ziadé89539132009-05-14 14:56:14 +0000212 def add_defaults(self):
Greg Ward23266fe2000-07-30 01:30:31 +0000213 """Add all the default files to self.filelist:
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000214 - README or README.txt
215 - setup.py
216 - test/test*.py
217 - all pure Python modules mentioned in setup script
Tarek Ziadé7dd53392009-02-16 21:38:01 +0000218 - all files pointed by package_data (build_py)
219 - all files defined in data_files.
220 - all files defined as scripts.
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000221 - all C sources listed as part of extensions or C libraries
222 in the setup script (doesn't catch C headers!)
223 Warns if (README or README.txt) or setup.py are missing; everything
224 else is optional.
225 """
Greg Ward14c8d052000-06-08 01:22:48 +0000226
Greg Wardd3b76a82000-09-06 02:08:24 +0000227 standards = [('README', 'README.txt'), self.distribution.script_name]
Greg Warda82122b2000-02-17 23:56:15 +0000228 for fn in standards:
Tarek Ziadé2b66da72009-12-21 01:22:46 +0000229 if isinstance(fn, tuple):
Greg Warda82122b2000-02-17 23:56:15 +0000230 alts = fn
Greg Ward48401122000-02-24 03:17:43 +0000231 got_it = 0
Greg Warda82122b2000-02-17 23:56:15 +0000232 for fn in alts:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000233 if os.path.exists(fn):
Greg Warda82122b2000-02-17 23:56:15 +0000234 got_it = 1
Greg Wardcb1f4c42000-09-30 18:27:54 +0000235 self.filelist.append(fn)
Greg Warda82122b2000-02-17 23:56:15 +0000236 break
237
238 if not got_it:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000239 self.warn("standard file not found: should have one of " +
240 string.join(alts, ', '))
Greg Warda82122b2000-02-17 23:56:15 +0000241 else:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000242 if os.path.exists(fn):
243 self.filelist.append(fn)
Greg Warda82122b2000-02-17 23:56:15 +0000244 else:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000245 self.warn("standard file '%s' not found" % fn)
Greg Warda82122b2000-02-17 23:56:15 +0000246
Greg Ward14c8d052000-06-08 01:22:48 +0000247 optional = ['test/test*.py', 'setup.cfg']
Greg Warda82122b2000-02-17 23:56:15 +0000248 for pattern in optional:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000249 files = filter(os.path.isfile, glob(pattern))
Greg Warda82122b2000-02-17 23:56:15 +0000250 if files:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000251 self.filelist.extend(files)
Greg Warda82122b2000-02-17 23:56:15 +0000252
Tarek Ziadé7dd53392009-02-16 21:38:01 +0000253 # build_py is used to get:
254 # - python modules
255 # - files defined in package_data
256 build_py = self.get_finalized_command('build_py')
257
258 # getting python files
Greg Ward578c10d2000-03-31 02:50:04 +0000259 if self.distribution.has_pure_modules():
Greg Wardcb1f4c42000-09-30 18:27:54 +0000260 self.filelist.extend(build_py.get_source_files())
Greg Warda82122b2000-02-17 23:56:15 +0000261
Tarek Ziadé7dd53392009-02-16 21:38:01 +0000262 # getting package_data files
263 # (computed in build_py.data_files by build_py.finalize_options)
264 for pkg, src_dir, build_dir, filenames in build_py.data_files:
265 for filename in filenames:
266 self.filelist.append(os.path.join(src_dir, filename))
267
268 # getting distribution.data_files
269 if self.distribution.has_data_files():
Tarek Ziadéf68b5b82009-02-17 09:42:44 +0000270 for item in self.distribution.data_files:
271 if isinstance(item, str): # plain file
272 item = convert_path(item)
273 if os.path.isfile(item):
274 self.filelist.append(item)
275 else: # a (dirname, filenames) tuple
276 dirname, filenames = item
277 for f in filenames:
Tarek Ziadé0e5001e2009-02-17 23:06:51 +0000278 f = convert_path(f)
Tarek Ziadéf68b5b82009-02-17 09:42:44 +0000279 if os.path.isfile(f):
280 self.filelist.append(f)
Tarek Ziadé7dd53392009-02-16 21:38:01 +0000281
Greg Ward578c10d2000-03-31 02:50:04 +0000282 if self.distribution.has_ext_modules():
Greg Wardcb1f4c42000-09-30 18:27:54 +0000283 build_ext = self.get_finalized_command('build_ext')
284 self.filelist.extend(build_ext.get_source_files())
Greg Warda82122b2000-02-17 23:56:15 +0000285
Greg Ward60908f12000-04-09 03:51:40 +0000286 if self.distribution.has_c_libraries():
Greg Wardcb1f4c42000-09-30 18:27:54 +0000287 build_clib = self.get_finalized_command('build_clib')
288 self.filelist.extend(build_clib.get_source_files())
Greg Ward60908f12000-04-09 03:51:40 +0000289
Fred Drake4b498232004-03-25 22:04:52 +0000290 if self.distribution.has_scripts():
291 build_scripts = self.get_finalized_command('build_scripts')
292 self.filelist.extend(build_scripts.get_source_files())
293
Tarek Ziadé89539132009-05-14 14:56:14 +0000294 def read_template(self):
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000295 """Read and parse manifest template file named by self.template.
Greg Warda82122b2000-02-17 23:56:15 +0000296
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000297 (usually "MANIFEST.in") The parsing and processing is done by
298 'self.filelist', which updates itself accordingly.
Greg Ward23266fe2000-07-30 01:30:31 +0000299 """
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000300 log.info("reading manifest template '%s'", self.template)
Greg Wardcb1f4c42000-09-30 18:27:54 +0000301 template = TextFile(self.template,
302 strip_comments=1,
303 skip_blanks=1,
304 join_lines=1,
305 lstrip_ws=1,
306 rstrip_ws=1,
307 collapse_join=1)
Greg Warda82122b2000-02-17 23:56:15 +0000308
Greg Warda82122b2000-02-17 23:56:15 +0000309 while 1:
Greg Warda82122b2000-02-17 23:56:15 +0000310 line = template.readline()
311 if line is None: # end of file
312 break
313
Greg Ward6b24dff2000-07-30 01:47:16 +0000314 try:
315 self.filelist.process_template_line(line)
316 except DistutilsTemplateError, msg:
317 self.warn("%s, line %d: %s" % (template.filename,
318 template.current_line,
319 msg))
Greg Warda82122b2000-02-17 23:56:15 +0000320
Tarek Ziadé89539132009-05-14 14:56:14 +0000321 def prune_file_list(self):
Greg Wardce15c6c2000-06-08 01:06:02 +0000322 """Prune off branches that might slip into the file list as created
Greg Ward499822d2000-06-29 02:06:29 +0000323 by 'read_template()', but really don't belong there:
324 * the build tree (typically "build")
325 * the release tree itself (only an issue if we ran "sdist"
Greg Wardaf64aed2000-09-25 01:51:01 +0000326 previously with --keep-temp, or it aborted)
Georg Brandl1df03402008-03-06 06:47:18 +0000327 * any RCS, CVS, .svn, .hg, .git, .bzr, _darcs directories
Greg Wardce15c6c2000-06-08 01:06:02 +0000328 """
329 build = self.get_finalized_command('build')
330 base_dir = self.distribution.get_fullname()
Greg Wardce15c6c2000-06-08 01:06:02 +0000331
Greg Ward23266fe2000-07-30 01:30:31 +0000332 self.filelist.exclude_pattern(None, prefix=build.build_base)
333 self.filelist.exclude_pattern(None, prefix=base_dir)
Greg Wardf8b9e202000-06-08 00:08:14 +0000334
Tarek Ziadé85d6fb52009-01-04 00:04:49 +0000335 # pruning out vcs directories
336 # both separators are used under win32
Tarek Ziadéd81780b2009-01-04 10:37:52 +0000337 if sys.platform == 'win32':
338 seps = r'/|\\'
339 else:
340 seps = '/'
341
342 vcs_dirs = ['RCS', 'CVS', r'\.svn', r'\.hg', r'\.git', r'\.bzr',
343 '_darcs']
Tarek Ziadé85d6fb52009-01-04 00:04:49 +0000344 vcs_ptrn = r'(^|%s)(%s)(%s).*' % (seps, '|'.join(vcs_dirs), seps)
345 self.filelist.exclude_pattern(vcs_ptrn, is_regex=1)
Greg Wardf8b9e202000-06-08 00:08:14 +0000346
Tarek Ziadé89539132009-05-14 14:56:14 +0000347 def write_manifest(self):
Greg Ward23266fe2000-07-30 01:30:31 +0000348 """Write the file list in 'self.filelist' (presumably as filled in
349 by 'add_defaults()' and 'read_template()') to the manifest file
350 named by 'self.manifest'.
Greg Warde0c8c2f2000-06-08 00:24:01 +0000351 """
Greg Wardab3a0f32000-08-05 01:31:54 +0000352 self.execute(file_util.write_file,
Greg Ward23266fe2000-07-30 01:30:31 +0000353 (self.manifest, self.filelist.files),
Greg Wardf8b9e202000-06-08 00:08:14 +0000354 "writing manifest file '%s'" % self.manifest)
Greg Warda82122b2000-02-17 23:56:15 +0000355
Tarek Ziadé89539132009-05-14 14:56:14 +0000356 def read_manifest(self):
Greg Warde0c8c2f2000-06-08 00:24:01 +0000357 """Read the manifest file (named by 'self.manifest') and use it to
Greg Ward23266fe2000-07-30 01:30:31 +0000358 fill in 'self.filelist', the list of files to include in the source
Greg Warde0c8c2f2000-06-08 00:24:01 +0000359 distribution.
360 """
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000361 log.info("reading manifest file '%s'", self.manifest)
Greg Wardcb1f4c42000-09-30 18:27:54 +0000362 manifest = open(self.manifest)
Greg Warda82122b2000-02-17 23:56:15 +0000363 while 1:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000364 line = manifest.readline()
Greg Warda82122b2000-02-17 23:56:15 +0000365 if line == '': # end of file
366 break
367 if line[-1] == '\n':
368 line = line[0:-1]
Greg Wardcb1f4c42000-09-30 18:27:54 +0000369 self.filelist.append(line)
Andrew M. Kuchling2d6c13e2008-02-21 14:23:38 +0000370 manifest.close()
Greg Warda82122b2000-02-17 23:56:15 +0000371
Tarek Ziadé89539132009-05-14 14:56:14 +0000372 def make_release_tree(self, base_dir, files):
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000373 """Create the directory tree that will become the source
374 distribution archive. All directories implied by the filenames in
375 'files' are created under 'base_dir', and then we hard link or copy
376 (if hard linking is unavailable) those files into place.
377 Essentially, this duplicates the developer's source tree, but in a
378 directory named after the distribution, containing only the files
379 to be distributed.
380 """
Greg Ward578c10d2000-03-31 02:50:04 +0000381 # Create all the directories under 'base_dir' necessary to
Greg Ward5fad2682000-09-06 02:18:59 +0000382 # put 'files' there; the 'mkpath()' is just so we don't die
383 # if the manifest happens to be empty.
384 self.mkpath(base_dir)
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000385 dir_util.create_tree(base_dir, files, dry_run=self.dry_run)
Greg Warda82122b2000-02-17 23:56:15 +0000386
387 # And walk over the list of files, either making a hard link (if
388 # os.link exists) to each one that doesn't already exist in its
389 # corresponding location under 'base_dir', or copying each file
390 # that's out-of-date in 'base_dir'. (Usually, all files will be
391 # out-of-date, because by default we blow away 'base_dir' when
392 # we're done making the distribution archives.)
Fred Drake21d45352001-12-06 21:01:19 +0000393
Greg Wardcb1f4c42000-09-30 18:27:54 +0000394 if hasattr(os, 'link'): # can make hard links on this system
Greg Ward578c10d2000-03-31 02:50:04 +0000395 link = 'hard'
Greg Warda82122b2000-02-17 23:56:15 +0000396 msg = "making hard links in %s..." % base_dir
Greg Ward578c10d2000-03-31 02:50:04 +0000397 else: # nope, have to copy
398 link = None
Greg Warda82122b2000-02-17 23:56:15 +0000399 msg = "copying files to %s..." % base_dir
400
Greg Ward5fad2682000-09-06 02:18:59 +0000401 if not files:
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000402 log.warn("no files to distribute -- empty manifest?")
Greg Ward5fad2682000-09-06 02:18:59 +0000403 else:
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000404 log.info(msg)
Greg Warda82122b2000-02-17 23:56:15 +0000405 for file in files:
Greg Ward5fad2682000-09-06 02:18:59 +0000406 if not os.path.isfile(file):
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000407 log.warn("'%s' not a regular file -- skipping" % file)
Greg Ward5fad2682000-09-06 02:18:59 +0000408 else:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000409 dest = os.path.join(base_dir, file)
410 self.copy_file(file, dest, link=link)
Greg Warda82122b2000-02-17 23:56:15 +0000411
Andrew M. Kuchlinga7f225d2001-03-22 03:10:05 +0000412 self.distribution.metadata.write_pkg_info(base_dir)
Fred Drake21d45352001-12-06 21:01:19 +0000413
Tarek Ziadé89539132009-05-14 14:56:14 +0000414 def make_distribution(self):
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000415 """Create the source distribution(s). First, we create the release
416 tree with 'make_release_tree()'; then, we create all required
417 archive files (according to 'self.formats') from the release tree.
418 Finally, we clean up by blowing away the release tree (unless
Greg Wardaf64aed2000-09-25 01:51:01 +0000419 'self.keep_temp' is true). The list of archive files created is
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000420 stored so it can be retrieved later by 'get_archive_files()'.
421 """
Greg Ward578c10d2000-03-31 02:50:04 +0000422 # Don't warn about missing meta-data here -- should be (and is!)
423 # done elsewhere.
Greg Ward0ae7f762000-04-22 02:51:25 +0000424 base_dir = self.distribution.get_fullname()
Greg Wardc0614102000-07-05 03:06:46 +0000425 base_name = os.path.join(self.dist_dir, base_dir)
Greg Warda82122b2000-02-17 23:56:15 +0000426
Greg Wardcb1f4c42000-09-30 18:27:54 +0000427 self.make_release_tree(base_dir, self.filelist.files)
Greg Wardd87eb732000-06-01 01:10:56 +0000428 archive_files = [] # remember names of files we create
Tarek Ziadéaaedcef2009-01-25 23:34:00 +0000429 # tar archive must be created last to avoid overwrite and remove
430 if 'tar' in self.formats:
431 self.formats.append(self.formats.pop(self.formats.index('tar')))
432
Greg Warda82122b2000-02-17 23:56:15 +0000433 for fmt in self.formats:
Tarek Ziadé1b486712009-10-02 23:49:48 +0000434 file = self.make_archive(base_name, fmt, base_dir=base_dir,
435 owner=self.owner, group=self.group)
Greg Wardd87eb732000-06-01 01:10:56 +0000436 archive_files.append(file)
Martin v. Löwis98da5622005-03-23 18:54:36 +0000437 self.distribution.dist_files.append(('sdist', '', file))
Greg Wardd87eb732000-06-01 01:10:56 +0000438
439 self.archive_files = archive_files
Greg Warda82122b2000-02-17 23:56:15 +0000440
Greg Wardaf64aed2000-09-25 01:51:01 +0000441 if not self.keep_temp:
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000442 dir_util.remove_tree(base_dir, dry_run=self.dry_run)
Greg Warda82122b2000-02-17 23:56:15 +0000443
Tarek Ziadé89539132009-05-14 14:56:14 +0000444 def get_archive_files(self):
Greg Wardd87eb732000-06-01 01:10:56 +0000445 """Return the list of archive files created when the command
446 was run, or None if the command hasn't run yet.
447 """
448 return self.archive_files