| 1 |  |  package org.apache.ojb.broker.metadata; | 
  | 2 |  |   | 
  | 3 |  |   | 
  | 4 |  |   | 
  | 5 |  |   | 
  | 6 |  |   | 
  | 7 |  |   | 
  | 8 |  |   | 
  | 9 |  |   | 
  | 10 |  |   | 
  | 11 |  |   | 
  | 12 |  |   | 
  | 13 |  |   | 
  | 14 |  |   | 
  | 15 |  |   | 
  | 16 |  |   | 
  | 17 |  |   | 
  | 18 |  |  import java.util.List; | 
  | 19 |  |  import java.util.Map; | 
  | 20 |  |  import java.util.HashMap; | 
  | 21 |  |   | 
  | 22 |  |  import org.apache.commons.lang.SystemUtils; | 
  | 23 |  |  import org.apache.ojb.broker.metadata.fieldaccess.AnonymousPersistentField; | 
  | 24 |  |  import org.apache.ojb.broker.metadata.fieldaccess.PersistentField; | 
  | 25 |  |  import org.apache.ojb.broker.metadata.fieldaccess.PersistentFieldFactory; | 
  | 26 |  |  import org.apache.ojb.broker.util.ClassHelper; | 
  | 27 |  |  import org.apache.ojb.broker.util.logging.Logger; | 
  | 28 |  |  import org.apache.ojb.broker.util.logging.LoggerFactory; | 
  | 29 |  |   | 
  | 30 |  |   | 
  | 31 |  |   | 
  | 32 |  |   | 
  | 33 |  |   | 
  | 34 |  |   | 
  | 35 |  |   | 
  | 36 |  |  public class SuperReferenceDescriptor extends ObjectReferenceDescriptor | 
  | 37 |  |  { | 
  | 38 |  |      private transient Logger log; | 
  | 39 |  |   | 
  | 40 |  |      public static final String SUPER_FIELD_INTERNAL_NAME = "ojbSuperFieldInternal"; | 
  | 41 |  |      public static final String SUPER_FIELD_NAME = RepositoryElements.TAG_SUPER; | 
  | 42 |  |   | 
  | 43 |  |      private Boolean javaInheritance; | 
  | 44 |  |      private Map declaredInheritanceFields = new HashMap(); | 
  | 45 |  |   | 
  | 46 |  |      public SuperReferenceDescriptor(ClassDescriptor descriptor) | 
  | 47 |  |      { | 
  | 48 |  |          super(descriptor); | 
  | 49 |  |           | 
  | 50 |  |          super.setPersistentField(new SuperReferenceField(this)); | 
  | 51 |  |           | 
  | 52 |  |          super.setLazy(false); | 
  | 53 |  |          super.setCascadeRetrieve(true); | 
  | 54 |  |          super.setCascadingStore(CASCADE_OBJECT); | 
  | 55 |  |          super.setCascadingDelete(CASCADE_OBJECT); | 
  | 56 |  |      } | 
  | 57 |  |   | 
  | 58 |  |      public boolean isSuperReferenceDescriptor() | 
  | 59 |  |      { | 
  | 60 |  |          return true; | 
  | 61 |  |      } | 
  | 62 |  |   | 
  | 63 |  |      public void setItemClass(Class c) | 
  | 64 |  |      { | 
  | 65 |  |          super.setItemClass(c); | 
  | 66 |  |          getClassDescriptor().setBaseClass(c.getName()); | 
  | 67 |  |      } | 
  | 68 |  |   | 
  | 69 |  |       | 
  | 70 |  |   | 
  | 71 |  |   | 
  | 72 |  |   | 
  | 73 |  |      public void setPersistentField(Class c, String fieldname) | 
  | 74 |  |      { | 
  | 75 |  |           | 
  | 76 |  |      } | 
  | 77 |  |   | 
  | 78 |  |       | 
  | 79 |  |   | 
  | 80 |  |   | 
  | 81 |  |   | 
  | 82 |  |      public void setPersistentField(PersistentField pf) | 
  | 83 |  |      { | 
  | 84 |  |           | 
  | 85 |  |      } | 
  | 86 |  |   | 
  | 87 |  |      public void setLazy(boolean lazy) | 
  | 88 |  |      { | 
  | 89 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 90 |  |      } | 
  | 91 |  |   | 
  | 92 |  |      public void setCascadeRetrieve(boolean b) | 
  | 93 |  |      { | 
  | 94 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 95 |  |      } | 
  | 96 |  |   | 
  | 97 |  |      public void setCascadingStore(int cascade) | 
  | 98 |  |      { | 
  | 99 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 100 |  |      } | 
  | 101 |  |   | 
  | 102 |  |      public void setCascadingStore(String value) | 
  | 103 |  |      { | 
  | 104 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 105 |  |      } | 
  | 106 |  |   | 
  | 107 |  |      public void setCascadingDelete(int cascade) | 
  | 108 |  |      { | 
  | 109 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 110 |  |      } | 
  | 111 |  |   | 
  | 112 |  |      public void setCascadingDelete(String value) | 
  | 113 |  |      { | 
  | 114 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 115 |  |      } | 
  | 116 |  |   | 
  | 117 |  |      public void setCascadeStore(boolean cascade) | 
  | 118 |  |      { | 
  | 119 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 120 |  |      } | 
  | 121 |  |   | 
  | 122 |  |      public void setCascadeDelete(boolean cascade) | 
  | 123 |  |      { | 
  | 124 |  |          getLog().info("Not allowed to change this property, will ignore setting"); | 
  | 125 |  |      } | 
  | 126 |  |   | 
  | 127 |  |      public SuperReferenceField getInheritanceField() | 
  | 128 |  |      { | 
  | 129 |  |          return (SuperReferenceField) getPersistentField(); | 
  | 130 |  |      } | 
  | 131 |  |   | 
  | 132 |  |       | 
  | 133 |  |   | 
  | 134 |  |   | 
  | 135 |  |   | 
  | 136 |  |   | 
  | 137 |  |   | 
  | 138 |  |      public boolean isJavaInheritance() | 
  | 139 |  |      { | 
  | 140 |  |          if(javaInheritance == null) | 
  | 141 |  |          { | 
  | 142 |  |              javaInheritance = getClassDescriptor().getSuperClassDescriptor().getClassOfObject() | 
  | 143 |  |                      .isAssignableFrom(getClassDescriptor().getClassOfObject()) ? Boolean.TRUE : Boolean.FALSE; | 
  | 144 |  |          } | 
  | 145 |  |          return javaInheritance.booleanValue(); | 
  | 146 |  |      } | 
  | 147 |  |   | 
  | 148 |  |      synchronized PersistentField getDeclaredInheritanceField(Class target, String name) | 
  | 149 |  |      { | 
  | 150 |  |          Map fields = (HashMap) declaredInheritanceFields.get(target); | 
  | 151 |  |          if(fields == null) | 
  | 152 |  |          { | 
  | 153 |  |              fields = new HashMap(); | 
  | 154 |  |              declaredInheritanceFields.put(target, fields); | 
  | 155 |  |          } | 
  | 156 |  |          PersistentField pf = (PersistentField) fields.get(name); | 
  | 157 |  |          if(pf == null) | 
  | 158 |  |          { | 
  | 159 |  |              pf = PersistentFieldFactory.createPersistentField(target, name); | 
  | 160 |  |               | 
  | 161 |  |              fields.put(name, pf); | 
  | 162 |  |          } | 
  | 163 |  |          return pf; | 
  | 164 |  |      } | 
  | 165 |  |   | 
  | 166 |  |      private Logger getLog() | 
  | 167 |  |      { | 
  | 168 |  |          if(log == null) | 
  | 169 |  |          { | 
  | 170 |  |              log = LoggerFactory.getLogger(SuperReferenceField.class); | 
  | 171 |  |          } | 
  | 172 |  |          return log; | 
  | 173 |  |      } | 
  | 174 |  |   | 
  | 175 |  |   | 
  | 176 |  |       | 
  | 177 |  |       | 
  | 178 |  |       | 
  | 179 |  |   | 
  | 180 |  |      public static final class SuperReferenceField extends AnonymousPersistentField | 
  | 181 |  |      { | 
  | 182 |  |          private transient Logger log; | 
  | 183 |  |   | 
  | 184 |  |          private SuperReferenceDescriptor superRef; | 
  | 185 |  |   | 
  | 186 |  |          public SuperReferenceField(SuperReferenceDescriptor superRef) | 
  | 187 |  |          { | 
  | 188 |  |              super(SUPER_FIELD_INTERNAL_NAME); | 
  | 189 |  |              this.superRef = superRef; | 
  | 190 |  |          } | 
  | 191 |  |   | 
  | 192 |  |          private Logger getLog() | 
  | 193 |  |          { | 
  | 194 |  |              if(log == null) | 
  | 195 |  |              { | 
  | 196 |  |                  log = LoggerFactory.getLogger(SuperReferenceField.class); | 
  | 197 |  |              } | 
  | 198 |  |              return log; | 
  | 199 |  |          } | 
  | 200 |  |   | 
  | 201 |  |           | 
  | 202 |  |   | 
  | 203 |  |   | 
  | 204 |  |   | 
  | 205 |  |   | 
  | 206 |  |   | 
  | 207 |  |   | 
  | 208 |  |   | 
  | 209 |  |          public synchronized void set(Object target, Object value) throws MetadataException | 
  | 210 |  |          { | 
  | 211 |  |               | 
  | 212 |  |              ClassDescriptor superCld = superRef.getClassDescriptor().getSuperClassDescriptor(); | 
  | 213 |  |              if(superRef.isJavaInheritance()) | 
  | 214 |  |              { | 
  | 215 |  |                  copyFields(superCld, target, superCld, value, true, true); | 
  | 216 |  |              } | 
  | 217 |  |              else | 
  | 218 |  |              { | 
  | 219 |  |                  copyFields(superRef.getClassDescriptor(), target, superCld, value, false, false); | 
  | 220 |  |              } | 
  | 221 |  |          } | 
  | 222 |  |   | 
  | 223 |  |           | 
  | 224 |  |   | 
  | 225 |  |   | 
  | 226 |  |   | 
  | 227 |  |   | 
  | 228 |  |   | 
  | 229 |  |   | 
  | 230 |  |   | 
  | 231 |  |   | 
  | 232 |  |   | 
  | 233 |  |   | 
  | 234 |  |          public synchronized Object get(Object obj) throws MetadataException | 
  | 235 |  |          { | 
  | 236 |  |              if(obj == null) return null; | 
  | 237 |  |              if(superRef.isJavaInheritance()) | 
  | 238 |  |              { | 
  | 239 |  |                  return obj; | 
  | 240 |  |              } | 
  | 241 |  |              else | 
  | 242 |  |              { | 
  | 243 |  |                  return getObjectWithDeclaredSuperClass(obj); | 
  | 244 |  |              } | 
  | 245 |  |          } | 
  | 246 |  |   | 
  | 247 |  |          private Object getObjectWithDeclaredSuperClass(Object obj) | 
  | 248 |  |          { | 
  | 249 |  |              Object value = getFromFieldCache(obj); | 
  | 250 |  |              if(value == null) | 
  | 251 |  |              { | 
  | 252 |  |                  ClassDescriptor baseCld = null; | 
  | 253 |  |                  try | 
  | 254 |  |                  { | 
  | 255 |  |                      baseCld = superRef.getClassDescriptor().getSuperClassDescriptor(); | 
  | 256 |  |                      value = ClassHelper.buildNewObjectInstance(baseCld); | 
  | 257 |  |                  } | 
  | 258 |  |                  catch(Exception e) | 
  | 259 |  |                  { | 
  | 260 |  |                      throw new MetadataException("Can't create new base class object for '" | 
  | 261 |  |                              + (baseCld != null ? baseCld.getClassNameOfObject() : null) + "'", e); | 
  | 262 |  |                  } | 
  | 263 |  |                  copyFields(baseCld, value, superRef.getClassDescriptor(), obj, true, false); | 
  | 264 |  |                  putToFieldCache(obj, value); | 
  | 265 |  |              } | 
  | 266 |  |              return value; | 
  | 267 |  |          } | 
  | 268 |  |   | 
  | 269 |  |          void copyFields(ClassDescriptor targetCld, Object target, ClassDescriptor sourceCld, Object source, boolean targetIsSuper, boolean javaInheritance) | 
  | 270 |  |          { | 
  | 271 |  |              if(getLog().isDebugEnabled()) | 
  | 272 |  |              { | 
  | 273 |  |                  String msg = ("Copy fields from " + SystemUtils.LINE_SEPARATOR | 
  | 274 |  |                          + "source object '" + (source != null ? source.getClass().getName() : null) + "'" + SystemUtils.LINE_SEPARATOR | 
  | 275 |  |                          + "using source fields declared in '" + sourceCld.getClassNameOfObject() + "'" + SystemUtils.LINE_SEPARATOR | 
  | 276 |  |                          + "to target object '" + (target != null ? target.getClass().getName() : null) + "'" + SystemUtils.LINE_SEPARATOR | 
  | 277 |  |                          + "using target fields declared in '" + targetCld.getClassNameOfObject() + "'" + SystemUtils.LINE_SEPARATOR | 
  | 278 |  |                          + "the fields to copy are declared in '" + (targetIsSuper ? targetCld.getClassNameOfObject() : sourceCld.getClassNameOfObject()) + "' class" + SystemUtils.LINE_SEPARATOR | 
  | 279 |  |                          + "the used classes are associated by java inheritance: " + javaInheritance + SystemUtils.LINE_SEPARATOR); | 
  | 280 |  |                  getLog().debug(msg); | 
  | 281 |  |              } | 
  | 282 |  |               | 
  | 283 |  |   | 
  | 284 |  |   | 
  | 285 |  |   | 
  | 286 |  |   | 
  | 287 |  |   | 
  | 288 |  |   | 
  | 289 |  |   | 
  | 290 |  |   | 
  | 291 |  |   | 
  | 292 |  |              FieldDescriptor[] fields = targetIsSuper ? targetCld.getFieldDescriptions() : sourceCld.getFieldDescriptions(); | 
  | 293 |  |              for(int i = 0; i < fields.length; i++) | 
  | 294 |  |              { | 
  | 295 |  |                  FieldDescriptor field = fields[i]; | 
  | 296 |  |                  if(!field.isAnonymous()) | 
  | 297 |  |                  { | 
  | 298 |  |                      performFieldCopy(target,  targetCld, source, sourceCld, | 
  | 299 |  |                                  field.getPersistentField(), targetIsSuper, javaInheritance); | 
  | 300 |  |                  } | 
  | 301 |  |              } | 
  | 302 |  |              List refs = targetIsSuper ? targetCld.getCollectionDescriptors() : sourceCld.getCollectionDescriptors(); | 
  | 303 |  |              for(int i = 0; i < refs.size(); i++) | 
  | 304 |  |              { | 
  | 305 |  |                  CollectionDescriptor col = (CollectionDescriptor) refs.get(i); | 
  | 306 |  |                  PersistentField pf = col.getPersistentField(); | 
  | 307 |  |                  performFieldCopy(target,  targetCld, source, sourceCld, pf, targetIsSuper, javaInheritance); | 
  | 308 |  |              } | 
  | 309 |  |   | 
  | 310 |  |              refs = targetIsSuper ? targetCld.getObjectReferenceDescriptors() : sourceCld.getObjectReferenceDescriptors(); | 
  | 311 |  |              for(int i = 0; i < refs.size(); i++) | 
  | 312 |  |              { | 
  | 313 |  |                  ObjectReferenceDescriptor ord = (ObjectReferenceDescriptor) refs.get(i); | 
  | 314 |  |                  PersistentField pf = ord.getPersistentField(); | 
  | 315 |  |                  performFieldCopy(target,  targetCld, source, sourceCld, pf, targetIsSuper, javaInheritance); | 
  | 316 |  |              } | 
  | 317 |  |          } | 
  | 318 |  |   | 
  | 319 |  |          private void performFieldCopy(Object target, ClassDescriptor targetCld, Object source, | 
  | 320 |  |                                   ClassDescriptor sourceCld, PersistentField pf, boolean targetIsSuper, boolean javaInheritance) | 
  | 321 |  |          { | 
  | 322 |  |              if(javaInheritance) | 
  | 323 |  |              { | 
  | 324 |  |                  pf.set(target, pf.get(source)); | 
  | 325 |  |              } | 
  | 326 |  |              else | 
  | 327 |  |              { | 
  | 328 |  |                  if(targetIsSuper) | 
  | 329 |  |                  { | 
  | 330 |  |                      if(pf instanceof SuperReferenceField) | 
  | 331 |  |                      { | 
  | 332 |  |                          log.error("Declared inheritance doesn't support nested super references, target '" | 
  | 333 |  |                                  + targetCld.getClassNameOfObject() + "' has super reference"); | 
  | 334 |  |                      } | 
  | 335 |  |                      else | 
  | 336 |  |                      { | 
  | 337 |  |                          PersistentField tmp = superRef.getDeclaredInheritanceField(sourceCld.getClassOfObject(), pf.getName()); | 
  | 338 |  |                          pf.set(target, tmp.get(source)); | 
  | 339 |  |                      } | 
  | 340 |  |                  } | 
  | 341 |  |                  else | 
  | 342 |  |                  { | 
  | 343 |  |                      PersistentField tmp = superRef.getDeclaredInheritanceField(targetCld.getClassOfObject(), pf.getName()); | 
  | 344 |  |                      tmp.set(target, pf.get(source)); | 
  | 345 |  |                  } | 
  | 346 |  |              } | 
  | 347 |  |          } | 
  | 348 |  |      } | 
  | 349 |  |  } | 
  | 350 |  |   |