OmegaChartのソースコードの保守
Rev. | abec95183e38adccdd3ae834f0e303862ebeff62 |
---|---|
サイズ | 4,490 バイト |
日時 | 2022-12-15 22:48:19 |
作者 | panacoran |
ログメッセージ | Yahooファイナンスからの株価取得が途中で止まるのを回避
|
/*
* Copyright (c) Daisuke OKAJIMA All rights reserved.
*
* $Id$
*/
using System;
using System.Diagnostics;
using Zanetti.Indicators;
namespace Zanetti.Arithmetic
{
//あるExpressionが有効になるために必要な日付範囲を表現する。
//たとえば begin=-4, end=1のときは、過去5日間が必要であることを意味する。
//両方0のときは常に有効になる。
internal class DateRange {
public int begin;
public int end;
public static DateRange Empty {
get {
DateRange r = new DateRange();
r.begin = 0;
r.end = 0;
return r;
}
}
public static DateRange Today {
get {
DateRange r = new DateRange();
r.begin = 0;
r.end = 1;
return r;
}
}
public DateRange Merge(DateRange r) {
if(r==null || (r.begin==0 && r.end==0)) return this;
if(begin==0 && end==0) {
begin = r.begin;
end = r.end;
}
else {
begin = Math.Min(begin, r.begin);
end = Math.Max(end, r.end);
}
return this;
}
}
/// <summary>
/// Expressionがある日付からの相対値でどの日付範囲を必要としているかを評価する
/// </summary>
internal class DateRangeChecker : BasicVisitor {
private EvalResult[] _args;
public DateRangeChecker(string fn, EvalResult[] args) : base(fn) {
_args = args;
}
public EvalResult[] Args {
get {
return _args;
}
}
protected virtual Evaluator CreateEvaluator(string name) {
return new Evaluator(name);
}
public override object Primitive(PrimitiveExpression expr) {
if(expr.Args==null || expr.Args.Length==0) {
return DateRange.Today;
}
else {
if(expr.Args.Length==1) {
DateRange r = DateRange.Today;
r.begin = -EvalPrimitiveExpressionArgs(expr.Args[0]) + 1;
return r;
}
else if(expr.Args.Length==2) {
DateRange r = DateRange.Today;
r.begin = -EvalPrimitiveExpressionArgs(expr.Args[0]) + 1;
r.end = -EvalPrimitiveExpressionArgs(expr.Args[1]) + 1;
return r;
}
else {
ThrowArithmeticException("argument count mismatch");
return null;
}
}
}
private int EvalPrimitiveExpressionArgs(Expression ex) {
Evaluator ev = CreateEvaluator(_functionName);
ev.Args = _args;
//BaseIndex, Farmは不要なはず
return (int)((EvalResult)ex.Apply(ev)).DoubleVal;
}
public override object Function(FunctionExpression expr) {
int i;
//ここでIndicator namespaceが必要になるのは意味的にちょっとおかしい
ExpressionIndicator ind = Env.CurrentIndicators.FindFromAscii(expr.Name) as ExpressionIndicator;
if(ind!=null) {
//引数0個のときはデフォルトのパラメータを使うので引数の個数チェックをしない
if(expr.Args==null || expr.Args.Length==0) {
return (DateRange)ind.Expression.Apply(new DateRangeChecker(_functionName, ind.Args));
}
else {
if(ind.ArgCount!=Util.SafeArgLength(expr.Args))
ThrowArithmeticException(String.Format("{0} の引数は {1} 個でなければなりませんが、{2} 個です。", expr.Name, ind.ArgCount, Util.SafeArgLength(expr.Args)));
EvalResult[] p = null;
if(expr.Args!=null) {
p = new EvalResult[expr.Args.Length];
for(i=0; i<p.Length; i++)
p[i] = new EvalResult((double)EvalPrimitiveExpressionArgs(expr.Args[i]));
}
return (DateRange)ind.Expression.Apply(new DateRangeChecker(_functionName, p));
}
}
else if((i = Env.ArithmeticLibrary.FindLaneID(expr.Name))!=-1) {
Evaluator ev = CreateEvaluator(_functionName);
ev.Args = _args;
return Env.ArithmeticLibrary.CalcDateRange(i, this, ev, expr.Args);
}
else
return null;
}
public override object Constant(ConstantExpression expr) {
return DateRange.Empty;
}
public override object Parameter(ParameterExpression expr) {
return DateRange.Empty;
}
public override object UnaryOp(UnaryOpExpression expr) {
return expr.Content.Apply(this);
}
public override object BinaryOp(BinaryOpExpression expr) {
DateRange r = (DateRange)expr.Left.Apply(this);
if(r==null)
r = (DateRange)expr.Right.Apply(this);
else
r.Merge((DateRange)expr.Right.Apply(this));
return r;
}
//これはEvaluatorと同じ
public void EnsureArgIndex(int index) {
EvalResult[] newval = new EvalResult[Math.Max(index+1, _args==null? 0 : _args.Length)];
if(_args!=null) {
for(int i=0; i<_args.Length; i++)
newval[i] = _args[i];
}
_args = newval;
}
}
}