2017-02-01 4 views
11

Возможно ли запустить comserver без необходимости повышения.Portable Python com server с использованием pywin32

Например, я могу запускать код с Python.TestServer (см. Ниже), но для этого требуется возвышение.

Python.TestServer код по адресу: Consuming Python COM Server from .NET

Можно ли запустить сервер ком, который не требует возвышения, так что я могу запустить COM-объект без пароля администратора.

, например

import pythoncom 
from win32com.server import localserver 

class demoObj(object): 
    _reg_clsctx_ = pythoncom.CLSCTX_LOCAL_SERVER 
    _reg_clsid_ = "{FA501660-8BB0-42F6-842B-A757FA3098DC}" 
    _reg_desc_ = "Demo COM server" 
    _reg_progid_ = "Python.Demo" 
    _public_methods_ = ['hello'] 

def hello(self, who): 
    return "Hellow " + who 

localserver.serve('B83DD222-7750-413D-A9AD-01B37021B24B') 

Я попытался выше код, но он говорит pywintypes.com_error: (-2147221005, 'Invalid class string', None, None)

как сделать правильную строку класса для локального сервера?

Пример VBA:

Sub demodemo() 
    Set obj = CreateObject("Python.Demo") 
    Debug.Print obj.Hello("World") 
End Sub 
+0

COM-сервер будет потребляться VBA. В чем проблема с vba? – Rahul

+0

Из любопытства, что вы пытаетесь сделать? Я знаю, что в vba есть другие альтернативы для ссылки excel и python – Kelvin

+0

вы пробовали xlwings? – denfromufa

ответ

7

Можно зарегистрировать и использовать класс без привилегий. Класс должен быть зарегистрирован у текущего пользователя, а не у всех пользователей. Опция не указана, поэтому вам необходимо зарегистрировать ее самостоятельно, написав ключи в HKCU\SOFTWARE\Classes.

Вот рабочий пример:

import os, sys, win32api, win32con, win32com.server.register 

class HelloWorld(object): 
    _reg_progid_ = "Python.TestServer" 
    _reg_clsid_ = "{7CC9F362-486D-11D1-BB48-0000E838A65F}" 
    _reg_desc_ = "Python Test COM Server" 
    _public_methods_ = ['Hello'] 

    def Hello(self): 
    return "Hello!" 



def RegisterClass(cls): 
    file = sys.modules[cls.__module__].__file__ 
    folder = os.path.dirname(file) 
    module = os.path.splitext(os.path.basename(file))[0] 
    python = win32com.server.register._find_localserver_exe(1) 
    python = win32api.GetShortPathName(python) 
    server = win32com.server.register._find_localserver_module() 
    command = '%s "%s" %s' % (python, server, cls._reg_clsid_) 
    typename = module + "." + cls.__name__ 

    def write(path, value): 
    win32api.RegSetValue(win32con.HKEY_CURRENT_USER, path, win32con.REG_SZ, value) 

    write("SOFTWARE\\Classes\\" + cls._reg_progid_ + '\\CLSID', cls._reg_clsid_) 
    write("SOFTWARE\\Classes\\AppID\\" + cls._reg_clsid_, cls._reg_progid_) 
    write("SOFTWARE\\Classes\\CLSID\\" + cls._reg_clsid_, cls._reg_desc_) 
    write("SOFTWARE\\Classes\\CLSID\\" + cls._reg_clsid_ + '\\LocalServer32', command) 
    write("SOFTWARE\\Classes\\CLSID\\" + cls._reg_clsid_ + '\\ProgID', cls._reg_progid_) 
    write("SOFTWARE\\Classes\\CLSID\\" + cls._reg_clsid_ + '\\PythonCOMPath', folder) 
    write("SOFTWARE\\Classes\\CLSID\\" + cls._reg_clsid_ + '\\PythonCOM', typename) 
    write("SOFTWARE\\Classes\\CLSID\\" + cls._reg_clsid_ + '\\Debugging', "0") 

    print("Registered %s" % cls.__name__) 

if __name__ == '__main__': 
    RegisterClass(HelloWorld) 
+0

Похоже на работу. Извините за то, что вы поздно ответили. Я был в отпуске – Rahul