Die Ressourcen sind geladen. Beförderung...

FMZ-Plattform: Erforschung von Signalempfangsstrategien für externe Netzwerke

Schriftsteller:Die Erfinder quantifizieren - Kleine Träume, Erstellt: 2024-12-17 11:44:07, Aktualisiert: 2024-12-17 16:08:11

[TOC]

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

Vorwort

In einem früheren Artikel"Untersuchung der FMZ-Plattform für den Empfang externer Signale: Erweiterung der API vs. Strategien für den eingebauten HTTP-Dienst"In der Diskussion über FMZ haben wir zwei verschiedene Arten des Empfangs von externen Signalen verglichen, um die Transaktion zu programmieren, und analysierten die Details. Die Methode zur Empfangung von externen Signalen mit der FMZ-Plattform erweitert die API.

Strategie umgesetzt

Wir kopieren die bisherigen Strategien, wie wir FMZ Extension API für den Zugriff auf Trading View-Signale nutzen, und machen einfache Änderungen an den Strategien, wie wir die vorherigen Nachrichtenformate, Nachrichtenverarbeitungsmethoden usw. verwenden.

Da die eingebauten Dienstleistungen von Richtlinien über HTTP oder HTTPS verfügbar sind, haben wir, um es einfach zu demonstrieren, den HTTP-Protokoll verwendet, um IP-Weißlisten-Authentifizierung und Passwort-Authentifizierung hinzuzufügen. Wenn Sie die Sicherheit noch weiter erhöhen möchten, können Sie die eingebauten Dienstleistungen von Richtlinien als Https-Dienste gestalten.

//信号结构
var Template = {
    Flag: "45M103Buy",     // 标识,可随意指定
    Exchange: 1,           // 指定交易所交易对
    Currency: "BTC_USDT",  // 交易对
    ContractType: "spot",  // 合约类型,swap,quarter,next_quarter,现货填写spot
    Price: "{{close}}",    // 开仓或者平仓价格,-1为市价
    Action: "buy",         // 交易类型[ buy:现货买入 , sell:现货卖出 , long:期货做多 , short:期货做空 , closesell:期货买入平空 , closebuy:期货卖出平多]
    Amount: "1",           // 交易量
}

var Success = "#5cb85c"    // 成功颜色
var Danger = "#ff0000"     // 危险颜色
var Warning = "#f0ad4e"    // 警告颜色
var buffSignal = []

// Http服务
function serverFunc(ctx, ipWhiteList, passPhrase) {
    var path = ctx.path()
    if (path == "/CommandRobot") {
        // 校验IP地址
        var fromIP = ctx.remoteAddr().split(":")[0]        
        if (ipWhiteList && ipWhiteList.length > 0) {
            var ipList = ipWhiteList.split(",")
            if (!ipList.includes(fromIP)) {
                ctx.setStatus(500)
                ctx.write("IP address not in white list")
                Log("500 Error: IP address not in white list", "#FF0000")
                return 
            }
        }

        // 校验口令
        var pass = ctx.rawQuery().length > 0 ? ctx.query("passPhrase") : ""
        if (passPhrase && passPhrase.length > 0) {
            if (pass != passPhrase) {
                ctx.setStatus(500)
                ctx.write("Authentication failed")
                Log("500 Error: Authentication failed", "#FF0000")
                return 
            }
        }

        var body = JSON.parse(ctx.body())
        threading.mainThread().postMessage(JSON.stringify(body))
        ctx.write("OK")
        // 200
    } else {
        ctx.setStatus(404)
    }
}

// 校验信号消息格式
function DiffObject(object1, object2) {
    const keys1 = Object.keys(object1)
    const keys2 = Object.keys(object2)
    if (keys1.length !== keys2.length) {
        return false
    }
    for (let i = 0; i < keys1.length; i++) {
        if (keys1[i] !== keys2[i]) {
            return false
        }
    }
    return true
}

