001/* 002 * Copyright 2015-2024 the original author or authors 003 * 004 * This software is licensed under the Apache License, Version 2.0, 005 * the GNU Lesser General Public License version 2 or later ("LGPL") 006 * and the WTFPL. 007 * You may choose either license to govern your use of this software only 008 * upon the condition that you accept all of the terms of either 009 * the Apache License 2.0, the LGPL 2.1+ or the WTFPL. 010 */ 011package org.minidns.util; 012 013import java.util.ArrayList; 014import java.util.Collection; 015import java.util.Collections; 016import java.util.LinkedList; 017import java.util.List; 018import java.util.SortedMap; 019import java.util.TreeMap; 020 021import org.minidns.dnsname.DnsName; 022import org.minidns.record.SRV; 023 024public class SrvUtil { 025 026 /** 027 * Sort the given collection of {@link SRV} resource records by their priority and weight. 028 * <p> 029 * Sorting by priority is easy. Sorting the buckets of SRV records with the same priority by weight requires to choose those records 030 * randomly but taking the weight into account. 031 * </p> 032 * 033 * @param srvRecords 034 * a collection of SRV records. 035 * @return a sorted list of the given records. 036 */ 037 public static List<SRV> sortSrvRecords(Collection<SRV> srvRecords) { 038 // RFC 2782, Usage rules: "If there is precisely one SRV RR, and its Target is "." 039 // (the root domain), abort." 040 if (srvRecords.size() == 1 && srvRecords.iterator().next().target.equals(DnsName.ROOT)) { 041 return Collections.emptyList(); 042 } 043 044 // Create the priority buckets. 045 SortedMap<Integer, List<SRV>> buckets = new TreeMap<>(); 046 for (SRV srvRecord : srvRecords) { 047 Integer priority = srvRecord.priority; 048 List<SRV> bucket = buckets.get(priority); 049 if (bucket == null) { 050 bucket = new LinkedList<>(); 051 buckets.put(priority, bucket); 052 } 053 bucket.add(srvRecord); 054 } 055 056 List<SRV> sortedSrvRecords = new ArrayList<>(srvRecords.size()); 057 058 for (List<SRV> bucket : buckets.values()) { 059 // The list of buckets will be sorted by priority, thanks to SortedMap. We now have determine the order of 060 // the SRV records with the same priority, i.e., within the same bucket, by their weight. This is done by 061 // creating an array 'totals' which reflects the percentage of the SRV RRs weight by the total weight of all 062 // SRV RRs in the bucket. For every entry in the bucket, we choose one using a random number and the sum of 063 // all weights left in the bucket. We then select RRs position based on the according index of the selected 064 // value in the 'total' array. This ensures that its weight is taken into account. 065 int bucketSize; 066 while ((bucketSize = bucket.size()) > 0) { 067 int[] totals = new int[bucketSize]; 068 069 int zeroWeight = 1; 070 for (SRV srv : bucket) { 071 if (srv.weight > 0) { 072 zeroWeight = 0; 073 break; 074 } 075 } 076 077 int bucketWeightSum = 0, count = 0; 078 for (SRV srv : bucket) { 079 bucketWeightSum += srv.weight + zeroWeight; 080 totals[count++] = bucketWeightSum; 081 } 082 083 int selectedPosition; 084 if (bucketWeightSum == 0) { 085 // If total priority is 0, then the sum of all weights in this priority bucket is 0. So we simply 086 // select one of the weights randomly as the other algorithm performed in the else block is unable 087 // to handle this case. 088 selectedPosition = (int) (Math.random() * bucketSize); 089 } else { 090 double rnd = Math.random() * bucketWeightSum; 091 selectedPosition = bisect(totals, rnd); 092 } 093 094 SRV choosenSrvRecord = bucket.remove(selectedPosition); 095 sortedSrvRecords.add(choosenSrvRecord); 096 } 097 } 098 099 return sortedSrvRecords; 100 } 101 102 // TODO This is not yet really bisection just a stupid linear search. 103 private static int bisect(int[] array, double value) { 104 int pos = 0; 105 for (int element : array) { 106 if (value < element) 107 break; 108 pos++; 109 } 110 return pos; 111 } 112 113}