使用pyinstaller打包twistd插件

时间:2012-04-16 21:18:24

标签: python twisted pyinstaller twistd

我创建了一个很棒的python Twisted应用程序,其中包含一个扭曲的跑步者插件,如Twisted Documentation:http://twistedmatrix.com/documents/current/core/howto/tap.html中所述。我在使用PyInstaller打包时遇到问题:在执行冻结的应用程序时找不到我的扭曲插件。

为了运送我的项目,我使用twistd runner模块创建了我自己的顶级启动脚本,例如。

#!/usr/bin/env python
from twisted.scripts.twistd import run
from sys import argv
argv[1:] = [
  '--pidfile', '/var/run/myapp.pid',
  '--logfile', '/var/run/myapp.log',
  'myapp_plugin'
]
run()

接下来,我使用PyInstaller将其冻结为单个目录部署。执行上面的冻结脚本失败,因为它无法找到我的扭曲插件(为简洁而编辑):

~/pyinstall/dist/bin/mystartup?16632/twisted/python/modules.py:758:
UserWarning: ~/pyinstall/dist/mystartup?16632 (for module twisted.plugins)
not in path importer cache (PEP 302 violation - check your local configuration).

~/pyinstall/dist/bin/mystartup: Unknown command: myapp_plugin

通常,Twistd检查Python系统路径以在twisted / plugins / myapp_plugin.py中发现我的插件。如果我在启动脚本中打印扭曲插件列表,则列表在PyInstaller生成的可执行文件中为空,例如

from twisted.plugin import IPlugin, getPlugins
plugins = list(getPlugins(IPlugin))
print "Twistd plugins=%s" % plugins

我使用了一个默认的PyInstaller spec文件,没有指定隐藏的导入或导入钩子。

我喜欢使用日志记录,pid文件等进行扭曲的功能,因此我希望避免完全放弃扭曲的运行程序来规避插件问题。 有没有办法确保在冻结的可执行文件中找到我的扭曲插件?

1 个答案:

答案 0 :(得分:2)

我通过对一些扭曲的代码进行逆向工程找到了解决方法。在这里我硬编码插件导入。对我来说,这适用于PyInstaller。

#!/usr/bin/env python
import sys

from twisted.application import app
from twisted.scripts.twistd import runApp, ServerOptions

import myapp_plugin as myplugin


plug = myplugin.serviceMaker


class MyServerOptions(ServerOptions):
    """
    See twisted.application.app.ServerOptions.subCommands().
    Override to specify a single plugin subcommand and load the plugin
    explictly.
    """
    def subCommands(self):
        self.loadedPlugins = {plug.tapname:plug}
        yield (plug.tapname,
               None,
               # Avoid resolving the options attribute right away, in case
               # it's a property with a non-trivial getter (eg, one which
               # imports modules).
               lambda plug=plug: plug.options(),
               plug.description)

    subCommands = property(subCommands)


def run():
    """
    Replace twisted.application.app.run()
    To use our ServerOptions.
    """
    app.run(runApp, MyServerOptions)


sys.argv[1:] = [
    '--pidfile', '/var/run/myapp.pid',
    '--logfile', '/var/run/myapp.log',
    plug.tapname] + sys.argv[1:]

run()