/* * QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals. * Lean Algorithmic Trading Engine v2.0. Copyright 2014 QuantConnect Corporation. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ using System; using System.Collections.Generic; using System.Linq; using Moq; using NodaTime; using NUnit.Framework; using Python.Runtime; using QuantConnect.Algorithm; using QuantConnect.Algorithm.Framework.Execution; using QuantConnect.Algorithm.Framework.Portfolio; using QuantConnect.Data; using QuantConnect.Data.Market; using QuantConnect.Interfaces; using QuantConnect.Orders; using QuantConnect.Securities; using QuantConnect.Tests.Common.Data.UniverseSelection; using QuantConnect.Tests.Engine.DataFeeds; using QuantConnect.Util; namespace QuantConnect.Tests.Algorithm.Framework.Execution { [TestFixture] public class VolumeWeightedAveragePriceExecutionModelTests { [TestCase(Language.CSharp)] [TestCase(Language.Python)] public void OrdersAreNotSubmittedWhenNoTargetsToExecute(Language language) { var actualOrdersSubmitted = new List(); var orderProcessor = new Mock(); orderProcessor.Setup(m => m.Process(It.IsAny())) .Returns((OrderTicket)null) .Callback((OrderRequest request) => actualOrdersSubmitted.Add((SubmitOrderRequest)request)); var algorithm = new QCAlgorithm(); algorithm.SetPandasConverter(); algorithm.Transactions.SetOrderProcessor(orderProcessor.Object); var model = GetExecutionModel(language); algorithm.SetExecution(model); var changes = SecurityChangesTests.CreateNonInternal(Enumerable.Empty(), Enumerable.Empty()); model.OnSecuritiesChanged(algorithm, changes); model.Execute(algorithm, new IPortfolioTarget[0]); Assert.AreEqual(0, actualOrdersSubmitted.Count); } [TestCase(Language.CSharp, new[] { 270d, 260d, 250d }, 5000, 1, 10)] [TestCase(Language.Python, new[] { 270d, 260d, 250d }, 5000, 1, 10)] [TestCase(Language.CSharp, new[] { 270d, 260d, 250d }, 500, 1, 5)] [TestCase(Language.Python, new[] { 270d, 260d, 250d }, 500, 1, 5)] [TestCase(Language.CSharp, new[] { 270d, 260d, 250d }, 50, 0, 0)] [TestCase(Language.Python, new[] { 270d, 260d, 250d }, 50, 0, 0)] [TestCase(Language.CSharp, new[] { 230d, 240d, 250d }, 50000, 0, 0)] [TestCase(Language.Python, new[] { 230d, 240d, 250d }, 50000, 0, 0)] public void OrdersAreSubmittedWhenRequiredForTargetsToExecute( Language language, double[] historicalPrices, decimal lastVolume, int expectedOrdersSubmitted, decimal expectedTotalQuantity) { var actualOrdersSubmitted = new List(); var time = new DateTime(2018, 8, 2, 16, 0, 0); var historyProvider = new Mock(); historyProvider.Setup(m => m.GetHistory(It.IsAny>(), It.IsAny())) .Returns(historicalPrices.Select((x, i) => new Slice(time.AddMinutes(i), new List { new TradeBar { Time = time.AddMinutes(i), Symbol = Symbols.AAPL, Open = Convert.ToDecimal(x), High = Convert.ToDecimal(x), Low = Convert.ToDecimal(x), Close = Convert.ToDecimal(x), Volume = 100m } }, time.AddMinutes(i)))); var algorithm = new QCAlgorithm(); algorithm.SubscriptionManager.SetDataManager(new DataManagerStub(algorithm)); algorithm.SetPandasConverter(); algorithm.SetHistoryProvider(historyProvider.Object); algorithm.SetDateTime(time.AddMinutes(5)); var security = algorithm.AddEquity(Symbols.AAPL.Value); security.SetMarketPrice(new TradeBar { Value = 250, Volume = lastVolume }); algorithm.SetFinishedWarmingUp(); var orderProcessor = ImmediateExecutionModelTests.GetAndSetBrokerageTransactionHandler(algorithm, out var brokerage); try { var model = GetExecutionModel(language); algorithm.SetExecution(model); var changes = SecurityChangesTests.CreateNonInternal(new[] { security }, Enumerable.Empty()); model.OnSecuritiesChanged(algorithm, changes); algorithm.History(new List { security.Symbol }, historicalPrices.Length, Resolution.Minute) .PushThroughConsolidators(symbol => algorithm.Securities[symbol].Subscriptions.Single(s => s.TickType == LeanData.GetCommonTickType(SecurityType.Equity)).Consolidators.First()); var targets = new IPortfolioTarget[] { new PortfolioTarget(security.Symbol, 10) }; model.Execute(algorithm, targets); var orders = orderProcessor.GetOrders().ToList(); Assert.AreEqual(expectedOrdersSubmitted, orders.Count); Assert.AreEqual(expectedTotalQuantity, orders.Sum(x => x.Quantity)); if (expectedOrdersSubmitted == 1) { var order = orders[0]; Assert.AreEqual(expectedTotalQuantity, order.Quantity); Assert.AreEqual(algorithm.UtcTime, order.Time); } } finally { brokerage?.Dispose(); } } private static IExecutionModel GetExecutionModel(Language language) { if (language == Language.Python) { using (Py.GIL()) { const string name = nameof(VolumeWeightedAveragePriceExecutionModel); var instance = Py.Import(name).GetAttr(name).Invoke(); return new ExecutionModelPythonWrapper(instance); } } return new VolumeWeightedAveragePriceExecutionModel(); } } }