/driver-core/src/main/com/mongodb/selector/LatencyMinimizingServerSelector.java

http://github.com/mongodb/mongo-java-driver · Java · 103 lines · 59 code · 12 blank · 32 comment · 9 complexity · de3727bb2d07d2e97b34a680b471aa33 MD5 · raw file

  1. /*
  2. * Copyright (c) 2008-2014 MongoDB, Inc.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.mongodb.selector;
  17. import com.mongodb.connection.ClusterDescription;
  18. import com.mongodb.connection.ServerDescription;
  19. import java.util.ArrayList;
  20. import java.util.List;
  21. import java.util.concurrent.TimeUnit;
  22. import static com.mongodb.connection.ClusterConnectionMode.MULTIPLE;
  23. import static java.util.concurrent.TimeUnit.MILLISECONDS;
  24. import static java.util.concurrent.TimeUnit.NANOSECONDS;
  25. /**
  26. * A server selector that accepts only servers within the given ping-time latency difference from the faster of the servers.
  27. *
  28. * @since 3.0
  29. */
  30. public class LatencyMinimizingServerSelector implements ServerSelector {
  31. private final long acceptableLatencyDifferenceNanos;
  32. /**
  33. *
  34. * @param acceptableLatencyDifference the maximum difference in ping-time latency between the fastest ping time and the slowest of
  35. * the chosen servers
  36. * @param timeUnit the time unit of the acceptableLatencyDifference
  37. */
  38. public LatencyMinimizingServerSelector(final long acceptableLatencyDifference, final TimeUnit timeUnit) {
  39. this.acceptableLatencyDifferenceNanos = NANOSECONDS.convert(acceptableLatencyDifference, timeUnit);
  40. }
  41. /**
  42. * Gets the acceptable latency difference.
  43. *
  44. * @param timeUnit the time unit to get it in.
  45. * @return the acceptable latency difference in the specified time unit
  46. */
  47. public long getAcceptableLatencyDifference(final TimeUnit timeUnit) {
  48. return timeUnit.convert(acceptableLatencyDifferenceNanos, NANOSECONDS);
  49. }
  50. @Override
  51. @SuppressWarnings("deprecation")
  52. public List<ServerDescription> select(final ClusterDescription clusterDescription) {
  53. if (clusterDescription.getConnectionMode() != MULTIPLE) {
  54. return clusterDescription.getAny();
  55. } else {
  56. return getServersWithAcceptableLatencyDifference(clusterDescription.getAny(),
  57. getFastestRoundTripTimeNanos(clusterDescription.getServerDescriptions()));
  58. }
  59. }
  60. @Override
  61. public String toString() {
  62. return "LatencyMinimizingServerSelector{"
  63. + "acceptableLatencyDifference=" + MILLISECONDS.convert(acceptableLatencyDifferenceNanos, NANOSECONDS) + " ms"
  64. + '}';
  65. }
  66. private long getFastestRoundTripTimeNanos(final List<ServerDescription> members) {
  67. long fastestRoundTripTime = Long.MAX_VALUE;
  68. for (final ServerDescription cur : members) {
  69. if (!cur.isOk()) {
  70. continue;
  71. }
  72. if (cur.getRoundTripTimeNanos() < fastestRoundTripTime) {
  73. fastestRoundTripTime = cur.getRoundTripTimeNanos();
  74. }
  75. }
  76. return fastestRoundTripTime;
  77. }
  78. private List<ServerDescription> getServersWithAcceptableLatencyDifference(final List<ServerDescription> servers,
  79. final long bestPingTime) {
  80. List<ServerDescription> goodSecondaries = new ArrayList<ServerDescription>(servers.size());
  81. for (final ServerDescription cur : servers) {
  82. if (!cur.isOk()) {
  83. continue;
  84. }
  85. if (cur.getRoundTripTimeNanos() - acceptableLatencyDifferenceNanos <= bestPingTime) {
  86. goodSecondaries.add(cur);
  87. }
  88. }
  89. return goodSecondaries;
  90. }
  91. }