using System; using System.Collections.Generic; using System.Linq; using static FrostFS.SDK.FrostFsNetmapSnapshot; namespace FrostFS.SDK.Client.Models.Netmap.Placement; public static class Tools { internal static ulong Distance(ulong x, ulong y) { var acc = x ^ y; acc ^= acc >> 33; acc *= 0xff51afd7ed558ccd; acc ^= acc >> 33; acc *= 0xc4ceb9fe1a85ec53; acc ^= acc >> 33; return acc; } internal static double ReverceNormalize(double r, double w) { return (r + 1) / (w + 1); } internal static double Normalize(double r, double w) { if (r == 0) { return 0; } var x = w / r; return x / (1 + x); } internal static void AppendWeightsTo(FrostFsNodeInfo[] nodes, Func wf, ref double[] weights) { if (weights.Length < nodes.Length) { weights = new double[nodes.Length]; } for (int i = 0; i < nodes.Length; i++) { weights[i] = wf(nodes[i]); } } internal static List SortHasherSliceByWeightValue(List nodes, Span weights, ulong hash) where T : IHasher { if (nodes.Count == 0) { return nodes; } var allEquals = true; if (weights.Length > 1) { for (int i = 1; i < weights.Length; i++) { if (weights[i] != weights[0]) { allEquals = false; break; } } } var dist = new double[nodes.Count]; if (allEquals) { for (int i = 0; i < dist.Length; i++) { var x = nodes[i].Hash(); dist[i] = Distance(x, hash); } return SortHasherByDistance(nodes, dist, true); } for (int i = 0; i < dist.Length; i++) { var d = Distance(nodes[i].Hash(), hash); dist[i] = (ulong.MaxValue - d) * weights[i]; } return SortHasherByDistance(nodes, dist, false); } internal static List SortHasherByDistance(List nodes, N[] dist, bool asc) { IndexedValue[] indexes = new IndexedValue[nodes.Count]; for (int i = 0; i < dist.Length; i++) { indexes[i] = new IndexedValue() { nodeInfo = nodes[i], dist = dist[i] }; } if (asc) { return new List(indexes .OrderBy(x => x.dist) .Select(x => x.nodeInfo).ToArray()); } else { return new List(indexes .OrderByDescending(x => x.dist) .Select(x => x.nodeInfo)); } } internal static Func DefaultWeightFunc(IReadOnlyList nodes) { MeanAgg mean = new(); MinAgg minV = new(); foreach (var node in nodes) { mean.Add(node.GetCapacity()); minV.Add(node.Price); } return NewWeightFunc( NewSigmoidNorm(mean.Compute()), NewReverseMinNorm(minV.Compute())); } private struct IndexedValue { internal T nodeInfo; internal N dist; } }