/src/main/com/mongodb/ReflectionDBObject.java

https://github.com/fes/mongo-java-driver · Java · 249 lines · 183 code · 47 blank · 19 comment · 33 complexity · b9d67bd87963e282990722b01da888d0 MD5 · raw file

  1. // ReflectionDBObject.java
  2. /**
  3. * Copyright (C) 2008 10gen Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. package com.mongodb;
  18. import java.util.*;
  19. import java.lang.reflect.*;
  20. import com.mongodb.util.*;
  21. public abstract class ReflectionDBObject implements DBObject {
  22. public Object get( String key ){
  23. return getWrapper().get( this , key );
  24. }
  25. public Set<String> keySet(){
  26. return getWrapper().keySet();
  27. }
  28. /**
  29. * @deprecated
  30. */
  31. public boolean containsKey( String s ){
  32. return containsField( s );
  33. }
  34. public boolean containsField( String s ){
  35. return getWrapper().containsKey( s );
  36. }
  37. public Object put( String key , Object v ){
  38. return getWrapper().set( this , key , v );
  39. }
  40. public void putAll( Map m ){
  41. for ( Object k : m.keySet() ){
  42. put( k.toString() , m.get( k ) );
  43. }
  44. }
  45. public void putAll( DBObject o ){
  46. for ( String k : o.keySet() ){
  47. put( k , o.get( k ) );
  48. }
  49. }
  50. public ObjectId get_id(){
  51. return _id;
  52. }
  53. public void set_id( ObjectId id ){
  54. _id = id;
  55. }
  56. public boolean isPartialObject(){
  57. return false;
  58. }
  59. public Map toMap() {
  60. Map m = new HashMap();
  61. Iterator i = this.keySet().iterator();
  62. while (i.hasNext()) {
  63. Object s = i.next();
  64. m.put(s, this.get(s+""));
  65. }
  66. return m;
  67. }
  68. public void markAsPartialObject(){
  69. throw new RuntimeException( "ReflectionDBObjects can't be partial" );
  70. }
  71. public Object removeField( String key ){
  72. throw new RuntimeException( "can't remove from a ReflectionDBObject" );
  73. }
  74. JavaWrapper getWrapper(){
  75. if ( _wrapper != null )
  76. return _wrapper;
  77. _wrapper = getWrapper( this.getClass() );
  78. return _wrapper;
  79. }
  80. JavaWrapper _wrapper;
  81. ObjectId _id;
  82. public static class JavaWrapper {
  83. JavaWrapper( Class c ){
  84. _class = c;
  85. _name = c.getName();
  86. _fields = new TreeMap<String,FieldInfo>();
  87. for ( Method m : c.getMethods() ){
  88. if ( ! ( m.getName().startsWith( "get" ) || m.getName().startsWith( "set" ) ) )
  89. continue;
  90. String name = m.getName().substring(3);
  91. if ( name.length() == 0 || IGNORE_FIELDS.contains( name ) )
  92. continue;
  93. Class type = m.getName().startsWith( "get" ) ? m.getReturnType() : m.getParameterTypes()[0];
  94. FieldInfo fi = _fields.get( name );
  95. if ( fi == null ){
  96. fi = new FieldInfo( name , type );
  97. _fields.put( name , fi );
  98. }
  99. if ( m.getName().startsWith( "get" ) )
  100. fi._getter = m;
  101. else
  102. fi._setter = m;
  103. }
  104. Set<String> names = new HashSet<String>( _fields.keySet() );
  105. for ( String name : names )
  106. if ( ! _fields.get( name ).ok() )
  107. _fields.remove( name );
  108. _keys = Collections.unmodifiableSet( _fields.keySet() );
  109. }
  110. public Set<String> keySet(){
  111. return _keys;
  112. }
  113. public boolean containsKey( String key ){
  114. return _keys.contains( key );
  115. }
  116. public Object get( ReflectionDBObject t , String name ){
  117. FieldInfo i = _fields.get( name );
  118. if ( i == null )
  119. return null;
  120. try {
  121. return i._getter.invoke( t );
  122. }
  123. catch ( Exception e ){
  124. throw new RuntimeException( "could not invoke getter for [" + name + "] on [" + _name + "]" , e );
  125. }
  126. }
  127. public Object set( ReflectionDBObject t , String name , Object val ){
  128. if ( IGNORE_SETS.contains( name ) )
  129. return null;
  130. FieldInfo i = _fields.get( name );
  131. if ( i == null )
  132. throw new IllegalArgumentException( "no field [" + name + "] on [" + _name + "]" );
  133. try {
  134. return i._setter.invoke( t , val );
  135. }
  136. catch ( Exception e ){
  137. throw new RuntimeException( "could not invoke setter for [" + name + "] on [" + _name + "]" , e );
  138. }
  139. }
  140. Class getInternalClass( String path ){
  141. String cur = path;
  142. String next = null;
  143. final int idx = path.indexOf( "." );
  144. if ( idx >= 0 ){
  145. cur = path.substring( 0 , idx );
  146. next = path.substring( idx + 1 );
  147. }
  148. FieldInfo fi = _fields.get( cur );
  149. if ( fi == null )
  150. return null;
  151. if ( next == null )
  152. return fi._class;
  153. JavaWrapper w = getWrapperIfReflectionObject( fi._class );
  154. if ( w == null )
  155. return null;
  156. return w.getInternalClass( next );
  157. }
  158. final Class _class;
  159. final String _name;
  160. final Map<String,FieldInfo> _fields;
  161. final Set<String> _keys;
  162. }
  163. static class FieldInfo {
  164. FieldInfo( String name , Class c ){
  165. _name = name;
  166. _class = c;
  167. }
  168. boolean ok(){
  169. return
  170. _getter != null &&
  171. _setter != null;
  172. }
  173. final String _name;
  174. final Class _class;
  175. Method _getter;
  176. Method _setter;
  177. }
  178. public static JavaWrapper getWrapperIfReflectionObject( Class c ){
  179. if ( ReflectionDBObject.class.isAssignableFrom( c ) )
  180. return getWrapper( c );
  181. return null;
  182. }
  183. public static JavaWrapper getWrapper( Class c ){
  184. JavaWrapper w = _wrappers.get( c );
  185. if ( w == null ){
  186. w = new JavaWrapper( c );
  187. _wrappers.put( c , w );
  188. }
  189. return w;
  190. }
  191. private static final Map<Class,JavaWrapper> _wrappers = Collections.synchronizedMap( new HashMap<Class,JavaWrapper>() );
  192. private static final Set<String> IGNORE_FIELDS = new HashSet<String>();
  193. static {
  194. IGNORE_FIELDS.add( "Int" );
  195. }
  196. private static final Set<String> IGNORE_SETS = new HashSet<String>();
  197. static {
  198. IGNORE_SETS.add( "_save" );
  199. IGNORE_SETS.add( "_update" );
  200. IGNORE_SETS.add( "_ns" );
  201. }
  202. }