blob: c66d82713d74eefad68d6809fe4a7d9cd5129c86 [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
Benjamin Peterson960cf0f2009-01-09 04:11:44 +00005import os
Benjamin Peterson960cf0f2009-01-09 04:11:44 +00006import sys
Greg Warda82122b2000-02-17 23:56:15 +00007from glob import glob
Tarek Ziadé5af55c62009-05-16 16:52:13 +00008from warnings import warn
9
Greg Warda82122b2000-02-17 23:56:15 +000010from distutils.core import Command
Jason R. Coombs7677ff22016-10-14 14:08:28 -040011from distutils import dir_util
12from distutils import file_util
13from distutils import archive_util
Greg Warda82122b2000-02-17 23:56:15 +000014from distutils.text_file import TextFile
Greg Ward4571ac12000-07-30 01:05:02 +000015from distutils.filelist import FileList
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +000016from distutils import log
Tarek Ziadéa30337a2009-02-17 09:47:25 +000017from distutils.util import convert_path
Jason R. Coombsc9bba2b2016-10-14 14:06:28 -040018from distutils.errors import DistutilsTemplateError, DistutilsOptionError
19
Greg Warda82122b2000-02-17 23:56:15 +000020
Tarek Ziadéaec039a2009-05-14 12:45:48 +000021def show_formats():
Greg Ward34593812000-06-24 01:23:37 +000022 """Print all possible values for the 'formats' option (used by
23 the "--help-formats" command-line option).
24 """
25 from distutils.fancy_getopt import FancyGetopt
26 from distutils.archive_util import ARCHIVE_FORMATS
Tarek Ziadéaec039a2009-05-14 12:45:48 +000027 formats = []
Greg Ward34593812000-06-24 01:23:37 +000028 for format in ARCHIVE_FORMATS.keys():
29 formats.append(("formats=" + format, None,
30 ARCHIVE_FORMATS[format][2]))
31 formats.sort()
Tarek Ziadéaec039a2009-05-14 12:45:48 +000032 FancyGetopt(formats).print_help(
Greg Ward34593812000-06-24 01:23:37 +000033 "List of available source distribution formats:")
34
Tarek Ziadéda0dc2e2009-05-14 15:21:26 +000035class sdist(Command):
Greg Warda82122b2000-02-17 23:56:15 +000036
37 description = "create a source distribution (tarball, zip file, etc.)"
38
Tarek Ziadé5af55c62009-05-16 16:52:13 +000039 def checking_metadata(self):
40 """Callable used for the check sub-command.
41
42 Placed here so user_options can view it"""
43 return self.metadata_check
44
Greg Wardbbeceea2000-02-18 00:25:39 +000045 user_options = [
46 ('template=', 't',
47 "name of manifest template file [default: MANIFEST.in]"),
48 ('manifest=', 'm',
49 "name of manifest file [default: MANIFEST]"),
50 ('use-defaults', None,
51 "include the default file set in the manifest "
52 "[default; disable with --no-defaults]"),
Greg Ward499822d2000-06-29 02:06:29 +000053 ('no-defaults', None,
54 "don't include the default file set"),
55 ('prune', None,
56 "specifically exclude files/directories that should not be "
57 "distributed (build tree, RCS/CVS dirs, etc.) "
58 "[default; disable with --no-prune]"),
59 ('no-prune', None,
60 "don't automatically exclude anything"),
Greg Ward839d5322000-04-26 01:14:33 +000061 ('manifest-only', 'o',
Greg Wardc3c8c6e2000-06-08 00:46:45 +000062 "just regenerate the manifest and then stop "
63 "(implies --force-manifest)"),
Greg Ward839d5322000-04-26 01:14:33 +000064 ('force-manifest', 'f',
Tarek Ziadé42db5c32010-05-17 11:04:41 +000065 "forcibly regenerate the manifest and carry on as usual. "
66 "Deprecated: now the manifest is always regenerated."),
Greg Wardbbeceea2000-02-18 00:25:39 +000067 ('formats=', None,
Greg Ward2ff78872000-06-24 00:23:20 +000068 "formats for source distribution (comma-separated list)"),
Greg Wardaf64aed2000-09-25 01:51:01 +000069 ('keep-temp', 'k',
Greg Wardbbeceea2000-02-18 00:25:39 +000070 "keep the distribution tree around after creating " +
71 "archive file(s)"),
Greg Wardc0614102000-07-05 03:06:46 +000072 ('dist-dir=', 'd',
73 "directory to put the source distribution archive(s) in "
74 "[default: dist]"),
Éric Araujo40f7e112010-09-11 15:30:19 +000075 ('metadata-check', None,
Tarek Ziadé5af55c62009-05-16 16:52:13 +000076 "Ensure that all required elements of meta-data "
77 "are supplied. Warn if any missing. [default]"),
Andrew Kuchling5e2d4562013-11-15 13:01:52 -050078 ('owner=', 'u',
79 "Owner name used when creating a tar file [default: current user]"),
80 ('group=', 'g',
81 "Group name used when creating a tar file [default: current group]"),
Greg Wardbbeceea2000-02-18 00:25:39 +000082 ]
Greg Wardf1fe1032000-06-08 00:14:18 +000083
Greg Ward99b032e2000-09-25 01:41:15 +000084 boolean_options = ['use-defaults', 'prune',
85 'manifest-only', 'force-manifest',
Tarek Ziadé5af55c62009-05-16 16:52:13 +000086 'keep-temp', 'metadata-check']
Greg Wardf1fe1032000-06-08 00:14:18 +000087
Greg Ward9d17a7a2000-06-07 03:00:06 +000088 help_options = [
89 ('help-formats', None,
Greg Ward2ff78872000-06-24 00:23:20 +000090 "list available distribution formats", show_formats),
Greg Wardfa9ff762000-10-14 04:06:40 +000091 ]
Greg Ward9d17a7a2000-06-07 03:00:06 +000092
Greg Ward499822d2000-06-29 02:06:29 +000093 negative_opt = {'no-defaults': 'use-defaults',
94 'no-prune': 'prune' }
Greg Warda82122b2000-02-17 23:56:15 +000095
Tarek Ziadé5af55c62009-05-16 16:52:13 +000096 sub_commands = [('check', checking_metadata)]
97
Jason R. Coombs10d5eef2016-10-14 14:44:11 -040098 READMES = 'README', 'README.txt'
99
Collin Winter5b7e9d72007-08-30 03:52:21 +0000100 def initialize_options(self):
Greg Warda82122b2000-02-17 23:56:15 +0000101 # 'template' and 'manifest' are, respectively, the names of
102 # the manifest template and manifest file.
103 self.template = None
104 self.manifest = None
105
106 # 'use_defaults': if true, we will include the default file set
107 # in the manifest
108 self.use_defaults = 1
Greg Ward499822d2000-06-29 02:06:29 +0000109 self.prune = 1
Greg Warda82122b2000-02-17 23:56:15 +0000110
111 self.manifest_only = 0
112 self.force_manifest = 0
113
Jason R. Coombsa47ca082016-08-20 17:31:07 -0400114 self.formats = ['gztar']
Greg Wardaf64aed2000-09-25 01:51:01 +0000115 self.keep_temp = 0
Greg Wardc0614102000-07-05 03:06:46 +0000116 self.dist_dir = None
Greg Warda82122b2000-02-17 23:56:15 +0000117
Greg Wardd87eb732000-06-01 01:10:56 +0000118 self.archive_files = None
Tarek Ziadé5af55c62009-05-16 16:52:13 +0000119 self.metadata_check = 1
Andrew Kuchling5e2d4562013-11-15 13:01:52 -0500120 self.owner = None
121 self.group = None
Greg Wardd87eb732000-06-01 01:10:56 +0000122
Collin Winter5b7e9d72007-08-30 03:52:21 +0000123 def finalize_options(self):
Greg Warda82122b2000-02-17 23:56:15 +0000124 if self.manifest is None:
125 self.manifest = "MANIFEST"
126 if self.template is None:
127 self.template = "MANIFEST.in"
128
Greg Ward62d5a572000-06-04 15:12:51 +0000129 self.ensure_string_list('formats')
Greg Warda82122b2000-02-17 23:56:15 +0000130
Greg Wardcb1f4c42000-09-30 18:27:54 +0000131 bad_format = archive_util.check_archive_formats(self.formats)
Greg Ward6a9a5452000-04-22 03:11:55 +0000132 if bad_format:
Collin Winter5b7e9d72007-08-30 03:52:21 +0000133 raise DistutilsOptionError(
134 "unknown archive format '%s'" % bad_format)
Greg Ward6a9a5452000-04-22 03:11:55 +0000135
Greg Wardc0614102000-07-05 03:06:46 +0000136 if self.dist_dir is None:
137 self.dist_dir = "dist"
138
Collin Winter5b7e9d72007-08-30 03:52:21 +0000139 def run(self):
Greg Ward23266fe2000-07-30 01:30:31 +0000140 # 'filelist' contains the list of files that will make up the
141 # manifest
142 self.filelist = FileList()
Fred Drake21d45352001-12-06 21:01:19 +0000143
Tarek Ziadé5af55c62009-05-16 16:52:13 +0000144 # Run sub commands
145 for cmd_name in self.get_sub_commands():
146 self.run_command(cmd_name)
Greg Warda82122b2000-02-17 23:56:15 +0000147
148 # Do whatever it takes to get the list of files to process
149 # (process the manifest template, read an existing manifest,
Greg Ward23266fe2000-07-30 01:30:31 +0000150 # whatever). File list is accumulated in 'self.filelist'.
Greg Wardcb1f4c42000-09-30 18:27:54 +0000151 self.get_file_list()
Greg Warda82122b2000-02-17 23:56:15 +0000152
153 # If user just wanted us to regenerate the manifest, stop now.
154 if self.manifest_only:
155 return
156
157 # Otherwise, go ahead and create the source distribution tarball,
158 # or zipfile, or whatever.
Greg Wardcb1f4c42000-09-30 18:27:54 +0000159 self.make_distribution()
Greg Warda82122b2000-02-17 23:56:15 +0000160
Collin Winter5b7e9d72007-08-30 03:52:21 +0000161 def check_metadata(self):
Tarek Ziadé5af55c62009-05-16 16:52:13 +0000162 """Deprecated API."""
163 warn("distutils.command.sdist.check_metadata is deprecated, \
164 use the check command instead", PendingDeprecationWarning)
165 check = self.distribution.get_command_obj('check')
166 check.ensure_finalized()
167 check.run()
Greg Warda82122b2000-02-17 23:56:15 +0000168
Collin Winter5b7e9d72007-08-30 03:52:21 +0000169 def get_file_list(self):
Greg Warda82122b2000-02-17 23:56:15 +0000170 """Figure out the list of files to include in the source
Greg Ward23266fe2000-07-30 01:30:31 +0000171 distribution, and put it in 'self.filelist'. This might involve
Greg Warde0c8c2f2000-06-08 00:24:01 +0000172 reading the manifest template (and writing the manifest), or just
173 reading the manifest, or just using the default file set -- it all
Tarek Ziadédd4671a2010-05-17 10:48:29 +0000174 depends on the user's options.
Greg Warde0c8c2f2000-06-08 00:24:01 +0000175 """
Éric Araujoab7c1b32011-07-31 04:06:12 +0200176 # new behavior when using a template:
Ezio Melottib5bc3532013-08-17 16:11:40 +0300177 # the file list is recalculated every time because
Tarek Ziadédd4671a2010-05-17 10:48:29 +0000178 # even if MANIFEST.in or setup.py are not changed
179 # the user might have added some files in the tree that
180 # need to be included.
181 #
Éric Araujoab7c1b32011-07-31 04:06:12 +0200182 # This makes --force the default and only behavior with templates.
Greg Ward23266fe2000-07-30 01:30:31 +0000183 template_exists = os.path.isfile(self.template)
Éric Araujoab7c1b32011-07-31 04:06:12 +0200184 if not template_exists and self._manifest_is_not_generated():
185 self.read_manifest()
186 self.filelist.sort()
187 self.filelist.remove_duplicates()
188 return
189
Tarek Ziadédd4671a2010-05-17 10:48:29 +0000190 if not template_exists:
191 self.warn(("manifest template '%s' does not exist " +
192 "(using default file list)") %
193 self.template)
194 self.filelist.findall()
195
196 if self.use_defaults:
197 self.add_defaults()
198
Greg Warda82122b2000-02-17 23:56:15 +0000199 if template_exists:
Tarek Ziadédd4671a2010-05-17 10:48:29 +0000200 self.read_template()
Greg Warda82122b2000-02-17 23:56:15 +0000201
Tarek Ziadédd4671a2010-05-17 10:48:29 +0000202 if self.prune:
203 self.prune_file_list()
Greg Wardb2db0eb2000-06-21 03:29:57 +0000204
Tarek Ziadédd4671a2010-05-17 10:48:29 +0000205 self.filelist.sort()
206 self.filelist.remove_duplicates()
207 self.write_manifest()
Greg Warda82122b2000-02-17 23:56:15 +0000208
Collin Winter5b7e9d72007-08-30 03:52:21 +0000209 def add_defaults(self):
Greg Ward23266fe2000-07-30 01:30:31 +0000210 """Add all the default files to self.filelist:
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000211 - README or README.txt
212 - setup.py
213 - test/test*.py
214 - all pure Python modules mentioned in setup script
Tarek Ziadé0d0506e2009-02-16 21:49:12 +0000215 - all files pointed by package_data (build_py)
216 - all files defined in data_files.
217 - all files defined as scripts.
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000218 - all C sources listed as part of extensions or C libraries
219 in the setup script (doesn't catch C headers!)
220 Warns if (README or README.txt) or setup.py are missing; everything
221 else is optional.
222 """
Jason R. Coombs00e3cea2016-10-14 14:53:32 -0400223 self._add_defaults_standards()
224 self._add_defaults_optional()
225 self._add_defaults_python()
226 self._add_defaults_data_files()
227 self._add_defaults_ext()
228 self._add_defaults_c_libs()
229 self._add_defaults_scripts()
230
231 def _add_defaults_standards(self):
Jason R. Coombs10d5eef2016-10-14 14:44:11 -0400232 standards = [self.READMES, self.distribution.script_name]
Greg Warda82122b2000-02-17 23:56:15 +0000233 for fn in standards:
Collin Winter5b7e9d72007-08-30 03:52:21 +0000234 if isinstance(fn, tuple):
Greg Warda82122b2000-02-17 23:56:15 +0000235 alts = fn
Collin Winter5b7e9d72007-08-30 03:52:21 +0000236 got_it = False
Greg Warda82122b2000-02-17 23:56:15 +0000237 for fn in alts:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000238 if os.path.exists(fn):
Collin Winter5b7e9d72007-08-30 03:52:21 +0000239 got_it = True
Greg Wardcb1f4c42000-09-30 18:27:54 +0000240 self.filelist.append(fn)
Greg Warda82122b2000-02-17 23:56:15 +0000241 break
242
243 if not got_it:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000244 self.warn("standard file not found: should have one of " +
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000245 ', '.join(alts))
Greg Warda82122b2000-02-17 23:56:15 +0000246 else:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000247 if os.path.exists(fn):
248 self.filelist.append(fn)
Greg Warda82122b2000-02-17 23:56:15 +0000249 else:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000250 self.warn("standard file '%s' not found" % fn)
Greg Warda82122b2000-02-17 23:56:15 +0000251
Jason R. Coombs00e3cea2016-10-14 14:53:32 -0400252 def _add_defaults_optional(self):
Greg Ward14c8d052000-06-08 01:22:48 +0000253 optional = ['test/test*.py', 'setup.cfg']
Greg Warda82122b2000-02-17 23:56:15 +0000254 for pattern in optional:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000255 files = filter(os.path.isfile, glob(pattern))
Georg Brandlcaa78fe2010-08-01 19:07:28 +0000256 self.filelist.extend(files)
Greg Warda82122b2000-02-17 23:56:15 +0000257
Jason R. Coombs00e3cea2016-10-14 14:53:32 -0400258 def _add_defaults_python(self):
Tarek Ziadé0d0506e2009-02-16 21:49:12 +0000259 # build_py is used to get:
260 # - python modules
261 # - files defined in package_data
262 build_py = self.get_finalized_command('build_py')
263
264 # getting python files
Greg Ward578c10d2000-03-31 02:50:04 +0000265 if self.distribution.has_pure_modules():
Greg Wardcb1f4c42000-09-30 18:27:54 +0000266 self.filelist.extend(build_py.get_source_files())
Greg Warda82122b2000-02-17 23:56:15 +0000267
Tarek Ziadé0d0506e2009-02-16 21:49:12 +0000268 # getting package_data files
269 # (computed in build_py.data_files by build_py.finalize_options)
270 for pkg, src_dir, build_dir, filenames in build_py.data_files:
271 for filename in filenames:
272 self.filelist.append(os.path.join(src_dir, filename))
273
Jason R. Coombs00e3cea2016-10-14 14:53:32 -0400274 def _add_defaults_data_files(self):
Tarek Ziadé0d0506e2009-02-16 21:49:12 +0000275 # getting distribution.data_files
276 if self.distribution.has_data_files():
Tarek Ziadéa30337a2009-02-17 09:47:25 +0000277 for item in self.distribution.data_files:
Jason R. Coombsc581ce42016-10-14 14:10:07 -0400278 if isinstance(item, str):
279 # plain file
Tarek Ziadéa30337a2009-02-17 09:47:25 +0000280 item = convert_path(item)
281 if os.path.isfile(item):
282 self.filelist.append(item)
Jason R. Coombsc581ce42016-10-14 14:10:07 -0400283 else:
284 # a (dirname, filenames) tuple
Tarek Ziadéa30337a2009-02-17 09:47:25 +0000285 dirname, filenames = item
286 for f in filenames:
Tarek Ziadé14d34a02009-02-17 23:10:18 +0000287 f = convert_path(f)
Tarek Ziadéa30337a2009-02-17 09:47:25 +0000288 if os.path.isfile(f):
289 self.filelist.append(f)
Tarek Ziadé0d0506e2009-02-16 21:49:12 +0000290
Jason R. Coombs00e3cea2016-10-14 14:53:32 -0400291 def _add_defaults_ext(self):
Greg Ward578c10d2000-03-31 02:50:04 +0000292 if self.distribution.has_ext_modules():
Greg Wardcb1f4c42000-09-30 18:27:54 +0000293 build_ext = self.get_finalized_command('build_ext')
294 self.filelist.extend(build_ext.get_source_files())
Greg Warda82122b2000-02-17 23:56:15 +0000295
Jason R. Coombs00e3cea2016-10-14 14:53:32 -0400296 def _add_defaults_c_libs(self):
Greg Ward60908f12000-04-09 03:51:40 +0000297 if self.distribution.has_c_libraries():
Greg Wardcb1f4c42000-09-30 18:27:54 +0000298 build_clib = self.get_finalized_command('build_clib')
299 self.filelist.extend(build_clib.get_source_files())
Greg Ward60908f12000-04-09 03:51:40 +0000300
Jason R. Coombs00e3cea2016-10-14 14:53:32 -0400301 def _add_defaults_scripts(self):
Fred Drake4b498232004-03-25 22:04:52 +0000302 if self.distribution.has_scripts():
303 build_scripts = self.get_finalized_command('build_scripts')
304 self.filelist.extend(build_scripts.get_source_files())
305
Collin Winter5b7e9d72007-08-30 03:52:21 +0000306 def read_template(self):
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000307 """Read and parse manifest template file named by self.template.
Greg Warda82122b2000-02-17 23:56:15 +0000308
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000309 (usually "MANIFEST.in") The parsing and processing is done by
310 'self.filelist', which updates itself accordingly.
Greg Ward23266fe2000-07-30 01:30:31 +0000311 """
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000312 log.info("reading manifest template '%s'", self.template)
Collin Winter5b7e9d72007-08-30 03:52:21 +0000313 template = TextFile(self.template, strip_comments=1, skip_blanks=1,
314 join_lines=1, lstrip_ws=1, rstrip_ws=1,
Greg Wardcb1f4c42000-09-30 18:27:54 +0000315 collapse_join=1)
Greg Warda82122b2000-02-17 23:56:15 +0000316
Éric Araujodcb22a32011-04-17 14:27:07 +0200317 try:
318 while True:
319 line = template.readline()
320 if line is None: # end of file
321 break
Greg Warda82122b2000-02-17 23:56:15 +0000322
Éric Araujodcb22a32011-04-17 14:27:07 +0200323 try:
324 self.filelist.process_template_line(line)
Éric Araujocfbd6302011-09-03 00:42:04 +0200325 # the call above can raise a DistutilsTemplateError for
326 # malformed lines, or a ValueError from the lower-level
327 # convert_path function
328 except (DistutilsTemplateError, ValueError) as msg:
Éric Araujodcb22a32011-04-17 14:27:07 +0200329 self.warn("%s, line %d: %s" % (template.filename,
330 template.current_line,
331 msg))
332 finally:
333 template.close()
Greg Warda82122b2000-02-17 23:56:15 +0000334
Collin Winter5b7e9d72007-08-30 03:52:21 +0000335 def prune_file_list(self):
Greg Wardce15c6c2000-06-08 01:06:02 +0000336 """Prune off branches that might slip into the file list as created
Greg Ward499822d2000-06-29 02:06:29 +0000337 by 'read_template()', but really don't belong there:
338 * the build tree (typically "build")
339 * the release tree itself (only an issue if we ran "sdist"
Greg Wardaf64aed2000-09-25 01:51:01 +0000340 previously with --keep-temp, or it aborted)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000341 * any RCS, CVS, .svn, .hg, .git, .bzr, _darcs directories
Greg Wardce15c6c2000-06-08 01:06:02 +0000342 """
343 build = self.get_finalized_command('build')
344 base_dir = self.distribution.get_fullname()
Greg Wardce15c6c2000-06-08 01:06:02 +0000345
Greg Ward23266fe2000-07-30 01:30:31 +0000346 self.filelist.exclude_pattern(None, prefix=build.build_base)
347 self.filelist.exclude_pattern(None, prefix=base_dir)
Greg Wardf8b9e202000-06-08 00:08:14 +0000348
Benjamin Peterson960cf0f2009-01-09 04:11:44 +0000349 if sys.platform == 'win32':
350 seps = r'/|\\'
351 else:
352 seps = '/'
353
354 vcs_dirs = ['RCS', 'CVS', r'\.svn', r'\.hg', r'\.git', r'\.bzr',
355 '_darcs']
356 vcs_ptrn = r'(^|%s)(%s)(%s).*' % (seps, '|'.join(vcs_dirs), seps)
357 self.filelist.exclude_pattern(vcs_ptrn, is_regex=1)
358
Tarek Ziadéda0dc2e2009-05-14 15:21:26 +0000359 def write_manifest(self):
Greg Ward23266fe2000-07-30 01:30:31 +0000360 """Write the file list in 'self.filelist' (presumably as filled in
361 by 'add_defaults()' and 'read_template()') to the manifest file
362 named by 'self.manifest'.
Greg Warde0c8c2f2000-06-08 00:24:01 +0000363 """
Éric Araujoab7c1b32011-07-31 04:06:12 +0200364 if self._manifest_is_not_generated():
365 log.info("not writing to manually maintained "
366 "manifest file '%s'" % self.manifest)
367 return
Éric Araujo60a95b72010-08-14 02:36:26 +0000368
369 content = self.filelist.files[:]
370 content.insert(0, '# file GENERATED by distutils, do NOT edit')
371 self.execute(file_util.write_file, (self.manifest, content),
Greg Wardf8b9e202000-06-08 00:08:14 +0000372 "writing manifest file '%s'" % self.manifest)
Greg Warda82122b2000-02-17 23:56:15 +0000373
Éric Araujoab7c1b32011-07-31 04:06:12 +0200374 def _manifest_is_not_generated(self):
375 # check for special comment used in 3.1.3 and higher
376 if not os.path.isfile(self.manifest):
377 return False
378
379 fp = open(self.manifest)
380 try:
381 first_line = fp.readline()
382 finally:
383 fp.close()
384 return first_line != '# file GENERATED by distutils, do NOT edit\n'
385
Collin Winter5b7e9d72007-08-30 03:52:21 +0000386 def read_manifest(self):
Greg Warde0c8c2f2000-06-08 00:24:01 +0000387 """Read the manifest file (named by 'self.manifest') and use it to
Greg Ward23266fe2000-07-30 01:30:31 +0000388 fill in 'self.filelist', the list of files to include in the source
Greg Warde0c8c2f2000-06-08 00:24:01 +0000389 distribution.
390 """
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000391 log.info("reading manifest file '%s'", self.manifest)
Greg Wardcb1f4c42000-09-30 18:27:54 +0000392 manifest = open(self.manifest)
Éric Araujoab7c1b32011-07-31 04:06:12 +0200393 for line in manifest:
394 # ignore comments and blank lines
395 line = line.strip()
396 if line.startswith('#') or not line:
397 continue
Greg Wardcb1f4c42000-09-30 18:27:54 +0000398 self.filelist.append(line)
Christian Heimesc3f30c42008-02-22 16:37:40 +0000399 manifest.close()
Greg Warda82122b2000-02-17 23:56:15 +0000400
Collin Winter5b7e9d72007-08-30 03:52:21 +0000401 def make_release_tree(self, base_dir, files):
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000402 """Create the directory tree that will become the source
403 distribution archive. All directories implied by the filenames in
404 'files' are created under 'base_dir', and then we hard link or copy
405 (if hard linking is unavailable) those files into place.
406 Essentially, this duplicates the developer's source tree, but in a
407 directory named after the distribution, containing only the files
408 to be distributed.
409 """
Greg Ward578c10d2000-03-31 02:50:04 +0000410 # Create all the directories under 'base_dir' necessary to
Greg Ward5fad2682000-09-06 02:18:59 +0000411 # put 'files' there; the 'mkpath()' is just so we don't die
412 # if the manifest happens to be empty.
413 self.mkpath(base_dir)
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000414 dir_util.create_tree(base_dir, files, dry_run=self.dry_run)
Greg Warda82122b2000-02-17 23:56:15 +0000415
416 # And walk over the list of files, either making a hard link (if
417 # os.link exists) to each one that doesn't already exist in its
418 # corresponding location under 'base_dir', or copying each file
419 # that's out-of-date in 'base_dir'. (Usually, all files will be
420 # out-of-date, because by default we blow away 'base_dir' when
421 # we're done making the distribution archives.)
Fred Drake21d45352001-12-06 21:01:19 +0000422
Greg Wardcb1f4c42000-09-30 18:27:54 +0000423 if hasattr(os, 'link'): # can make hard links on this system
Greg Ward578c10d2000-03-31 02:50:04 +0000424 link = 'hard'
Greg Warda82122b2000-02-17 23:56:15 +0000425 msg = "making hard links in %s..." % base_dir
Greg Ward578c10d2000-03-31 02:50:04 +0000426 else: # nope, have to copy
427 link = None
Greg Warda82122b2000-02-17 23:56:15 +0000428 msg = "copying files to %s..." % base_dir
429
Greg Ward5fad2682000-09-06 02:18:59 +0000430 if not files:
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000431 log.warn("no files to distribute -- empty manifest?")
Greg Ward5fad2682000-09-06 02:18:59 +0000432 else:
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000433 log.info(msg)
Greg Warda82122b2000-02-17 23:56:15 +0000434 for file in files:
Greg Ward5fad2682000-09-06 02:18:59 +0000435 if not os.path.isfile(file):
Vinay Sajipdd917f82016-08-31 08:22:29 +0100436 log.warn("'%s' not a regular file -- skipping", file)
Greg Ward5fad2682000-09-06 02:18:59 +0000437 else:
Greg Wardcb1f4c42000-09-30 18:27:54 +0000438 dest = os.path.join(base_dir, file)
439 self.copy_file(file, dest, link=link)
Greg Warda82122b2000-02-17 23:56:15 +0000440
Andrew M. Kuchlinga7f225d2001-03-22 03:10:05 +0000441 self.distribution.metadata.write_pkg_info(base_dir)
Fred Drake21d45352001-12-06 21:01:19 +0000442
Collin Winter5b7e9d72007-08-30 03:52:21 +0000443 def make_distribution(self):
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000444 """Create the source distribution(s). First, we create the release
445 tree with 'make_release_tree()'; then, we create all required
446 archive files (according to 'self.formats') from the release tree.
447 Finally, we clean up by blowing away the release tree (unless
Greg Wardaf64aed2000-09-25 01:51:01 +0000448 'self.keep_temp' is true). The list of archive files created is
Greg Wardc3c8c6e2000-06-08 00:46:45 +0000449 stored so it can be retrieved later by 'get_archive_files()'.
450 """
Greg Ward578c10d2000-03-31 02:50:04 +0000451 # Don't warn about missing meta-data here -- should be (and is!)
452 # done elsewhere.
Greg Ward0ae7f762000-04-22 02:51:25 +0000453 base_dir = self.distribution.get_fullname()
Greg Wardc0614102000-07-05 03:06:46 +0000454 base_name = os.path.join(self.dist_dir, base_dir)
Greg Warda82122b2000-02-17 23:56:15 +0000455
Greg Wardcb1f4c42000-09-30 18:27:54 +0000456 self.make_release_tree(base_dir, self.filelist.files)
Greg Wardd87eb732000-06-01 01:10:56 +0000457 archive_files = [] # remember names of files we create
Tarek Ziadé0302cf52009-01-26 17:23:20 +0000458 # tar archive must be created last to avoid overwrite and remove
459 if 'tar' in self.formats:
460 self.formats.append(self.formats.pop(self.formats.index('tar')))
461
Greg Warda82122b2000-02-17 23:56:15 +0000462 for fmt in self.formats:
Andrew Kuchling5e2d4562013-11-15 13:01:52 -0500463 file = self.make_archive(base_name, fmt, base_dir=base_dir,
464 owner=self.owner, group=self.group)
Greg Wardd87eb732000-06-01 01:10:56 +0000465 archive_files.append(file)
Martin v. Löwis98da5622005-03-23 18:54:36 +0000466 self.distribution.dist_files.append(('sdist', '', file))
Greg Wardd87eb732000-06-01 01:10:56 +0000467
468 self.archive_files = archive_files
Greg Warda82122b2000-02-17 23:56:15 +0000469
Greg Wardaf64aed2000-09-25 01:51:01 +0000470 if not self.keep_temp:
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +0000471 dir_util.remove_tree(base_dir, dry_run=self.dry_run)
Greg Warda82122b2000-02-17 23:56:15 +0000472
Collin Winter5b7e9d72007-08-30 03:52:21 +0000473 def get_archive_files(self):
Greg Wardd87eb732000-06-01 01:10:56 +0000474 """Return the list of archive files created when the command
475 was run, or None if the command hasn't run yet.
476 """
477 return self.archive_files