最近自己的这个命令行工具管理的项目差不多做到了尾声,需要转移到虚拟环境中进行打包,涉及到了使用pyinstaller,引出了环境切换,资源存放,查找模块等一系列问题,做一下记录。
(一)首先进入到项目根目录,创建一个名为venv的虚拟环境
python3 -m venv venv
然后启用虚拟环境
source venv/bin/activate
安装PyQt5
pip3 install pyqt5 pyinstaller
现在可以在终端的虚拟环境下尝试运行程序,或者在pycharm添加并配置虚拟环境的解释器,尝试运行。
哦然而这并不会成功,并抛出了异常:
(二)参考stackoverflow上的一条建议,添加QT_QPA_PLATFORM_PLUGIN_PATH环境变量,在Pycharm中配置路径:venv/lib/python3.9/site-packages/PyQt5/Qt5/plugins
程序就切换完环境可以跑起来了:
(三)然后使用pyinstaller进行打包,打包之前先检查一下pyinstaller的位置是不是在刚安装的虚拟环境中,如果不是可以将其他环境中的先uninstall。
which pyinstaller
然后使用pyinstaller进行打包。
pyinstaller -Fw mac_app.py
查看dist目录下生成的两个文件
运行.app发现在dock栏跳两下之后闪退,然后在终端运行可执行程序mac_app,发现有报错
可以看到找不到文件 ,这里代码中使用的是sys.path[0],看来不能这么用,查资料。。
Not a directory: '/var/folders/zl/3r_pj56x0z1gdcsmh5bv6ytm0000gn/T/_MEIIfnbhF/base_library.zip/Json/configs.json'
(四)然后就涉及到pyinstaller的资源打包相关的问题,由于打包后的应用放置的位置不确定,不能在代码中直接使用绝对路径,所以需要处理一下项目中用到的资源路径。先看下笔者这个项目的资源路径,可以看到第二栏为项目地址,为了整洁将.py文件集体放到了Project下,而资源又放到了其中的Resource文件中。
首先来看,相对路径根据打包与否,转换成不同绝对路径的方法:这里参考网络上流传着如下这么一个神奇的代码来处理相对路径到绝对路径。原理就不深究了,以下代码的效果大概是:这里if打包后的base_pass打印出为.app/contents/resources/,else未打包base_pass打印出为QuickTerminal/Project/
def resource_path(relative_path):
if getattr(sys, 'frozen', False): # 打包后的路径
base_path = sys._MEIPASS
print(base_path)
else:
base_path = os.path.abspath(".") # 未打包的路径
base_path = os.path.join(base_path, "Project") # 是否需要这一步要看资源位于项目中的路径
print(base_path)
return os.path.join(base_path, relative_path)
CONFIGS_PATH = resource_path("Resource/Json/configs.json")
SCRIPTS_PATH = resource_path("Resource/Scripts/")
当然上面的代码只是告诉程序去bundle的哪里寻找资源,还需要告诉pyinstaller打包时要把资源放到bundle的那个位置去,这个是靠配置文件完成的,首先获得那个配置文件(带资源不能使用-F):
pinstaller -w mac_app.py
可以得到项目目录下面的一个mac_app.spec,大概算是pyinstaller进行文件打包的一个配置文件,使用文本编辑器打开,编辑下面标红的data,格式为datas=[(‘项目根目录下要存入的文件或文件夹', '存入.app的Resource文件夹中的位置'), ('···', '···')]
以本项目为例则是
datas=[('Project/Resource/Json', './Resource/Json'),('Project/Resource/Scripts', './Resource/Scripts')],
按照上图编辑完之后执行pyinstaller mac_app.spec即可在dist目录下查看生成的.app文件,查看包内容就可以看到保存到.app内的资源文件了。
(五)如果.app文件运行的过程中出了啥问题闪退了,可以使用终端运行dist同级目录下的同名可执行文件,会有报错的文件显示在终端,可以进行debug,大部分都是因为找不到什么文件导致的问题。
于是我就遇到了另一个问题。。找不到模块。
排查之后发现,由于我之前的.py文件都是直接散在QuickTerminal根目录下的,将他们移动到Project文件夹的时候,使用Pycharm自动进行了Refactor,导致模块的导入都变成了这个样子:
解决方法是去掉前面的Project.使用Pycharm仍然可以编辑运行。但是pyinstaller需要设置导入这些模块,也就是添加参数-p,加Project文件夹(终端里我使用的都是绝对路径,已隐去)
pyinstaller -w /Project/mac_app.py -p /Project
其他就是pyinstaller再添加一些参数来设定程序和程序名:
pyinstaller -w /Project/mac_app.py -i /Project/Resource/icon.ico -n QuickTerminal -p Project
最终,此文涉及到的,完整的.sepc打包配置文件需要注意的地方大概标出来如下: