public Node FindNode(string id)
 {
     using (var db = ApiFab.GetDbContext())
     {
         return(db.Nodes.Find(id));
     }
 }
        private void OnTimer(object state)
        {
            foreach (var network in Network.Networks)
            {
                for (var numOfTryes = 0; numOfTryes < 3; numOfTryes++)
                {
                    try
                    {
                        using (var client = ApiFab.CreateSignalApi(network.SignalIp))
                        {
                            var result = client.GetActiveNodes();
                            var nodes  = result.Nodes.Distinct((n1, n2) => n1.Ip.Equals(n2.Ip)).ToArray();
                            using (var db = ApiFab.GetDbContext())
                            {
                                var dbNodes = db.Nodes.ToArray();
                                foreach (var serverNode in nodes)
                                {
                                    var ip = serverNode.Ip;
                                    if (string.IsNullOrEmpty(ip))
                                    {
                                        continue;
                                    }
                                    var exnode = dbNodes.FirstOrDefault(n => n.Ip.Equals(ip));
                                    if (exnode != null)
                                    {
                                        exnode.ModifyTime = DateTime.Now;
                                        exnode.Version    = serverNode.Version;
                                        db.Nodes.Update(exnode);
                                    }
                                    else
                                    {
                                        var node = new Node
                                        {
                                            Ip      = ip,
                                            Version = serverNode.Version,
                                            Network = network.Id
                                        };
                                        db.Nodes.Add(node);
                                    }

                                    db.SaveChanges();
                                }
                            }
                        }
                        break;
                    }
                    catch (Exception e)
                    {
                        _logger.LogError(e, $"Exception in NodesSource NumOfTryes={numOfTryes}");
                        Thread.Sleep(1000);
                    }
                }
            }

            _timer.Change(Period, 0);
        }
예제 #3
0
        private void OnTimer(object state)
        {
            try
            {
                foreach (var network in Network.Networks)
                {
                    using (var client = ApiFab.CreateSignalApi(network.SignalIp))
                    {
                        var result = client.GetActiveNodes();
                        var nodes  = result.Nodes.Distinct((n1, n2) => n1.Ip.Equals(n2.Ip)).ToArray();
                        using (var db = GetDbContext())
                        {
                            var dbNodes = db.Nodes.ToArray();
                            foreach (var serverNode in nodes)
                            {
                                var ip = serverNode.Ip;
                                if (string.IsNullOrEmpty(ip))
                                {
                                    continue;
                                }
                                var exnode = dbNodes.FirstOrDefault(n => n.Ip.Equals(ip));
                                if (exnode != null)
                                {
                                    exnode.ModifyTime = DateTime.Now;
                                    exnode.Version    = serverNode.Version;
                                    db.Nodes.Update(exnode);
                                }
                                else
                                {
                                    //var uri = new Uri($"https://ipapi.co/{ip}/json/");
                                    //var nodestr = await GetAsync(uri);
                                    //var node = JsonConvert.DeserializeObject<Node>(nodestr);
                                    //if (!node.Ip.Equals(ip)) continue;
                                    var node = new Node
                                    {
                                        Ip      = ip,
                                        Version = serverNode.Version,
                                        Network = network.Id
                                    };
                                    db.Nodes.Add(node);
                                }

                                db.SaveChanges();
                            }
                        }
                    }
                }
            }
            catch (Exception e)
            {
                _logger.LogError(e, "Exception in NodesSource");
            }
            _timer.Change(Period, 0);
        }
        public NodesData GetNodes(string network)
        {
            var net = Network.GetById(network);

            using (var db = ApiFab.GetDbContext())
            {
                var result = new NodesData
                {
                    Nodes = db.Nodes.Where(n => (n.Network == network) &&
                                           (net.RandomNodes || (n.ModifyTime.AddMinutes(LiveTimeMinutes) >= DateTime.Now)))
                            .OrderBy(n => n.ModifyTime)
                            .Take(1000).ToList()
                };
                return(result);
            }
        }
예제 #5
0
        public NodesData GetNodes(string network)
        {
            var net = Network.GetById(network);

            using (var db = ApiFab.GetDbContext())
            {
                var now   = DateTime.Now;
                var nodes = db.Nodes.Where(n => n.Network == network &&
                                           (net.RandomNodes || n.ModifyTime.AddSeconds(Settings.NodesLivePeriodSec) >= now))
                            .OrderBy(n => n.ModifyTime)
                            .Take(1000).ToList();
                var result = new NodesData {
                    Nodes = nodes
                };
                return(result);
            }
        }
예제 #6
0
        public ContractsData GetContracts(int page)
        {
            if (page <= 0)
            {
                page = 1;
            }
            var result = new ContractsData {
                Page = page, LastPage = 1
            };

            using (var db = ApiFab.GetDbContext())
            {
                var i = 1;
                foreach (var s in db.Smarts.Where(s => s.Network == Net))
                {
                    result.Contracts.Add(new ContractLinkInfo(i++, s.Address));
                }
            }
            result.NumStr = result.Contracts.Any() ? $"{result.Contracts.First().Index} - {result.Contracts.Last().Index}" : "0";
            return(result);
        }
예제 #7
0
        public ContractsData GetContracts(int page)
        {
            if (page <= 0)
            {
                page = 1;
            }
            var result = new ContractsData {
                Page = page
            };

            if (Net == "tetris")
            {
                using (var client = CreateApi())
                {
                    var offset = result.NumPerPage * (page - 1);
                    var res    = client.SmartContractsAllListGet(offset, result.NumPerPage + 1);
                    result.HaveNextPage = res.SmartContractsList.Count > result.NumPerPage;
                    var count = Math.Min(result.NumPerPage, res.SmartContractsList.Count);
                    for (var i = 0; i < count; i++)
                    {
                        var c     = res.SmartContractsList[i];
                        var cInfo = new ContractLinkInfo(i + offset + 1, c.Address);
                        result.Contracts.Add(cInfo);
                    }
                }
            }
            else
            {
                using (var db = ApiFab.GetDbContext())
                {
                    var i = 1;
                    foreach (var s in db.Smarts.Where(s => s.Network == Net))
                    {
                        result.Contracts.Add(new ContractLinkInfo(i++, s.Address));
                    }
                }
            }
            return(result);
        }
예제 #8
0
        public async Task InvokeAsync(HttpContext context)
        {
            var token = context.Request.Path.ToString();

            if (!context.Request.IsHttps && token.EndsWith("/getLastBlock", StringComparison.InvariantCultureIgnoreCase))
            {
                var net = Network.GetById(token.Split('/')[1]);
                if (net == null)
                {
                    context.Response.Redirect("/");
                    return;
                }
                using (var client = ApiFab.CreateReleaseApi(net.Ip))
                {
                    var pools = client.PoolListGet(0, 1);
                    context.Response.StatusCode = 200;
                    await context.Response.WriteAsync(pools.Pools[0].PoolNumber.ToString());
                }
            }
            else
            {
                await _next.Invoke(context);
            }
        }
예제 #9
0
 private API.Client CreateApi()
 {
     return(ApiFab.CreateReleaseApi(Network.GetById(Net).Ip));
 }
예제 #10
0
 private API.Client CreateApi()
 {
     return(ApiFab.CreateNodeApi(Network.GetById(RouteData.Values["network"].ToString()).Ip));
 }
예제 #11
0
        private void OnCacheTimer(object state)
        {
            var tpState = (IndexServiceState)state;

            try
            {
                if (tpState.Net.Api.EndsWith("/Api"))
                {
                    using (var client = ApiFab.CreateNodeApi(tpState.Net.Ip))
                    {
                        // Service available
                        if (tpState.Net.Updating)
                        {
                            tpState.Net.Updating = false;
                        }

                        // Request blocks
                        if ((!tpState.PoolsOut.Any() && !tpState.PoolsIn.Any()))
                        {
                            var result = client.PoolListGet(0, SizeOut);
                            tpState.PoolsOut = result.Pools.Select(p => new PoolInfo(p)).ToList();
                        }
                        else
                        {
                            var result = client.PoolListGet(0, 20);
                            lock (tpState.PoolsLock)
                            {
                                var firstPoolNum = tpState.PoolsIn.Any()
                                    ? tpState.PoolsIn[0].Number
                                    : tpState.PoolsOut[0].Number;
                                var nPools = result.Pools.TakeWhile(p => (p.PoolNumber > firstPoolNum) || (p.PoolNumber < firstPoolNum - 1000)).Select(p => new PoolInfo(p)).ToList();
                                tpState.PoolsIn = nPools.Concat(tpState.PoolsIn).ToList();
                            }
                        }

                        // Request stats
                        if (tpState.StatRequestCounter == 0)
                        {
                            var stats = client.StatsGet();
                            if (stats != null && stats.Stats.Count >= 4)
                            {
                                var statsSorted = stats.Stats.OrderBy(s => s.PeriodDuration).ToList();
                                var statData    = new StatData();
                                for (var i = 0; i < 4; i++)
                                {
                                    statData.Pdata[i] = new PeriodData(statsSorted[i]);
                                }
                                tpState.StatData = statData;
                            }
                        }
                    }
                }
                else if (tpState.Net.Api.EndsWith("/TestApi"))
                {
                    using (var client = ApiFab.CreateTestApi(tpState.Net.Ip))
                    {
                        // Service available
                        if (tpState.Net.Updating)
                        {
                            tpState.Net.Updating = false;
                        }

                        // Request blocks
                        if ((!tpState.PoolsOut.Any() && !tpState.PoolsIn.Any()))
                        {
                            var result = client.PoolListGet(0, SizeOut);
                            tpState.PoolsOut = result.Pools.Where(p => p.PoolNumber > 0).Select(p => new PoolInfo(p)).ToList();
                        }
                        else
                        {
                            var result = client.PoolListGet(0, 20);
                            lock (tpState.PoolsLock)
                            {
                                var firstPoolNum = tpState.PoolsIn.Any()
                                    ? tpState.PoolsIn[0].Number
                                    : tpState.PoolsOut[0].Number;
                                var nPools = result.Pools.Where(p => p.PoolNumber > 0).TakeWhile(p => (p.PoolNumber > firstPoolNum) || (p.PoolNumber < firstPoolNum - 1000)).Select(p => new PoolInfo(p)).ToList();
                                tpState.PoolsIn = nPools.Concat(tpState.PoolsIn).ToList();
                            }
                        }

                        // Request stats
                        if (tpState.StatRequestCounter == 0)
                        {
                            var stats = client.StatsGet();
                            if (stats != null && stats.Stats.Count >= 4)
                            {
                                var statsSorted = stats.Stats.OrderBy(s => s.PeriodDuration).ToList();
                                var statData    = new StatData();
                                for (var i = 0; i < 4; i++)
                                {
                                    statData.Pdata[i] = new PeriodData(statsSorted[i]);
                                }
                                tpState.StatData = statData;
                            }
                        }
                    }
                }
                else if (tpState.Net.Api.EndsWith("/ReleaseApi"))
                {
                    using (var client = ApiFab.CreateReleaseApi(tpState.Net.Ip))
                    {
                        // Service available
                        if (tpState.Net.Updating)
                        {
                            tpState.Net.Updating = false;
                        }

                        // Request blocks
                        if ((!tpState.PoolsOut.Any() && !tpState.PoolsIn.Any()))
                        {
                            var result = client.PoolListGet(0, SizeOut);
                            tpState.PoolsOut = result.Pools.Where(p => p.PoolNumber > 0).Select(p => new PoolInfo(p)).ToList();
                        }
                        else
                        {
                            var result = client.PoolListGet(0, 20);

                            var firstPoolNum = tpState.PoolsIn.Any()
                                ? tpState.PoolsIn[0].Number
                                : tpState.PoolsOut[0].Number;

                            var newPools = result.Pools
                                           .Where(p => p.PoolNumber > 0)
                                           .TakeWhile(p => p.PoolNumber > firstPoolNum || p.PoolNumber < firstPoolNum - 1000)
                                           .Select(p => new PoolInfo(p)).ToList();

                            lock (tpState.PoolsLock)
                                tpState.PoolsIn = newPools.Concat(tpState.PoolsIn).ToList();
                        }

                        // Request stats
                        if (tpState.StatRequestCounter == 0)
                        {
                            var stats = client.StatsGet();
                            if (stats != null && stats.Stats.Count >= 4)
                            {
                                var statsSorted = stats.Stats.OrderBy(s => s.PeriodDuration).ToList();
                                var statData    = new StatData();
                                for (var i = 0; i < 4; i++)
                                {
                                    statData.Pdata[i] = new PeriodData(statsSorted[i]);
                                }
                                tpState.StatData = statData;
                            }
                        }
                    }
                }
                if (tpState.StatRequestCounter < Settings.UpdStatsPeriodSec * 1000 / Period)
                {
                    tpState.StatRequestCounter++;
                }
                else
                {
                    tpState.StatRequestCounter = 0;
                }
            }
            catch (Thrift.Transport.TTransportException e)
            {
                tpState.Net.Updating = true;
                _logger.LogError(e, "");
            }
            catch (Exception e)
            {
                _logger.LogError(e, "");
            }
            tpState.TimerForCache.Change(Period, 0);
        }
예제 #12
0
        private void OnTimer(object state)
        {
            var tpState = (TpsServiceState)state;

            try
            {
                using (var client = ApiFab.CreateNodeApi(tpState.Net.Ip))
                {
                    List <PoolInfo> lastPools;
                    if (tpState.Points == null)
                    {
                        var points = GetPoints(client, 1000, out lastPools);
                        if (points.Length > 2)
                        {
                            tpState.Points = new ConcurrentQueue <Point>(points.Skip(1).Take(points.Length - 2));
                        }
                    }
                    else
                    {
                        while (tpState.Points.Count > 100)
                        {
                            tpState.Points.TryDequeue(out _);
                        }
                        var points   = GetPoints(client, 100, out lastPools);
                        var lasttime = tpState.Points.Last().X;
                        if (points.Length > 1)
                        {
                            foreach (var point in points.Take(points.Length - 1).Where(p => p.X > lasttime))
                            {
                                tpState.Points.Enqueue(point);
                            }
                        }
                    }
                    var indexData = new IndexData {
                        LastBlocks = lastPools
                    };
                    if (lastPools.Any())
                    {
                        var lastPool = lastPools.First();
                        indexData.LastBlock        = lastPool.Number;
                        indexData.LastTime         = lastPool.Age.Equals("0") ? DateTime.Now.ToString("G") : lastPool.Time.ToString("G");
                        indexData.LastBlockTxCount = lastPool.TxCount;

                        var last10Pools = lastPools.Take(10).ToList();
                        if ((DateTime.Now - last10Pools.First().Time).TotalSeconds < 2)
                        {
                            var time10Pools = (last10Pools.First().Time - last10Pools.Last().Time).TotalSeconds;
                            indexData.Pps = (int)(time10Pools > 0 ? last10Pools.Count / time10Pools : 4);
                        }
                    }

                    if (tpState.StatRequestCounter == 0)
                    {
                        var stats = client.StatsGet();
                        if (stats != null && stats.Stats.Count >= 4)
                        {
                            var statsSorted = stats.Stats.OrderBy(s => s.PeriodDuration).ToArray();
                            for (var i = 0; i < 4; i++)
                            {
                                indexData.SetPData(i, new PeriodData(statsSorted[i]));
                            }
                            //indexData.CorrectTotal();
                        }
                    }
                    else if (tpState.IndexData != null)
                    {
                        indexData.Pdata = tpState.IndexData.Pdata;
                    }

                    tpState.IndexData = indexData;

                    if (tpState.StatRequestCounter < 100)
                    {
                        tpState.StatRequestCounter++;
                    }
                    else
                    {
                        tpState.StatRequestCounter = 0;
                    }
                }
            }
            catch (Exception e)
            {
                _logger.LogError(e, "Exception in TpsSource");

                try
                {
                    if (tpState.IndexData != null)
                    {
                        tpState.IndexData.Pps = 0;
                        var now = DateTime.Now;
                        foreach (var block in tpState.IndexData.LastBlocks)
                        {
                            block.RefreshAge(now);
                        }
                    }
                }
                catch (Exception ex)
                {
                    _logger.LogError(ex, "Secondary Exception in TpsSource");
                }
            }
            tpState.Timer.Change(Period, 0);
        }
