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

http://github.com/mongodb/mongo-java-driver · Java · 206 lines · 166 code · 25 blank · 15 comment · 30 complexity · 922cb8057f2de778e547452b0e5b54d1 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.connection;
  17. import com.mongodb.async.SingleResultCallback;
  18. import org.bson.BsonDocument;
  19. import org.bson.BsonInt32;
  20. import java.util.List;
  21. import java.util.concurrent.atomic.AtomicInteger;
  22. import static com.mongodb.assertions.Assertions.notNull;
  23. import static com.mongodb.connection.CommandHelper.executeCommand;
  24. import static com.mongodb.connection.CommandHelper.executeCommandAsync;
  25. import static com.mongodb.connection.CommandHelper.executeCommandWithoutCheckingForFailure;
  26. import static com.mongodb.connection.DescriptionHelper.createConnectionDescription;
  27. class InternalStreamConnectionInitializer implements InternalConnectionInitializer {
  28. private final List<Authenticator> authenticators;
  29. InternalStreamConnectionInitializer(final List<Authenticator> authenticators) {
  30. this.authenticators = notNull("authenticators", authenticators);
  31. }
  32. @Override
  33. public ConnectionDescription initialize(final InternalConnection internalConnection) {
  34. notNull("internalConnection", internalConnection);
  35. ConnectionDescription connectionDescription = initializeConnectionDescription(internalConnection);
  36. authenticateAll(internalConnection, connectionDescription);
  37. return completeConnectionDescriptionInitialization(internalConnection, connectionDescription);
  38. }
  39. @Override
  40. public void initializeAsync(final InternalConnection internalConnection, final SingleResultCallback<ConnectionDescription> callback) {
  41. initializeConnectionDescriptionAsync(internalConnection, createConnectionDescriptionCallback(internalConnection, callback));
  42. }
  43. private SingleResultCallback<ConnectionDescription>
  44. createConnectionDescriptionCallback(final InternalConnection internalConnection,
  45. final SingleResultCallback<ConnectionDescription> callback) {
  46. return new SingleResultCallback<ConnectionDescription>() {
  47. @Override
  48. public void onResult(final ConnectionDescription connectionDescription, final Throwable t) {
  49. if (t != null) {
  50. callback.onResult(null, t);
  51. } else {
  52. new CompoundAuthenticator(internalConnection, connectionDescription,
  53. new SingleResultCallback<Void>() {
  54. @Override
  55. public void onResult(final Void result, final Throwable t) {
  56. if (t != null) {
  57. callback.onResult(null, t);
  58. } else {
  59. completeConnectionDescriptionInitializationAsync(internalConnection,
  60. connectionDescription,
  61. callback);
  62. }
  63. }
  64. })
  65. .start();
  66. }
  67. }
  68. };
  69. }
  70. private ConnectionDescription initializeConnectionDescription(final InternalConnection internalConnection) {
  71. BsonDocument isMasterResult = executeCommand("admin", new BsonDocument("ismaster", new BsonInt32(1)), internalConnection);
  72. BsonDocument buildInfoResult = executeCommand("admin", new BsonDocument("buildinfo", new BsonInt32(1)), internalConnection);
  73. return createConnectionDescription(internalConnection.getDescription().getConnectionId(), isMasterResult, buildInfoResult);
  74. }
  75. private ConnectionDescription completeConnectionDescriptionInitialization(final InternalConnection internalConnection,
  76. final ConnectionDescription connectionDescription) {
  77. return applyGetLastErrorResult(executeCommandWithoutCheckingForFailure("admin",
  78. new BsonDocument("getlasterror", new BsonInt32(1)),
  79. internalConnection),
  80. connectionDescription);
  81. }
  82. private void authenticateAll(final InternalConnection internalConnection, final ConnectionDescription connectionDescription) {
  83. if (connectionDescription.getServerType() != ServerType.REPLICA_SET_ARBITER) {
  84. for (final Authenticator cur : authenticators) {
  85. cur.authenticate(internalConnection, connectionDescription);
  86. }
  87. }
  88. }
  89. private void initializeConnectionDescriptionAsync(final InternalConnection internalConnection,
  90. final SingleResultCallback<ConnectionDescription> callback) {
  91. executeCommandAsync("admin", new BsonDocument("ismaster", new BsonInt32(1)), internalConnection,
  92. new SingleResultCallback<BsonDocument>() {
  93. @Override
  94. public void onResult(final BsonDocument isMasterResult, final Throwable t) {
  95. if (t != null) {
  96. callback.onResult(null, t);
  97. } else {
  98. executeCommandAsync("admin", new BsonDocument("buildinfo", new BsonInt32(1)), internalConnection,
  99. new SingleResultCallback<BsonDocument>() {
  100. @Override
  101. public void onResult(final BsonDocument buildInfoResult,
  102. final Throwable t) {
  103. if (t != null) {
  104. callback.onResult(null, t);
  105. } else {
  106. ConnectionId connectionId = internalConnection.getDescription()
  107. .getConnectionId();
  108. callback.onResult(createConnectionDescription(connectionId,
  109. isMasterResult,
  110. buildInfoResult),
  111. null);
  112. }
  113. }
  114. });
  115. }
  116. }
  117. });
  118. }
  119. private void completeConnectionDescriptionInitializationAsync(final InternalConnection internalConnection,
  120. final ConnectionDescription connectionDescription,
  121. final SingleResultCallback<ConnectionDescription> callback) {
  122. executeCommandAsync("admin", new BsonDocument("getlasterror", new BsonInt32(1)),
  123. internalConnection,
  124. new SingleResultCallback<BsonDocument>() {
  125. @Override
  126. public void onResult(final BsonDocument result, final Throwable t) {
  127. if (result == null) {
  128. callback.onResult(connectionDescription, null);
  129. } else {
  130. callback.onResult(applyGetLastErrorResult(result, connectionDescription), null);
  131. }
  132. }
  133. });
  134. }
  135. private ConnectionDescription applyGetLastErrorResult(final BsonDocument getLastErrorResult,
  136. final ConnectionDescription connectionDescription) {
  137. ConnectionId connectionId;
  138. if (getLastErrorResult.containsKey("connectionId")) {
  139. connectionId = connectionDescription.getConnectionId().withServerValue(getLastErrorResult.getNumber("connectionId").intValue());
  140. } else {
  141. connectionId = connectionDescription.getConnectionId();
  142. }
  143. return connectionDescription.withConnectionId(connectionId);
  144. }
  145. private class CompoundAuthenticator implements SingleResultCallback<Void> {
  146. private final InternalConnection internalConnection;
  147. private final ConnectionDescription connectionDescription;
  148. private final SingleResultCallback<Void> callback;
  149. private final AtomicInteger currentAuthenticatorIndex = new AtomicInteger(-1);
  150. public CompoundAuthenticator(final InternalConnection internalConnection, final ConnectionDescription connectionDescription,
  151. final SingleResultCallback<Void> callback) {
  152. this.internalConnection = internalConnection;
  153. this.connectionDescription = connectionDescription;
  154. this.callback = callback;
  155. }
  156. @Override
  157. public void onResult(final Void result, final Throwable t) {
  158. if (t != null) {
  159. callback.onResult(null, t);
  160. } else if (completedAuthentication()) {
  161. callback.onResult(null, null);
  162. } else {
  163. authenticateNext();
  164. }
  165. }
  166. public void start() {
  167. if (connectionDescription.getServerType() == ServerType.REPLICA_SET_ARBITER || authenticators.isEmpty()) {
  168. callback.onResult(null, null);
  169. } else {
  170. authenticateNext();
  171. }
  172. }
  173. private boolean completedAuthentication() {
  174. return currentAuthenticatorIndex.get() == authenticators.size() - 1;
  175. }
  176. private void authenticateNext() {
  177. authenticators.get(currentAuthenticatorIndex.incrementAndGet())
  178. .authenticateAsync(internalConnection, connectionDescription, this);
  179. }
  180. }
  181. }