Differences
This shows you the differences between two versions of the page.
Next revision | Previous revision | ||
research:software:reflex:documentation:tracing_aspect [2007/07/30 02:15] – created admin | research:software:reflex:documentation:tracing_aspect [2007/08/24 21:01] (current) – rtoledo | ||
---|---|---|---|
Line 1: | Line 1: | ||
+ | ====== Tracing Aspect ====== | ||
+ | |||
+ | One of the most simple and well known aspects is the tracing aspect: an aspect that traces the method invocations that occur in the application. First we will see how to implement this using the Reflex standard metaobject protocol (MOP) making use of a special metaobject, and then, how to implement this tracing aspect using the the standard System.out object. | ||
+ | |||
+ | |||
+ | ===== Using the Standard MOP ===== | ||
+ | |||
+ | As in every Reflex configuration we need to define the hookset(s), the metaobject definition(s) and the behavioral link(s). | ||
+ | First, we need a hookset that matches all method receptions in our application: | ||
+ | |||
+ | <code java> | ||
+ | Hookset theHookset = new PrimitiveHookset( | ||
+ | MsgReceive.class, | ||
+ | new NameCS(" | ||
+ | new DeclaredInOS(" | ||
+ | ); | ||
+ | </ | ||
+ | |||
+ | Here we use two utility subclasses of [[http:// | ||
+ | * NameCS: selects classes based on the name(s) passed as parameter(s). | ||
+ | * DeclaredInOS: | ||
+ | |||
+ | Notice that because Reflex acts at loadtime, we always refer to classes using their name and not usingFibonacci.class.getName() because it triggers the loading of the class. | ||
+ | |||
+ | In our case, we are interested in message receives (MsgReceive.class) in the class Fibonacci (new NameCS(" | ||
+ | |||
+ | Second, we need the metaobject definition. In this example, we will use the standard MOP and for this purpose, we will use an utility metaobject: [[http:// | ||
+ | |||
+ | The more straightforward way to declare a metaobject definition is to use SharedMO, as it takes a pre-existent object as parameter: | ||
+ | |||
+ | <code java> | ||
+ | MODefinition theMO = new MODefinition.SharedMO(new SimpleLogger()); | ||
+ | </ | ||
+ | |||
+ | Finally, we need to declare the behavioral link between the previous two elements: | ||
+ | <code java> | ||
+ | BLink theLink = Links.get(theHookset, | ||
+ | </ | ||
+ | |||
+ | That is all the configuration we need. The complete source code of the link provider that adds this link is [[http:// | ||
+ | |||
+ | < | ||
+ | Windows | ||
+ | %java -classpath | ||
+ | " | ||
+ | reflex.examples.tutorial.StdMOPLoggingAspect | ||
+ | |||
+ | Linux | ||
+ | %java -classpath | ||
+ | " | ||
+ | reflex.examples.tutorial.StdMOPLoggingAspect | ||
+ | </ | ||
+ | |||
+ | If everything went ok, you should see something like the following output: | ||
+ | |||
+ | < | ||
+ | Before message receive: | ||
+ | Before message receive: main([5]) | ||
+ | Before message receive: get([5]) | ||
+ | Before message receive: get([4]) | ||
+ | Before message receive: get([3]) | ||
+ | Before message receive: get([2]) | ||
+ | Before message receive: get([1]) | ||
+ | After message receive: get is returning: [1] | ||
+ | Before message receive: get([0]) | ||
+ | ... | ||
+ | After message receive: get is returning: [5] | ||
+ | The fibonacci[5] is 5 | ||
+ | |||
+ | Before message receive: get([6]) | ||
+ | Before message receive: get([5]) | ||
+ | Before message receive: get([4]) | ||
+ | Before message receive: get([3]) | ||
+ | Before message receive: get([2]) | ||
+ | Before message receive: get([1]) | ||
+ | After message receive: get is returning: [1] | ||
+ | Before message receive: get([0]) | ||
+ | ... | ||
+ | After message receive: get is returning: [8] | ||
+ | The fibonacci[6] is 8 | ||
+ | After message receive: main is returning: [null] | ||
+ | </ | ||
+ | |||
+ | As you can see, the first line corresponds to the main method invocation, then, a lot of recursive calls of the get method are traced. Finally, the exit of the main method is traced (the last line). | ||
+ | |||
+ | In the following section we will see how to implement this without any auxiliary class: direct calls to '' | ||
+ | |||
+ | ===== Using Call Descriptors and Parameters ===== | ||
+ | |||
+ | In this section we will view how to use a powerful feature of Reflex: call descriptors. Call descriptors allows us to describe the exact method that should be called in the metaobject (ie, the exact MOP we will use), as well as the parameters to use. A detailed review of these elements can be found [[call_descriptors_parameters|here]]. | ||
+ | |||
+ | In this example we will use System.out as our metaobject. | ||
+ | |||
+ | As we want to trace the same operations as before, the hookset remains the same: | ||
+ | |||
+ | <code java> | ||
+ | Hookset theHookset = new PrimitiveHookset(MsgReceive.class, | ||
+ | new NameCS(" | ||
+ | new DeclaredInOS(" | ||
+ | </ | ||
+ | |||
+ | But, the metaobject definition changes as we are using '' | ||
+ | <code java> | ||
+ | MODefinition theMO = new MODefinition.SharedMO(System.out); | ||
+ | </ | ||
+ | Here we create a SharedMO with the metaobject we want to use: '' | ||
+ | |||
+ | Then, we create the behavioral link: | ||
+ | |||
+ | <code java> | ||
+ | BLink theLink = Links.get(theHookset, | ||
+ | </ | ||
+ | |||
+ | Finally comes the most important part of the configuration, | ||
+ | |||
+ | <code java> | ||
+ | theLink.setControl(Control.BEFORE_AFTER); | ||
+ | theLink.setCall(Control.BEFORE, | ||
+ | theLink.setCall(Control.AFTER, | ||
+ | </ | ||
+ | |||
+ | Here we are specifing three things: | ||
+ | * First, the control attribute of the link is '' | ||
+ | * Second, for the '' | ||
+ | * And Finally, for the '' | ||
+ | |||
+ | Here is the implementation of BeforeParameter: | ||
+ | <code java> | ||
+ | private static class BeforeParameter extends StdParameters.StringParameter{ | ||
+ | |||
+ | public String evaluate(Operation aOperation){ | ||
+ | MsgReceive theMsgReceive = (MsgReceive) aOperation; | ||
+ | return "" | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | The code above defines a subclass of StdParameters.StringParameter, | ||
+ | Notice that we use a reification of the occurring operation (aOperation) to obtain the name of the method being invoked. At the same time, we use the special variable $args that allows us to access the actual parameters of the method (various special variables are present, see the Javassist tutorial for more information). Also we use '' | ||
+ | |||
+ | The code of the AfterParameter is: | ||
+ | <code java> | ||
+ | private static class AfterParameter extends StdParameters.StringParameter{ | ||
+ | |||
+ | public String evaluate(Operation aOperation){ | ||
+ | MsgReceive theMsgReceive = (MsgReceive) aOperation; | ||
+ | return "" | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | The code above is similar to the one we used to define the BeforeParameter. There is just one difference: we use $_ to access to the return value (here we are using [[http:// | ||
+ | The complete code of the link provider that adds this link to the Reflex configuration is [[http:// | ||
+ | |||
+ | < | ||
+ | Windows | ||
+ | %java -classpath " | ||
+ | reflex.examples.tutorial.CallDescriptorLoggingAspect | ||
+ | |||
+ | Linux | ||
+ | %java -classpath " | ||
+ | reflex.examples.tutorial.CallDescriptorLoggingAspect | ||
+ | </ | ||
+ | |||
+ | If everything went ok, you should see the same output (line by line) as in the previous example: | ||
+ | < | ||
+ | Before message receive: main([5]) | ||
+ | Before message receive: get([5]) | ||
+ | Before message receive: get([4]) | ||
+ | Before message receive: get([3]) | ||
+ | Before message receive: get([2]) | ||
+ | Before message receive: get([1]) | ||
+ | After message receive: get is returning: [1] | ||
+ | Before message receive: get([0]) | ||
+ | ... | ||
+ | After message receive: get is returning: [5] | ||
+ | The fibonacci[5] is 5 | ||
+ | |||
+ | Before message receive: get([6]) | ||
+ | Before message receive: get([5]) | ||
+ | Before message receive: get([4]) | ||
+ | Before message receive: get([3]) | ||
+ | Before message receive: get([2]) | ||
+ | Before message receive: get([1]) | ||
+ | After message receive: get is returning: [1] | ||
+ | Before message receive: get([0]) | ||
+ | ... | ||
+ | After message receive: get is returning: [8] | ||
+ | The fibonacci[6] is 8 | ||
+ | After message receive: main is returning: [null] | ||
+ | </ |