資源の読み込みに... 荷物...

FMZ Quant の新しい機能: _Serve 機能を使用して HTTP サービスを簡単に作成する

作者: リン・ハーンFMZ~リディア作成日:2024年11月13日 11:59:30 更新日:2024年11月15日 09:57:05

New Feature of FMZ Quant: Use _Serve Function to Create HTTP Services Easily

定量取引および自動化された戦略開発では,httpサービスが時々使用されます. FMZ Quant Trading Platformは機能を追加しました_Serve()最近,この機能は,ユーザに柔軟な HTTP,HTTPS,TCP サービス作成機能を提供しています.この機能により,開発者はサービスの構成プロセスを簡素化し,定量的な環境でよりカスタマイズされたサービスを実装し,戦略設計をよりスムーズかつ便利にします.この記事では,機能の使用シナリオと基本的な操作を紹介します._Serve()FMZ Quantのこの新しい機能で 早く始められるようにします

プラットフォームのAPIのドキュメント_Serve()更新されました:

https://www.fmz.com/syntax-guide/fun/global/__serve

需要

プラットフォームは機能をアップグレードしました_Serve()この機能は,ネットワークサービスを作成する機能を持つことを可能にします.この機能に基づいて,多くの機能を開発し,多くの問題を解決できます.例えば,戦略は外部インターフェース,データ転送,およびプラットフォームのカスタムプロトコル機能と協力して,FMZプラットフォームがサポートしていない交換をシームレスにカプセルすることができます.

この記事では,FMZプラットフォームがサポートしていない交換をシームレスにカプセルするために,プラットフォームのカスタムプロトコル機能と協力する要求を例として使用します.カスタムプロトコルガイド,我々はPython言語を使用して,OX交換のAPIをスポットモードでカプセル化しました (FMZ自体はOKXをサポートしているため,OKXはここで例として使用され,FMZプラットフォームに接続されていない他の交換にも適用されます).この記事ではPythonのカスタムプロトコルプログラムは別々に実行する必要があります.JavaScript言語が関数をサポートする場合_Serve()JavaScript 言語戦略がカスタム プロトコルにアクセスする方が簡単です

交換インターフェースのカスタムプロトコルを,テンプレートライブラリとしてカスタムして,戦略に直接統合し,戦略がFMZでサポートされていない交換にシームレスにアクセスできるようにします.ここでカスタムプロトコル交換オブジェクトの設定方法については詳細には触れません.

https://www.fmz.com/bbs-topic/10527

  • プラットフォーム上のカスタムプロトコル交換構成は以下のとおりである.

New Feature of FMZ Quant: Use _Serve Function to Create HTTP Services Easily

テンプレートを設計する際には/OKX設定されたカスタムプロトコル交換オブジェクトがどの交換所属かを識別する.

カスタム プロトコル テンプレートの実装

まず FMZ Quant Trading Platform で新しい戦略を作成し,戦略タイプをテンプレートライブラリに設定し,戦略言語を JavaScript に設定します.

テンプレートパラメータ設計

作成された戦略テンプレートに 3 つのパラメータを追加します.

New Feature of FMZ Quant: Use _Serve Function to Create HTTP Services Easily

プログラムテンプレートの設計とコードを書くことができます

コード実施

TSスタイルで書かれています.$.startService()function は,カスタム プロトコル サービスを起動するために使用されるテンプレート インターフェース 機能です.

// @ts-check

