You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

SimpleAOPParser.java 10KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. /*******************************************************************************
  2. * Copyright (c) 2011 Contributors.
  3. * All rights reserved.
  4. * This program and the accompanying materials are made available
  5. * under the terms of the Eclipse Public License v1.0
  6. * which accompanies this distribution and is available at
  7. * http://eclipse.org/legal/epl-v10.html
  8. *
  9. * Contributors:
  10. * Abraham Nevado - Lucierna initial implementation
  11. * Just a slight variation of current DocumentParser.java from Alexandre Vasseur.
  12. *******************************************************************************/
  13. package org.aspectj.weaver.loadtime.definition;
  14. import java.io.InputStream;
  15. import java.io.InputStreamReader;
  16. import java.net.URL;
  17. import java.util.ArrayList;
  18. import java.util.Map;
  19. import org.aspectj.util.LangUtil;
  20. import org.aspectj.weaver.loadtime.definition.Definition.AdviceKind;
  21. import org.aspectj.weaver.loadtime.definition.Definition.DeclareAnnotationKind;
  22. import org.xml.sax.SAXException;
  23. /**
  24. * This class has been created to avoid deadlocks when instrumenting SAXParser.
  25. * So it is used as a wrapper for the ligthweigh XML parser LightXMLParser.
  26. *
  27. * @author A. Nevado
  28. */
  29. public class SimpleAOPParser {
  30. private final static String ASPECTJ_ELEMENT = "aspectj";
  31. private final static String WEAVER_ELEMENT = "weaver";
  32. private final static String DUMP_ELEMENT = "dump";
  33. private final static String DUMP_BEFOREANDAFTER_ATTRIBUTE = "beforeandafter";
  34. private final static String DUMP_PERCLASSLOADERDIR_ATTRIBUTE = "perclassloaderdumpdir";
  35. private final static String INCLUDE_ELEMENT = "include";
  36. private final static String EXCLUDE_ELEMENT = "exclude";
  37. private final static String OPTIONS_ATTRIBUTE = "options";
  38. private final static String ASPECTS_ELEMENT = "aspects";
  39. private final static String ASPECT_ELEMENT = "aspect";
  40. private final static String CONCRETE_ASPECT_ELEMENT = "concrete-aspect";
  41. private final static String NAME_ATTRIBUTE = "name";
  42. private final static String SCOPE_ATTRIBUTE = "scope";
  43. private final static String REQUIRES_ATTRIBUTE = "requires";
  44. private final static String EXTEND_ATTRIBUTE = "extends";
  45. private final static String PRECEDENCE_ATTRIBUTE = "precedence";
  46. private final static String PERCLAUSE_ATTRIBUTE = "perclause";
  47. private final static String POINTCUT_ELEMENT = "pointcut";
  48. private final static String WITHIN_ATTRIBUTE = "within";
  49. private final static String EXPRESSION_ATTRIBUTE = "expression";
  50. private static final String DECLARE_ANNOTATION = "declare-annotation";
  51. private static final String ANNONATION_TAG = "annotation";
  52. private static final String ANNO_KIND_TYPE = "type";
  53. private static final String ANNO_KIND_METHOD = "method";
  54. private static final String ANNO_KIND_FIELD = "field";
  55. private final static String BEFORE_ELEMENT = "before";
  56. private final static String AFTER_ELEMENT = "after";
  57. private final static String AROUND_ELEMENT = "around";
  58. private final Definition m_definition;
  59. private boolean m_inAspectJ;
  60. private boolean m_inWeaver;
  61. private boolean m_inAspects;
  62. private Definition.ConcreteAspect m_lastConcreteAspect;
  63. private SimpleAOPParser() {
  64. m_definition = new Definition();
  65. }
  66. public static Definition parse(final URL url) throws Exception {
  67. // FileReader freader = new FileReader("/tmp/aop.xml");
  68. InputStream in = url.openStream();
  69. LightXMLParser xml = new LightXMLParser();
  70. xml.parseFromReader(new InputStreamReader(in));
  71. SimpleAOPParser sap = new SimpleAOPParser();
  72. traverse(sap, xml);
  73. return sap.m_definition;
  74. }
  75. private void startElement(String qName, Map attrMap) throws Exception {
  76. if (ASPECT_ELEMENT.equals(qName)) {
  77. String name = (String) attrMap.get(NAME_ATTRIBUTE);
  78. String scopePattern = replaceXmlAnd((String) attrMap
  79. .get(SCOPE_ATTRIBUTE));
  80. String requiredType = (String) attrMap.get(REQUIRES_ATTRIBUTE);
  81. if (!isNull(name)) {
  82. m_definition.getAspectClassNames().add(name);
  83. if (scopePattern != null) {
  84. m_definition.addScopedAspect(name, scopePattern);
  85. }
  86. if (requiredType != null) {
  87. m_definition.setAspectRequires(name, requiredType);
  88. }
  89. }
  90. } else if (WEAVER_ELEMENT.equals(qName)) {
  91. String options = (String) attrMap.get(OPTIONS_ATTRIBUTE);
  92. if (!isNull(options)) {
  93. m_definition.appendWeaverOptions(options);
  94. }
  95. m_inWeaver = true;
  96. } else if (CONCRETE_ASPECT_ELEMENT.equals(qName)) {
  97. String name = (String) attrMap.get(NAME_ATTRIBUTE);
  98. String extend = (String) attrMap.get(EXTEND_ATTRIBUTE);
  99. String precedence = (String) attrMap.get(PRECEDENCE_ATTRIBUTE);
  100. String perclause = (String) attrMap.get(PERCLAUSE_ATTRIBUTE);
  101. if (!isNull(name)) {
  102. m_lastConcreteAspect = new Definition.ConcreteAspect(name,
  103. extend, precedence, perclause);
  104. m_definition.getConcreteAspects().add(m_lastConcreteAspect);
  105. }
  106. } else if (POINTCUT_ELEMENT.equals(qName)
  107. && m_lastConcreteAspect != null) {
  108. String name = (String) attrMap.get(NAME_ATTRIBUTE);
  109. String expression = (String) attrMap.get(EXPRESSION_ATTRIBUTE);
  110. if (!isNull(name) && !isNull(expression)) {
  111. m_lastConcreteAspect.pointcuts.add(new Definition.Pointcut(
  112. name, replaceXmlAnd(expression)));
  113. }
  114. } else if (ASPECTJ_ELEMENT.equals(qName)) {
  115. if (m_inAspectJ) {
  116. throw new Exception("Found nested <aspectj> element");
  117. }
  118. m_inAspectJ = true;
  119. } else if (ASPECTS_ELEMENT.equals(qName)) {
  120. m_inAspects = true;
  121. } else if (INCLUDE_ELEMENT.equals(qName) && m_inWeaver) {
  122. String typePattern = getWithinAttribute(attrMap);
  123. if (!isNull(typePattern)) {
  124. m_definition.getIncludePatterns().add(typePattern);
  125. }
  126. } else if (EXCLUDE_ELEMENT.equals(qName) && m_inWeaver) {
  127. String typePattern = getWithinAttribute(attrMap);
  128. if (!isNull(typePattern)) {
  129. m_definition.getExcludePatterns().add(typePattern);
  130. }
  131. } else if (DUMP_ELEMENT.equals(qName) && m_inWeaver) {
  132. String typePattern = getWithinAttribute(attrMap);
  133. if (!isNull(typePattern)) {
  134. m_definition.getDumpPatterns().add(typePattern);
  135. }
  136. String beforeAndAfter = (String) attrMap
  137. .get(DUMP_BEFOREANDAFTER_ATTRIBUTE);
  138. if (isTrue(beforeAndAfter)) {
  139. m_definition.setDumpBefore(true);
  140. }
  141. String perWeaverDumpDir = (String) attrMap
  142. .get(DUMP_PERCLASSLOADERDIR_ATTRIBUTE);
  143. if (isTrue(perWeaverDumpDir)) {
  144. m_definition.setCreateDumpDirPerClassloader(true);
  145. }
  146. } else if (EXCLUDE_ELEMENT.equals(qName) && m_inAspects) {
  147. String typePattern = getWithinAttribute(attrMap);
  148. if (!isNull(typePattern)) {
  149. m_definition.getAspectExcludePatterns().add(typePattern);
  150. }
  151. } else if (INCLUDE_ELEMENT.equals(qName) && m_inAspects) {
  152. String typePattern = getWithinAttribute(attrMap);
  153. if (!isNull(typePattern)) {
  154. m_definition.getAspectIncludePatterns().add(typePattern);
  155. }
  156. }else if (DECLARE_ANNOTATION.equals(qName) && m_inAspects) {
  157. String anno = (String) attrMap.get(ANNONATION_TAG);
  158. if (!isNull(anno)){
  159. String pattern = (String) attrMap.get(ANNO_KIND_FIELD);
  160. if (pattern != null){
  161. m_lastConcreteAspect.declareAnnotations.add(new Definition.DeclareAnnotation(
  162. DeclareAnnotationKind.Field, pattern, anno));
  163. }
  164. else{
  165. pattern = (String) attrMap.get(ANNO_KIND_METHOD);
  166. if (pattern != null){
  167. m_lastConcreteAspect.declareAnnotations.add(new Definition.DeclareAnnotation(
  168. DeclareAnnotationKind.Method, pattern, anno));
  169. }
  170. else{
  171. pattern = (String) attrMap.get(ANNO_KIND_TYPE);
  172. if (pattern != null){
  173. m_lastConcreteAspect.declareAnnotations.add(new Definition.DeclareAnnotation(
  174. DeclareAnnotationKind.Type, pattern, anno));
  175. }
  176. }
  177. }
  178. }
  179. }
  180. else if (BEFORE_ELEMENT.equals(qName) && m_inAspects ) {
  181. String pointcut = (String) attrMap.get(POINTCUT_ELEMENT);
  182. String adviceClass = (String) attrMap.get("invokeClass");
  183. String adviceMethod = (String) attrMap.get("invokeMethod");
  184. if (!isNull(pointcut) && !isNull(adviceClass) && !isNull(adviceMethod)) {
  185. m_lastConcreteAspect.pointcutsAndAdvice.add(new Definition.PointcutAndAdvice(AdviceKind.Before,
  186. replaceXmlAnd(pointcut), adviceClass, adviceMethod));
  187. } else {
  188. throw new SAXException("Badly formed <before> element");
  189. }
  190. } else if (AFTER_ELEMENT.equals(qName) && m_inAspects) {
  191. String pointcut = (String) attrMap.get(POINTCUT_ELEMENT);
  192. String adviceClass = (String) attrMap.get("invokeClass");
  193. String adviceMethod = (String) attrMap.get("invokeMethod");
  194. if (!isNull(pointcut) && !isNull(adviceClass) && !isNull(adviceMethod)) {
  195. m_lastConcreteAspect.pointcutsAndAdvice.add(new Definition.PointcutAndAdvice(AdviceKind.After,
  196. replaceXmlAnd(pointcut), adviceClass, adviceMethod));
  197. } else {
  198. throw new SAXException("Badly formed <after> element");
  199. }
  200. } else if (AROUND_ELEMENT.equals(qName) && m_inAspects) {
  201. String pointcut = (String) attrMap.get(POINTCUT_ELEMENT);
  202. String adviceClass = (String) attrMap.get("invokeClass");
  203. String adviceMethod = (String) attrMap.get("invokeMethod");
  204. if (!isNull(pointcut) && !isNull(adviceClass) && !isNull(adviceMethod)) {
  205. m_lastConcreteAspect.pointcutsAndAdvice.add(new Definition.PointcutAndAdvice(AdviceKind.Around,
  206. replaceXmlAnd(pointcut), adviceClass, adviceMethod));
  207. }
  208. }
  209. else {
  210. throw new Exception(
  211. "Unknown element while parsing <aspectj> element: " + qName);
  212. }
  213. }
  214. private void endElement(String qName) throws Exception {
  215. if (CONCRETE_ASPECT_ELEMENT.equals(qName)) {
  216. m_lastConcreteAspect = null;
  217. } else if (ASPECTJ_ELEMENT.equals(qName)) {
  218. m_inAspectJ = false;
  219. } else if (WEAVER_ELEMENT.equals(qName)) {
  220. m_inWeaver = false;
  221. } else if (ASPECTS_ELEMENT.equals(qName)) {
  222. m_inAspects = false;
  223. }
  224. }
  225. private String getWithinAttribute(Map attributes) {
  226. return replaceXmlAnd((String) attributes.get(WITHIN_ATTRIBUTE));
  227. }
  228. private static String replaceXmlAnd(String expression) {
  229. // TODO AV do we need to handle "..)AND" or "AND(.." ?
  230. return LangUtil.replace(expression, " AND ", " && ");
  231. }
  232. private boolean isNull(String s) {
  233. return (s == null || s.length() <= 0);
  234. }
  235. private boolean isTrue(String s) {
  236. return (s != null && s.equals("true"));
  237. }
  238. private static void traverse(SimpleAOPParser sap, LightXMLParser xml)
  239. throws Exception {
  240. sap.startElement(xml.getName(), xml.getAttributes());
  241. ArrayList childrens = xml.getChildrens();
  242. for (int i = 0; i < childrens.size(); i++) {
  243. LightXMLParser child = (LightXMLParser) childrens.get(i);
  244. traverse(sap, child);
  245. }
  246. sap.endElement(xml.getName());
  247. }
  248. }