예제 #13
0
        private void OnCacheTimer(object state)
        {
            var tpState = (IndexServiceState)state;

            try
            {
                using (var client = ApiFab.CreateReleaseApi(tpState.Net.Ip))
                {
                    // Service available
                    if (tpState.Net.Updating)
                    {
                        tpState.Net.Updating = false;
                    }

                    // Request blocks
                    if (!tpState.PoolsOut.Any() && !tpState.PoolsIn.Any())
                    {
                        var result = client.PoolListGet(0, SizeOut);
                        tpState.PoolsOut = result.Pools.Where(p => p.PoolNumber > 0).Select(p => new BlockInfo(p)).ToList();
                    }
                    else
                    {
                        // Get last 20 blocks from API
                        var result = client.PoolListGet(0, 20);

                        // Get last block number (first from the top)
                        var firstPoolNum = tpState.PoolsIn.Any()
                            ? tpState.PoolsIn[0].Number
                            : tpState.PoolsOut[0].Number;

                        // Network reset detection
                        var newNetWorkPools = result.Pools
                                              .Count(p => p.PoolNumber > 0 && p.PoolNumber < firstPoolNum - 200);

                        // Reset cache, if network reset
                        if (newNetWorkPools > 0)
                        {
                            lock (tpState.PoolsLock)
                            {
                                tpState.PoolsOut = new List <BlockInfo>();
                                tpState.PoolsIn  = new List <BlockInfo>();
                            }
                            firstPoolNum = 0;

                            // Delete Transactions per second statistics data
                            TpsService.Reset(tpState.Net.Id);
                        }

                        // Prepare list of new blocks
                        var newPools = result.Pools
                                       .Where(p => p.PoolNumber > 0)
                                       .TakeWhile(p => p.PoolNumber > firstPoolNum).ToList();

                        // Get Txs of new blocks
                        var newTx = new List <TransactionInfo>();
                        foreach (var pool in newPools)
                        {
                            var poolTr    = client.PoolTransactionsGet(pool.Hash, 0, BlockTxLimit);
                            var newPoolTx = poolTr.Transactions.Select((t, i) => new TransactionInfo(i, t.Id, t.Trxn)
                            {
                                Color = (int)(pool.PoolNumber % 10)
                            }).ToList();
                            newTx = newPoolTx.Concat(newTx).ToList();
                        }

                        // Append new blocks and txs to the input cache
                        lock (tpState.PoolsLock)
                        {
                            tpState.PoolsIn = newPools.Select(p => new BlockInfo(p)).Concat(tpState.PoolsIn).ToList();
                            tpState.TxIn    = newTx.Concat(tpState.TxIn).ToList();
                        }
                    }

                    // Request stats
                    if (tpState.StatRequestCounter == 0)
                    {
                        var stats = client.StatsGet();
                        if (stats != null && stats.Stats.Count >= 4)
                        {
                            var statsSorted = stats.Stats.OrderBy(s => s.PeriodDuration).ToList();
                            var statData    = new StatData();
                            for (var i = 0; i < 4; i++)
                            {
                                statData.Pdata[i] = new PeriodData(statsSorted[i]);
                            }
                            //statData.CorrectTotalValue();
                            tpState.StatData = statData;
                        }
                    }
                }

                // Increment statistics time counter (or reset if it's time)
                if (tpState.StatRequestCounter < Settings.UpdStatsPeriodSec * 1000 / Period)
                {
                    tpState.StatRequestCounter++;
                }
                else
                {
                    tpState.StatRequestCounter = 0;
                }
            }
            catch (Thrift.Transport.TTransportException e)
            {
                // Set up network updating flag in case of no connection to the node
                tpState.Net.Updating = true;
                _logger.LogError(e, "");
            }
            catch (Exception e)
            {
                // Log other errors
                _logger.LogError(e, "");
            }

            // Set up next timer tick
            tpState.TimerForCache.Change(Period, 0);
        }
