众所周知由于商品期货交易时间规则限制,白天只有4个小时的交易时间,夜盘只有2~4个小时的交易时间,再加上某些品种还没有夜盘交易时间,综合下来平均一天只有6个小时的交易时间。如果使用优宽量化机器人24小时不停运行策略,则有一半多的时间是在无谓的消耗。所以很多人定个闹钟来启动和停止机器人,但这么做并不能保证每次的准时,而且每次需要登录网页并不方便。本篇就创建一个管理机器人的机器人教大家如何四折使用优宽量化。
由于商品期货除了周六周末闭市外,在国家法定节假日也会闭市,所以在日期的选择上有不确定性。那么折中的办法是:结合当前热门合约的成交量变动情况,就可以判断当前是否开市。如果当前热门合约的成交量一直在累加,则说明交易所处于开市状态;如果当前热门合约的成交量不动,则说明交易所处于闭市状态。如下面的代码:
function isTrading() {
let records = exchange.GetRecords();
if (!records) return;
let newVolume = records[records.length - 1].Volume;
if (newVolume == lastVolume) {
return;
} else if (lastVolume == 0) {
lastVolume = newVolume;
return;
}
lastVolume = newVolume;
return true;
}
在优宽量化的扩展API中,可以使用StopRobot()函数来停止正在运行的机器人,使用RestartRobot()函数来启动已经停止的机器人,但是在这之前需要请求 https://www.youquant.com/api/v1? URL,以及对“secretKey”和“accessKey”进行md5加密操作,才能调用StopRobot()函数和RestartRobot()。如下面的代码:
// 获取参数对象
function getParam(version, ak, method, args){
return {
'version': version,
'access_key': ak,
'method': method,
'args': JSON.stringify(args),
'nonce': new Date().getTime()
}
}
// md5加密
function md5(param){
let paramUrl = param.version + "|" + param.method + "|" + param.args + "|" + param.nonce + "|" + secretKey
return Hash("md5", "hex", paramUrl);
}
// 获取请求URL
function getFinalUrl(param){
let url = "https://www.youquant.com/api/v1?";
return url + "access_key=" + accessKey + "&nonce=" + param.nonce + "&args=" + param.args + "&sign=" + param.sign + "&version=" + param.version + "&method=" + param.method;
}
// 获取API信息
function getAPIInfo(method, dateInfo){
let param = getParam("1.0.0", accessKey, method, dateInfo);
let md5Result = md5(param);
param.sign = md5Result;
let finalUrl = getFinalUrl(param);
let info = HttpQuery(finalUrl);
return JSON.parse(info);
}
// 用arguments关键字获取参数数组
function getArgs(){
return [].slice.call(arguments);
}
注:以上是本策略的简单原理和核心代码,无需再做修改即可使用。
第1步:申请优宽量化API 打开该链接 https://www.youquant.com/m/account 选择API接口标签,然后点击右上方创建新的ApiKey,默认“*”号即开启了所有权限。创建完毕后页面就会显示“AccessKey”和“SecretKey”。
第2步:修改策略中的全局变量 该策略一共有4个可以修改的全局变量,分别是:secretKey、accessKey、botId、code。其中secretKey和accessKey变量需要修改为刚才申请优宽量化API“AccessKey”和“SecretKey”,注意是字符串类型。全局变量botId则是指定的机器人ID号,数字类型。全局变量code则是热门合约,可以设置为:中金所的“IF000”、上期所的“rb000”、大商所的“i000”、郑商所的“RM000”等等。如下面的代码:
var secretKey = "f88b07cbf2023937aa8a4894702e0a09";
var accessKey = "ddbbeaf65a8a4fc9cc12c405aa3b247e";
var botId = 266250;
var code = 'rb000';
在实盘应用中,就像创建常规机器人一样部署即可。如下图所示:名字为“CTA策略之商品期货定时启停机器人”机器人管理者ID为266250的机器人。
作为抛砖引玉,本篇文章管理机器人策略只能管理一个机器人,不过相信你通读策略代码后,可以对该策略进行升级改进,可以升级为管理多个机器人,以及对优宽量化策略、托管者等诸多功能加以扩充,具体可以参考 https://www.youquant.com/api#优宽平台扩展API 。
// 设置全局变量参数 var secretKey = "f88b07cbf2023937aa8a4894702e0a09"; var accessKey = "ddbbeaf65a8a4fc9cc12c405aa3b247e"; var botId = 266250; var code = 'rb000'; // 固定参数 var lastVolume = 0; // 获取参数对象 function getParam(version, ak, method, args){ return { 'version': version, 'access_key': ak, 'method': method, 'args': JSON.stringify(args), 'nonce': new Date().getTime() } } // md5加密 function md5(param){ let paramUrl = param.version + "|" + param.method + "|" + param.args + "|" + param.nonce + "|" + secretKey return Hash("md5", "hex", paramUrl); } // 获取请求URL function getFinalUrl(param){ let url = "https://www.fmz.com/api/v1?"; return url + "access_key=" + accessKey + "&nonce=" + param.nonce + "&args=" + param.args + "&sign=" + param.sign + "&version=" + param.version + "&method=" + param.method; } // 获取API信息 function getAPIInfo(method, dateInfo){ let param = getParam("1.0.0", accessKey, method, dateInfo); let md5Result = md5(param); param.sign = md5Result; let finalUrl = getFinalUrl(param); let info = HttpQuery(finalUrl); return JSON.parse(info); } // 用arguments关键字获取参数数组 function getArgs(){ return [].slice.call(arguments); } // 根据合约成交量判断是否开市 function isTrading() { let records = _C(exchange.GetRecords); let newVolume = records[records.length - 1].Volume; if (newVolume == lastVolume) { return; } else if (lastVolume == 0) { lastVolume = newVolume; return; } lastVolume = newVolume; return true; } // 策略入口函数 function main() { SetErrorFilter('not login'); while (true) { if (!exchange.IO("status")) { Sleep(10000); continue; } if (exchange.SetContractType(code)) break; } while (true) { let info = getAPIInfo('GetRobotDetail', getArgs(botId)); if (isTrading()) { if (info.data.result.robot.status == 4) { getAPIInfo('RestartRobot', getArgs(botId)); Log('启动策略') } } else { if (info.data.result.robot.status == 1) { getAPIInfo('StopRobot', getArgs(botId)); Log('停止策略') } } Sleep(10000); } }