using System;
using System.IO;
using System.Collections.Generic;
using NLog;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
using DataClient.Struct;
using System.Net;
using System.Threading.Tasks;
namespace DataClient
{
///
/// Класс для связи с СТП. Содержит все инструменты, для получения данных.
///
public class STPClient
{
//Переменные
Logger log = LogManager.GetCurrentClassLogger();
string confDir = Path.Combine(Directory.GetCurrentDirectory(), "Config");
private List servers = new List();
private Dictionary confAnalog = new Dictionary();
private List<(int num, string name)> vdpList = new List<(int num, string name)>();
int serverDefault = -1;
//Геттеры и сеттеры
/// Номер сервера, используемого по умолчанию.
public int UseServer
{
get { return serverDefault; }
set {
serverDefault =
(servers.Count == 0) ? -1 :
(value < 0) ? 0 :
(value >= servers.Count) ? serverDefault = servers.Count - 1 :
value;
}
}
/// Массив серверов, инициализированных в экземпляре класса.
public server[] ListServers { get { return servers.ToArray(); } }
//Функции
/// Получение номера сервера по его названию.
/// Название сервера.
/// Номер сервера в списке серверов.
public int GetServerIdByName(string name)
{
return servers.FindIndex(x => x.name == name);
}
/// Получение название сервера по его номеру.
/// Номер сервера в списке серверов.
/// Название сервера.
public string GetServerNameById(int id)
{
var res = (id >= servers.Count || id < 0) ? null : servers[id].name;
return res;
}
/// Проверка параметров серевера перед их добавлением в список.
/// IP адресс сервера.
/// Порт сервера.
/// Путь к архиву.
/// Результат выполнения функции.
private bool CheckDataServer(string ip, int port, string dir)
{
if (!IPAddress.TryParse(ip, out _))
{
log.Warn("Ip address incorrect: " + ip + " .");
return false;
}
if (port < 1 && port > 65535)
{
log.Warn("Port incorrect: " + port.ToString() + ".");
return false;
}
if (!Directory.Exists(dir))
{
log.Warn("Directory incorrect or not exist: " + dir + " .");
return false;
}
return true;
}
/// Добавление сервера в список серверов.
/// Название сервера
/// IP адресс сервера.
/// Порт сервера.
/// Путь к архиву.
/// Результат выполнения функции.
public bool AddServer(string name, string ip, int port, string dir)
{
if (!CheckDataServer(ip, port, dir))
{
log.Warn("Server not added: Data incorrect.");
return false;
}
if (string.IsNullOrEmpty(name))
{
log.Warn("Server not added: Incorrect name server.");
return false;
}
if (servers.FindIndex(x => x.name == name) != -1)
{
log.Warn("Server not added: Server with this name exist.");
return false;
}
servers.Add(new server(name, ip, port, dir));
UseServer = 0;
log.Trace("Server (name=" + name + ";ip=" + ip + ";port=" + port + ";dir=" + dir + ") added.");
return true;
}
/// Изменение сервера в списке серверов по его номеру.
/// Номер сервера в списке серверов.
/// Название сервера.
/// IP адресс сервера.
/// Порт сервера.
/// Путь к архиву.
/// Результат выполнения функции.
public bool ChangeServerById(int id, string name, string ip, int port, string dir)
{
if (servers.Count == 0 || id >= servers.Count || id < 0)
{
log.Warn("Server not changed: Id incorrect.");
return false;
}
if (!CheckDataServer(ip, port, dir))
{
log.Warn("Server not changed: Data incorrect.");
return false;
}
var tmpId = servers.FindIndex(x => x.name == name);
if (string.IsNullOrEmpty(name) || (tmpId != -1 && tmpId != id))
{
log.Warn("Server not changed: Name incorrect.");
return false;
}
var tmp = new server(name, ip, port, dir);
log.Trace("Server id=" + id + " changed: name=" + name + ";ip=" + ip + ";port=" + port.ToString() + ";dir=" + dir + ";.");
servers[id] = tmp;
return true;
}
/// Изменение сервера в списке серверов по его названию.
/// Название сервера.
/// IP адресс сервера.
/// Порт сервера.
/// Путь к архиву.
/// Результат выполнения функции.
public bool ChangeServerByName(string name, string ip, int port, string dir)
{
var id = servers.FindIndex(x => x.name == name);
if (id == -1)
{
log.Warn("Server not changed: Name incorrect or server not exist.");
return false;
}
return ChangeServerById(id, name, ip, port, dir);
}
/// Удаление сервера из списка серверов по его номеру.
/// Номер сервера в списке серверов.
/// Результат выполнения функции.
public bool RemoveServerById(int id)
{
if (servers.Count == 0 || id >= servers.Count || id < 0)
{
log.Warn("Server not removed: Id incorrect or list server are empty.");
return false;
}
log.Trace("Server id=" + id.ToString() + ";name=" + servers[id].name + ";ip=" + servers[id].ip + ";port=" + servers[id].port.ToString() + ";dir=" + servers[id].dir + "; removed.");
servers.RemoveAt(id);
UseServer = 0;
return true;
}
/// Удаление сервера из списка серверов по его названию.
/// Название сервера.
/// Результат выполнения функции.
public bool RemoveServerByName(string name)
{
var id = servers.FindIndex(x => x.name == name);
if (id == -1)
{
log.Warn("Server not removed: Name incorrect or server not exist.");
return false;
}
return RemoveServerById(id);
}
/// Чтение конфигурационнного файла и установка параметров по умолчанию.
private void Config()
{
string jsonString;
var confFile = Path.Combine(confDir, "config.json");
try
{
jsonString = File.ReadAllText(confFile);
}
catch (Exception e)
{
log.Warn(e, "Can't read config file from " + confDir + ".");
return;
}
var conf = (JObject)JsonConvert.DeserializeObject(jsonString);
if (!conf.HasValues && !conf["servers"].HasValues && conf["servers"].Type != JTokenType.Array)
{
log.Warn("Doesn't exist \"servers\" object in config file.");
return;
}
var count = -1;
foreach (var s in conf["servers"])
{
count++;
var name = (s["name"].Type == JTokenType.String) ? (string)s["name"] : "default";
var ip = (s["ip"].Type == JTokenType.String) ? (string)s["ip"] : "";
var port = (s["port"].Type == JTokenType.Integer) ? (int)s["port"] : -1;
var dir = (s["dir"].Type == JTokenType.String) ? (string)s["dir"] : "";
var res = false;
if (GetServerIdByName(name) == -1)
res = AddServer(name, ip, port, dir);
else
res = ChangeServerByName(name, ip, port, dir);
if (!res)
log.Trace("Can't add or change server #" + count + " from conf file.");
}
//init vdp
for (var i = 0; i < 49; i++)
vdpList.Add(new (i, i == 0 ? "Общие" : "Печь " + i.ToString("D2")));
for (var i = 91; i < 95; i++)
vdpList.Add((i, "Печь " + i.ToString("D2")));
//init analog
var tmpvdp = new List();
foreach (var v in vdpList) tmpvdp.Add(v.num);
confAnalog = Configuration.InitAnalogList(Path.Combine(confDir, "analog"), tmpvdp.ToArray());
}
/// Создание экземпляра класса.
public STPClient()
{
log.Trace("Create instance class.");
Config();
}
/// Создание экземпляра класса с указанием конфигурационного файла.
/// Путь к файлу конфигурации.
public STPClient(string confDir)
{
log.Trace("Create instance class with dir:" + confDir + ".");
this.confDir = confDir;
Config();
}
public TechCycle[] GetTechCycles(int vdp, DateTime tS, DateTime tE)
{
//Use File with ID = 3
var res = new List();
return res.ToArray();
}
public Analog[] GetAnalogs(int vdp, DateTime tS, DateTime tE)
{
var res = new List();
var net = new NETClient(servers[serverDefault].ip, servers[serverDefault].port);
if (!net.Connect()) return Array.Empty();
var s1 = new List>();
for (var d = new DateTime(tS.Year, tS.Month, tS.Day); d.Year <= tE.Year && d.Month <= tE.Month && d.Day <= tE.Day; d = d.AddDays(1))
{
var t = new Task(() => net.Full_Download(d, vdp, 01));
t.Start();
s1.Add(t);
}
var s2 = new List>();
while(s1.Count > 0)
{
for(var i = 0; i < s1.Count; i++)
if (s1[i].IsCompleted)
{
var t = new Task( () => {
if (!s1[i].IsCompletedSuccessfully)
return new Analog[0];
return s1[i].Result
});
}
}
return res.ToArray();
}
}
}