forked from tslab-hub/handlers
-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathTradeStatisticsBaseExtendedBarsHandler.cs
More file actions
162 lines (136 loc) · 7.73 KB
/
TradeStatisticsBaseExtendedBarsHandler.cs
File metadata and controls
162 lines (136 loc) · 7.73 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
namespace TSLab.Script.Handlers
{
//[HandlerCategory(HandlerCategories.ClusterAnalysis)]
//[Input(0, TemplateTypes.TRADE_STATISTICS | TemplateTypes.LAST_CONTRACTS_TRADE_STATISTICS)]
// Не стоит навешивать на абстрактные классы атрибуты категорий и описания входов/выходов. Это снижает гибкость управления в перспективе.
public abstract class TradeStatisticsBaseExtendedBarsHandler : INeedVariableId, IContextUses
{
protected struct TrimContext
{
public TrimContext(bool useTrimValue, double trimValue, ComparisonMode trimComparisonMode)
{
UseTrimValue = useTrimValue;
TrimValue = trimValue;
TrimComparisonMode = trimComparisonMode;
}
public bool UseTrimValue { get; }
public double TrimValue { get; }
public ComparisonMode TrimComparisonMode { get; }
}
public string VariableId { get; set; }
public IContext Context { get; set; }
protected IList<double> Execute(
IBaseTradeStatisticsWithKind tradeStatistics,
TrimContext tradesCountTrimContext,
TrimContext quantityTrimContext,
TrimContext askQuantityTrimContext,
TrimContext bidQuantityTrimContext,
TrimContext deltaAskBidQuantityTrimContext,
TrimContext relativeDeltaAskBidQuantityPercentTrimContext)
{
var histograms = tradeStatistics.GetHistograms();
var tradeHistogramsCache = tradeStatistics.TradeHistogramsCache;
var barsCount = tradeHistogramsCache.Bars.Count;
const double DefaultValue = double.NaN;
if (histograms.Count == 0 ||
histograms.All(item => item.Bars.Count == 0) ||
IsInvalid(tradesCountTrimContext) ||
IsInvalid(quantityTrimContext) ||
IsInvalid(askQuantityTrimContext) ||
IsInvalid(bidQuantityTrimContext) ||
IsInvalid(deltaAskBidQuantityTrimContext) ||
IsInvalid(relativeDeltaAskBidQuantityPercentTrimContext))
{
return new ConstGenBase<double>(barsCount, DefaultValue);
}
var isInRangeFuncs = new List<Func<ITradeHistogramBar, bool>>();
if (tradesCountTrimContext.UseTrimValue)
isInRangeFuncs.Add(GetIsInRangeFunc(tradeStatistics, TradeStatisticsKind.TradesCount, tradesCountTrimContext));
if (quantityTrimContext.UseTrimValue)
isInRangeFuncs.Add(GetIsInRangeFunc(tradeStatistics, TradeStatisticsKind.Quantity, quantityTrimContext));
if (askQuantityTrimContext.UseTrimValue)
isInRangeFuncs.Add(GetIsInRangeFunc(tradeStatistics, TradeStatisticsKind.AskQuantity, askQuantityTrimContext));
if (bidQuantityTrimContext.UseTrimValue)
isInRangeFuncs.Add(GetIsInRangeFunc(tradeStatistics, TradeStatisticsKind.BidQuantity, bidQuantityTrimContext));
if (deltaAskBidQuantityTrimContext.UseTrimValue)
isInRangeFuncs.Add(GetIsInRangeFunc(tradeStatistics, TradeStatisticsKind.DeltaAskBidQuantity, deltaAskBidQuantityTrimContext));
if (relativeDeltaAskBidQuantityPercentTrimContext.UseTrimValue)
isInRangeFuncs.Add(GetIsInRangeFunc(tradeStatistics, TradeStatisticsKind.RelativeDeltaAskBidQuantityPercent, relativeDeltaAskBidQuantityPercentTrimContext));
double[] results = null;
var runtime = Context?.Runtime;
var canBeCached = tradeStatistics.HasStaticTimeline && barsCount > 1 && runtime != null;
string id = null, stateId = null;
DerivativeTradeStatisticsCacheContext context = null;
var cachedCount = 0;
if (canBeCached)
{
id = string.Join(".", runtime.TradeName, runtime.IsAgentMode, VariableId);
stateId = GetParametersStateId() + "." + tradeStatistics.StateId;
context = DerivativeTradeStatisticsCache.Instance.GetContext(id, stateId, tradeHistogramsCache);
if (context != null)
{
var cachedResults = context.Values;
cachedCount = Math.Min(cachedResults.Length, barsCount) - 1;
if (cachedResults.Length == barsCount)
results = cachedResults;
else
Buffer.BlockCopy(cachedResults, 0, results = new double[barsCount], 0, cachedCount * sizeof(double));
}
else
results = new double[barsCount];
}
else
results = Context?.GetArray<double>(barsCount) ?? new double[barsCount];
tradeStatistics.GetHistogramsBarIndexes(out var firstBarIndex, out var lastBarIndex);
for (var i = cachedCount; i < firstBarIndex; i++)
results[i] = DefaultValue;
lock (tradeStatistics.Source)
{
for (var i = Math.Max(cachedCount, firstBarIndex); i <= lastBarIndex; i++)
{
var bars = tradeStatistics.GetAggregatedHistogramBars(i);
var selectedBars = isInRangeFuncs.Count > 0
? bars.Where(bar => isInRangeFuncs.All(item => item(bar)))
: bars;
results[i] = GetResult(tradeStatistics, selectedBars);
}
}
for (var i = Math.Max(cachedCount, lastBarIndex + 1); i < barsCount; i++)
results[i] = DefaultValue;
if (canBeCached)
DerivativeTradeStatisticsCache.Instance.SetContext(id, stateId, tradeHistogramsCache, results, context);
return results;
}
private static bool IsInvalid(TrimContext trimContext)
{
return trimContext.UseTrimValue && double.IsNaN(trimContext.TrimValue);
}
private static Func<ITradeHistogramBar, bool> GetIsInRangeFunc(IBaseTradeStatisticsWithKind tradeStatistics, TradeStatisticsKind tradeStatisticsKind, TrimContext trimContext)
{
var trimComparisonMode = trimContext.TrimComparisonMode;
switch (trimComparisonMode)
{
case ComparisonMode.Greater:
return bar => tradeStatistics.GetValue(bar, tradeStatisticsKind) > trimContext.TrimValue;
case ComparisonMode.GreaterOrEqual:
return bar => tradeStatistics.GetValue(bar, tradeStatisticsKind) >= trimContext.TrimValue;
case ComparisonMode.Less:
return bar => tradeStatistics.GetValue(bar, tradeStatisticsKind) < trimContext.TrimValue;
case ComparisonMode.LessOrEqual:
return bar => tradeStatistics.GetValue(bar, tradeStatisticsKind) <= trimContext.TrimValue;
case ComparisonMode.AreEqual:
return bar => tradeStatistics.GetValue(bar, tradeStatisticsKind) == trimContext.TrimValue;
case ComparisonMode.AreNotEqual:
return bar => tradeStatistics.GetValue(bar, tradeStatisticsKind) != trimContext.TrimValue;
default:
throw new InvalidEnumArgumentException(nameof(trimComparisonMode), (int)trimComparisonMode, trimComparisonMode.GetType());
}
}
protected abstract string GetParametersStateId();
protected abstract double GetResult(IBaseTradeStatisticsWithKind tradeStatistics, IEnumerable<ITradeHistogramBar> bars);
}
}