/driver-core/src/main/com/mongodb/connection/ClusterDescription.java

http://github.com/mongodb/mongo-java-driver · Java · 336 lines · 177 code · 34 blank · 125 comment · 42 complexity · 934d6ea1c4e2efa4171ef8f0ae07c86a MD5 · raw file

  1. /*
  2. * Copyright 2008-present 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.connection;
  17. import com.mongodb.MongoException;
  18. import com.mongodb.ReadPreference;
  19. import com.mongodb.annotations.Immutable;
  20. import com.mongodb.internal.selector.ReadPreferenceServerSelector;
  21. import com.mongodb.internal.selector.WritableServerSelector;
  22. import com.mongodb.lang.Nullable;
  23. import java.util.ArrayList;
  24. import java.util.Collections;
  25. import java.util.List;
  26. import static com.mongodb.assertions.Assertions.notNull;
  27. import static com.mongodb.internal.connection.ClusterDescriptionHelper.getServersByPredicate;
  28. import static java.lang.String.format;
  29. /**
  30. * Immutable snapshot state of a cluster.
  31. *
  32. * @since 3.0
  33. */
  34. @Immutable
  35. public class ClusterDescription {
  36. private final ClusterConnectionMode connectionMode;
  37. private final ClusterType type;
  38. private final List<ServerDescription> serverDescriptions;
  39. private final ClusterSettings clusterSettings;
  40. private final ServerSettings serverSettings;
  41. private final MongoException srvResolutionException;
  42. /**
  43. * Creates a new ClusterDescription.
  44. *
  45. * @param connectionMode whether to connect directly to a single server or to multiple servers
  46. * @param type what sort of cluster this is
  47. * @param serverDescriptions the descriptions of all the servers currently in this cluster
  48. */
  49. public ClusterDescription(final ClusterConnectionMode connectionMode, final ClusterType type,
  50. final List<ServerDescription> serverDescriptions) {
  51. this(connectionMode, type, serverDescriptions, null, null);
  52. }
  53. /**
  54. * Creates a new ClusterDescription.
  55. *
  56. * @param connectionMode whether to connect directly to a single server or to multiple servers
  57. * @param type what sort of cluster this is
  58. * @param serverDescriptions the descriptions of all the servers currently in this cluster
  59. * @param clusterSettings the cluster settings
  60. * @param serverSettings the server settings
  61. * @since 3.4
  62. */
  63. public ClusterDescription(final ClusterConnectionMode connectionMode, final ClusterType type,
  64. final List<ServerDescription> serverDescriptions,
  65. final ClusterSettings clusterSettings,
  66. final ServerSettings serverSettings) {
  67. this(connectionMode, type, null, serverDescriptions, clusterSettings, serverSettings);
  68. }
  69. /**
  70. * Creates a new ClusterDescription.
  71. *
  72. * @param connectionMode whether to connect directly to a single server or to multiple servers
  73. * @param type what sort of cluster this is
  74. * @param srvResolutionException an exception resolving the SRV record
  75. * @param serverDescriptions the descriptions of all the servers currently in this cluster
  76. * @param clusterSettings the cluster settings
  77. * @param serverSettings the server settings
  78. * @since 3.10
  79. */
  80. public ClusterDescription(final ClusterConnectionMode connectionMode, final ClusterType type,
  81. final MongoException srvResolutionException,
  82. final List<ServerDescription> serverDescriptions,
  83. final ClusterSettings clusterSettings,
  84. final ServerSettings serverSettings) {
  85. notNull("all", serverDescriptions);
  86. this.connectionMode = notNull("connectionMode", connectionMode);
  87. this.type = notNull("type", type);
  88. this.srvResolutionException = srvResolutionException;
  89. this.serverDescriptions = new ArrayList<ServerDescription>(serverDescriptions);
  90. this.clusterSettings = clusterSettings;
  91. this.serverSettings = serverSettings;
  92. }
  93. /**
  94. * Gets the cluster settings, which may be null if not provided.
  95. *
  96. * @return the cluster settings
  97. * @since 3.4
  98. */
  99. public ClusterSettings getClusterSettings() {
  100. return clusterSettings;
  101. }
  102. /**
  103. * Gets the server settings, which may be null if not provided.
  104. *
  105. * @return the server settings
  106. * @since 3.4
  107. */
  108. public ServerSettings getServerSettings() {
  109. return serverSettings;
  110. }
  111. /**
  112. * Return whether all servers in the cluster are compatible with the driver.
  113. *
  114. * @return true if all servers in the cluster are compatible with the driver
  115. */
  116. public boolean isCompatibleWithDriver() {
  117. for (ServerDescription cur : serverDescriptions) {
  118. if (!cur.isCompatibleWithDriver()) {
  119. return false;
  120. }
  121. }
  122. return true;
  123. }
  124. /**
  125. * Return a server in the cluster that is incompatibly older than the driver.
  126. *
  127. * @return a server in the cluster that is incompatibly older than the driver, or null if there are none
  128. * @since 3.6
  129. */
  130. @Nullable
  131. public ServerDescription findServerIncompatiblyOlderThanDriver() {
  132. for (ServerDescription cur : serverDescriptions) {
  133. if (cur.isIncompatiblyOlderThanDriver()) {
  134. return cur;
  135. }
  136. }
  137. return null;
  138. }
  139. /**
  140. * Return a server in the cluster that is incompatibly newer than the driver.
  141. *
  142. * @return a server in the cluster that is incompatibly newer than the driver, or null if there are none
  143. * @since 3.6
  144. */
  145. @Nullable
  146. public ServerDescription findServerIncompatiblyNewerThanDriver() {
  147. for (ServerDescription cur : serverDescriptions) {
  148. if (cur.isIncompatiblyNewerThanDriver()) {
  149. return cur;
  150. }
  151. }
  152. return null;
  153. }
  154. /**
  155. * Returns true if this cluster has at least one server that satisfies the given read preference.
  156. *
  157. * @param readPreference the non-null read preference
  158. * @return whether this cluster has at least one server that satisfies the given read preference
  159. * @since 3.3
  160. */
  161. public boolean hasReadableServer(final ReadPreference readPreference) {
  162. notNull("readPreference", readPreference);
  163. return !new ReadPreferenceServerSelector(readPreference).select(this).isEmpty();
  164. }
  165. /**
  166. * Returns true if this cluster has at least one server that can be used for write operations.
  167. *
  168. * @return true if this cluster has at least one server that can be used for write operations
  169. * @since 3.3
  170. */
  171. public boolean hasWritableServer() {
  172. return !new WritableServerSelector().select(this).isEmpty();
  173. }
  174. /**
  175. * Gets whether this cluster is connecting to a single server or multiple servers.
  176. *
  177. * @return the ClusterConnectionMode for this cluster
  178. */
  179. public ClusterConnectionMode getConnectionMode() {
  180. return connectionMode;
  181. }
  182. /**
  183. * Gets the specific type of this cluster
  184. *
  185. * @return a ClusterType enum representing the type of this cluster
  186. */
  187. public ClusterType getType() {
  188. return type;
  189. }
  190. /**
  191. * Gets any exception encountered while resolving the SRV record for the initial host.
  192. *
  193. * @return any exception encountered while resolving the SRV record for the initial host, or null if none
  194. * @since 3.10
  195. */
  196. @Nullable
  197. public MongoException getSrvResolutionException() {
  198. return srvResolutionException;
  199. }
  200. /**
  201. * Returns an unmodifiable list of the server descriptions in this cluster description.
  202. *
  203. * @return an unmodifiable list of the server descriptions in this cluster description
  204. * @since 3.3
  205. */
  206. public List<ServerDescription> getServerDescriptions() {
  207. return Collections.unmodifiableList(serverDescriptions);
  208. }
  209. /**
  210. * Gets the logical session timeout in minutes, or null if at least one of the known servers does not support logical sessions.
  211. *
  212. * @return the logical session timeout in minutes, which may be null
  213. * @mongodb.server.release 3.6
  214. * @since 3.6
  215. */
  216. @Nullable
  217. public Integer getLogicalSessionTimeoutMinutes() {
  218. Integer retVal = null;
  219. for (ServerDescription cur : getServersByPredicate(this, serverDescription ->
  220. serverDescription.isPrimary() || serverDescription.isSecondary())) {
  221. Integer logicalSessionTimeoutMinutes = cur.getLogicalSessionTimeoutMinutes();
  222. if (logicalSessionTimeoutMinutes == null) {
  223. return null;
  224. }
  225. if (retVal == null) {
  226. retVal = logicalSessionTimeoutMinutes;
  227. } else {
  228. retVal = Math.min(retVal, logicalSessionTimeoutMinutes);
  229. }
  230. }
  231. return retVal;
  232. }
  233. @Override
  234. public boolean equals(final Object o) {
  235. if (this == o) {
  236. return true;
  237. }
  238. if (o == null || getClass() != o.getClass()) {
  239. return false;
  240. }
  241. ClusterDescription that = (ClusterDescription) o;
  242. if (connectionMode != that.connectionMode) {
  243. return false;
  244. }
  245. if (type != that.type) {
  246. return false;
  247. }
  248. if (serverDescriptions.size() != that.serverDescriptions.size()) {
  249. return false;
  250. }
  251. if (!serverDescriptions.containsAll(that.serverDescriptions)) {
  252. return false;
  253. }
  254. // Compare class equality and message as exceptions rarely override equals
  255. Class<?> thisExceptionClass = srvResolutionException != null ? srvResolutionException.getClass() : null;
  256. Class<?> thatExceptionClass = that.srvResolutionException != null ? that.srvResolutionException.getClass() : null;
  257. if (thisExceptionClass != null ? !thisExceptionClass.equals(thatExceptionClass) : thatExceptionClass != null) {
  258. return false;
  259. }
  260. String thisExceptionMessage = srvResolutionException != null ? srvResolutionException.getMessage() : null;
  261. String thatExceptionMessage = that.srvResolutionException != null ? that.srvResolutionException.getMessage() : null;
  262. if (thisExceptionMessage != null ? !thisExceptionMessage.equals(thatExceptionMessage) : thatExceptionMessage != null) {
  263. return false;
  264. }
  265. return true;
  266. }
  267. @Override
  268. public int hashCode() {
  269. int result = connectionMode.hashCode();
  270. result = 31 * result + type.hashCode();
  271. result = 31 * result + (srvResolutionException == null ? 0 : srvResolutionException.hashCode());
  272. result = 31 * result + serverDescriptions.hashCode();
  273. return result;
  274. }
  275. @Override
  276. public String toString() {
  277. return "ClusterDescription{"
  278. + "type=" + getType()
  279. + (srvResolutionException == null ? "" : ", srvResolutionException=" + srvResolutionException)
  280. + ", connectionMode=" + connectionMode
  281. + ", serverDescriptions=" + serverDescriptions
  282. + '}';
  283. }
  284. /**
  285. * Returns a short, pretty description for this ClusterDescription.
  286. *
  287. * @return a String describing this cluster.
  288. */
  289. public String getShortDescription() {
  290. StringBuilder serverDescriptions = new StringBuilder();
  291. String delimiter = "";
  292. for (final ServerDescription cur : this.serverDescriptions) {
  293. serverDescriptions.append(delimiter).append(cur.getShortDescription());
  294. delimiter = ", ";
  295. }
  296. if (srvResolutionException == null) {
  297. return format("{type=%s, servers=[%s]", type, serverDescriptions);
  298. } else {
  299. return format("{type=%s, srvResolutionException=%s, servers=[%s]", type, srvResolutionException, serverDescriptions);
  300. }
  301. }
  302. }