예제 #14
0
 private API.Client CreateApi()
 {
     return(ApiFab.CreateTestApi2(Network.GetById(Net).Ip));
 }
예제 #15
0
        // Updates the list of nodes, from signal server
        private async Task UpdateNetworkNodes(Network network)
        {
            // Create a connection to the signal server API
            using (var client = ApiFab.CreateSignalApi(network.SignalIp, network.SignalPort))
            {
                // Get the list of nodes from API
                var result = client.GetActiveNodes();

                // Convert nodes to nodeInfos
                var nodes = result.Nodes.Select(n => new NodeInfo(n)).ToList();

                // Try to get country and geo-location for all nodes
                try
                {
                    // Connect to DB
                    using (var db = CsmonDbContext.Create())
                    {
                        // Get nodes, stored in db
                        var dbNodes = db.Nodes.Where(n => n.Network == network.Id).ToList();

                        // Add new nodes into db and update existing
                        foreach (var node in nodes)
                        {
                            var dbNode = dbNodes.FirstOrDefault(n => n.PublicKey.Equals(node.PublicKey));
                            if (dbNode == null)
                            {
                                db.Nodes.Add(new Node(node, network.Id));
                            }
                            else if (!node.EqualsDbNode(dbNode))
                            {
                                db.Nodes.Update(node.UpdateDbNode(dbNode));
                            }
                        }
                        db.SaveChanges();

                        // Get Non-Active nodes from db
                        foreach (var node in dbNodes.Where(n => !nodes.Any(d => d.PublicKey.Equals(n.PublicKey))))
                        {
                            nodes.Add(new NodeInfo(node));
                        }

                        // Find geo data for nodes
                        foreach (var ip in nodes.Select(n => n.Ip).Distinct())
                        {
                            // Try to find node in db by ip address
                            var location = db.Locations.FirstOrDefault(l => l.Ip.Equals(ip));

                            // If not found, try to get info by ip using ipapi.co web service
                            if (location == null)
                            {
                                try
                                {
                                    var uri     = new Uri("https://ipapi.co/" + $"{ip}/json/");
                                    var nodeStr = await GetAsync(uri);

                                    nodeStr     = nodeStr.Replace("\"latitude\": null,", "");
                                    nodeStr     = nodeStr.Replace("\"longitude\": null,", "");
                                    location    = JsonConvert.DeserializeObject <Location>(nodeStr);
                                    location.Ip = ip;
                                    if (location.Org.Length > 64)
                                    {
                                        location.Org = location.Org.Substring(0, 64);
                                    }
                                    // Store data in db
                                    db.Locations.Add(location);
                                    db.SaveChanges();
                                }
                                catch (Exception e)
                                {
                                    _logger.LogError(e, "");
                                    continue;
                                }
                            }

                            // Set location data to nodes
                            foreach (var nodeInfo in nodes.Where(n => n.Ip.Equals(ip)))
                            {
                                nodeInfo.SetLocation(location);
                            }
                        }
                    }
                }
                catch (Exception e)
                {
                    _logger.LogError(e, "");
                }

                // Collect additional info about nodes from the network
                try
                {
                    using (var cnt = ApiFab.CreateReleaseApi(network.Ip))
                    {
                        var page = 0;
                        while (true)
                        {
                            var writers = cnt.WritersGet(page++);
                            if (!writers.Writers.Any())
                            {
                                break;
                            }
                            foreach (var writer in writers.Writers)
                            {
                                var key  = Base58Encoding.Encode(writer.Address);
                                var node = nodes.FirstOrDefault(n => n.PublicKey == key);
                                if (node == null)
                                {
                                    continue;
                                }
                                node.TotalFee    = ConvUtils.FormatAmount(writer.FeeCollected);
                                node.TimesWriter = writer.TimesWriter;
                            }
                        }
                    }
                }
                catch (Exception e)
                {
                    _logger.LogError(e, "");
                }

                // Hide Ip addresses before output
                foreach (var nodeInfo in nodes)
                {
                    nodeInfo.HideIp();
                }

                // Put the ordered list into the storage
                _states[network.Id].Nodes = nodes.OrderByDescending(n => n.Active).ThenBy(n => n.Ip).ToList();
            }
        }
