using System; using System.Collections.Generic; using System.Linq; using System.Net; using System.Threading; using Newtonsoft.Json; using Shadowsocks.Model; namespace Shadowsocks.Controller.Strategy { using Statistics = Dictionary>; internal class StatisticsStrategy : IStrategy, IDisposable { private readonly ShadowsocksController _controller; private Server _currentServer; private readonly Timer _timer; private Statistics _filteredStatistics; private AvailabilityStatistics Service => _controller.availabilityStatistics; private int ChoiceKeptMilliseconds => (int)TimeSpan.FromMinutes(_controller.StatisticsConfiguration.ChoiceKeptMinutes).TotalMilliseconds; public StatisticsStrategy(ShadowsocksController controller) { _controller = controller; var servers = controller.GetCurrentConfiguration().configs; var randomIndex = new Random().Next() % servers.Count; _currentServer = servers[randomIndex]; //choose a server randomly at first _timer = new Timer(ReloadStatisticsAndChooseAServer); } private void ReloadStatisticsAndChooseAServer(object obj) { Logging.Debug("Reloading statistics and choose a new server...."); var servers = _controller.GetCurrentConfiguration().configs; LoadStatistics(); ChooseNewServer(servers); } private void LoadStatistics() { _filteredStatistics = Service.FilteredStatistics ?? Service.RawStatistics ?? _filteredStatistics; } //return the score by data //server with highest score will be choosen private float GetScore(string serverName) { var config = _controller.StatisticsConfiguration; List records; if (_filteredStatistics == null || !_filteredStatistics.TryGetValue(serverName, out records)) return 0; float factor; float score = 0; var averageRecord = new StatisticsRecord(serverName, records.FindAll(record => record.MaxInboundSpeed != null).Select(record => record.MaxInboundSpeed.Value), records.FindAll(record => record.MaxOutboundSpeed != null).Select(record => record.MaxOutboundSpeed.Value), records.FindAll(record => record.AverageLatency != null).Select(record => record.AverageLatency.Value)); averageRecord.setResponse(records.Select(record => record.AverageResponse)); if (!config.Calculations.TryGetValue("PackageLoss", out factor)) factor = 0; score += averageRecord.PackageLoss*factor ?? 0; if (!config.Calculations.TryGetValue("AverageResponse", out factor)) factor = 0; score += averageRecord.AverageResponse*factor ?? 0; if (!config.Calculations.TryGetValue("MinResponse", out factor)) factor = 0; score += averageRecord.MinResponse*factor ?? 0; if (!config.Calculations.TryGetValue("MaxResponse", out factor)) factor = 0; score += averageRecord.MaxResponse*factor ?? 0; Logging.Debug($"{JsonConvert.SerializeObject(averageRecord, Formatting.Indented)}"); return score; } private void ChooseNewServer(List servers) { if (_filteredStatistics == null || servers.Count == 0) { return; } try { var bestResult = (from server in servers let name = server.FriendlyName() where _filteredStatistics.ContainsKey(name) select new { server, score = GetScore(name) } ).Aggregate((result1, result2) => result1.score > result2.score ? result1 : result2); LogWhenEnabled($"Switch to server: {bestResult.server.FriendlyName()} by statistics: score {bestResult.score}"); _currentServer = bestResult.server; } catch (Exception e) { Logging.LogUsefulException(e); } } private void LogWhenEnabled(string log) { if (_controller.GetCurrentStrategy()?.ID == ID) //output when enabled { Console.WriteLine(log); } } public string ID => "com.shadowsocks.strategy.scbs"; public string Name => I18N.GetString("Choose By Total Package Loss"); public Server GetAServer(IStrategyCallerType type, IPEndPoint localIPEndPoint) { if (_currentServer == null) { ChooseNewServer(_controller.GetCurrentConfiguration().configs); } return _currentServer; //current server cached for CachedInterval } public void ReloadServers() { ChooseNewServer(_controller.GetCurrentConfiguration().configs); _timer?.Change(0, ChoiceKeptMilliseconds); } public void SetFailure(Server server) { Logging.Debug($"failure: {server.FriendlyName()}"); } public void UpdateLastRead(Server server) { } public void UpdateLastWrite(Server server) { } public void UpdateLatency(Server server, TimeSpan latency) { } public void Dispose() { _timer.Dispose(); } } }