OCO订单#

发布版本``1.9.34.116``将OCO(即“One Cancel Others”)添加到回测工具中。

注意

此功能仅在回测中实现,尚未实现实盘经纪商支持。

注意

在版本``1.9.36.116``更新。Interactive Brokers支持``StopTrail``、StopTrailLimit``和``OCO

  • OCO 指定作为该组中第一个订单的参数``oco``。

  • StopTrailLimit:模拟经纪商和``IB``经纪商具有相同的行为。先指定``price``作为初始止损触发价格(还要指定``trailamount``),然后指定``plimi``作为初始限价。两者之间的差距将决定``limitoffset``(限价与止损触发价格之间的距离)。

使用模式尽可能保持用户友好。因此,如果策略逻辑决定是时候发出订单,可以按以下方式使用``OCO``:

def next(self):

… o1 = self.buy(…) … o2 = self.buy(…, oco=o1) … o3 = self.buy(…, oco=o1) # 或者甚至是oco=o2,o2已经在o1组中

简单明了。第一个订单``o1``将成为组的领导者。通过使用``oco``命名参数并将``o1``指定为``o2``和``o3``,它们将成为**OCO Group**的一部分。请注意代码片段中的注释表明,也可以通过指定``o2``(已经是组中的一部分)来将``o3``纳入到该组中。

有了组后,将发生以下情况:- 如果组内的任何一个订单被执行、取消或过期,其他订单将被取消。

下面的示例将“OCO”概念应用到实际操作中。带有绘图的标准执行过程:

$ ./oco.py –broker cash=50000 –plot

注意

现金增加到“50000”,因为资产达到“4000”,3个订单需要至少“12000”货币单位(经纪人的默认值是“10000”)。

下面是生成的图表。

这个图表实际上没有提供太多信息(它是一个标准的“SMA交叉”策略)。该示例执行以下操作:

  • 当快速SMA上穿慢速SMA时,发出3个订单。

  • “order1”是一个“限价”订单,将在“limdays”天内过期(策略的参数),限价为收盘价格减去一定比例。

  • “order2”是一个更长时间过期且限价更低的“限价”订单。- order3 是一个 Limit 订单,它进一步降低了限价

因此 order2order3 的执行不会发生 因为:

  • 首先会执行 order1,这应该触发其他订单的取消

或者

  • order1 会过期,这将触发其他订单的取消

系统会保留这3个订单的 ref 标识符,并且只会在 notify_order 中看到这三个 ref 标识符时才发出新的 buy 订单,状态为 CompletedCancelledMargin``或 ``Expired

退出仅仅是在持有某些柱时完成的。

为了尽量跟踪实际执行过程,会产生文本输出。其中一部分为:

2005-01-28: Oref 1 / 以 2941.11055 进行买入 2005-01-28: Oref 2 / 以 2896.7722 进行买入 2005-01-28: Oref 3 / 以 2822.87495 进行买入 2005-01-31: 订单 ref: 1 / 类型:买入 / 状态:已提交 2005-01-31: 订单 ref: 2 / 类型:买入 / 状态:已提交 2005-01-31: 订单 ref: 3 / 类型:买入 / 状态:已提交 2005-01-31: 订单 ref: 1 / 类型:买入 / 状态:已接受 2005-01-31: 订单 ref: 2 / 类型:买入 / 状态:已接受 2005-01-31: 订单 ref: 3 / 类型:买入 / 状态:已接受 2005-02-01: 订单 ref: 1 / 类型:买入 / 状态:已过期 2005-02-01: 订单 ref: 3 / 类型:买入 / 状态:已取消 2005-02-01: 订单 ref: 2 / 类型:买入 / 状态:已取消 … 2006-06-23: Oref 49 / 以 3532.39925 进行买入 2006-06-23: Oref 50 / 以 3479.147 进行买入 2006-06-23: Oref 51 / 以 3390.39325 进行买入 2006-06-26: 订单 ref: 49 / 类型:买入 / 状态:已提交 2006-06-26: 订单 ref: 50 / 类型:买入 / 状态:已提交 2006-06-26: 订单 ref: 51 / 类型:买入 / 状态:已提交 2006-06-26: 订单 ref: 49 / 类型:买入 / 状态:已接受 2006-06-26: 订单 ref: 50 / 类型:买入 / 状态:已接受 2006-06-26: 订单 ref: 51 / 类型:买入 / 状态:已接受 2006-06-26: 订单 ref: 49 / 类型:买入 / 状态:已完成 2006-06-26: 订单 ref: 51 / 类型:买入 / 状态:已取消 2006-06-26: 订单 ref: 50 / 类型:买入 / 状态:已取消 … 2006-11-10: 订单 ref: 61 / 类型:买入 / 状态:已取消 2006-12-11: Oref 63 / 以 4032.62555 进行买入 2006-12-11: Oref 64 / 以 3971.8322 进行买入 2006-12-11: Oref 65 / 以 3870.50995 进行买入 2006-12-12: 订单 ref: 63 / 类型:买入 / 状态:已提交 2006-12-12: 订单 ref: 64 / 类型:买入 / 状态:已提交 2006-12-12: 订单 ref: 65 / 类型:买入 / 状态:已提交 2006-12-12: 订单 ref: 63 / 类型:买入 / 状态:已接受 2006-12-12: 订单 ref: 64 / 类型:买入 / 状态:已接受 2006-12-12: 订单 ref: 65 / 类型:买入 / 状态:已接受 2006-12-15: 订单 ref: 63 / 类型:买入 / 状态:已过期 2006-12-15: 订单 ref: 65 / 类型:买入 / 状态:已取消 2006-12-15: 订单 ref: 64 / 类型:买入 / 状态:已取消