$.startService = function (address, port, proxyConfig) {
    __Serve(`http://${address}:${port}`, function (ctx, proxyConfig) {
        // interface
        interface IData {
            data: object
            raw: object
        }
        
        interface IError {
            error: any
        }
        
        // custom protocol for OKX
        class CustomProtocolOKX {
            apiBase: string = "https://www.okx.com"
            accessKey: string
            secretKey: string
            passphrase: string
            proxyConfig: string = ""
            simulate: boolean = false
            
            constructor(accessKey: string, secretKey: string, passphrase: string, simulate?: boolean, proxyConfig?: string) {
                this.accessKey = accessKey
                this.secretKey = secretKey
                this.passphrase = passphrase
                if (typeof(simulate) == "boolean") {
                    this.simulate = simulate
                }
                this.proxyConfig = proxyConfig
            }
            
            httpReq(method: string, path: string, query: string = "", params: {[key: string]: any} = {}, headers: {key: string, value: string | ArrayBuffer}[] = []): {[key: string]: any} {
                let ret = null
                let options = {
                    method: method,                    
                    headers: {
                        'User-Agent': 'Mozilla/5.0 (Windows; U; Windows NT 6.1; en-US; rv:1.9.1.6) Gecko/20091201 Firefox/3.5.6', 
                        'Content-Type': 'application/json; charset=UTF-8',
                        'x-simulated-trading': this.simulate ? "1" : "0"
                    },
                }
                
                // headers
                if (Array.isArray(headers) && headers.length > 0) {
                    for (var pair of headers) {
                        options.headers[pair.key] = pair.value
                    }
                }
                
                let url = ""
                if (method == "GET") {
                    if (typeof(query) == "string" && query.length > 0) {
                        url = `${this.apiBase}${path}?${query}`
                    } else {
                        url = `${this.apiBase}${path}`
                    }
                } else {
                    url = `${this.apiBase}${path}`
                    options.body = JSON.stringify(params)
                }

                // request
                try {
                    if (this.proxyConfig != "") {
                        url = `${this.proxyConfig}${url}`
                    }
                    ret = JSON.parse(HttpQuery(url, options))
                } catch(e) {
                    return null
                }
    
                return ret
            }

            callSignedAPI(method: string, path: string, query: string = "", params: {[key: string]: any} = {}): {[key: string]: any} {
                const strTime = new Date().toISOString().slice(0, -5) + 'Z'
                let jsonStr = ""
                if (method == "GET") {
                    jsonStr = Object.keys(params).length > 0 ? JSON.stringify(params) : ""
                } else {
                    jsonStr = Object.keys(params).length > 0 ? JSON.stringify(params) : "{}"
                }                
                let message = `${strTime}${method}${path}${jsonStr}`
                if (method === "GET" && query !== "") {
                    message = `${strTime}${method}${path}?${query}${jsonStr}`
                }

                const signature = Encode("sha256", "string", "base64", message, "string", this.secretKey)
                let headers = []
                headers.push({key: "OK-ACCESS-KEY", value: this.accessKey})
                headers.push({key: "OK-ACCESS-PASSPHRASE", value: this.passphrase})
                headers.push({key: "OK-ACCESS-TIMESTAMP", value: strTime})
                headers.push({key: "OK-ACCESS-SIGN", value: signature})                

                return this.httpReq(method, path, query, params, headers)
            }
            
            urlEncode(params: {[key: string]: string | number}): string {
                let encodeParams: string[] = []
                for (const [key, value] of Object.entries(params)) {
                    encodeParams.push(`${encodeURIComponent(key)}=${encodeURIComponent(value)}`)
                }

                return encodeParams.join("&")
            }

            symbol2Inst(symbol: string): string {
                let arr = symbol.split("_")
                if (arr.length >= 2) {
                    return `${arr[0]}-${arr[1]}`.toUpperCase()
                } else {
                    return `${arr[0]}-USDT`.toUpperCase()
                }
            }

            getSymbol(inst: string): string {
                let arr = inst.split("-")
                if (arr.length >= 2) {
                    return `${arr[0]}_${arr[1]}`.toUpperCase()
                } else {
                    return `${arr[0]}-USDT`.toUpperCase()
                }
            }
            
            // The following code encapsulates OKX's interface
            GetTicker(symbol: string): IData | IError {
                // GET /api/v5/market/ticker , param: instId 

                let inst = this.symbol2Inst(symbol)
                let ret = this.httpReq("GET", "/api/v5/market/ticker", `instId=${inst}`)

                let retData = {}
                for (var ele of ret["data"]) {
                    retData["symbol"] = this.getSymbol(ele["instId"])
                    retData["buy"] = ele["bidPx"]
                    retData["sell"] = ele["askPx"]
                    retData["high"] = ele["high24h"]
                    retData["low"] = ele["low24h"]
                    retData["open"] = ele["open24h"]
                    retData["last"] = ele["last"]
                    retData["vol"] = ele["vol24h"]
                    retData["time"] = ele["ts"]
                }

                return {data: retData, raw: ret}
            }
            
            GetAccount(): IData | IError {
                // GET /api/v5/account/balance
                
                let ret = this.callSignedAPI("GET", "/api/v5/account/balance")

                let retData = []
                for (var ele of ret["data"]) {
                    for (var detail of ele["details"]) {
                        let asset = {"currency": detail["ccy"], "free": detail["availEq"], "frozen": detail["ordFrozen"]}
                        if (detail["availEq"] == "") {
                            asset["free"] = detail["availBal"]
                        }
                        retData.push(asset)
                    }
                }
                
                return {data: retData, raw: ret}
            }

            IO(method: string, path: string, params: {[key: string]: any}): {[key: string]: any} {
                let ret = null 
                if (method == "GET") {
                    ret = this.callSignedAPI(method, path, this.urlEncode(params))
                } else {
                    ret = this.callSignedAPI(method, path, "", params)
                }

                return {data: ret}
            }
        }
        
        // protocol factory
        class ProtocolFactory {
            static createExWrapper(accessKey: string, secretKey: string, exName: string): any {
                let protocol = null
                if (exName == "/OKX") {
                    try {
                        let passphrase = ""
                        let simulate = false
                        let arrSecretKey = secretKey.split(",")

                        if (arrSecretKey.length == 2) {
                            secretKey = arrSecretKey[0]
                            passphrase = arrSecretKey[1]
                        } else if (arrSecretKey.length == 3) {
                            secretKey = arrSecretKey[0]
                            passphrase = arrSecretKey[1]
                            simulate = arrSecretKey[2] == "simulate" ? true : false 
                        } else {
                            return null
                        }
                        protocol = new CustomProtocolOKX(accessKey, secretKey, passphrase, simulate, proxyConfig)
                    } catch(e) {
                        Log("e.name:", e.name, "e.stack:", e.stack, "e.message:", e.message)
                        return null
                    }
                }
                return protocol
            }
        }
        
        // http service
        let resp = {}
        let reqMethod = ctx.method()
        let reqPath = ctx.path()
        let httpMethod = ctx.header("Http-Method")
        let reqBody = null

        try {
            reqBody = JSON.parse(ctx.body())
        } catch(e) {
            resp = {error: {name: e.name, stack: e.stack, message: e.message, errDesc: "JSON parse error."}}
        }

        // onPost
        if (reqMethod == "POST") {
            if (!["access_key", "secret_key", "method", "params"].every(key=> key in reqBody)) {
                resp = {error: {reqBody: reqBody, errDesc: "reqBody error."}}
            }

            if ("error" in resp) {
                ctx.write(JSON.stringify(resp))
                return 
            }

            let accessKey = reqBody["access_key"]
            let secretKey = reqBody["secret_key"]
            let method = reqBody["method"]
            let params = reqBody["params"]

            let protocol = ProtocolFactory.createExWrapper(accessKey, secretKey, reqPath)
            if (!protocol) {
                ctx.write(JSON.stringify({error: {errDesc: "createExWrapper error."}}))
                return 
            }

            // process GetTicker / GetAccount ...
            if (method == "ticker") {
                if (!["symbol"].every(key=> key in params)) {
                    resp = {error: {params: params, errDesc: "params error."}}
                } else {
                    let symbol = params["symbol"]
                    resp = protocol.GetTicker(symbol)
                }
            } else if (method == "accounts") {
                resp = protocol.GetAccount()
            } else if (method.slice(0, 6) == "__api_") {
                resp = protocol.IO(httpMethod, method.slice(6), params)
            } else {
                ctx.write(JSON.stringify({error: {method: method, errDesc: "method not support."}}))
                return 
            }

            ctx.write(JSON.stringify(resp))
        }
    }, proxyConfig)
}