function CheckSignal(Signal) {
    Signal.Price = parseFloat(Signal.Price)
    Signal.Amount = parseFloat(Signal.Amount)
    if (Signal.Exchange <= 0 || !Number.isInteger(Signal.Exchange)) {
        Log("交易所最小编号为1,并且为整数", Danger)
        return
    }
    if (Signal.Amount <= 0 || typeof(Signal.Amount) != "number") {
        Log("交易量不能小于0,并且为数值类型", typeof(Signal.Amount), Danger)
        return
    }
    if (typeof(Signal.Price) != "number") {
        Log("价格必须是数值", Danger)
        return
    }
    if (Signal.ContractType == "spot" && Signal.Action != "buy" && Signal.Action != "sell") {
        Log("指令为操作现货,Action错误,Action:", Signal.Action, Danger)
        return 
    }
    if (Signal.ContractType != "spot" && Signal.Action != "long" && Signal.Action != "short" && Signal.Action != "closesell" && Signal.Action != "closebuy") {
        Log("指令为操作期货,Action错误,Action:", Signal.Action, Danger)
        return 
    }
    return true
}

// 信号处理对象
function createManager() {
    var self = {}
    self.tasks = []
    
    self.process = function() {
        var processed = 0
        if (self.tasks.length > 0) {
            _.each(self.tasks, function(task) {
                if (!task.finished) {
                    processed++
                    self.pollTask(task)
                }
            })
            if (processed == 0) {
                self.tasks = []
            }
        }
    }
    
    self.newTask = function(signal) {
        // {"Flag":"45M103Buy","Exchange":1,"Currency":"BTC_USDT","ContractType":"swap","Price":"10000","Action":"buy","Amount":"0"}
        var task = {}
        task.Flag = signal["Flag"]
        task.Exchange = signal["Exchange"]
        task.Currency = signal["Currency"]
        task.ContractType = signal["ContractType"]
        task.Price = signal["Price"]
        task.Action = signal["Action"]
        task.Amount = signal["Amount"]
        task.exchangeIdx = signal["Exchange"] - 1
        task.pricePrecision = null
        task.amountPrecision = null 
        task.error = null 
        task.exchangeLabel = exchanges[task.exchangeIdx].GetLabel()
        task.finished = false 
        
        Log("创建任务:", task)
        self.tasks.push(task)
    }
    
    self.getPrecision = function(n) {
        var precision = null 
        var arr = n.toString().split(".")
        if (arr.length == 1) {
            precision = 0
        } else if (arr.length == 2) {
            precision = arr[1].length
        } 
        return precision
    }
    
    self.pollTask = function(task) {
        var e = exchanges[task.exchangeIdx]
        var name = e.GetName()
        var isFutures = true
        e.SetCurrency(task.Currency)
        if (task.ContractType != "spot" && name.indexOf("Futures_") != -1) {
            // 非现货,则设置合约
            e.SetContractType(task.ContractType)
        } else if (task.ContractType == "spot" && name.indexOf("Futures_") == -1) {
            isFutures = false 
        } else {
            task.error = "指令中的ContractType与配置的交易所对象类型不匹配"
            return 
        }
        
        var depth = e.GetDepth()
        if (!depth || !depth.Bids || !depth.Asks) {
            task.error = "订单薄数据异常"
            return 
        }
        
        if (depth.Bids.length == 0 && depth.Asks.length == 0) {
            task.error = "盘口无订单"
            return 
        }
        
        _.each([depth.Bids, depth.Asks], function(arr) {
            _.each(arr, function(order) {
                var pricePrecision = self.getPrecision(order.Price)
                var amountPrecision = self.getPrecision(order.Amount)
                if (Number.isInteger(pricePrecision) && !Number.isInteger(self.pricePrecision)) {
                    self.pricePrecision = pricePrecision
                } else if (Number.isInteger(self.pricePrecision) && Number.isInteger(pricePrecision) && pricePrecision > self.pricePrecision) {
                    self.pricePrecision = pricePrecision
                }
                if (Number.isInteger(amountPrecision) && !Number.isInteger(self.amountPrecision)) {
                    self.amountPrecision = amountPrecision
                } else if (Number.isInteger(self.amountPrecision) && Number.isInteger(amountPrecision) && amountPrecision > self.amountPrecision) {
                    self.amountPrecision = amountPrecision
                }
            })
        })

        if (!Number.isInteger(self.pricePrecision) || !Number.isInteger(self.amountPrecision)) {
            task.err = "获取精度失败"
            return 
        }
        
        e.SetPrecision(self.pricePrecision, self.amountPrecision)
        
        // buy:现货买入 , sell:现货卖出 , long:期货做多 , short:期货做空 , closesell:期货买入平空 , closebuy:期货卖出平多
        var direction = null 
        var tradeFunc = null 
        if (isFutures) {
            switch (task.Action) {
                case "long": 
                    direction = "buy"
                    tradeFunc = e.Buy 
                    break
                case "short": 
                    direction = "sell"
                    tradeFunc = e.Sell
                    break
                case "closesell": 
                    direction = "closesell"
                    tradeFunc = e.Buy 
                    break
                case "closebuy": 
                    direction = "closebuy"
                    tradeFunc = e.Sell
                    break
            }
            if (!direction || !tradeFunc) {
                task.error = "交易方向错误:" + task.Action
                return 
            }
            e.SetDirection(direction)
        } else {
            if (task.Action == "buy") {
                tradeFunc = e.Buy 
            } else if (task.Action == "sell") {
                tradeFunc = e.Sell 
            } else {
                task.error = "交易方向错误:" + task.Action
                return 
            }
        }
        var id = tradeFunc(task.Price, task.Amount)
        if (!id) {
            task.error = "下单失败"
        }
        
        task.finished = true
    }
    
    return self
}

