我将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。这导致ModuleNotFoundErrorcontrol.pycrs.pyvrt.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最初无法看到这些模块。但是,它确实可以立即解决此问题,并且代码可以正常编译。