py2exe / pyinstaller и DispatchWithEvents - PullRequest
3 голосов
/ 26 апреля 2010

У меня есть программа, которая использует библиотеку 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

Я знаю, что могу вручную добавить typelib в свой файл setup.py, но я бы хотел запускать код на компьютерах с разными версиями iTunes без перекомпиляции, поэтому я бы предпочел динамически создавать его. Если нет способа сделать это с помощью setup / spec, может быть, есть другой способ загрузить события? Спасибо.


Дополнительно:

Благодаря Райану я обнаружил, что могу взять сгенерированный файл 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 ]

2 голосов
/ 23 января 2012

Я столкнулся с точно такой же ошибкой. Эта ссылка направила меня в правильном направлении -> http://www.py2exe.org/index.cgi/UsingEnsureDispatch однако упоминается, что: NB. Вы должны убедиться, что каталог python ... \ win32com.client.gen_py не существует разрешить создание кеша в% temp% Что немного сбивало с толку. Для меня это решило переименование "C: \ Python26 \ Lib \ site-packages \ win32com \ gen_py" в "C: \ Python26 \ Lib \ site-packages \ win32com \ gen_pybak" (при запуске py2exe)

1 голос
/ 08 октября 2011

Это официальный способ сделать это.

(Изменить: дословно скопированный пример с этой ссылки)

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 )
0 голосов
/ 26 апреля 2010

Вместо того, чтобы зависеть от кеша, я бы порекомендовал перейти в каталог локального кеша, скопировать сгенерированный файл в файл локального проекта, назвать его чем-то вроде ITunesInterface.py и явно вызвать его. Это заставит py2exe вытянуть его в ваше скомпилированное приложение.

...