function main() {
    // 重置日志信息
    if (isResetLog) {
        LogReset(1)
    }

    Log("交易类型[ buy:现货买入 , sell:现货卖出 , long:期货做多 , short:期货做空 , closesell:期货买入平空 , closebuy:期货卖出平多]", Danger)
    Log("指令模板:", JSON.stringify(Template), Danger)    
    if (!passPhrase || passPhrase.length == 0) {
        Log("webhook url:", `http://${serverIP}:${port}/CommandRobot`)
    } else {
        Log("webhook url:", `http://${serverIP}:${port}/CommandRobot?passPhrase=${passPhrase}`)
    }

    // 创建Http内置服务
    __Serve("http://0.0.0.0:" + port, serverFunc, ipWhiteList, passPhrase)

    // 初始化执行的代码
    if (initCode && initCode.length > 0) {
        try {
            Log("执行初始化代码:", initCode)
            eval(initCode)
        } catch(error) {
            Log("e.name:", error.name, "e.stack:", error.stack, "e.message:", error.message)
        }
    }    

    // 创建信号管理对象
    var manager = createManager()
    
    while (true) {
        try {
            // 检测交互控件,用于测试
            var cmd = GetCommand()
            if (cmd) {
                // 发送Http请求,模拟测试
                var arrCmd = cmd.split(":", 2)
                if (arrCmd[0] == "TestSignal") {
                    // {"Flag":"TestSignal","Exchange":1,"Currency":"BTC_USDT","ContractType":"swap","Price":"10000","Action":"long","Amount":"1"}
                    var signal = cmd.replace("TestSignal:", "")
                    if (!passPhrase || passPhrase.length == 0) {
                        var ret = HttpQuery(`http://${serverIP}:${port}/CommandRobot`, {"method": "POST", "body": signal})
                        Log("测试请求的应答:", ret)
                    } else {
                        var ret = HttpQuery(`http://${serverIP}:${port}/CommandRobot?passPhrase=${passPhrase}`, {"method": "POST", "body": signal})
                        Log("测试请求的应答:", ret)
                    }                    
                }
            }

            // 检测内置Http服务收到请求之后通知主线程的消息,写入manager对象的任务队列
            var msg = threading.mainThread().peekMessage(-1)
            if (msg) {
                Log("收到消息 msg:", msg)
                var objSignal = JSON.parse(msg)
                if (DiffObject(Template, objSignal)) {
                    Log("接收到交易信号指令:", objSignal)
                    buffSignal.push(objSignal)
                    
                    // 检查交易量、交易所编号
                    if (!CheckSignal(objSignal)) {
                        continue
                    }
                    
                    // 创建任务
                    if (objSignal["Flag"] == "TestSignal") {
                        Log("收到测试消息:", JSON.stringify(objSignal))
                    } else {
                        manager.newTask(objSignal)
                    }                    
                } else {
                    Log("指令无法识别", signal)
                }
            } else {
                Sleep(1000 * SleepInterval)
            }

            // 处理任务
            manager.process()
            
            // 状态栏显示信号
            if (buffSignal.length > maxBuffSignalRowDisplay) {
                buffSignal.shift()
            }
            var buffSignalTbl = {
                "type" : "table",
                "title" : "信号记录",
                "cols" : ["Flag", "Exchange", "Currency", "ContractType", "Price", "Action", "Amount"],
                "rows" : []
            }
            for (var i = buffSignal.length - 1 ; i >= 0 ; i--) {
                buffSignalTbl.rows.push([buffSignal[i].Flag, buffSignal[i].Exchange, buffSignal[i].Currency, buffSignal[i].ContractType, buffSignal[i].Price, buffSignal[i].Action, buffSignal[i].Amount])
            }

            LogStatus(_D(), "\n", "`" + JSON.stringify(buffSignalTbl) + "`")            
        } catch (error) {
            Log("e.name:", error.name, "e.stack:", error.stack, "e.message:", error.message)
        }        
    }
}

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

  • Port-Parameter: Wenn Sie das HTTP-Protokoll verwenden, können Sie nur 80 Ports in der Trading View einstellen.
  • Server-IP-Parameter: Die öffentliche IP-Adresse des Servers wird ausgefüllt.
  • initCode Parameter: kann für den Umfeldtest an den Börsen verwendet werden.

