/src/main/java/com/velix/jmongo/CursorIterator.java

https://github.com/velna/jmongo · Java · 218 lines · 182 code · 17 blank · 19 comment · 46 complexity · 43a3829fb73ed51a11d43b038cc8d07e MD5 · raw file

  1. /**
  2. * JMongo is a mongodb driver writtern in java.
  3. * Copyright (C) 2010 Xiaohu Huang
  4. *
  5. * JMongo is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * JMongo is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with JMongo. If not, see <http://www.gnu.org/licenses/>.
  17. */
  18. package com.velix.jmongo;
  19. import java.io.IOException;
  20. import java.util.ArrayList;
  21. import java.util.Iterator;
  22. import java.util.List;
  23. import java.util.NoSuchElementException;
  24. import org.apache.log4j.Logger;
  25. import com.velix.bson.BSONDocument;
  26. import com.velix.jmongo.protocol.GetMoreMessage;
  27. import com.velix.jmongo.protocol.KillCursorsMessage;
  28. import com.velix.jmongo.protocol.QueryMessage;
  29. import com.velix.jmongo.protocol.ReplyMessage;
  30. public class CursorIterator<T extends BSONDocument> implements Iterator<T> {
  31. private final static Logger LOG = Logger.getLogger(CursorIterator.class);
  32. private Boolean hasNext = null;
  33. private ReplyMessage<T> replyMessage;
  34. private QueryMessage queryMessage;
  35. private Iterator<T> resultIterator;
  36. private ConnectionPool pool;
  37. private boolean getMore = false;
  38. private boolean closed = false;
  39. private Cursor<T> cursor;
  40. private int numberReturned;
  41. private int numberToReturn;
  42. public CursorIterator(ConnectionPool pool, Cursor<T> cursor) {
  43. this.pool = pool;
  44. this.cursor = cursor;
  45. prepareQueryMessage();
  46. }
  47. private void prepareQueryMessage() {
  48. // cal numberToReturn
  49. numberToReturn = cursor.getLimit();
  50. if (cursor.getBatchSize() > 0) {
  51. if (numberToReturn == 0) {
  52. numberToReturn = cursor.getBatchSize();
  53. } else {
  54. numberToReturn = Math
  55. .min(numberToReturn, cursor.getBatchSize());
  56. }
  57. }
  58. // prepare queryMessage
  59. queryMessage = new QueryMessage();
  60. queryMessage
  61. .setFullCollectionName(cursor.getCollection().getFullName());
  62. queryMessage.setNumberToReturn(numberToReturn);
  63. queryMessage.setNumberToSkip(cursor.getSkip());
  64. if (cursor.isExplain() || cursor.isSnapshot()
  65. || (null != cursor.getSort() && !cursor.getSort().isEmpty())
  66. || null != cursor.getHint()) {
  67. BSONDocument query = new MongoDocument();
  68. query.put("query", cursor.getQuery());
  69. if (null != cursor.getSort() && !cursor.getSort().isEmpty()) {
  70. query.put("orderby", cursor.getSort());
  71. }
  72. if (cursor.isExplain()) {
  73. query.put("$explain", true);
  74. }
  75. if (cursor.isSnapshot()) {
  76. query.put("$snapshot", true);
  77. }
  78. if (null != cursor.getHint()) {
  79. query.put("$hint", cursor.getHint());
  80. }
  81. queryMessage.setQuery(query);
  82. } else {
  83. queryMessage.setQuery(cursor.getQuery());
  84. }
  85. queryMessage.setReturnFieldSelector(cursor.getFields());
  86. queryMessage.setTailable(cursor.isTailableCursor());
  87. queryMessage.setNoCursorTimeout(cursor.isNoCursorTimeout());
  88. queryMessage.setSlaveOk(cursor.isSlaveOk());
  89. queryMessage.setAwaitData(cursor.isAwaitData());
  90. }
  91. @SuppressWarnings("unchecked")
  92. @Override
  93. public boolean hasNext() throws IllegalStateException {
  94. if (closed) {
  95. return false;
  96. }
  97. if (Boolean.FALSE == hasNext) {
  98. return false;
  99. }
  100. if (cursor.getLimit() > 0 && cursor.getLimit() <= this.numberReturned) {
  101. return false;
  102. }
  103. if (null != resultIterator) {
  104. if (resultIterator.hasNext()) {
  105. hasNext = Boolean.TRUE;
  106. } else if (replyMessage.getCursorID() == 0) {
  107. hasNext = Boolean.FALSE;
  108. }
  109. } else {
  110. try {
  111. Connection connection = getConnection();
  112. try {
  113. if (getMore) {
  114. GetMoreMessage getMoreMessage = new GetMoreMessage();
  115. getMoreMessage.setFullCollectionName(cursor
  116. .getCollection().getFullName());
  117. getMoreMessage.setNumberToReturn(numberToReturn);
  118. getMoreMessage.setCursorID(replyMessage.getCursorID());
  119. connection.send(getMoreMessage);
  120. } else {
  121. connection.send(queryMessage);
  122. getMore = true;
  123. }
  124. replyMessage = (ReplyMessage<T>) connection.receive(cursor
  125. .getCollection().getObjectClass());
  126. if (null != replyMessage) {
  127. resultIterator = replyMessage.getDocuments().iterator();
  128. } else {
  129. resultIterator = null;
  130. }
  131. if (null != resultIterator) {
  132. hasNext = resultIterator.hasNext();
  133. } else {
  134. hasNext = Boolean.FALSE;
  135. }
  136. } finally {
  137. connection.close();
  138. }
  139. } catch (IOException e) {
  140. throw new MongoException(e);
  141. }
  142. }
  143. return hasNext;
  144. }
  145. @Override
  146. public T next() throws IllegalStateException {
  147. check();
  148. if (hasNext()) {
  149. numberReturned++;
  150. T ret = resultIterator.next();
  151. if (ret instanceof MongoCollectionAware) {
  152. ((MongoCollectionAware) ret).setMongoCollection(cursor
  153. .getCollection());
  154. }
  155. return ret;
  156. }
  157. throw new NoSuchElementException("no more documents");
  158. }
  159. @Override
  160. public void remove() throws UnsupportedOperationException {
  161. throw new UnsupportedOperationException("remove() is not supported yet");
  162. }
  163. public void close() throws IOException {
  164. closed = true;
  165. killCursor();
  166. }
  167. private void killCursor() throws IOException {
  168. if (null != replyMessage && replyMessage.getCursorID() != 0) {
  169. Connection connection = getConnection();
  170. try {
  171. KillCursorsMessage message = new KillCursorsMessage();
  172. message.setNumberOfCursorIDs(1);
  173. List<Long> cursorIdList = new ArrayList<Long>(1);
  174. cursorIdList.add(replyMessage.getCursorID());
  175. message.setCursorIDs(cursorIdList);
  176. connection.send(message);
  177. } finally {
  178. connection.close();
  179. }
  180. }
  181. }
  182. private Connection getConnection() throws IOException {
  183. Connection connection = pool.getConnection();
  184. this.cursor.getCollection().getDB().authenticate(connection);
  185. return connection;
  186. }
  187. private void check() {
  188. if (closed) {
  189. throw new IllegalStateException("cursor is already closed");
  190. }
  191. }
  192. @Override
  193. protected void finalize() throws Throwable {
  194. super.finalize();
  195. try {
  196. killCursor();
  197. } catch (Throwable e) {
  198. LOG.error("error when kill cursors: ", e);
  199. }
  200. }
  201. }