예제 #16
0
        private async void OnTimer(object state)
        {
            foreach (var network in Network.Networks)
            {
                for (var numOfTryes = 0; numOfTryes < 3; numOfTryes++)
                {
                    try
                    {
                        if (string.IsNullOrEmpty(network.SignalIp))
                        {
                            break;
                        }

                        using (var client = ApiFab.CreateSignalApi(network.SignalIp))
                        {
                            var result = client.GetActiveNodes();
                            var nodes  = result.Nodes.Distinct((n1, n2) => n1.Ip.Equals(n2.Ip)).ToArray();
                            using (var db = ApiFab.GetDbContext())
                            {
                                var dbNodes = db.Nodes.ToArray();
                                foreach (var serverNode in nodes)
                                {
                                    var ip = serverNode.Ip;
                                    if (string.IsNullOrEmpty(ip))
                                    {
                                        continue;
                                    }
                                    var exnode = dbNodes.FirstOrDefault(n => n.Ip.Equals(ip));
                                    if (exnode != null)
                                    {
                                        exnode.ModifyTime = DateTime.Now;
                                        exnode.Version    = serverNode.Version;
                                        exnode.Platform   = serverNode.Platform;
                                        exnode.Size       = result.Nodes.Count(n => n.Ip.Equals(ip));
                                        db.Nodes.Update(exnode);
                                    }
                                    else
                                    {
                                        var uri     = new Uri($"https://ipapi.co/{ip}/json/");
                                        var nodestr = await GetAsync(uri);

                                        var node = JsonConvert.DeserializeObject <Node>(nodestr);
                                        node.Ip       = ip;
                                        node.Version  = serverNode.Version;
                                        node.Network  = network.Id;
                                        node.Platform = serverNode.Platform;
                                        node.Size     = result.Nodes.Count(n => n.Ip.Equals(ip));
                                        db.Nodes.Add(node);
                                    }
                                    db.SaveChanges();
                                }
                            }
                        }

                        break;
                    }
                    catch (Exception e)
                    {
                        _logger.LogError(e, $"Exception in NodesSource NumOfTryes={numOfTryes}");
                        Thread.Sleep(1000);
                    }
                }
            }

            _timer.Change(_period, 0);
        }