Im Vergleich zu den Strategien, die mit der erweiterten API-Methode auf externe Signale zugreifen, ist die Änderung der Strategie nicht groß, sondern nur eine Erweiterung.serverFuncHttp-Dienstverarbeitungsfunktionen, die eine neue Methode der Mehrthread-Nachrichtenübertragung auf der FMZ-Plattform verwenden:postMessage/peekMessageDie anderen Code wurden kaum verändert.

IP-Weißliste

Da die Anfragen von Webhooks für Trading View nur von folgenden IP-Adressen ausgehen:

52.89.214.238
34.212.75.30
54.218.53.128
52.32.178.7

Also haben wir eine Parameter hinzugefügt.ipWhiteList, um eine IP-Whitelist einzurichten und Anfragen, die nicht in dieser IP-Adress-Whitelist enthalten sind, zu ignorieren.

        // 校验IP地址
        var fromIP = ctx.remoteAddr().split(":")[0]        
        if (ipWhiteList && ipWhiteList.length > 0) {
            var ipList = ipWhiteList.split(",")
            if (!ipList.includes(fromIP)) {
                ctx.setStatus(500)
                ctx.write("IP address not in white list")
                Log("500 Error: IP address not in white list", "#FF0000")
                return 
            }
        }

Überprüfen Sie das Passwort

Fügen Sie einen Parameter zur Strategie hinzupassPhraseDie Webhook-URL-Einstellungen in Trading View ignorieren Anfragen, die nicht mit dem Verifizierungspasswort übereinstimmen.

Wir haben beispielsweise:test123456

        // 校验口令
        var pass = ctx.rawQuery().length > 0 ? ctx.query("passPhrase") : ""
        if (passPhrase && passPhrase.length > 0) {
            if (pass != passPhrase) {
                ctx.setStatus(500)
                ctx.write("Authentication failed")
                Log("500 Error: Authentication failed", "#FF0000")
                return 
            }
        }

Außensignale

Mit dem PINE-Skript der Trading View-Plattform als externer Signal-Trigger-Quelle wählen Sie in der offiziell von Trading View veröffentlichten PINE-Skript einfach einen:

//@version=6
strategy("MovingAvg Cross", overlay=true)
length = input(9)
confirmBars = input(1)
price = close
ma = ta.sma(price, length)
bcond = price > ma
bcount = 0
bcount := bcond ? nz(bcount[1]) + 1 : 0
if (bcount == confirmBars)
	strategy.entry("MACrossLE", strategy.long, comment="long")
scond = price < ma
scount = 0
scount := scond ? nz(scount[1]) + 1 : 0
if (scount == confirmBars)
	strategy.entry("MACrossSE", strategy.short, comment="short")

