#-----------------------------------------------------------------------------
|
# Copyright (c) 2013-2023, PyInstaller Development Team.
|
#
|
# Distributed under the terms of the GNU General Public License (version 2
|
# or later) with exception for distributing the bootloader.
|
#
|
# The full license is in the file COPYING.txt, distributed with this software.
|
#
|
# SPDX-License-Identifier: (GPL-2.0-or-later WITH Bootloader-exception)
|
#-----------------------------------------------------------------------------
|
|
from PyInstaller import isolated
|
|
|
def pre_safe_import_module(api):
|
"""
|
Add the `six.moves` module as a dynamically defined runtime module node and all modules mapped by
|
`six._SixMetaPathImporter` as aliased module nodes to the passed graph.
|
|
The `six.moves` module is dynamically defined at runtime by the `six` module and hence cannot be imported in the
|
standard way. Instead, this hook adds a placeholder node for the `six.moves` module to the graph,
|
which implicitly adds an edge from that node to the node for its parent `six` module. This ensures that the `six`
|
module will be frozen into the executable. (Phew!)
|
|
`six._SixMetaPathImporter` is a PEP 302-compliant module importer converting imports independent of the current
|
Python version into imports specific to that version (e.g., under Python 3, from `from six.moves import
|
tkinter_tix` to `import tkinter.tix`). For each such mapping, this hook adds a corresponding module alias to the
|
graph allowing PyInstaller to translate the former to the latter.
|
"""
|
@isolated.call
|
def real_to_six_module_name():
|
"""
|
Generate a dictionary from conventional module names to "six.moves" attribute names (e.g., from `tkinter.tix` to
|
`six.moves.tkinter_tix`).
|
"""
|
import six
|
|
# Iterate over the "six._moved_attributes" list rather than the "six._importer.known_modules" dictionary, as
|
# "urllib"-specific moved modules are overwritten in the latter with unhelpful "LazyModule" objects. If this is
|
# a moved module or attribute, map the corresponding module. In the case of moved attributes, the attribute's
|
# module is mapped while the attribute itself is mapped at runtime and hence ignored here.
|
return {
|
moved.mod: 'six.moves.' + moved.name
|
for moved in six._moved_attributes if isinstance(moved, (six.MovedModule, six.MovedAttribute))
|
}
|
|
# Add "six.moves" as a runtime package rather than module. Modules cannot physically contain submodules; only
|
# packages can. In "from"-style import statements (e.g., "from six.moves import queue"), this implies that:
|
# * Attributes imported from customary modules are guaranteed *NOT* to be submodules. Hence, ModuleGraph justifiably
|
# ignores these attributes. While some attributes declared by "six.moves" are ignorable non-modules (e.g.,
|
# functions, classes), others are non-ignorable submodules that must be imported. Adding "six.moves" as a runtime
|
# module causes ModuleGraph to ignore these submodules, which defeats the entire point.
|
# * Attributes imported from packages could be submodules. To disambiguate non-ignorable submodules from ignorable
|
# non-submodules (e.g., classes, variables), ModuleGraph first attempts to import these attributes as submodules.
|
# This is exactly what we want.
|
if isinstance(real_to_six_module_name, str):
|
raise SystemExit("pre-safe-import-module hook failed, needs fixing.")
|
api.add_runtime_package(api.module_name)
|
for real_module_name, six_module_name in real_to_six_module_name.items():
|
api.add_alias_module(real_module_name, six_module_name)
|