Build a More Powerful AOP Framework Based on AspectDNG's ILML Library - Teddy's Aspect Weaver. (Updated Introduction to Full Functions of Version 0.6 and Sample)
Return to Contents
7. Build a More Powerful AOP Framework Based on AspectDNG's ILML Library - Teddy's Aspect Weaver.
In this chapter, I'll show you my AOP framework - Teddy's Aspect Weaver which is based on AspectDNG's ILML library. I just do my best to make this framework easier for using and avoid AspectDNG's disadvantages and encumbering facts. Also this framework extends functions of AspectDNG and provides more supported types of advices. Let's take a closer look with the example within the Aspect Weaver V0.1 source code.
7.1 Command Line Usage
AspectWeaver.exe Path_to_AspectWeaverConfiguration.xml
7.2 Configuration Format
The full configuration format is defined in AspectWeaver/AspectWeaver.xsd with comments for each Advice's specific configuration. And the best way to learn how to use is to work through the example.
Full TestAspectConfiguration.xml source code
<Configuration logFile="LogWeaving.xml" cleanTempFiles="false">
<BaseAssembly>..\TestLib\bin\Debug\TestLib.dll</BaseAssembly>
<OutputAssembly>..\TestLib\bin\Debug\TestLib_output.dll</OutputAssembly>
<AspectAssemblies>
<AspectAssembly uniqueName="TestAspectLib.dll" path="bin\Debug\TestAspectLib.dll" />
</AspectAssemblies>
<AdviceFiles>
<AdviceFile>Advice-Test.xml</AdviceFile>
</AdviceFiles>
</Configuration>
Full Advice-Test.xml source code
<Advices>
<Delete>
<Pointcut targetXPath="//Type[@name='ClassToBeDeleted']" />
</Delete>
<Delete>
<Pointcut targetXPath="//Field[@name='fieldToBeDeleted']" />
</Delete>
<Delete>
<Pointcut targetXPath="//Method[@name='MethodToBeDeleted']" />
</Delete>
<Introduce>
<Pointcut targetXPath="//Module" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Type[@name='ClassToBeIntroduced']"></Code>
</Introduce>
<Introduce>
<Pointcut targetXPath="//Type[@name='TestClass']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Field[@name='fieldToBeIntroduced']"></Code>
</Introduce>
<Introduce>
<Pointcut targetXPath="//Type[@name='TestClass']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='MethodToBeIntroduced']"></Code>
</Introduce>
<SetBaseType>
<Pointcut targetXPath="//Type[@name='TestClass']" />
<BaseType>System.MarshalByRefObject</BaseType>
</SetBaseType>
<ImplementInterface>
<Pointcut targetXPath="//Type[@name='TestClass']" />
<InterfaceType>TestLib.InterfaceToBeImpelementInterface</InterfaceType>
</ImplementInterface>
<MakeSealed>
<Pointcut targetXPath="//Type[@name='TestClass']" />
</MakeSealed>
<MakeSerializable>
<Pointcut targetXPath="//Type[@name='TestClass']" />
</MakeSerializable>
<BeforeConstructorCall>
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeBeforeConstructorCall']"></Code>
</BeforeConstructorCall>
<AfterConstructorCall>
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAfterConstructorCall']"></Code>
</AfterConstructorCall>
<BeforeMethodCall>
<Pointcut targetXPath="//Method[@name='MethodUsedToTestMethodBeforeAfterCall']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeBeforeMethodCall']"></Code>
</BeforeMethodCall>
<AfterMethodCall>
<Pointcut targetXPath="//Method[@name='MethodUsedToTestMethodBeforeAfterCall']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAfterMethodCall']"></Code>
</AfterMethodCall>
<InlineAtStart>
<Pointcut targetXPath="//Method[@name='TestInlineAtStartBeforeReturn']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineAtStart']"></Code>
</InlineAtStart>
<InlineAtStart>
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineAtStart']"></Code>
</InlineAtStart>
<InlineBeforeReturn>
<Pointcut targetXPath="//Method[@name='TestInlineAtStartBeforeReturn']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineBeforeReturn']"></Code>
</InlineBeforeReturn>
<InlineBeforeReturn>
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineBeforeReturn']"></Code>
</InlineBeforeReturn>
<AroundBody>
<Pointcut targetXPath="//Method[@name='MethodToBeAroundedBody']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAroundedMethodBody']"></Code>
</AroundBody>
<AroundBody>
<Pointcut targetXPath="//Method[@name='set_PropertyToBeAroundedBody']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAroundedMethodBody']"></Code>
</AroundBody>
</Advices>
7.3 Advices Test Result
7.3.1 Delete
Only Type, Field or Method can be deleted. Below is the sample configuration.
<Pointcut targetXPath="//Type[@name='ClassToBeDeleted']" />
</Delete>
<Delete>
<Pointcut targetXPath="//Field[@name='fieldToBeDeleted']" />
</Delete>
<Delete>
<Pointcut targetXPath="//Method[@name='MethodToBeDeleted']" />
</Delete>
After weaving, the specific elements will be deleted from the base assembly.
7.3.2 Introduce
Introduce here means "Insert", we can insert Type, Field or Method from aspect assembly to base assembly. Below is the sample configuration.
<Pointcut targetXPath="//Module" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Type[@name='ClassToBeIntroduced']"></Code>
</Introduce>
<Introduce>
<Pointcut targetXPath="//Type[@name='TestClass']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Field[@name='fieldToBeIntroduced']"></Code>
</Introduce>
<Introduce>
<Pointcut targetXPath="//Type[@name='TestClass']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='MethodToBeIntroduced']"></Code>
</Introduce>
7.3.3 SetBaseType, ImplementInterface, MakeSealed, MakeUnsealed and MakeSerializable
These advices are some degree simpler so I put them as a example together.
Configuration
<Pointcut targetXPath="//Type[@name='TestClass']" />
<BaseType>System.MarshalByRefObject</BaseType>
</SetBaseType>
<ImplementInterface>
<Pointcut targetXPath="//Type[@name='TestClass']" />
<InterfaceType>TestLib.InterfaceToBeImpelementInterface</InterfaceType>
</ImplementInterface>
<MakeSealed>
<Pointcut targetXPath="//Type[@name='TestClass']" />
</MakeSealed>
<MakeSerializable>
<Pointcut targetXPath="//Type[@name='TestClass']" />
</MakeSerializable>
Code before weaving
{
}
Code after weaving (decompiled)
public sealed class TestClass : MarshalByRefObject, InterfaceToBeImpelementInterface
{
}
7.3.4 BeforeConstructorCall, AfterConstructorCall
These two advices insert code before/after where the constructors of specific Type be called.
Configuration
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeBeforeConstructorCall']"></Code>
</BeforeConstructorCall>
<AfterConstructorCall>
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAfterConstructorCall']"></Code>
</AfterConstructorCall>
Code before weaving
{
public TestClass()
{
}
public TestClass(object obj)
{
}
public void TestBeforeAfterConstructorCall()
{
object obj1 = new TestClass();
new TestClass("test");
}
}
{
public void CodeBeforeConstructorCall()
{
Console.Write("CodeBeforeConstructorCall");
}
public void CodeAfterConstructorCall()
{
Console.Write("CodeAfterConstructorCall");
}
}
Code after weaving (decompiled)
{
public TestClass()
{
}
public TestClass(object )
{
}
public void TestBeforeAfterConstructorCall()
{
Console.Write("CodeBeforeConstructorCall");
object local = new TestClass();
Console.Write("CodeAfterConstructorCall");
Console.Write("CodeBeforeConstructorCall");
new TestClass("test");
Console.Write("CodeAfterConstructorCall");
}
}
7.3.5 BeforeMethodCall and AfterMethodCall
These two advices are similar to the ConstructorCall ones but for MethodCall.
Configuration
<Pointcut targetXPath="//Method[@name='MethodUsedToTestMethodBeforeAfterCall']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeBeforeMethodCall']"></Code>
</BeforeMethodCall>
<AfterMethodCall>
<Pointcut targetXPath="//Method[@name='MethodUsedToTestMethodBeforeAfterCall']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAfterMethodCall']"></Code>
</AfterMethodCall>
Code before weaving
{
public int MethodUsedToTestMethodBeforeAfterCall()
{
return 0;
}
public void TestBeforeAfterMethodCall()
{
int i = MethodUsedToTestMethodBeforeAfterCall();
MethodUsedToTestMethodBeforeAfterCall();
}
}
{
public void CodeBeforeMethodCall()
{
Console.Write("CodeBeforeMethodCall");
}
public void CodeAfterMethodCall()
{
Console.Write("CodeAfterMethodCall");
}
}
Code after weaving (decompiled)
{
public int MethodUsedToTestMethodBeforeAfterCall()
{
return 0;
}
public void TestBeforeAfterMethodCall()
{
Console.Write("CodeBeforeMethodCall");
Console.Write("CodeAfterMethodCall");
int i = MethodUsedToTestMethodBeforeAfterCall();
Console.Write("CodeBeforeMethodCall");
MethodUsedToTestMethodBeforeAfterCall();
Console.Write("CodeAfterMethodCall");
}
}
7.3.6 InlineAtStart and InlineBeforeEnd
These advices insert code in to constructor or methods' body.
Configuration
<Pointcut targetXPath="//Method[@name='TestInlineAtStartBeforeReturn']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineAtStart']"></Code>
</InlineAtStart>
<InlineAtStart>
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineAtStart']"></Code>
</InlineAtStart>
<InlineBeforeReturn>
<Pointcut targetXPath="//Method[@name='TestInlineAtStartBeforeReturn']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineBeforeReturn']"></Code>
</InlineBeforeReturn>
<InlineBeforeReturn>
<Pointcut targetXPath="//Type[@name='TestClass']/Constructor" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeInlineBeforeReturn']"></Code>
</InlineBeforeReturn>
Code before weaving
{
public TestClass()
{
}
public TestClass(object obj)
{
}
public void TestInlineAtStartBeforeReturn()
{
}
}
{
public void CodeInlineAtStart()
{
Console.Write("CodeInlineAtStart");
}
public void CodeInlineBeforeReturn()
{
Console.Write("CodeInlineBeforeReturn");
}
}
Code after weaving (decompiled)
{
public TestClass()
{
Console.Write("CodeInlineAtStart");
Console.Write("CodeInlineBeforeReturn");
}
public TestClass(object )
{
Console.Write("CodeInlineAtStart");
Console.Write("CodeInlineBeforeReturn");
}
public void TestInlineAtStartBeforeReturn()
{
Console.Write("CodeInlineAtStart");
Console.Write("CodeInlineBeforeReturn");
}
}
7.3.7 AroundBody
The advice AroundBody can only be apply to Method that means uses specific Method code body in aspect assembly to replace Methodcode bodies in base assembly.
Configuration
<Pointcut targetXPath="//Method[@name='MethodToBeAroundedBody']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAroundedMethodBody']"></Code>
</AroundBody>
<AroundBody>
<Pointcut targetXPath="//Method[@name='set_PropertyToBeAroundedBody']" />
<Code aspectAssemblyUniqueName="TestAspectLib.dll" aspectXPath="//Method[@name='CodeAroundedMethodBody']"></Code>
</AroundBody>
Code before weaving
{
public void MethodToBeAroundedBody()
{
Console.Write("MethodToBeAroundedBody");
}
public string PropertyToBeAroundedBody
{
set
{
}
}
}
{
public void CodeAroundedMethodBody()
{
Console.Write("CodeAroundedMethodBody");
}
}
Code after weaving (decompiled)
{
public void MethodToBeAroundedBody()
{
Console.Write("CodeAroundedMethodBody");
}
public string PropertyToBeAroundedBody
{
set
{
Console.Write("CodeAroundedMethodBody");
}
}
}
7.4 Additional Custom Attribute Configuration Way
TestAspectClass.cs
using AspectWeaver.Apects;
namespace TestAspectLib
{
public class TestAspectClass
{
[Introduce("//Type[@name='TestClass']")]
private string fieldToBeIntroducedByCustomAttribute = "fieldToBeIntroducedByCustomAttribute";
[Introduce("//Type[@name='TestClass']")]
public void MethodToBeIntrodcuedByCustomAttribute()
{
}
}
[Introduce()]
public class ClassToBeIntroducedByCustomAttribute
{
}
}
This sample is part of sample with both meta way and custom attribute way configuration within AspectWeaver source code. Be careful, don't define the same advice both in meta xml and custom attribute, or that may cause error. And only the advices with a code element can be configurated with the new custom attribute way. Advices can be configurated by the new way are: Introduce, BeforeConstructor/AfterConstructor, BeforeMethodCall/AfterMethodCall, InlineAtStart, InlineBeforeReturn and AroundBody.
7.5 How to Get Runtime Method Context Info and Method Arguments
TestClass.cs
namespace TestLib
{
public class TestClass
{
public void MethodToBeTestedGetContextInfoAndGetArgumments(object p1, string p2, int p3)
{
}
public void MethodToTestGetContextInfoAndGetArgumments()
{
MethodToBeTestedGetContextInfoAndGetArgumments("p1", "p2", 3);
}
}
}
TestAspectClass.cs
using AspectWeaver.Aspects;
namespace TestAspectLib
{
public class TestAspectClass : Aspect
{
[InlineAtStart("//Method[@name='MethodToBeTestedGetContextInfoAndGetArgumments']")]
public void CodeToTestGetContextInfoAndGetArgumments()
{
string contextInfo = GetContextInfo();
Console.Write(contextInfo + "\n\n");
object[] args = GetArguments();
Console.Write(string.Format("Arguments: {0}, {1}, {2}\n\n", args));
}
}
}
After Weaving Code (decompiled)
{
public void MethodToBeTestedGetContextInfoAndGetArgumments(object p1, string p2, int p3)
{
String string1 = "MethodToBeTestedGetContextInfoAndGetArgumments(System.Object,System.String,System.Int32)";
Console.Write(String.Concat(contextInfo, "\n\n"));
object[] locals = new object[]{p1, p2, p3};
Console.Write(String.Format("Arguments: {0}, {1}, {2}\n\n", locals));
}
public void MethodToTestGetContextInfoAndGetArgumments()
{
MethodToBeTestedGetContextInfoAndGetArgumments("p1", "p2", 3);
}
}
7.6 Download Source Code And Sample (Updated: 2005/10/05)
AspectWeaver0.6.zip
AspectWeaverSample1.0.zip [More About AspectWeaverSample1.0]
[More about Teddy's Aspect Weaver]