Открыть на GitHub

Стратегия IsConnected

Краткое описание

  • Источник: конвертирована из скрипта MetaTrader 5 IsConnected.mq5 (каталог MQL/35056).
  • Назначение: непрерывный контроль статуса подключения коннектора и фиксация переходов между состояниями online/offline с отметками времени.
  • Тип: сервисная стратегия, ориентированная на мониторинг инфраструктуры без торговой логики.

Логика работы

  1. При запуске стратегия выводит сообщение о старте мониторинга и сохраняет текущее состояние соединения.
  2. Фоновый таймер проверяет флаг Connector.IsConnected каждые CheckIntervalSeconds секунд (по умолчанию 1 секунда).
  3. При обнаружении изменения состояния стратегия:
    • Запоминает момент перехода, используя CurrentTime стратегии.
    • Записывает новое состояние (Online либо Offline).
    • Сообщает, сколько длилось предыдущее состояние (время в онлайне до разрыва либо время офлайна до восстановления).
  4. При остановке стратегия отменяет таймер и сообщает, в каком состоянии находилось соединение в момент завершения.

Параметры

Имя Тип Значение по умолчанию Описание
CheckIntervalSeconds int 1 Интервал между проверками соединения в секундах. Значение должно быть больше нуля.

Особенности логирования

  • Все сообщения пишутся методом LogInfo на английском языке, что соответствует оригиналу, использовавшему Print.
  • Интервалы выводятся с указанием времени начала события и длительности предыдущего состояния.

Отличия от оригинальной реализации

  • Цикл с Sleep заменён управляемым таймером, который не блокирует поток стратегии.
  • Вместо дублирующихся сообщений выводятся структурированные записи с длительностью аптайма/даунтайма.
  • При остановке и сбросе таймер корректно останавливается, что исключает «висячие» фоновые задачи.
namespace StockSharp.Samples.Strategies;

using System;
using Ecng.Common;
using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.Messages;

/// <summary>
/// IsConnected strategy: Parabolic SAR trend following.
/// Buys when close above SAR, sells when close below SAR.
/// </summary>
public class IsConnectedStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;
	private readonly StrategyParam<decimal> _acceleration;
	private readonly StrategyParam<decimal> _accelerationMax;

	private decimal _prevSar;
	private decimal _prevClose;
	private bool _hasPrev;

	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
	public decimal Acceleration { get => _acceleration.Value; set => _acceleration.Value = value; }
	public decimal AccelerationMax { get => _accelerationMax.Value; set => _accelerationMax.Value = value; }

	public IsConnectedStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(60).TimeFrame())
			.SetDisplay("Candle Type", "Candle timeframe", "General");
		_acceleration = Param(nameof(Acceleration), 0.01m)
			.SetDisplay("Acceleration", "SAR acceleration factor", "Indicators");
		_accelerationMax = Param(nameof(AccelerationMax), 0.1m)
			.SetDisplay("Acceleration Max", "SAR max acceleration", "Indicators");
	}

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();
		_prevSar = 0;
		_prevClose = 0;
		_hasPrev = false;
	}

	/// <inheritdoc />
	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);
		_prevSar = 0;
		_prevClose = 0;
		_hasPrev = false;
		var sar = new ParabolicSar { Acceleration = Acceleration, AccelerationMax = AccelerationMax };
		var subscription = SubscribeCandles(CandleType);
		subscription.Bind(sar, ProcessCandle).Start();
	}

	private void ProcessCandle(ICandleMessage candle, decimal sarValue)
	{
		if (candle.State != CandleStates.Finished) return;

		if (_hasPrev)
		{
			if (_prevClose <= _prevSar && candle.ClosePrice > sarValue && Position <= 0)
				BuyMarket();
			else if (_prevClose >= _prevSar && candle.ClosePrice < sarValue && Position >= 0)
				SellMarket();
		}

		_prevClose = candle.ClosePrice;
		_prevSar = sarValue;
		_hasPrev = true;
	}
}