我将rasterio捆绑到可执行文件中时出了点问题
问题内容:
预期行为和实际行为。
我希望使用将脚本使用编译rasterio
为可执行文件pyinstaller
。该脚本可以在我的python环境中正常运行。但是我不能使用将其冻结为可执行文件PyInstaller
。
重现此问题的步骤。
我有一个脚本workflow_3.py
,其中包含以下内容:
import rasterio
而已。我尝试编译,然后使用pyinstaller如下运行:
(wps_env36) D:\11202750-002_RA2CE\Basis>pyinstaller workflow_3.py (wps_env36) D:\11202750-002_RA2CE\Basis>dist\workflow_3\workflow_3.exe
编译似乎已完成,但是当我运行可执行文件时,出现以下错误:
(wps_env36) D:\11202750-002_RA2CE\Basis>dist\workflow_3\workflow_3.exe
Traceback (most recent call last):
File "workflow_3.py", line 1, in <module>
import rasterio
File "c:\programdata\anaconda2\envs\wps_env36\lib\site-packages\PyInstaller\loader\pyimod03_i
mporters.py", line 627, in exec_module
exec(bytecode, module.__dict__)
File "site-packages\rasterio\__init__.py", line 23, in <module>
File "rasterio\_base.pyx", line 1, in init rasterio._base
ModuleNotFoundError: No module named 'rasterio._shim'
[17536] Failed to execute script workflow_3
尝试解决问题
我通过显式添加'rasterio._shim'
到hidden-imports
变量包含的列表中来修改了规范文件。然后我跑了pyinstaller workflow_3.spec
。这导致ModuleNotFoundError
了control
.pycrs.py
和vrt.py
。模块的其他问题。
将它们添加到hidden-imports
成功可以消除该ModuleNotFoundError
特定软件包的,但它仍在寻找其他软件包,所有这些软件包都包含在中C:\ProgramData\Anaconda2\envs\wps_env36\Lib\site- packages\rasterio
。此目录中大约有40个模块。将目录中的每个文件名都添加到hidden- imports
变量中似乎太多余了。实际上,我什至不知道它是否行得通。
因此,我还尝试将整个目录添加到我的pathex
变量中,以便它可以扩展PYTHONPATH
它。但是,这会导致另一个问题:
File "c:\programdata\anaconda2\envs\wps_env36\lib\traceback.py", line 5, in <module> File "c:\programdata\anaconda2\envs\wps_env36\lib\linecache.py", line 11, in <module> File "c:\programdata\anaconda2\envs\wps_env36\lib\tokenize.py", line 27, in <module> ImportError: cannot import name 'open' pre-safe-import-module hook failed, needs fixing.
操作系统
Windows 7的
Rasterio版本和出处
raconio版本是1.0.8,来自conda-forge python版本是3.6.6
我有两个版本的pyinstaller
pyinstaller 3.4 py36h7602738_0 conda-forge
PyInstaller 3.5.dev0+b13e6b30b <pip>
题
如何PyInstaller
冻结应用程序rasterio
?
问题答案:
我想出的当前解决方案是强制使用该程序包中的hidden- imports
所有模块供稿变量。在我的规格文件中,我添加了一些python代码来执行此操作:C:\ProgramData\Anaconda2\envs\wps_env36\Lib\sitepackages\rasterio``glob
# -*- mode: python -*-
block_cipher = None
import glob, os
rasterio_imports_paths = glob.glob(r'C:\ProgramData\Anaconda2\envs\wps_env36\Lib\site-packages\rasterio\*.py')
rasterio_imports = ['rasterio._shim']
for item in rasterio_imports_paths:
current_module_filename = os.path.split(item)[-1]
current_module_filename = 'rasterio.'+current_module_filename.replace('.py', '')
rasterio_imports.append(current_module_filename)
a = Analysis(['workflow_3.py'],
pathex=['D:\\11202750-002_RA2CE\\Basis'],
binaries=[],
datas=[],
hiddenimports=rasterio_imports,
hookspath=[],
runtime_hooks=[],
excludes=[],
win_no_prefer_redirects=False,
win_private_assemblies=False,
cipher=block_cipher,
noarchive=False)
pyz = PYZ(a.pure, a.zipped_data,
cipher=block_cipher)
exe = EXE(pyz,
a.scripts,
[],
exclude_binaries=True,
name='workflow_3',
debug=True,
bootloader_ignore_signals=False,
strip=False,
upx=True,
console=True )
coll = COLLECT(exe,
a.binaries,
a.zipfiles,
a.datas,
strip=False,
upx=True,
name='workflow_3')
不幸的是,这并不能解释为什么pyinstaller
最初无法看到这些模块。但是,它确实可以立即解决此问题,并且代码可以正常编译。