/spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/convert/DBObjectAccessor.java

http://github.com/SpringSource/spring-data-mongodb · Java · 123 lines · 56 code · 24 blank · 43 comment · 9 complexity · 1b5abacd8693ab7c05fd23e47fd0b939 MD5 · raw file

  1. /*
  2. * Copyright 2013 the original author or authors.
  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 org.springframework.data.mongodb.core.convert;
  17. import java.util.Arrays;
  18. import java.util.Iterator;
  19. import java.util.Map;
  20. import org.springframework.data.mongodb.core.mapping.MongoPersistentProperty;
  21. import org.springframework.util.Assert;
  22. import com.mongodb.BasicDBObject;
  23. import com.mongodb.DBObject;
  24. /**
  25. * Wrapper value object for a {@link BasicDBObject} to be able to access raw values by {@link MongoPersistentProperty}
  26. * references. The accessors will transparently resolve nested document values that a {@link MongoPersistentProperty}
  27. * might refer to through a path expression in field names.
  28. *
  29. * @author Oliver Gierke
  30. */
  31. class DBObjectAccessor {
  32. private final DBObject dbObject;
  33. /**
  34. * Creates a new {@link DBObjectAccessor} for the given {@link DBObject}.
  35. *
  36. * @param dbObject must be a {@link BasicDBObject} effectively, must not be {@literal null}.
  37. */
  38. public DBObjectAccessor(DBObject dbObject) {
  39. Assert.notNull(dbObject, "DBObject must not be null!");
  40. Assert.isInstanceOf(BasicDBObject.class, dbObject, "Given DBObject must be a BasicDBObject!");
  41. this.dbObject = dbObject;
  42. }
  43. /**
  44. * Puts the given value into the backing {@link DBObject} based on the coordinates defined through the given
  45. * {@link MongoPersistentProperty}. By default this will be the plain field name. But field names might also consist
  46. * of path traversals so we might need to create intermediate {@link BasicDBObject}s.
  47. *
  48. * @param prop must not be {@literal null}.
  49. * @param value
  50. */
  51. public void put(MongoPersistentProperty prop, Object value) {
  52. Assert.notNull(prop, "MongoPersistentProperty must not be null!");
  53. String fieldName = prop.getFieldName();
  54. Iterator<String> parts = Arrays.asList(fieldName.split("\\.")).iterator();
  55. DBObject dbObject = this.dbObject;
  56. while (parts.hasNext()) {
  57. String part = parts.next();
  58. if (parts.hasNext()) {
  59. BasicDBObject nestedDbObject = new BasicDBObject();
  60. dbObject.put(part, nestedDbObject);
  61. dbObject = nestedDbObject;
  62. } else {
  63. dbObject.put(part, value);
  64. }
  65. }
  66. }
  67. /**
  68. * Returns the value the given {@link MongoPersistentProperty} refers to. By default this will be a direct field but
  69. * the method will also transparently resolve nested values the {@link MongoPersistentProperty} might refer to through
  70. * a path expression in the field name metadata.
  71. *
  72. * @param property must not be {@literal null}.
  73. * @return
  74. */
  75. @SuppressWarnings("unchecked")
  76. public Object get(MongoPersistentProperty property) {
  77. String fieldName = property.getFieldName();
  78. Iterator<String> parts = Arrays.asList(fieldName.split("\\.")).iterator();
  79. Map<Object, Object> source = this.dbObject.toMap();
  80. Object result = null;
  81. while (source != null && parts.hasNext()) {
  82. result = source.get(parts.next());
  83. if (parts.hasNext()) {
  84. source = getAsMap(result);
  85. }
  86. }
  87. return result;
  88. }
  89. @SuppressWarnings("unchecked")
  90. private Map<Object, Object> getAsMap(Object source) {
  91. if (source instanceof BasicDBObject) {
  92. return ((DBObject) source).toMap();
  93. }
  94. if (source instanceof Map) {
  95. return (Map<Object, Object>) source;
  96. }
  97. return null;
  98. }
  99. }