py2exe / pyinstaller和DispatchWithEvents

时间:2010-04-26 14:38:30

标签: python py2exe win32com pyinstaller

我有一个使用win32com库来控制iTunes的程序,但是在编译成可执行文件时遇到了一些问题。问题似乎围绕使用DispatchWithEvents而不是Dispatch。我创建了一个非常简单的程序来说明我的问题:

import win32com.client
win32com.client.gencache.is_readonly = False #From py2exe wiki

class ITunesEvents(object):
    def __init__(self): self.comEnabled = True
    def OnCOMCallsDisabledEvent(self, reason): self.comEnabled = False
    def OnCOMCallsEnabledEvent(self): self.comEnabled = True

# The first line works in the exe, the second doesn't.
itunes = win32com.client.Dispatch("iTunes.Application")
#itunes = win32com.client.DispatchWithEvents("iTunes.Application", ITunesEvents)

lib = getattr(itunes, "LibraryPlaylist")
src = getattr(lib, "Source")
playlists = getattr(src, "Playlists")

print "Found %i playlists." % getattr(playlists, "Count")

使用Dispatch,程序可以正确编译和运行。使用DispatchWithEvents,程序在从命令行调用时运行正常,但在运行exe时会产生以下错误:

Traceback (most recent call last):
File "sandbox.py", line 16, in <module>
  itunes = win32com.client.DispatchWithEvents("iTunes.Application", ITunesEvents)
File "win32com\client\__init__.pyc", line 252, in DispatchWithEvents
File "win32com\client\gencache.pyc", line 520, in EnsureModule
File "win32com\client\gencache.pyc", line 287, in MakeModuleForTypelib
File "win32com\client\makepy.pyc", line 259, in GenerateFromTypeLibSpec
File "win32com\client\gencache.pyc", line 141, in GetGeneratePath
IOError: [Errno 2] No such file or directory: '[distDir]\\library.zip\\win32com\\gen_py\\__init__.py'

我也尝试过使用PyInstaller,它会产生类似的错误:

File "<string>", line 16, in <module>
File "[outDir]/win32com.client", line 252, in DispatchWithEvents
File "[outDir]/win32com.client.gencache", line 520, in EnsureModule
File "[outDir]/win32com.client.gencache", line 287, in MakeModuleForTypelib
File "[outDir]/win32com.client.makepy", line 286, in GenerateFromTypeLibSpec
File "[outDir]/win32com.client.gencache", line 550, in AddModuleToCache
File "[outDir]/win32com.client.gencache", line 629, in _GetModule
File "[pyinstallerDir]\iu.py", line 455, in importHook
    raise ImportError, "No module named %s" % fqname
ImportError: No module named win32com.gen_py.9E93C96F-CF0D-43F6-8BA8-B807A3370712x0x1x13

我知道我可以在setup.py文件中手动添加typelib,但是我想在不同版本的iTunes的计算机上运行代码而不重新编译,所以我更愿意动态创建它。如果使用setup / spec无法做到这一点,可能还有另一种加载事件的方法吗?感谢。


增加:

感谢Ryan,我发现我可以使用生成的py文件,经过一番挖掘后,能够得到以下内容。

获取生成的py文件(来自makepy.py)并将其重命名为cominterface.py。然后,您需要执行以下操作以使用事件处理程序实际创建COM对象。

import cominterface
from types import ClassType
from win32com.client import EventsProxy, _event_setattr_

class ItunesEvents:
    '''iTunes events class. See cominterface for details.'''
    def OnPlayerPlayEvent(self, t):print "Playing..."
    def OnPlayerStopEvent(self, t): print "Stopping..."

itunes = cominterface.iTunesApp()
rClass = ClassType("COMEventClass", (itunes.__class__, itunes.default_source, ItunesEvents), {'__setattr__': _event_setattr_})
instance = rClass(itunes._oleobj_)
itunes.default_source.__init__(instance, instance)
#ItunesEvents.__init__(instance) #Uncomment this line if your events class has __init__.
itunes = EventsProxy(instance)

然后你就可以开展业务了。

3 个答案:

答案 0 :(得分:2)

我遇到了完全相同的错误。 这个链接让我朝着正确的方向前进 - &gt; http://www.py2exe.org/index.cgi/UsingEnsureDispatch 但它提到: 注意:您必须确保python ... \ win32com.client.gen_py目录不存在 允许在%temp%中创建缓存 这有点令人困惑。 为我解决的是重命名&#34; C:\ Python26 \ Lib \ site-packages \ win32com \ gen_py&#34;到&#34; C:\ Python26 \ Lib \ site-packages \ win32com \ gen_pybak&#34; (运行py2exe时)

答案 1 :(得分:1)

This是正式的做法。

(编辑:从上面的链接中逐字复制的示例)

import win32com.client
if win32com.client.gencache.is_readonly == True:

    #allow gencache to create the cached wrapper objects
    win32com.client.gencache.is_readonly = False

    # under p2exe the call in gencache to __init__() does not happen
    # so we use Rebuild() to force the creation of the gen_py folder
    win32com.client.gencache.Rebuild()

    # NB You must ensure that the python...\win32com.client.gen_py dir does not exist
    # to allow creation of the cache in %temp%

# Use SAPI speech through IDispatch
from win32com.client.gencache import EnsureDispatch
from win32com.client import constants
voice = EnsureDispatch("Sapi.SpVoice", bForDemand=0)
voice.Speak( "Hello World.", constants.SVSFlagsAsync )

答案 2 :(得分:-1)

我建议不要依赖缓存,而是进入本地缓存目录,将生成的文件复制到本地项目文件中,并将其命名为ITunesInterface.py,并明确地调用它。这将使py2exe将其拉入已编译的应用程序。