2017年7月31日 星期一

用 pywin32 COM makpy utiltiy 找群益api com 元件 GUID 碼

安裝pywin32
裝好後會在 python目錄/lib/site-pacjages/pythonwin 下看到一個 pythonwin IDE,執行後如下
選擇 Tools => COM makpy Uitility => Select library 選SKCOMlib後,會在 ~/lib\site-packages\win32com\gen_py\ 產生長串檔名的.py
打開後找 class ISKCenterLib, ISKOSQuoteLib 等的 coclass_clsid像是SKCenterLibclass_clsid 
{AC30BAB5-194A-4515-A8D3-6260749F8577}
之後就可以用這些class_clsid comtypes.client.CreatObject COM元件,請參考


群益 API in pyhton 報價範例

群益api更新至2.13.8後,改一下順序,先呼叫 SKReplyLib_ConnectByID,再 EnterMonitor,才會可以收到 OnConnection 3003 的回報,然後就可以求報價了。本範例示範如何用python 接群益api的報價,希望能拋磚引玉, 讓更多能手參與。小弟python自學,COM 元件的運作也是一知半解,東拼西湊下還算可用,程式碼很醜請見諒。過程遇到一些錯誤,很多我無法解決,以下是我試誤各種組合後, 可以運作的一組方法,,若有更好的解法,請各位大大多加分享。

另外是事件處理,我用comtypes 中的 showevents 也有錯誤產生,於是用 pyhoncom.PumpWaitingMessages() 來代替

使用comtpyes 若出現
"comtypes com object's method returns: 'tuple' object has no attribute 'ctypes_from_outparam'",需要做一些修正 ,請參考
https://easontseng.blogspot.com/2017/07/install-pythoncom-tuple-object-has-no.html
#這個bug好像在最新版的comtypes中被修正了

弄個FB社團,大家來討論好了。
https://www.facebook.com/groups/1805224676441902/

運行環境:
winXP, Anaconda python3.4 32 bit, CapitalAPI 2.13.8, pywin32 build221, comtypes 1.1.3

參考文獻:
#pythoncom 是安裝 pywin32 附帶的
https://sourceforge.net/projects/pywin32/
#用pywin makepy utility 找 GUID
https://easontseng.blogspot.tw/2017/07/pywin32-com-makpy-utiltiy-api-com.html
#python 元大期貨 api,
http://hlfutures.blogspot.tw/2016/08/api-in-python_8.html

#comtpyes:
https://pythonhosted.org/comtypes/
http://starship.python.net/crew/theller/comtypes/
https://easontseng.blogspot.tw/2017_07_07_archive.html

###########################################################
# pythoncom 是安裝 pywin32 附的,如果 event 沒反應才需要用到
# 第一次使用cc.GetModule, 會在 comtypes\gen\ 下產生幾個.py檔,像是
# _75AAD71C_8F4F_4F1F_9AEE_3D41A8C9BA5E_0_1_0.py
# 如果有 dll 錯誤產生,請檢查 SKCOM.dll 路徑有沒有錯誤
# 如果發生 gen 下沒有 SKCOMLib 錯誤訊息,請將python關閉
# AttributeError: module 'comtypes.gen.SKCOMLib' has no attribute 'SKCenterLib'
# 重新啟動python後,應該就可以找到了
# 如果還是不行,請重新安裝 capital API 元件,注意 API 與python 版本要一致,
# API安裝要以管理員權限執行

# 20190506,程式碼簡化
import pythoncom, time
import comtypes.client as cc

cc.GetModule('C:\\SKCOM\\x86\\SKCOM.dll')
import comtypes.gen.SKCOMLib as sk

ts=sk.SKSTOCK()

skC=cc.CreateObject(sk.SKCenterLib,interface=sk.ISKCenterLib)
skQ=cc.CreateObject(sk.SKQuoteLib,interface=sk.ISKQuoteLib)

#Some Configure
ID='身分證'
PW='密碼'

#想取得報價的股票代碼
strStocks='TSEA'