发生了以下情况: - 第一批订单已发布。订单1过期,2和3被取消。正如预期的那样。

  • 几个月后,另一批由3个订单组成。在这种情况下,订单49被标记为“已完成”,而50和51立即被取消。

  • 最后一批订单与第一批相同。

现在让我们检查没有“OCO”的行为:

$ ./oco.py –strat do_oco=False –broker cash=50000

2005-01-28: 订单引用1 / 以2941.11055购买 2005-01-28: 订单引用2 / 以2896.7722购买 2005-01-28: 订单引用3 / 以2822.87495购买 2005-01-31: 订单引用1 / 类型买入 / 状态已提交 2005-01-31: 订单引用2 / 类型买入 / 状态已提交 2005-01-31: 订单引用3 / 类型买入 / 状态已提交 2005-01-31: 订单引用1 / 类型买入 / 状态已接受 2005-01-31: 订单引用2 / 类型买入 / 状态已接受 2005-01-31: 订单引用3 / 类型买入 / 状态已接受 2005-02-01: 订单引用1 / 类型买入 / 状态已过期

就是这样,没什么(没有订单执行,也没有太大的需要绘制图表的需求)

  • 订单组被发布

  • 订单1过期,但因为策略已将参数``do_oco=False``设置为假,订单2和3不纳入“OCO”组合

  • 因此,订单2和3不会被取消,并且由于默认到期时间是在``1000``天后,它们不会在数据样本(两年的数据)中过期

  • 系统从未发布第二批订单。示例用法


$ ./oco.py --help
用法:oco.py [-h] [--data0 DATA0] [--fromdate FROMDATE] [--todate TODATE]
              [--cerebro kwargs] [--broker kwargs] [--sizer kwargs]
              [--strat kwargs] [--plot [kwargs]]

示例模板
可选参数:
-h, --help

显示帮助信息并退出

--data0 DATA0

要读取的数据(默认值:../../datas/2005-2006-day-001.txt)

--fromdate FROMDATE

日期[时间]格式为YYYY-MM-DD[THH:MM:SS] (默认值:)

--todate TODATE

日期[时间]格式为YYYY-MM-DD[THH:MM:SS] (默认值:)

--cerebro kwargs

以键=值的格式提供kwargs(默认值:)

--broker kwargs

以键=值的格式提供kwargs(默认值:)

--sizer kwargs

以键=值的格式提供kwargs(默认值:)

--strat kwargs

以键=值的格式提供kwargs(默认值:)

–plot [kwargs] 以键=值的格式提供kwargs(默认值:)

示例代码#

from __future__ import (absolute_import, division, print_function,
                        unicode_literals)


import argparse
import datetime

import backtrader as bt


