/driver-async/src/main/com/mongodb/async/client/ClientSessionBinding.java

https://github.com/mongodb/mongo-java-driver · Java · 228 lines · 181 code · 32 blank · 15 comment · 24 complexity · c7abf464f7b8f3fde86105568f0bd8d2 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.async.client;
  17. import com.mongodb.ReadConcern;
  18. import com.mongodb.ReadPreference;
  19. import com.mongodb.async.SingleResultCallback;
  20. import com.mongodb.binding.AsyncClusterBinding;
  21. import com.mongodb.binding.AsyncConnectionSource;
  22. import com.mongodb.binding.AsyncReadWriteBinding;
  23. import com.mongodb.binding.AsyncSingleServerBinding;
  24. import com.mongodb.connection.AsyncConnection;
  25. import com.mongodb.connection.ClusterType;
  26. import com.mongodb.connection.Server;
  27. import com.mongodb.connection.ServerDescription;
  28. import com.mongodb.internal.session.ClientSessionContext;
  29. import com.mongodb.selector.ReadPreferenceServerSelector;
  30. import com.mongodb.session.SessionContext;
  31. import static com.mongodb.assertions.Assertions.notNull;
  32. class ClientSessionBinding implements AsyncReadWriteBinding {
  33. private final AsyncClusterBinding wrapped;
  34. private final ClientSession session;
  35. private final boolean ownsSession;
  36. private final ClientSessionContext sessionContext;
  37. ClientSessionBinding(final ClientSession session, final boolean ownsSession, final AsyncReadWriteBinding wrapped) {
  38. this.wrapped = notNull("wrapped", ((AsyncClusterBinding) wrapped));
  39. this.ownsSession = ownsSession;
  40. this.session = notNull("session", session);
  41. this.sessionContext = new AsyncClientSessionContext(session);
  42. }
  43. @Override
  44. public ReadPreference getReadPreference() {
  45. return wrapped.getReadPreference();
  46. }
  47. @Override
  48. public void getReadConnectionSource(final SingleResultCallback<AsyncConnectionSource> callback) {
  49. wrapped.getReadConnectionSource(new SingleResultCallback<AsyncConnectionSource>() {
  50. @Override
  51. public void onResult(final AsyncConnectionSource result, final Throwable t) {
  52. if (t != null) {
  53. callback.onResult(null, t);
  54. } else {
  55. wrapConnectionSource(result, callback);
  56. }
  57. }
  58. });
  59. }
  60. public void getWriteConnectionSource(final SingleResultCallback<AsyncConnectionSource> callback) {
  61. wrapped.getWriteConnectionSource(new SingleResultCallback<AsyncConnectionSource>() {
  62. @Override
  63. public void onResult(final AsyncConnectionSource result, final Throwable t) {
  64. if (t != null) {
  65. callback.onResult(null, t);
  66. } else {
  67. wrapConnectionSource(result, callback);
  68. }
  69. }
  70. });
  71. }
  72. @Override
  73. public SessionContext getSessionContext() {
  74. return sessionContext;
  75. }
  76. private void wrapConnectionSource(final AsyncConnectionSource connectionSource,
  77. final SingleResultCallback<AsyncConnectionSource> callback) {
  78. if (isActiveShardedTxn()) {
  79. if (session.getPinnedServerAddress() == null) {
  80. wrapped.getCluster().selectServerAsync(
  81. new ReadPreferenceServerSelector(wrapped.getReadPreference()),
  82. new SingleResultCallback<Server>() {
  83. @Override
  84. public void onResult(final Server server, final Throwable t) {
  85. if (t != null) {
  86. callback.onResult(null, t);
  87. } else {
  88. session.setPinnedServerAddress(server.getDescription().getAddress());
  89. setSingleServerBindingConnectionSource(callback);
  90. }
  91. }
  92. });
  93. } else {
  94. setSingleServerBindingConnectionSource(callback);
  95. }
  96. } else {
  97. callback.onResult(new SessionBindingAsyncConnectionSource(connectionSource), null);
  98. }
  99. }
  100. private void setSingleServerBindingConnectionSource(final SingleResultCallback<AsyncConnectionSource> callback) {
  101. final AsyncSingleServerBinding binding =
  102. new AsyncSingleServerBinding(wrapped.getCluster(), session.getPinnedServerAddress(), wrapped.getReadPreference());
  103. binding.getWriteConnectionSource(new SingleResultCallback<AsyncConnectionSource>() {
  104. @Override
  105. public void onResult(final AsyncConnectionSource result, final Throwable t) {
  106. binding.release();
  107. if (t != null) {
  108. callback.onResult(null, t);
  109. } else {
  110. callback.onResult(new SessionBindingAsyncConnectionSource(result), null);
  111. }
  112. }
  113. });
  114. }
  115. @Override
  116. public int getCount() {
  117. return wrapped.getCount();
  118. }
  119. @Override
  120. public AsyncReadWriteBinding retain() {
  121. wrapped.retain();
  122. return this;
  123. }
  124. @Override
  125. public void release() {
  126. wrapped.release();
  127. closeSessionIfCountIsZero();
  128. }
  129. private void closeSessionIfCountIsZero() {
  130. if (getCount() == 0 && ownsSession) {
  131. session.close();
  132. }
  133. }
  134. private boolean isActiveShardedTxn() {
  135. return session.hasActiveTransaction() && wrapped.getCluster().getDescription().getType() == ClusterType.SHARDED;
  136. }
  137. private class SessionBindingAsyncConnectionSource implements AsyncConnectionSource {
  138. private AsyncConnectionSource wrapped;
  139. SessionBindingAsyncConnectionSource(final AsyncConnectionSource wrapped) {
  140. this.wrapped = wrapped;
  141. }
  142. @Override
  143. public ServerDescription getServerDescription() {
  144. return wrapped.getServerDescription();
  145. }
  146. @Override
  147. public SessionContext getSessionContext() {
  148. return sessionContext;
  149. }
  150. @Override
  151. public void getConnection(final SingleResultCallback<AsyncConnection> callback) {
  152. wrapped.getConnection(callback);
  153. }
  154. @Override
  155. public AsyncConnectionSource retain() {
  156. wrapped = wrapped.retain();
  157. return this;
  158. }
  159. @Override
  160. public int getCount() {
  161. return wrapped.getCount();
  162. }
  163. @Override
  164. public void release() {
  165. wrapped.release();
  166. closeSessionIfCountIsZero();
  167. }
  168. }
  169. private final class AsyncClientSessionContext extends ClientSessionContext implements SessionContext {
  170. private final ClientSession clientSession;
  171. AsyncClientSessionContext(final ClientSession clientSession) {
  172. super(clientSession);
  173. this.clientSession = clientSession;
  174. }
  175. @Override
  176. public boolean isImplicitSession() {
  177. return ownsSession;
  178. }
  179. @Override
  180. public boolean notifyMessageSent() {
  181. return clientSession.notifyMessageSent();
  182. }
  183. @Override
  184. public boolean hasActiveTransaction() {
  185. return clientSession.hasActiveTransaction();
  186. }
  187. @Override
  188. public ReadConcern getReadConcern() {
  189. if (clientSession.hasActiveTransaction()) {
  190. return clientSession.getTransactionOptions().getReadConcern();
  191. } else {
  192. return wrapped.getSessionContext().getReadConcern();
  193. }
  194. }
  195. }
  196. }