#define functions
def getStock(nMarket, nIndex, ts):
    skQ.SKQuoteLib_GetStockByIndex(nMarket, nIndex, ts)
    print(ts.bstrStockName, ts.bstrStockNo,  ts.nClose/10**ts.sDecimal)

#建立事件類別
class skQ_events:
    def OnConnection(self, nKind, nCode):
        if nCode == 0 :
            if nKind == 3001 :
                print("skQ OnConnection, nkind= ", nKind)
            elif (nKind == 3003):
                #等到回報3003 確定連線報價伺服器成功後,才登陸要報價的股票
                skQ.SKQuoteLib_RequestStocks(1, strStocks)
                print("skQ OnConnection, request stocks, nkind= ", nKind)
    def OnNotifyQuote(self, sMarketNo, sStockIdx):
        getStock(sMarketNo, sStockIdx, ts)

#Event sink
EventQ=skQ_events()
#make connection to event sink
ConnectionQ = cc.GetEvents(skQ, EventQ)        

#Login
print("Login,", skC.SKCenterLib_GetReturnCodeMessage(skC.SKCenterLib_Login(ID,PW)))
time.sleep(1)

#登錄報價伺服器
print("EnterMonitor,", skC.SKCenterLib_GetReturnCodeMessage(skQ.SKQuoteLib_EnterMonitor()))
#每秒 pump event 一次,這裡示範15秒

for i in range(15):
    time.sleep(1)
    pythoncom.PumpWaitingMessages()

2017年7月7日 星期五

python, COM, 群益 api

我剛開始自學python想做一些訊號分析的工作,想要用來分析金融商品,做自動化交易的工作。網路上很多人推薦群益的api,可以查詢報價,且伺服器相對穩定,取得速度又快,還不用費用。群益api是使用com元件的架構,我第一次使用的comtypes 這個module,但常常會莫名死機,不是很穩定。後來再google了許久後,發現可以使用win32com這個module,情況似乎好了點。以下是一些使用過程中的知識,有些用語其實我不知道正確的用法,就以我的理解寫下來了,拉拉雜雜,當作是一個筆記,先記上了,有空再來整理。希望可以有同好一起討論,我沒什麼程式經驗就是了,純粹自己興趣。


1. 什麼是 COM,先稍微了解一下com的架構
https://zh.wikipedia.org/wiki/%E7%BB%84%E4%BB%B6%E5%AF%B9%E8%B1%A1%E6%A8%A1%E5%9E%8B

2.安裝pywin32, 使用 makepy
參考網址HL的做法
http://hlfutures.blogspot.tw/2016/08/api-in-python_8.html

HL大用元大api的例子中可以找到# This CoClass is known by the name 'Yuanta.YuantaOrdCtrl.1'
from win32com.client import CoClassBaseClass
# This CoClass is known by the name 'Yuanta.YuantaOrdCtrl.1'
class YuantaOrd(CoClassBaseClass): # A CoClass

再把 Yuanta.YuantaOrdCtrl.1給win32com.client.Disptch('Yuanta.YuantaOrdCtrl.1') ,來使用
但群益的api不像元大api ,找不到name的訊息,後來我發現

class ISKCenterLib(DispatchBaseClass):
 CLSID = IID('{D61780D3-2239-4FD8-9C64-0E47B2E75464}')
 coclass_clsid = IID('{AC30BAB5-194A-4515-A8D3-6260749F8577}')
把coclass_clsid這串當作progid用也是可行的
所以輸入:
    
skC=win32com.client.Dispatch('{AC30BAB5-194A-4515-A8D3-6260749F8577}')

就可以用skC呼叫SKCenterLib底下的方法了

skC.SKCenterLib_GetReturnCodeMessage(skC.SKCenterLib_Login(ID,PW))
Out[5]: 'SK_SUCCESS'

這樣就可以登入群益伺服器主機了


3.取得報價,我不會!!  有人可以教我嗎?  要使用event 嗎? 不知道該怎麼處理event
我研究出來可以獲取報價的方式了,請參考
https://easontseng.blogspot.tw/2017/07/api-in-pyhton.html