class St(bt.Strategy):
    params = dict(
        ma=bt.ind.SMA,
        p1=5,
        p2=15,
        limit=0.005,
        limdays=3,
        limdays2=1000,
        hold=10,
        switchp1p2=False,  # switch prices of order1 and order2
        oco1oco2=False,  # False - use order1 as oco for order3, else order2
        do_oco=True,  # use oco or not
    )

    def notify_order(self, order):
        print('{}: Order ref: {} / Type {} / Status {}'.format(
            self.data.datetime.date(0),
            order.ref, 'Buy' * order.isbuy() or 'Sell',
            order.getstatusname()))

        if order.status == order.Completed:
            self.holdstart = len(self)

        if not order.alive() and order.ref in self.orefs:
            self.orefs.remove(order.ref)

    def __init__(self):
        ma1, ma2 = self.p.ma(period=self.p.p1), self.p.ma(period=self.p.p2)
        self.cross = bt.ind.CrossOver(ma1, ma2)

        self.orefs = list()

    def next(self):
        if self.orefs:
            return  # pending orders do nothing

        if not self.position:
            if self.cross > 0.0:  # crossing up

                p1 = self.data.close[0] * (1.0 - self.p.limit)
                p2 = self.data.close[0] * (1.0 - 2 * 2 * self.p.limit)
                p3 = self.data.close[0] * (1.0 - 3 * 3 * self.p.limit)

                if self.p.switchp1p2:
                    p1, p2 = p2, p1

                o1 = self.buy(exectype=bt.Order.Limit, price=p1,
                              valid=datetime.timedelta(self.p.limdays))

                print('{}: Oref {} / Buy at {}'.format(
                    self.datetime.date(), o1.ref, p1))

                oco2 = o1 if self.p.do_oco else None
                o2 = self.buy(exectype=bt.Order.Limit, price=p2,
                              valid=datetime.timedelta(self.p.limdays2),
                              oco=oco2)

                print('{}: Oref {} / Buy at {}'.format(
                    self.datetime.date(), o2.ref, p2))

                if self.p.do_oco:
                    oco3 = o1 if not self.p.oco1oco2 else oco2
                else:
                    oco3 = None

                o3 = self.buy(exectype=bt.Order.Limit, price=p3,
                              valid=datetime.timedelta(self.p.limdays2),
                              oco=oco3)

                print('{}: Oref {} / Buy at {}'.format(
                    self.datetime.date(), o3.ref, p3))

                self.orefs = [o1.ref, o2.ref, o3.ref]

        else:  # in the market
            if (len(self) - self.holdstart) >= self.p.hold:
                self.close()


def runstrat(args=None):
    args = parse_args(args)

    cerebro = bt.Cerebro()

    # Data feed kwargs
    kwargs = dict()

    # Parse from/to-date
    dtfmt, tmfmt = '%Y-%m-%d', 'T%H:%M:%S'
    for a, d in ((getattr(args, x), x) for x in ['fromdate', 'todate']):
        if a:
            strpfmt = dtfmt + tmfmt * ('T' in a)
            kwargs[d] = datetime.datetime.strptime(a, strpfmt)

    # Data feed
    data0 = bt.feeds.BacktraderCSVData(dataname=args.data0, **kwargs)
    cerebro.adddata(data0)

    # Broker
    cerebro.broker = bt.brokers.BackBroker(**eval('dict(' + args.broker + ')'))

    # Sizer
    cerebro.addsizer(bt.sizers.FixedSize, **eval('dict(' + args.sizer + ')'))

    # Strategy
    cerebro.addstrategy(St, **eval('dict(' + args.strat + ')'))

    # Execute
    cerebro.run(**eval('dict(' + args.cerebro + ')'))

    if args.plot:  # Plot if requested to
        cerebro.plot(**eval('dict(' + args.plot + ')'))


def parse_args(pargs=None):
    parser = argparse.ArgumentParser(
        formatter_class=argparse.ArgumentDefaultsHelpFormatter,
        description=(
            'Sample Skeleton'
        )
    )

    parser.add_argument('--data0', default='../../datas/2005-2006-day-001.txt',
                        required=False, help='Data to read in')

    # Defaults for dates
    parser.add_argument('--fromdate', required=False, default='',
                        help='Date[time] in YYYY-MM-DD[THH:MM:SS] format')

    parser.add_argument('--todate', required=False, default='',
                        help='Date[time] in YYYY-MM-DD[THH:MM:SS] format')

    parser.add_argument('--cerebro', required=False, default='',
                        metavar='kwargs', help='kwargs in key=value format')

    parser.add_argument('--broker', required=False, default='',
                        metavar='kwargs', help='kwargs in key=value format')

    parser.add_argument('--sizer', required=False, default='',
                        metavar='kwargs', help='kwargs in key=value format')

    parser.add_argument('--strat', required=False, default='',
                        metavar='kwargs', help='kwargs in key=value format')

    parser.add_argument('--plot', required=False, default='',
                        nargs='?', const='{}',
                        metavar='kwargs', help='kwargs in key=value format')

    return parser.parse_args(pargs)


if __name__ == '__main__':
    runstrat()