function init() {
    $.startService(address, port, proxyConfig)
    Log("Start the custom protocol service, address:", address, ",port:", port, "#FF0000")
    if (proxyConfig != "") {
        Log("Setting up the proxy:", proxyConfig, "#FF0000")
    }
}

空間が限られているため,すべてのインターフェースがここで実装されていません.市場調査 , 資産の問い合わせそしてIO コール設計が完了した後,テンプレートコードを保存し,テンプレート名を: TypeScript バージョンカスタムプロトコル example として保存します.

テスト戦略

OKX 交換器のAPIKEY,シークレットキー,パスワードフレーズなどを設定した後,テスト戦略を書きます.

戦略は設計されたテンプレートライブラリをチェックします.

New Feature of FMZ Quant: Use _Serve Function to Create HTTP Services Easily

試験戦略コード:

function main() {
    // Test GetTicker
    Log(`exchange.GetTicker():`, exchange.GetTicker())

    // Test GetAccount
    Log(`exchange.GetAccount():`, exchange.GetAccount())

    // Test exchange.IO
    Log(`exchange.IO("api", "POST", "/api/v5/trade/cancel-all-after", "timeOut=0"):`, exchange.IO("api", "POST", "/api/v5/trade/cancel-all-after", "timeOut=0"))

    // Output the exchange name added by the custom protocol
    Log(`exchange.GetName():`, exchange.GetName())

    // Output exchange tags added by the custom protocol
    Log(`exchange.GetLabel():`, exchange.GetLabel())
}

走行テスト

New Feature of FMZ Quant: Use _Serve Function to Create HTTP Services Easily

OKX取引所にシームレスなアクセスを達成するために,この戦略はテンプレートをチェックするだけです (OKX取引所は既にOKXをサポートしていますが,例えば,OKXはここでFMZがまだ接続していない取引所に置き換えられています).


もっと