Natürlich kann man auch die PINE-Skripte direkt auf der FMZ-Plattform laufen lassen, aber wenn man die Trading View-Plattform dazu bringen will, die PINE-Skripte zu laufen und Signale zu senden, dann kann man nur die von uns besprochenen Systeme verwenden.

Wir müssen uns auf die Unterordnungsfunktionen dieses Skripts konzentrieren, und um das PINE-Skript für unsere Webhook-Anfrage zu passen, müssen wir uns auf die Transaktionsfunktionen konzentrieren.commentIch möchte Ihnen sagen, dass ich das in meinem Artikel erwähnen werde.

Webhook Url und Anforderungs-Body

Die Einstellungen für Webhook Url und Requestbody entsprechen grundsätzlich der vorherigen API-Erweiterung, die die Zugriff auf externe Signale ermöglicht.

Webhook-Adresse

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

Wenn wir dieses PINE-Skript in ein Markt in der Trading View hinzufügen (wir testen den ETH_USDT-Forex-Markt für Bitcoin), dann sehen wir, dass das Skript anfängt zu funktionieren.

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

Webhook-URL-Einstellungen: In der Strategie-Code ist eine automatische Generierung der Webhook-URLs konzipiert, die wir nur in den Logs kopieren können, in denen die Strategie gestartet wurde.

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

http://xxx.xxx.xxx.xxx:80/CommandRobot?passPhrase=test123456

Trading View legt fest, dass Webhook-URLs nur für HTTP-Anfragen auf 80 Ports verwendet werden können, daher haben wir die Portparameter auch auf 80 gesetzt, so dass Sie sehen, dass die Linksport der Webhook-URL, die von der Strategie erzeugt wird, auch 80 ist.

Body Nachrichten

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

Dann setzen Sie die angeforderte Body-Nachricht im Bildschirm unter dem "Einstellungen"-Tag.

{
    "Flag":"{{strategy.order.id}}",
    "Exchange":1,
    "Currency":"ETH_USDT",
    "ContractType":"swap",
    "Price":"-1",
    "Action":"{{strategy.order.comment}}",
    "Amount":"{{strategy.order.contracts}}"
}

Erinnern Sie sich noch an den folgenden Code aus dem PINE-Skript, von dem wir gerade sprachen?

strategy.entry("MACrossLE", strategy.long, comment="long")

Die MACrossLE-File ist die Ausfüllung der {{strategy.order.id}}-File, die in Zukunft den Alarm auslösen soll.

Bei der Anzeige der Strategie ist es wichtig, dass die Strategie- und Anzeigetexte in der Strategie ausgefüllt werden, um einen Alarm auszulösen.

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

Daher müssen wir einheitlich sein. Hier setzen wir für die folgende Funktion die Long- und die Short-Klammern, die für mehr und mehr freie Signale stehen.

Das PINE-Skript gibt nicht an, wie viele Auftragsbestände pro Auftragsbestellung ausgefüllt werden, so dass Trading View die Default-Strategy.order.contracts-Abschnitte verwendet, um Alarmmeldungen zu senden.

Tests auf der Festplatte

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

Wenn ein PINE-Skript in der Trading View ausgeführt wird und eine Transaktionsfunktion ausgeführt wird, weil wir einen Webhook-URL-Alarm eingerichtet haben, sendet die Trading View-Plattform eine POST-Anfrage an den HTTP-Dienst, der in unserer Richtlinie integriert ist, die eine Passwortparameter für die Authentifizierung enthält.passPhraseDie tatsächliche Requestbody sieht so aus:

FMZ平台外部信号接收的探讨:策略内置Http服务接收信号的完整方案

Unsere Strategie ist dann, die entsprechenden Transaktionen nach den Nachrichten in diesem Body durchzuführen.

Sie können sehen, dass die Strategie im OKX-Simulationsumfeld synchronisierte Signal-Trading durchführt, basierend auf dem PINE-Skript in der Trading View.

Strategische Adresse

https://www.fmz.com/strategy/475235

Danke für die Aufmerksamkeit auf FMZ Quantity und danke fürs Lesen.


Mehr