/modello-1.5/modello-core/src/main/java/org/codehaus/modello/model/Model.java

# · Java · 465 lines · 339 code · 90 blank · 36 comment · 55 complexity · 0db62148777d5484a77a1e24cd34638a MD5 · raw file

  1. package org.codehaus.modello.model;
  2. /*
  3. * Copyright (c) 2004, Codehaus.org
  4. *
  5. * Permission is hereby granted, free of charge, to any person obtaining a copy of
  6. * this software and associated documentation files (the "Software"), to deal in
  7. * the Software without restriction, including without limitation the rights to
  8. * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
  9. * of the Software, and to permit persons to whom the Software is furnished to do
  10. * so, subject to the following conditions:
  11. *
  12. * The above copyright notice and this permission notice shall be included in all
  13. * copies or substantial portions of the Software.
  14. *
  15. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  16. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  17. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  18. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  19. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  20. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  21. * SOFTWARE.
  22. */
  23. import org.codehaus.modello.ModelloRuntimeException;
  24. import org.codehaus.modello.metadata.ModelMetadata;
  25. import org.codehaus.modello.plugin.model.ModelClassMetadata;
  26. import org.codehaus.plexus.util.StringUtils;
  27. import java.util.ArrayList;
  28. import java.util.HashMap;
  29. import java.util.List;
  30. import java.util.Map;
  31. /**
  32. * @author <a href="mailto:jason@modello.org">Jason van Zyl</a>
  33. * @author <a href="mailto:evenisse@codehaus.org">Emmanuel Venisse</a>
  34. * @version $Id$
  35. */
  36. public class Model
  37. extends BaseElement
  38. {
  39. private String id;
  40. private List<ModelClass> classes = new ArrayList<ModelClass>();
  41. private List<ModelDefault> defaults = new ArrayList<ModelDefault>();
  42. private List<ModelInterface> interfaces = new ArrayList<ModelInterface>();
  43. private transient Map<String,List<ModelClass>> classMap = new HashMap<String,List<ModelClass>>();
  44. private transient Map<String,ModelDefault> defaultMap = new HashMap<String,ModelDefault>();
  45. private transient Map<String,List<ModelInterface>> interfaceMap = new HashMap<String,List<ModelInterface>>();
  46. private VersionDefinition versionDefinition;
  47. public Model()
  48. {
  49. super( true );
  50. }
  51. public String getId()
  52. {
  53. return id;
  54. }
  55. public void setId( String id )
  56. {
  57. this.id = id;
  58. }
  59. public VersionDefinition getVersionDefinition()
  60. {
  61. return versionDefinition;
  62. }
  63. public void setVersionDefinition( VersionDefinition versionDefinition )
  64. {
  65. this.versionDefinition = versionDefinition;
  66. }
  67. public ModelMetadata getMetadata( String key )
  68. {
  69. return getMetadata( ModelMetadata.class, key );
  70. }
  71. public String getRoot( Version version )
  72. {
  73. List<ModelClass> classes = getClasses( version );
  74. String className = null;
  75. for ( ModelClass currentClass : classes )
  76. {
  77. ModelClassMetadata metadata = null;
  78. try
  79. {
  80. metadata = (ModelClassMetadata) currentClass.getMetadata( ModelClassMetadata.ID );
  81. }
  82. catch ( Exception e )
  83. {
  84. }
  85. if ( metadata != null && metadata.isRootElement() )
  86. {
  87. if ( className == null )
  88. {
  89. className = currentClass.getName();
  90. }
  91. else
  92. {
  93. throw new ModelloRuntimeException(
  94. "There are more than one class as root element for this version " + version + "." );
  95. }
  96. }
  97. }
  98. if ( className == null )
  99. {
  100. throw new ModelloRuntimeException( "There aren't root element for version " + version + "." );
  101. }
  102. return className;
  103. }
  104. /**
  105. * @deprecated This shouldn't be used, anything querying the model should read the
  106. * package of the class. Use getDefaultPackageName(..).
  107. */
  108. public String getPackageName( boolean withVersion, Version version )
  109. {
  110. return getDefaultPackageName( withVersion, version );
  111. }
  112. public List<ModelClass> getAllClasses()
  113. {
  114. return classes;
  115. }
  116. public List<ModelClass> getClasses( Version version )
  117. {
  118. List<ModelClass> classList = new ArrayList<ModelClass>();
  119. for ( ModelClass currentClass : classes )
  120. {
  121. if ( version.inside( currentClass.getVersionRange() ) )
  122. {
  123. classList.add( currentClass );
  124. }
  125. }
  126. return classList;
  127. }
  128. public ModelClass getClass( String type, Version version )
  129. {
  130. return getClass( type, new VersionRange( version ) );
  131. }
  132. public ModelClass getClass( String type, VersionRange versionRange )
  133. {
  134. ModelClass value = getModelClass( type, versionRange );
  135. if ( value != null )
  136. {
  137. return value;
  138. }
  139. throw new ModelloRuntimeException(
  140. "There is no class '" + type + "' in the version range '" + versionRange.toString() + "'." );
  141. }
  142. public boolean hasClass( String type, Version version )
  143. {
  144. ModelClass value = getModelClass( type, new VersionRange( version ) );
  145. return value != null;
  146. }
  147. private ModelClass getModelClass( String type, VersionRange versionRange )
  148. {
  149. List<ModelClass> classList = classMap.get( type );
  150. ModelClass value = null;
  151. if ( classList != null )
  152. {
  153. for ( ModelClass modelClass : classList )
  154. {
  155. if ( versionRange.getFromVersion().inside( modelClass.getVersionRange() )
  156. && versionRange.getToVersion().inside( modelClass.getVersionRange() ) )
  157. {
  158. value = modelClass;
  159. }
  160. }
  161. }
  162. return value;
  163. }
  164. public void addClass( ModelClass modelClass )
  165. {
  166. if ( classMap.containsKey( modelClass.getName() ) )
  167. {
  168. List<ModelClass> classList = classMap.get( modelClass.getName() );
  169. for ( ModelClass currentClass : classList )
  170. {
  171. if ( VersionUtil.isInConflict( modelClass.getVersionRange(), currentClass.getVersionRange() ) )
  172. {
  173. throw new ModelloRuntimeException( "Duplicate class: " + modelClass.getName() + "." );
  174. }
  175. }
  176. }
  177. else
  178. {
  179. List<ModelClass> classList = new ArrayList<ModelClass>();
  180. classMap.put( modelClass.getName(), classList );
  181. }
  182. getAllClasses().add( modelClass );
  183. classMap.get( modelClass.getName() ).add( modelClass );
  184. }
  185. // ----------------------------------------------------------------------
  186. // Defaults
  187. // ----------------------------------------------------------------------
  188. public List<ModelDefault> getDefaults()
  189. {
  190. return defaults;
  191. }
  192. public ModelDefault getDefault( String key )
  193. {
  194. ModelDefault modelDefault = (ModelDefault) defaultMap.get( key );
  195. if ( modelDefault == null )
  196. {
  197. try
  198. {
  199. modelDefault = ModelDefault.getDefault( key );
  200. }
  201. catch ( ModelValidationException mve )
  202. {
  203. throw new ModelloRuntimeException( mve.getMessage(), mve);
  204. }
  205. }
  206. return modelDefault;
  207. }
  208. public void addDefault( ModelDefault modelDefault )
  209. {
  210. if ( defaultMap.containsKey( modelDefault.getKey() ) )
  211. {
  212. throw new ModelloRuntimeException( "Duplicate default: " + modelDefault.getKey() + "." );
  213. }
  214. getDefaults().add( modelDefault );
  215. defaultMap.put( modelDefault.getKey(), modelDefault );
  216. }
  217. public String getDefaultPackageName( boolean withVersion, Version version )
  218. {
  219. String packageName = getDefault( ModelDefault.PACKAGE ).getValue();
  220. if ( withVersion )
  221. {
  222. packageName += "." + version.toString( "v", "_" );
  223. }
  224. return packageName;
  225. }
  226. // ----------------------------------------------------------------------
  227. //
  228. // ----------------------------------------------------------------------
  229. public List<ModelInterface> getAllInterfaces()
  230. {
  231. return interfaces;
  232. }
  233. public List<ModelInterface> getInterfaces( Version version )
  234. {
  235. List<ModelInterface> interfaceList = new ArrayList<ModelInterface>();
  236. for ( ModelInterface currentInterface : interfaces )
  237. {
  238. if ( version.inside( currentInterface.getVersionRange() ) )
  239. {
  240. interfaceList.add( currentInterface );
  241. }
  242. }
  243. return interfaceList;
  244. }
  245. public ModelInterface getInterface( String type, Version version )
  246. {
  247. return getInterface( type, new VersionRange( version ) );
  248. }
  249. public ModelInterface getInterface( String type, VersionRange versionRange )
  250. {
  251. ModelInterface value = getModelInterface( type, versionRange );
  252. if ( value != null )
  253. {
  254. return value;
  255. }
  256. throw new ModelloRuntimeException(
  257. "There is no interface '" + type + "' in the version range '" + versionRange.toString() + "'." );
  258. }
  259. private ModelInterface getModelInterface( String type, VersionRange versionRange )
  260. {
  261. List<ModelInterface> interfaceList = interfaceMap.get( type );
  262. if ( interfaceList != null )
  263. {
  264. for ( ModelInterface modelInterface : interfaceList )
  265. {
  266. if ( versionRange.getFromVersion().inside( modelInterface.getVersionRange() )
  267. && versionRange.getToVersion().inside( modelInterface.getVersionRange() ) )
  268. {
  269. return modelInterface;
  270. }
  271. }
  272. }
  273. return null;
  274. }
  275. public void addInterface( ModelInterface modelInterface )
  276. {
  277. if ( interfaceMap.containsKey( modelInterface.getName() ) )
  278. {
  279. List<ModelInterface> interfaceList = interfaceMap.get( modelInterface.getName() );
  280. for ( ModelInterface currentInterface : interfaceList )
  281. {
  282. if ( VersionUtil.isInConflict( modelInterface.getVersionRange(), currentInterface.getVersionRange() ) )
  283. {
  284. throw new ModelloRuntimeException( "Duplicate interface: " + modelInterface.getName() + "." );
  285. }
  286. }
  287. }
  288. else
  289. {
  290. List<ModelInterface> interfaceList = new ArrayList<ModelInterface>();
  291. interfaceMap.put( modelInterface.getName(), interfaceList );
  292. }
  293. getAllInterfaces().add( modelInterface );
  294. interfaceMap.get( modelInterface.getName() ).add( modelInterface );
  295. }
  296. public ModelType getType( String type, Version version )
  297. {
  298. return getType( type, new VersionRange( version ) );
  299. }
  300. public ModelType getType( String type, VersionRange versionRange )
  301. {
  302. ModelType value = getModelClass( type, versionRange );
  303. if ( value != null )
  304. {
  305. return value;
  306. }
  307. value = getModelInterface( type, versionRange );
  308. if ( value != null )
  309. {
  310. return value;
  311. }
  312. throw new ModelloRuntimeException(
  313. "There is no class or interface '" + type + "' in the version range '" + versionRange.toString() + "'." );
  314. }
  315. public void initialize()
  316. {
  317. for ( ModelClass modelClass : classes )
  318. {
  319. modelClass.initialize( this );
  320. }
  321. for ( ModelInterface modelInterface : interfaces )
  322. {
  323. modelInterface.initialize( this );
  324. }
  325. }
  326. public void validateElement()
  327. {
  328. }
  329. public ModelClass getLocationTracker( Version version )
  330. {
  331. List<ModelClass> modelClasses = getClasses( version );
  332. ModelClass locationTracker = null;
  333. for ( ModelClass modelClass : modelClasses )
  334. {
  335. ModelClassMetadata metadata = (ModelClassMetadata) modelClass.getMetadata( ModelClassMetadata.ID );
  336. if ( metadata != null && StringUtils.isNotEmpty( metadata.getLocationTracker() ) )
  337. {
  338. if ( locationTracker == null )
  339. {
  340. locationTracker = modelClass;
  341. }
  342. else
  343. {
  344. throw new ModelloRuntimeException( "There are multiple location tracker classes ("
  345. + locationTracker.getName() + " vs. " + modelClass.getName() + ") for this version " + version
  346. + "." );
  347. }
  348. }
  349. }
  350. return locationTracker;
  351. }
  352. public ModelClass getSourceTracker( Version version )
  353. {
  354. List<ModelClass> modelClasses = getClasses( version );
  355. ModelClass sourceTracker = null;
  356. for ( ModelClass modelClass : modelClasses )
  357. {
  358. ModelClassMetadata metadata = (ModelClassMetadata) modelClass.getMetadata( ModelClassMetadata.ID );
  359. if ( metadata != null && StringUtils.isNotEmpty( metadata.getSourceTracker() ) )
  360. {
  361. if ( sourceTracker == null )
  362. {
  363. sourceTracker = modelClass;
  364. }
  365. else
  366. {
  367. throw new ModelloRuntimeException( "There are multiple source tracker classes ("
  368. + sourceTracker.getName() + " vs. " + modelClass.getName() + ") for this version " + version
  369. + "." );
  370. }
  371. }
  372. }
  373. return sourceTracker;
  374. }
  375. }