/driver-core/src/main/com/mongodb/internal/connection/ByteBufBsonDocument.java

http://github.com/mongodb/mongo-java-driver · Java · 173 lines · 139 code · 18 blank · 16 comment · 8 complexity · 05b92305f8aa63ff27558c286d3fdfb6 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.internal.connection;
  17. import org.bson.BsonBinaryReader;
  18. import org.bson.BsonDocument;
  19. import org.bson.BsonReader;
  20. import org.bson.BsonType;
  21. import org.bson.ByteBuf;
  22. import org.bson.ByteBufNIO;
  23. import org.bson.RawBsonDocument;
  24. import org.bson.codecs.BsonDocumentCodec;
  25. import org.bson.codecs.DecoderContext;
  26. import org.bson.io.ByteBufferBsonInput;
  27. import org.bson.json.JsonMode;
  28. import org.bson.json.JsonWriter;
  29. import org.bson.json.JsonWriterSettings;
  30. import java.io.InvalidObjectException;
  31. import java.io.ObjectInputStream;
  32. import java.io.StringWriter;
  33. import java.nio.ByteOrder;
  34. import java.util.ArrayList;
  35. import java.util.List;
  36. class ByteBufBsonDocument extends AbstractByteBufBsonDocument {
  37. private static final long serialVersionUID = 2L;
  38. private final transient ByteBuf byteBuf;
  39. static List<ByteBufBsonDocument> createList(final ResponseBuffers responseBuffers) {
  40. int numDocuments = responseBuffers.getReplyHeader().getNumberReturned();
  41. ByteBuf documentsBuffer = responseBuffers.getBodyByteBuffer();
  42. documentsBuffer.order(ByteOrder.LITTLE_ENDIAN);
  43. List<ByteBufBsonDocument> documents = new ArrayList<ByteBufBsonDocument>(numDocuments);
  44. while (documents.size() < numDocuments) {
  45. int documentSizeInBytes = documentsBuffer.getInt();
  46. documentsBuffer.position(documentsBuffer.position() - 4);
  47. ByteBuf documentBuffer = documentsBuffer.duplicate();
  48. documentBuffer.limit(documentBuffer.position() + documentSizeInBytes);
  49. documents.add(new ByteBufBsonDocument(new ByteBufNIO(documentBuffer.asNIO())));
  50. documentBuffer.release();
  51. documentsBuffer.position(documentsBuffer.position() + documentSizeInBytes);
  52. }
  53. return documents;
  54. }
  55. static List<ByteBufBsonDocument> createList(final ByteBufferBsonOutput bsonOutput, final int startPosition) {
  56. List<ByteBuf> duplicateByteBuffers = bsonOutput.getByteBuffers();
  57. CompositeByteBuf outputByteBuf = new CompositeByteBuf(duplicateByteBuffers);
  58. outputByteBuf.position(startPosition);
  59. List<ByteBufBsonDocument> documents = new ArrayList<ByteBufBsonDocument>();
  60. int curDocumentStartPosition = startPosition;
  61. while (outputByteBuf.hasRemaining()) {
  62. int documentSizeInBytes = outputByteBuf.getInt();
  63. ByteBuf slice = outputByteBuf.duplicate();
  64. slice.position(curDocumentStartPosition);
  65. slice.limit(curDocumentStartPosition + documentSizeInBytes);
  66. documents.add(new ByteBufBsonDocument(slice));
  67. curDocumentStartPosition += documentSizeInBytes;
  68. outputByteBuf.position(outputByteBuf.position() + documentSizeInBytes - 4);
  69. }
  70. for (ByteBuf byteBuffer : duplicateByteBuffers) {
  71. byteBuffer.release();
  72. }
  73. return documents;
  74. }
  75. static ByteBufBsonDocument createOne(final ByteBufferBsonOutput bsonOutput, final int startPosition) {
  76. List<ByteBuf> duplicateByteBuffers = bsonOutput.getByteBuffers();
  77. CompositeByteBuf outputByteBuf = new CompositeByteBuf(duplicateByteBuffers);
  78. outputByteBuf.position(startPosition);
  79. int documentSizeInBytes = outputByteBuf.getInt();
  80. ByteBuf slice = outputByteBuf.duplicate();
  81. slice.position(startPosition);
  82. slice.limit(startPosition + documentSizeInBytes);
  83. for (ByteBuf byteBuffer : duplicateByteBuffers) {
  84. byteBuffer.release();
  85. }
  86. return new ByteBufBsonDocument(slice);
  87. }
  88. @Override
  89. @SuppressWarnings("deprecation")
  90. public String toJson() {
  91. return toJson(JsonWriterSettings.builder().outputMode(JsonMode.RELAXED).build());
  92. }
  93. @Override
  94. public String toJson(final JsonWriterSettings settings) {
  95. StringWriter stringWriter = new StringWriter();
  96. JsonWriter jsonWriter = new JsonWriter(stringWriter, settings);
  97. ByteBuf duplicate = byteBuf.duplicate();
  98. BsonBinaryReader reader = new BsonBinaryReader(new ByteBufferBsonInput(duplicate));
  99. try {
  100. jsonWriter.pipe(reader);
  101. return stringWriter.toString();
  102. } finally {
  103. duplicate.release();
  104. reader.close();
  105. }
  106. }
  107. @Override
  108. public BsonReader asBsonReader() {
  109. return new BsonBinaryReader(new ByteBufferBsonInput(byteBuf.duplicate()));
  110. }
  111. <T> T findInDocument(final Finder<T> finder) {
  112. ByteBuf duplicateByteBuf = byteBuf.duplicate();
  113. BsonBinaryReader bsonReader = new BsonBinaryReader(new ByteBufferBsonInput(byteBuf.duplicate()));
  114. try {
  115. bsonReader.readStartDocument();
  116. while (bsonReader.readBsonType() != BsonType.END_OF_DOCUMENT) {
  117. T found = finder.find(bsonReader);
  118. if (found != null) {
  119. return found;
  120. }
  121. }
  122. bsonReader.readEndDocument();
  123. } finally {
  124. duplicateByteBuf.release();
  125. bsonReader.close();
  126. }
  127. return finder.notFound();
  128. }
  129. @Override
  130. public BsonDocument clone() {
  131. byte[] clonedBytes = new byte[byteBuf.remaining()];
  132. byteBuf.get(byteBuf.position(), clonedBytes);
  133. return new RawBsonDocument(clonedBytes);
  134. }
  135. int getSizeInBytes() {
  136. return byteBuf.getInt(byteBuf.position());
  137. }
  138. BsonDocument toBaseBsonDocument() {
  139. ByteBuf duplicateByteBuf = byteBuf.duplicate();
  140. BsonBinaryReader bsonReader = new BsonBinaryReader(new ByteBufferBsonInput(duplicateByteBuf));
  141. try {
  142. return new BsonDocumentCodec().decode(bsonReader, DecoderContext.builder().build());
  143. } finally {
  144. duplicateByteBuf.release();
  145. bsonReader.close();
  146. }
  147. }
  148. ByteBufBsonDocument(final ByteBuf byteBuf) {
  149. this.byteBuf = byteBuf;
  150. }
  151. // see https://docs.oracle.com/javase/6/docs/platform/serialization/spec/input.html
  152. private void readObject(final ObjectInputStream stream) throws InvalidObjectException {
  153. throw new InvalidObjectException("Proxy required");
  154. }
  155. }