使用Pyinstaller或Cython从Python模块创建可执行文件

时间:2018-09-04 16:03:41

标签: python cython cythonize

我想转换为具有模块结构的可执行this python 2.7项目:

(.venv) ip-192-168-22-127:indictrans loretoparisi$ tree -L 1
.
├── __init__.py
├── __init__.pyc
├── __init__.spec
├── _decode
├── _utils
├── base.py
├── build
├── mappings
├── models
├── script_transliterate.py
├── tests
├── transliterator.py
└── trunk

第一阶段,我正在为此使用pyinstaller

pyinstall --onefile __init__.py

我得到了一个可执行文件:

192 INFO: PyInstaller: 3.3.1
192 INFO: Python: 2.7.10
201 INFO: Platform: Darwin-17.7.0-x86_64-i386-64bit
202 INFO: wrote /Users/loretoparisi/Documents/Projects/AI/indic-trans/indictrans/__init__.spec
208 INFO: UPX is not available.
209 INFO: Extending PYTHONPATH with paths
['/Users/loretoparisi/Documents/Projects/AI/indic-trans',
 '/Users/loretoparisi/Documents/Projects/AI/indic-trans/indictrans']
210 INFO: checking Analysis
218 INFO: checking PYZ
223 INFO: checking PKG
224 INFO: Bootloader /Users/loretoparisi/Documents/Projects/AI/indic-trans/.venv/lib/python2.7/site-packages/PyInstaller/bootloader/Darwin-64bit/run
224 INFO: checking EXE
225 INFO: Rebuilding out00-EXE.toc because __init__ missing
225 INFO: Building EXE from out00-EXE.toc
225 INFO: Appending archive to EXE /Users/loretoparisi/Documents/Projects/AI/indic-trans/indictrans/dist/__init__
230 INFO: Fixing EXE for code signing /Users/loretoparisi/Documents/Projects/AI/indic-trans/indictrans/dist/__init__
234 INFO: Building EXE from out00-EXE.toc completed successfully.

但是当我运行它时,出现导入错误

Traceback (most recent call last):
  File "indictrans/__init__.py", line 9, in <module>
ValueError: Attempted relative import in non-package
[30629] Failed to execute script __init__

此库是通过cythonize安装程序使用 Cython 构建的,因此另一种选择是使用--embed Cython选项构建可执行的嵌入式模块。

我的setup.py是:

#!/usr/bin/env python

import os

from setuptools import setup
from setuptools.extension import Extension
from Cython.Build import cythonize

import numpy


os.environ['PBR_VERSION'] = '1.2.3'
os.environ['SKIP_WRITE_GIT_CHANGELOG'] = '1'
os.environ['SKIP_GENERATE_AUTHORS'] = '1'


extensions = [
    Extension(
        "indictrans._decode.beamsearch",
        [
            "indictrans/_decode/beamsearch.pyx"
        ],
        include_dirs=[numpy.get_include()]
    ),
    Extension(
        "indictrans._decode.viterbi",
        [
            "indictrans/_decode/viterbi.pyx"
        ],
        include_dirs=[numpy.get_include()]
    ),
    Extension(
        "indictrans._utils.ctranxn",
        [
            "indictrans/_utils/ctranxn.pyx"
        ],
        include_dirs=[numpy.get_include()]
    ),
    Extension(
        "indictrans._utils.sparseadd",
        [
            "indictrans/_utils/sparseadd.pyx"
        ],
        include_dirs=[numpy.get_include()]
    )

]

setup(
    setup_requires=['pbr'],
    pbr=True,
    ext_modules=cythonize(extensions)
)

尽管使用--embed option编译一个简单的python文件很容易,但有关更多信息,请参见here,但我不知道如何在{{1 }},以摆脱项目中的所有依赖关系。

1 个答案:

答案 0 :(得分:1)

我将使用setup.py文件作为 package 目录引用根软件包目录,在您的情况下为indic-trans。我将第一级源目录称为 module 目录,在您的情况下为indic-trans/indictrans


如果我正确理解了您的设置,则可能是因为尝试在 module 目录而不是 package 目录中的脚本上创建.exe目录。这使当前目录成为内部模块目录,并且相对引用将不起作用。

我在类似情况下解决此问题的方法是在主 package 目录(与run.py相同的文件夹)中创建一个setup.py脚本,该脚本会导入该软件包,然后在 module 目录中运行要运行的任何脚本。

您没有发布__init__.py文件(顺便说一句,在__init.py__中通常没有任何真实代码...),所以我在下面假设您要在main()脚本中运行base.py函数。在这种情况下,请使run.py类似:

# This will import everything from the module __init__.py based on it's "all" setup
# You likely want to limit it to just what is needed or just the script you are targeting
# Or you can use the "import indictrans" format and let __init__.py handle it
from indictrans import * 

def main():
    indictrans.base.main()

if __name__ == '__main__':
    main()

现在,您可以从命令行仅运行run.py或将其用作运行软件包的调试目标。

请注意,使用PyInstaller,您可以定位run.py(或run.spec)并使用--name参数将名称更改为其他名称。

pyinstaller --onefile --name indictrans run.spec

这将在indictrans.exe目录中创建dist


将程序包作为模块运行

请注意,在 module 目录中,您还可以创建一个__main__.py文件,该文件基本上是run.py的副本,并且执行相同的操作,但是将以如果模块是本地安装的,则由Python可执行文件提供。

python -m indictrans将使用__main__.py作为入口点运行您的软件包。