462 lines
16 KiB
Python
462 lines
16 KiB
Python
|
"""setuptools.command.bdist_egg
|
||
|
|
||
|
Build .egg distributions"""
|
||
|
|
||
|
from distutils.dir_util import remove_tree, mkpath
|
||
|
from distutils import log
|
||
|
from types import CodeType
|
||
|
import sys
|
||
|
import os
|
||
|
import re
|
||
|
import textwrap
|
||
|
import marshal
|
||
|
|
||
|
from setuptools.extension import Library
|
||
|
from setuptools import Command
|
||
|
from .._path import ensure_directory
|
||
|
|
||
|
from sysconfig import get_path, get_python_version
|
||
|
|
||
|
|
||
|
def _get_purelib():
|
||
|
return get_path("purelib")
|
||
|
|
||
|
|
||
|
def strip_module(filename):
|
||
|
if '.' in filename:
|
||
|
filename = os.path.splitext(filename)[0]
|
||
|
if filename.endswith('module'):
|
||
|
filename = filename[:-6]
|
||
|
return filename
|
||
|
|
||
|
|
||
|
def sorted_walk(dir):
|
||
|
"""Do os.walk in a reproducible way,
|
||
|
independent of indeterministic filesystem readdir order
|
||
|
"""
|
||
|
for base, dirs, files in os.walk(dir):
|
||
|
dirs.sort()
|
||
|
files.sort()
|
||
|
yield base, dirs, files
|
||
|
|
||
|
|
||
|
def write_stub(resource, pyfile):
|
||
|
_stub_template = textwrap.dedent(
|
||
|
"""
|
||
|
def __bootstrap__():
|
||
|
global __bootstrap__, __loader__, __file__
|
||
|
import sys, pkg_resources, importlib.util
|
||
|
__file__ = pkg_resources.resource_filename(__name__, %r)
|
||
|
__loader__ = None; del __bootstrap__, __loader__
|
||
|
spec = importlib.util.spec_from_file_location(__name__,__file__)
|
||
|
mod = importlib.util.module_from_spec(spec)
|
||
|
spec.loader.exec_module(mod)
|
||
|
__bootstrap__()
|
||
|
"""
|
||
|
).lstrip()
|
||
|
with open(pyfile, 'w') as f:
|
||
|
f.write(_stub_template % resource)
|
||
|
|
||
|
|
||
|
class bdist_egg(Command):
|
||
|
description = "create an \"egg\" distribution"
|
||
|
|
||
|
user_options = [
|
||
|
('bdist-dir=', 'b', "temporary directory for creating the distribution"),
|
||
|
(
|
||
|
'plat-name=',
|
||
|
'p',
|
||
|
"platform name to embed in generated filenames "
|
||
|
"(by default uses `pkg_resources.get_build_platform()`)",
|
||
|
),
|
||
|
('exclude-source-files', None, "remove all .py files from the generated egg"),
|
||
|
(
|
||
|
'keep-temp',
|
||
|
'k',
|
||
|
"keep the pseudo-installation tree around after "
|
||
|
+ "creating the distribution archive",
|
||
|
),
|
||
|
('dist-dir=', 'd', "directory to put final built distributions in"),
|
||
|
('skip-build', None, "skip rebuilding everything (for testing/debugging)"),
|
||
|
]
|
||
|
|
||
|
boolean_options = ['keep-temp', 'skip-build', 'exclude-source-files']
|
||
|
|
||
|
def initialize_options(self):
|
||
|
self.bdist_dir = None
|
||
|
self.plat_name = None
|
||
|
self.keep_temp = 0
|
||
|
self.dist_dir = None
|
||
|
self.skip_build = 0
|
||
|
self.egg_output = None
|
||
|
self.exclude_source_files = None
|
||
|
|
||
|
def finalize_options(self):
|
||
|
ei_cmd = self.ei_cmd = self.get_finalized_command("egg_info")
|
||
|
self.egg_info = ei_cmd.egg_info
|
||
|
|
||
|
if self.bdist_dir is None:
|
||
|
bdist_base = self.get_finalized_command('bdist').bdist_base
|
||
|
self.bdist_dir = os.path.join(bdist_base, 'egg')
|
||
|
|
||
|
if self.plat_name is None:
|
||
|
from pkg_resources import get_build_platform
|
||
|
|
||
|
self.plat_name = get_build_platform()
|
||
|
|
||
|
self.set_undefined_options('bdist', ('dist_dir', 'dist_dir'))
|
||
|
|
||
|
if self.egg_output is None:
|
||
|
# Compute filename of the output egg
|
||
|
basename = ei_cmd._get_egg_basename(
|
||
|
py_version=get_python_version(),
|
||
|
platform=self.distribution.has_ext_modules() and self.plat_name,
|
||
|
)
|
||
|
|
||
|
self.egg_output = os.path.join(self.dist_dir, basename + '.egg')
|
||
|
|
||
|
def do_install_data(self):
|
||
|
# Hack for packages that install data to install's --install-lib
|
||
|
self.get_finalized_command('install').install_lib = self.bdist_dir
|
||
|
|
||
|
site_packages = os.path.normcase(os.path.realpath(_get_purelib()))
|
||
|
old, self.distribution.data_files = self.distribution.data_files, []
|
||
|
|
||
|
for item in old:
|
||
|
if isinstance(item, tuple) and len(item) == 2:
|
||
|
if os.path.isabs(item[0]):
|
||
|
realpath = os.path.realpath(item[0])
|
||
|
normalized = os.path.normcase(realpath)
|
||
|
if normalized == site_packages or normalized.startswith(
|
||
|
site_packages + os.sep
|
||
|
):
|
||
|
item = realpath[len(site_packages) + 1 :], item[1]
|
||
|
# XXX else: raise ???
|
||
|
self.distribution.data_files.append(item)
|
||
|
|
||
|
try:
|
||
|
log.info("installing package data to %s", self.bdist_dir)
|
||
|
self.call_command('install_data', force=0, root=None)
|
||
|
finally:
|
||
|
self.distribution.data_files = old
|
||
|
|
||
|
def get_outputs(self):
|
||
|
return [self.egg_output]
|
||
|
|
||
|
def call_command(self, cmdname, **kw):
|
||
|
"""Invoke reinitialized command `cmdname` with keyword args"""
|
||
|
for dirname in INSTALL_DIRECTORY_ATTRS:
|
||
|
kw.setdefault(dirname, self.bdist_dir)
|
||
|
kw.setdefault('skip_build', self.skip_build)
|
||
|
kw.setdefault('dry_run', self.dry_run)
|
||
|
cmd = self.reinitialize_command(cmdname, **kw)
|
||
|
self.run_command(cmdname)
|
||
|
return cmd
|
||
|
|
||
|
def run(self): # noqa: C901 # is too complex (14) # FIXME
|
||
|
# Generate metadata first
|
||
|
self.run_command("egg_info")
|
||
|
# We run install_lib before install_data, because some data hacks
|
||
|
# pull their data path from the install_lib command.
|
||
|
log.info("installing library code to %s", self.bdist_dir)
|
||
|
instcmd = self.get_finalized_command('install')
|
||
|
old_root = instcmd.root
|
||
|
instcmd.root = None
|
||
|
if self.distribution.has_c_libraries() and not self.skip_build:
|
||
|
self.run_command('build_clib')
|
||
|
cmd = self.call_command('install_lib', warn_dir=0)
|
||
|
instcmd.root = old_root
|
||
|
|
||
|
all_outputs, ext_outputs = self.get_ext_outputs()
|
||
|
self.stubs = []
|
||
|
to_compile = []
|
||
|
for p, ext_name in enumerate(ext_outputs):
|
||
|
filename, ext = os.path.splitext(ext_name)
|
||
|
pyfile = os.path.join(self.bdist_dir, strip_module(filename) + '.py')
|
||
|
self.stubs.append(pyfile)
|
||
|
log.info("creating stub loader for %s", ext_name)
|
||
|
if not self.dry_run:
|
||
|
write_stub(os.path.basename(ext_name), pyfile)
|
||
|
to_compile.append(pyfile)
|
||
|
ext_outputs[p] = ext_name.replace(os.sep, '/')
|
||
|
|
||
|
if to_compile:
|
||
|
cmd.byte_compile(to_compile)
|
||
|
if self.distribution.data_files:
|
||
|
self.do_install_data()
|
||
|
|
||
|
# Make the EGG-INFO directory
|
||
|
archive_root = self.bdist_dir
|
||
|
egg_info = os.path.join(archive_root, 'EGG-INFO')
|
||
|
self.mkpath(egg_info)
|
||
|
if self.distribution.scripts:
|
||
|
script_dir = os.path.join(egg_info, 'scripts')
|
||
|
log.info("installing scripts to %s", script_dir)
|
||
|
self.call_command('install_scripts', install_dir=script_dir, no_ep=1)
|
||
|
|
||
|
self.copy_metadata_to(egg_info)
|
||
|
native_libs = os.path.join(egg_info, "native_libs.txt")
|
||
|
if all_outputs:
|
||
|
log.info("writing %s", native_libs)
|
||
|
if not self.dry_run:
|
||
|
ensure_directory(native_libs)
|
||
|
libs_file = open(native_libs, 'wt')
|
||
|
libs_file.write('\n'.join(all_outputs))
|
||
|
libs_file.write('\n')
|
||
|
libs_file.close()
|
||
|
elif os.path.isfile(native_libs):
|
||
|
log.info("removing %s", native_libs)
|
||
|
if not self.dry_run:
|
||
|
os.unlink(native_libs)
|
||
|
|
||
|
write_safety_flag(os.path.join(archive_root, 'EGG-INFO'), self.zip_safe())
|
||
|
|
||
|
if os.path.exists(os.path.join(self.egg_info, 'depends.txt')):
|
||
|
log.warn(
|
||
|
"WARNING: 'depends.txt' will not be used by setuptools 0.6!\n"
|
||
|
"Use the install_requires/extras_require setup() args instead."
|
||
|
)
|
||
|
|
||
|
if self.exclude_source_files:
|
||
|
self.zap_pyfiles()
|
||
|
|
||
|
# Make the archive
|
||
|
make_zipfile(
|
||
|
self.egg_output,
|
||
|
archive_root,
|
||
|
verbose=self.verbose,
|
||
|
dry_run=self.dry_run,
|
||
|
mode=self.gen_header(),
|
||
|
)
|
||
|
if not self.keep_temp:
|
||
|
remove_tree(self.bdist_dir, dry_run=self.dry_run)
|
||
|
|
||
|
# Add to 'Distribution.dist_files' so that the "upload" command works
|
||
|
getattr(self.distribution, 'dist_files', []).append((
|
||
|
'bdist_egg',
|
||
|
get_python_version(),
|
||
|
self.egg_output,
|
||
|
))
|
||
|
|
||
|
def zap_pyfiles(self):
|
||
|
log.info("Removing .py files from temporary directory")
|
||
|
for base, dirs, files in walk_egg(self.bdist_dir):
|
||
|
for name in files:
|
||
|
path = os.path.join(base, name)
|
||
|
|
||
|
if name.endswith('.py'):
|
||
|
log.debug("Deleting %s", path)
|
||
|
os.unlink(path)
|
||
|
|
||
|
if base.endswith('__pycache__'):
|
||
|
path_old = path
|
||
|
|
||
|
pattern = r'(?P<name>.+)\.(?P<magic>[^.]+)\.pyc'
|
||
|
m = re.match(pattern, name)
|
||
|
path_new = os.path.join(base, os.pardir, m.group('name') + '.pyc')
|
||
|
log.info("Renaming file from [%s] to [%s]" % (path_old, path_new))
|
||
|
try:
|
||
|
os.remove(path_new)
|
||
|
except OSError:
|
||
|
pass
|
||
|
os.rename(path_old, path_new)
|
||
|
|
||
|
def zip_safe(self):
|
||
|
safe = getattr(self.distribution, 'zip_safe', None)
|
||
|
if safe is not None:
|
||
|
return safe
|
||
|
log.warn("zip_safe flag not set; analyzing archive contents...")
|
||
|
return analyze_egg(self.bdist_dir, self.stubs)
|
||
|
|
||
|
def gen_header(self):
|
||
|
return 'w'
|
||
|
|
||
|
def copy_metadata_to(self, target_dir):
|
||
|
"Copy metadata (egg info) to the target_dir"
|
||
|
# normalize the path (so that a forward-slash in egg_info will
|
||
|
# match using startswith below)
|
||
|
norm_egg_info = os.path.normpath(self.egg_info)
|
||
|
prefix = os.path.join(norm_egg_info, '')
|
||
|
for path in self.ei_cmd.filelist.files:
|
||
|
if path.startswith(prefix):
|
||
|
target = os.path.join(target_dir, path[len(prefix) :])
|
||
|
ensure_directory(target)
|
||
|
self.copy_file(path, target)
|
||
|
|
||
|
def get_ext_outputs(self):
|
||
|
"""Get a list of relative paths to C extensions in the output distro"""
|
||
|
|
||
|
all_outputs = []
|
||
|
ext_outputs = []
|
||
|
|
||
|
paths = {self.bdist_dir: ''}
|
||
|
for base, dirs, files in sorted_walk(self.bdist_dir):
|
||
|
for filename in files:
|
||
|
if os.path.splitext(filename)[1].lower() in NATIVE_EXTENSIONS:
|
||
|
all_outputs.append(paths[base] + filename)
|
||
|
for filename in dirs:
|
||
|
paths[os.path.join(base, filename)] = paths[base] + filename + '/'
|
||
|
|
||
|
if self.distribution.has_ext_modules():
|
||
|
build_cmd = self.get_finalized_command('build_ext')
|
||
|
for ext in build_cmd.extensions:
|
||
|
if isinstance(ext, Library):
|
||
|
continue
|
||
|
fullname = build_cmd.get_ext_fullname(ext.name)
|
||
|
filename = build_cmd.get_ext_filename(fullname)
|
||
|
if not os.path.basename(filename).startswith('dl-'):
|
||
|
if os.path.exists(os.path.join(self.bdist_dir, filename)):
|
||
|
ext_outputs.append(filename)
|
||
|
|
||
|
return all_outputs, ext_outputs
|
||
|
|
||
|
|
||
|
NATIVE_EXTENSIONS = dict.fromkeys('.dll .so .dylib .pyd'.split())
|
||
|
|
||
|
|
||
|
def walk_egg(egg_dir):
|
||
|
"""Walk an unpacked egg's contents, skipping the metadata directory"""
|
||
|
walker = sorted_walk(egg_dir)
|
||
|
base, dirs, files = next(walker)
|
||
|
if 'EGG-INFO' in dirs:
|
||
|
dirs.remove('EGG-INFO')
|
||
|
yield base, dirs, files
|
||
|
yield from walker
|
||
|
|
||
|
|
||
|
def analyze_egg(egg_dir, stubs):
|
||
|
# check for existing flag in EGG-INFO
|
||
|
for flag, fn in safety_flags.items():
|
||
|
if os.path.exists(os.path.join(egg_dir, 'EGG-INFO', fn)):
|
||
|
return flag
|
||
|
if not can_scan():
|
||
|
return False
|
||
|
safe = True
|
||
|
for base, dirs, files in walk_egg(egg_dir):
|
||
|
for name in files:
|
||
|
if name.endswith('.py') or name.endswith('.pyw'):
|
||
|
continue
|
||
|
elif name.endswith('.pyc') or name.endswith('.pyo'):
|
||
|
# always scan, even if we already know we're not safe
|
||
|
safe = scan_module(egg_dir, base, name, stubs) and safe
|
||
|
return safe
|
||
|
|
||
|
|
||
|
def write_safety_flag(egg_dir, safe):
|
||
|
# Write or remove zip safety flag file(s)
|
||
|
for flag, fn in safety_flags.items():
|
||
|
fn = os.path.join(egg_dir, fn)
|
||
|
if os.path.exists(fn):
|
||
|
if safe is None or bool(safe) != flag:
|
||
|
os.unlink(fn)
|
||
|
elif safe is not None and bool(safe) == flag:
|
||
|
f = open(fn, 'wt')
|
||
|
f.write('\n')
|
||
|
f.close()
|
||
|
|
||
|
|
||
|
safety_flags = {
|
||
|
True: 'zip-safe',
|
||
|
False: 'not-zip-safe',
|
||
|
}
|
||
|
|
||
|
|
||
|
def scan_module(egg_dir, base, name, stubs):
|
||
|
"""Check whether module possibly uses unsafe-for-zipfile stuff"""
|
||
|
|
||
|
filename = os.path.join(base, name)
|
||
|
if filename[:-1] in stubs:
|
||
|
return True # Extension module
|
||
|
pkg = base[len(egg_dir) + 1 :].replace(os.sep, '.')
|
||
|
module = pkg + (pkg and '.' or '') + os.path.splitext(name)[0]
|
||
|
skip = 16 # skip magic & reserved? & date & file size
|
||
|
f = open(filename, 'rb')
|
||
|
f.read(skip)
|
||
|
code = marshal.load(f)
|
||
|
f.close()
|
||
|
safe = True
|
||
|
symbols = dict.fromkeys(iter_symbols(code))
|
||
|
for bad in ['__file__', '__path__']:
|
||
|
if bad in symbols:
|
||
|
log.warn("%s: module references %s", module, bad)
|
||
|
safe = False
|
||
|
if 'inspect' in symbols:
|
||
|
for bad in [
|
||
|
'getsource',
|
||
|
'getabsfile',
|
||
|
'getsourcefile',
|
||
|
'getfile' 'getsourcelines',
|
||
|
'findsource',
|
||
|
'getcomments',
|
||
|
'getframeinfo',
|
||
|
'getinnerframes',
|
||
|
'getouterframes',
|
||
|
'stack',
|
||
|
'trace',
|
||
|
]:
|
||
|
if bad in symbols:
|
||
|
log.warn("%s: module MAY be using inspect.%s", module, bad)
|
||
|
safe = False
|
||
|
return safe
|
||
|
|
||
|
|
||
|
def iter_symbols(code):
|
||
|
"""Yield names and strings used by `code` and its nested code objects"""
|
||
|
yield from code.co_names
|
||
|
for const in code.co_consts:
|
||
|
if isinstance(const, str):
|
||
|
yield const
|
||
|
elif isinstance(const, CodeType):
|
||
|
yield from iter_symbols(const)
|
||
|
|
||
|
|
||
|
def can_scan():
|
||
|
if not sys.platform.startswith('java') and sys.platform != 'cli':
|
||
|
# CPython, PyPy, etc.
|
||
|
return True
|
||
|
log.warn("Unable to analyze compiled code on this platform.")
|
||
|
log.warn(
|
||
|
"Please ask the author to include a 'zip_safe'"
|
||
|
" setting (either True or False) in the package's setup.py"
|
||
|
)
|
||
|
return False
|
||
|
|
||
|
|
||
|
# Attribute names of options for commands that might need to be convinced to
|
||
|
# install to the egg build directory
|
||
|
|
||
|
INSTALL_DIRECTORY_ATTRS = ['install_lib', 'install_dir', 'install_data', 'install_base']
|
||
|
|
||
|
|
||
|
def make_zipfile(zip_filename, base_dir, verbose=0, dry_run=0, compress=True, mode='w'):
|
||
|
"""Create a zip file from all the files under 'base_dir'. The output
|
||
|
zip file will be named 'base_dir' + ".zip". Uses either the "zipfile"
|
||
|
Python module (if available) or the InfoZIP "zip" utility (if installed
|
||
|
and found on the default search path). If neither tool is available,
|
||
|
raises DistutilsExecError. Returns the name of the output zip file.
|
||
|
"""
|
||
|
import zipfile
|
||
|
|
||
|
mkpath(os.path.dirname(zip_filename), dry_run=dry_run)
|
||
|
log.info("creating '%s' and adding '%s' to it", zip_filename, base_dir)
|
||
|
|
||
|
def visit(z, dirname, names):
|
||
|
for name in names:
|
||
|
path = os.path.normpath(os.path.join(dirname, name))
|
||
|
if os.path.isfile(path):
|
||
|
p = path[len(base_dir) + 1 :]
|
||
|
if not dry_run:
|
||
|
z.write(path, p)
|
||
|
log.debug("adding '%s'", p)
|
||
|
|
||
|
compression = zipfile.ZIP_DEFLATED if compress else zipfile.ZIP_STORED
|
||
|
if not dry_run:
|
||
|
z = zipfile.ZipFile(zip_filename, mode, compression=compression)
|
||
|
for dirname, dirs, files in sorted_walk(base_dir):
|
||
|
visit(z, dirname, files)
|
||
|
z.close()
|
||
|
else:
|
||
|
for dirname, dirs, files in sorted_walk(base_dir):
|
||
|
visit(None, dirname, files)
|
||
|
return zip_filename
|