/driver-reactive-streams/src/main/com/mongodb/reactivestreams/client/internal/GridFSDownloadPublisherImpl.java

http://github.com/mongodb/mongo-java-driver · Java · 295 lines · 244 code · 34 blank · 17 comment · 31 complexity · 3b2c804a01bc6b7b2c8cb7af00128db5 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.reactivestreams.client.internal;
  17. import com.mongodb.Block;
  18. import com.mongodb.client.gridfs.model.GridFSFile;
  19. import com.mongodb.internal.async.SingleResultCallback;
  20. import com.mongodb.internal.async.client.gridfs.AsyncGridFSDownloadStream;
  21. import com.mongodb.reactivestreams.client.gridfs.GridFSDownloadPublisher;
  22. import org.reactivestreams.Publisher;
  23. import org.reactivestreams.Subscriber;
  24. import org.reactivestreams.Subscription;
  25. import java.nio.Buffer;
  26. import java.nio.ByteBuffer;
  27. public class GridFSDownloadPublisherImpl implements GridFSDownloadPublisher {
  28. private final AsyncGridFSDownloadStream gridFSDownloadStream;
  29. private int bufferSizeBytes;
  30. GridFSDownloadPublisherImpl(final AsyncGridFSDownloadStream gridFSDownloadStream) {
  31. this.gridFSDownloadStream = gridFSDownloadStream;
  32. }
  33. @Override
  34. public Publisher<GridFSFile> getGridFSFile() {
  35. return Publishers.publish(gridFSDownloadStream::getGridFSFile);
  36. }
  37. @Override
  38. public GridFSDownloadPublisher bufferSizeBytes(final int bufferSizeBytes) {
  39. this.bufferSizeBytes = bufferSizeBytes;
  40. return this;
  41. }
  42. @Override
  43. public void subscribe(final Subscriber<? super ByteBuffer> s) {
  44. s.onSubscribe(new GridFSDownloadSubscription(s));
  45. }
  46. class GridFSDownloadSubscription implements Subscription {
  47. private final Subscriber<? super ByteBuffer> outerSubscriber;
  48. /* protected by `this` */
  49. private GridFSFile gridFSFile;
  50. private long sizeRead = 0;
  51. private long requested = 0;
  52. private boolean unsubscribed;
  53. private int currentBatchSize = 0;
  54. private Action currentAction = Action.WAITING;
  55. /* protected by `this` */
  56. GridFSDownloadSubscription(final Subscriber<? super ByteBuffer> outerSubscriber) {
  57. this.outerSubscriber = outerSubscriber;
  58. }
  59. private final Subscriber<GridFSFile> gridFSFileSubscriber = new Subscriber<GridFSFile>() {
  60. @Override
  61. public void onSubscribe(final Subscription s) {
  62. s.request(1);
  63. }
  64. @Override
  65. public void onNext(final GridFSFile result) {
  66. synchronized (GridFSDownloadSubscription.this) {
  67. gridFSFile = result;
  68. }
  69. }
  70. @Override
  71. public void onError(final Throwable t) {
  72. outerSubscriber.onError(t);
  73. terminate();
  74. }
  75. @Override
  76. public void onComplete() {
  77. synchronized (GridFSDownloadSubscription.this) {
  78. currentAction = Action.WAITING;
  79. }
  80. tryProcess();
  81. }
  82. };
  83. class GridFSDownloadStreamSubscriber implements Subscriber<Integer> {
  84. private final ByteBuffer byteBuffer;
  85. GridFSDownloadStreamSubscriber(final ByteBuffer byteBuffer) {
  86. this.byteBuffer = byteBuffer;
  87. }
  88. @Override
  89. public void onSubscribe(final Subscription s) {
  90. s.request(1);
  91. }
  92. @Override
  93. public void onNext(final Integer integer) {
  94. synchronized (GridFSDownloadSubscription.this) {
  95. sizeRead += integer;
  96. }
  97. }
  98. @Override
  99. public void onError(final Throwable t) {
  100. terminate();
  101. outerSubscriber.onError(t);
  102. }
  103. @Override
  104. public void onComplete() {
  105. if (byteBuffer.remaining() > 0) {
  106. Publishers.publish((Block<SingleResultCallback<Integer>>) callback -> gridFSDownloadStream.read(byteBuffer, callback))
  107. .subscribe(new GridFSDownloadStreamSubscriber(byteBuffer));
  108. } else {
  109. boolean hasTerminated;
  110. synchronized (GridFSDownloadSubscription.this) {
  111. hasTerminated = currentAction == Action.TERMINATE || currentAction == Action.FINISHED;
  112. if (!hasTerminated) {
  113. currentAction = Action.WAITING;
  114. if (sizeRead == gridFSFile.getLength()) {
  115. currentAction = Action.COMPLETE;
  116. }
  117. }
  118. }
  119. if (!hasTerminated) {
  120. ((Buffer) byteBuffer).flip();
  121. outerSubscriber.onNext(byteBuffer);
  122. tryProcess();
  123. }
  124. }
  125. }
  126. }
  127. @Override
  128. public void request(final long n) {
  129. boolean isUnsubscribed;
  130. synchronized (this) {
  131. isUnsubscribed = unsubscribed;
  132. if (!isUnsubscribed && n < 1) {
  133. currentAction = Action.FINISHED;
  134. } else {
  135. requested += n;
  136. }
  137. }
  138. if (!isUnsubscribed && n < 1) {
  139. outerSubscriber.onError(new IllegalArgumentException("3.9 While the Subscription is not cancelled, "
  140. + "Subscription.request(long n) MUST throw a java.lang.IllegalArgumentException if the "
  141. + "argument is <= 0."));
  142. return;
  143. }
  144. tryProcess();
  145. }
  146. @Override
  147. public void cancel() {
  148. synchronized (this) {
  149. unsubscribed = true;
  150. }
  151. terminate();
  152. }
  153. private void tryProcess() {
  154. NextStep nextStep;
  155. synchronized (this) {
  156. switch (currentAction) {
  157. case WAITING:
  158. if (requested == 0) {
  159. nextStep = NextStep.DO_NOTHING;
  160. } else if (gridFSFile == null) {
  161. nextStep = NextStep.GET_FILE;
  162. currentAction = Action.IN_PROGRESS;
  163. } else if (sizeRead == gridFSFile.getLength()) {
  164. nextStep = NextStep.COMPLETE;
  165. currentAction = Action.FINISHED;
  166. } else {
  167. requested--;
  168. nextStep = NextStep.READ;
  169. currentAction = Action.IN_PROGRESS;
  170. }
  171. break;
  172. case COMPLETE:
  173. nextStep = NextStep.COMPLETE;
  174. currentAction = Action.FINISHED;
  175. break;
  176. case TERMINATE:
  177. nextStep = NextStep.TERMINATE;
  178. currentAction = Action.FINISHED;
  179. break;
  180. case IN_PROGRESS:
  181. case FINISHED:
  182. default:
  183. nextStep = NextStep.DO_NOTHING;
  184. break;
  185. }
  186. }
  187. switch (nextStep) {
  188. case GET_FILE:
  189. getGridFSFile().subscribe(gridFSFileSubscriber);
  190. break;
  191. case READ:
  192. int chunkSize;
  193. long remaining;
  194. synchronized (this) {
  195. chunkSize = gridFSFile.getChunkSize();
  196. remaining = gridFSFile.getLength() - sizeRead;
  197. }
  198. int byteBufferSize = Math.max(chunkSize, bufferSizeBytes);
  199. if (remaining < Integer.MAX_VALUE) {
  200. byteBufferSize = Math.min(Long.valueOf(remaining).intValue(), byteBufferSize);
  201. }
  202. ByteBuffer byteBuffer = ByteBuffer.allocate(byteBufferSize);
  203. if (currentBatchSize == 0) {
  204. currentBatchSize = Math.max(byteBufferSize / chunkSize, 1);
  205. gridFSDownloadStream.batchSize(currentBatchSize);
  206. }
  207. Publishers.publish((Block<SingleResultCallback<Integer>>) callback -> gridFSDownloadStream.read(byteBuffer, callback))
  208. .subscribe(new GridFSDownloadStreamSubscriber(byteBuffer));
  209. break;
  210. case COMPLETE:
  211. case TERMINATE:
  212. final boolean propagateToOuter = nextStep == NextStep.COMPLETE;
  213. Publishers.publish(gridFSDownloadStream::close).subscribe(new Subscriber<Void>() {
  214. @Override
  215. public void onSubscribe(final Subscription s) {
  216. s.request(1);
  217. }
  218. @Override
  219. public void onNext(final Void result) {
  220. }
  221. @Override
  222. public void onError(final Throwable t) {
  223. if (propagateToOuter) {
  224. outerSubscriber.onError(t);
  225. }
  226. }
  227. @Override
  228. public void onComplete() {
  229. if (propagateToOuter) {
  230. outerSubscriber.onComplete();
  231. }
  232. }
  233. });
  234. break;
  235. case DO_NOTHING:
  236. default:
  237. break;
  238. }
  239. }
  240. private void terminate() {
  241. synchronized (this) {
  242. currentAction = Action.TERMINATE;
  243. }
  244. tryProcess();
  245. }
  246. }
  247. enum Action {
  248. WAITING,
  249. IN_PROGRESS,
  250. TERMINATE,
  251. COMPLETE,
  252. FINISHED
  253. }
  254. enum NextStep {
  255. GET_FILE,
  256. READ,
  257. COMPLETE,
  258. TERMINATE,
  259. DO_NOTHING
  260. }
  261. }