- Advice
- Advice Types
- Joinpoint
- Advice Order
- Advising Methods
- Removing Method Advice
- Advising Functions
- Advising Constructors
Advice is additional functionality that can be added to a method or function noninvasively rather than being coded directly into the original. This has the advantage of allowing the original to be simpler and to be concerned only with its primary purpose. It also allows the advice to be applied to many methods or functions, to be easily enabled or disabled, as well as to be maintained and tested separately.
Some common examples of advice are:
- Logging function parameters and return values for debugging.
- Collecting statistics about execution time
- Caching/memoization
- Transaction demarcation
Here is a simple example of using meld.js to add caching to an existing method.
// A simple object with a method whose results we'd like to
// cache/memoize
var myObject = {
doSomething: function(a, b) {
if(arguments.length < 2) {
throw new Error('doSomething must be called with 2 arguments');
}
// This is a trivial computation, not really worth
// caching, but could easily be something more complex.
return a + b;
}
};
var cache = {};
meld.around(myObject, 'doSomething', function(methodCall) {
var cacheKey, result;
// Create a simple cache key from the arguments passed
// to the original method call.
cacheKey = methodCall.args.join();
if(cacheKey in cache) {
// If the cacheKey was found, return the cached result
result = cache[cacheKey];
} else {
// If not, allow the original method to be executed,
// with it's original arguments, and compute the result.
result = methodCall.proceed();
cache[cacheKey] = result;
}
return result;
});
var result;
// No results cached yet, so this will compute the result
// and cache it.
result = myObject.doSomething(1, 2);
// The previous call caused the result of 1 + 2 to be cached,
// so this call will simply return the cached result.
result = myObject.doSomething(1, 2);
Before executes before another function, and receives the same parameters.
On advice executes, conceptually, at the same time as another function, and receives the same parameters. In practice, on advice will execute immediately after the advised function, before around advice code that comes after joinpoint.proceed
. See Around advice and Advice Order for more information on the exact ordering of around and on advice.
AfterReturning advice executes after another function returns, and receives the return value as its parameter.
AfterThrowing advice executes after another function throws, and receives the thrown exception as its parameter.
After advice executes after another function returns or throws, and receives either the return value or the thrown exception as its parameter.
Around is the most powerful type of advice and, as its name implies, executes around another function. It may execute code before and after the original function, modify the arguments passed to the original, or even elect to prevent the original function from executing at all (see the Simple Caching Example above).
Around advice receives an Around Joinpoint as its only parameter, which it can use to access the original arguments, context, and method name of the original function, as well as allow the original function to continue, with either its original arguments, or a modified list of arguments.
A joinpoint is the point at which a method or function was intercepted. Think of it as a description of the original function invocation. It provides the arguments with which the original method was called, the method name (in the case of an object method), and the context with which the method was called.
A joinpoint has the following properties:
- target - context with which the original method was called
- method - String name of the method that was called
- args - Array of arguments that were passed to the method
Around advice always receives the current around joinpoint as its only argument.
From inside any other advice type (before, after, etc.) you may access the current joinpoint by calling meld.joinpoint()
. It will return the joinpoint within which the current advice function is executing.
When called from outside an advice function, meld.joinpoint()
always returns undefined.
IMPORTANT: The returned joinpoint is only valid within the advice function where it was retrieved by calling meld.joinpoint
. You should not cache the joinpoint returned by meld.joinpoint()
and attempt to use it outside of an advice function, or in different advice functions from that which it was originally retrieved.
Around advice receives as its only argument, an augmented joinpoint, sometimes called a "proceeding joinpoint". It has the same fields as a regular joinpoint, with the addition of "proceed" methods for allowing the original method call to continue and getting its result.
An around joinpoint has the following additional properties:
- proceed() - function that around advice should call to proceed to the original method invocation. You may pass arguments to
proceed()
and they will be used instead of the original arguments. It will return the result of the original method, or throw an exception if the original method throws. - proceedApply:
Function
- similar toproceed()
but accepts an Array of new arguments, e.g.proceedApply([1, 2, 3])
is equivalent toproceed(1, 2, 3)
- proceedCount() - function that returns the number of times that
proceed()
and/orproceedApply()
have been called within the current around advice.
When multiple advices are added to the same method or function, they run in the following order. Note that before advice runs in LIFO order, while all other advice types run in FIFO order.
- Before advice in LIFO order
- Around advice:
- All code up to calling
joinpoint.proceed()
in LIFO order - On advice in FIFO order
- All code after calling
joinpoint.proceed()
in FIFO order
- All code up to calling
- AfterReturning or AfterThrowing advice in FIFO order
- After (finally) advice in FIFO order
var myObject = {
doSomething: function(x) {
console.log(x);
}
};
// Before advice executes in LIFO order
meld.before(myObject, 'doSomething', function(x) {
console.log(2);
});
meld.before(myObject, 'doSomething', function(x) {
console.log(1);
});
// Around advice wraps the intercepted method in layers
// The first around is immediately around the intercepted method
meld.around(myObject, 'doSomething', function(joinpoint) {
// code before joinpoint.proceed() executes LIFO
console.log(4);
var result = joinpoint.proceed();
// code after joinpoint.proceed() executes FIFO
console.log(8);
return result;
});
// The second around advice is layered around the first!
meld.around(myObject, 'doSomething', function(joinpoint) {
// code before joinpoint.proceed() executes LIFO
console.log(3);
var result = joinpoint.proceed();
// code after joinpoint.proceed() executes FIFO
console.log(9);
return result;
});
// On advice executes in FIFO order inside around advice
meld.on(myObject, 'doSomething', function(x) {
console.log(6);
});
meld.on(myObject, 'doSomething', function(x) {
console.log(7);
});
// After Returning advice executes in FIFO order
meld.afterReturning(myObject, 'doSomething', function(x) {
console.log(10);
});
meld.afterReturning(myObject, 'doSomething', function(x) {
console.log(11);
});
// After advice executes in FIFO order
meld.after(myObject, 'doSomething', function(x) {
console.log(12);
});
meld.after(myObject, 'doSomething', function(x) {
console.log(13);
});
myObject.doSomething(5);
Output:
1 // before
2 // before
3 // around, before joinpoint.proceed
4 // around, before joinpoint.proceed
5 // **original myObject.doSomething**
6 // on
7 // on
8 // around, after joinpoint.proceed
9 // around, after joinpoint.proceed
10 // afterReturning
11 // afterReturning
12 // after
13 // after
Adding advice to a method:
var myObject = {
doSomething: function(a, b) {
if(arguments.length < 2) {
throw new Error('doSomething must be called with 2 arguments');
}
// This is a trivial computation, not really worth
// caching, but could easily be something more complex.
return a + b;
}
};
// Arrange for a function to be called before myObject.doSomething
meld.before(myObject, 'doSomething', function() {
console.log(arguments.length);
});
// Logs:
// 2
myObject.doSomething(1, 2);
// Multiple advices may be added
meld.after(myObject, 'doSomething', function(result) {
console.log(result);
});
// Logs:
// 2
// 3
myObject.doSomething(1, 2);
// Logs:
// 1
// Error: doSomething must be called with 2 arguments
myObject.doSomething(1);
In addition to passing a String to specify what method to advise, you can also pass an Array, a RegExp, or a Function as the second parameter, and meld will apply the supplied advice to each method that matches.
The simplest type of method name matching that matches only a method with the exact name specified in the String. See the example above.
Passing an Array of Strings will match each method name in the Array.
// Use an Array to call a function before particular methods
// of myObject
meld.before(myObject, ['doSomething', 'doSomethingElse'], function() {
console.log(arguments.length);
});
myObject.doSomething(1, 2, 3); // Logs: "3"
myObject.doSomethingElse(1, 2); // Logs: "2"
myObject.someOtherMethod(1, 2); // Doesn't log anything
Passing a RegExp will match each method name that matches the RegExp.
// Use a RegExp to call a function before methods of myObject
// whose name matches.
meld.before(myObject, /^doSomething/, function() {
console.log(arguments.length);
});
myObject.doSomething(1, 2, 3); // Logs: "3"
myObject.doSomethingElse(1, 2); // Logs: "2"
myObject.someOtherMethod(1, 2); // Doesn't log anything
// Use a Function matcher to call a function before
// particular methods of myObject
meld.before(myObject,
function(targetObject) {
// Matcher function must return an Array of method names
// In this case, match all owned method, whose name is
// shorter than 15 characters
return Object.keys(targetObject).filter(function(name) {
return name.length < 15;
});
},
function() {
console.log(arguments.length);
}
);
myObject.doSomething(1, 2, 3); // Logs: "3"
myObject.doSomethingElse(1, 2); // Doesn't log anything
myObject.someOtherMethod(1, 2); // Doesn't log anything
When adding advice to object methods, meld returns an object with a remove()
method that can be used to remove the advice that was just added.
// Call a function before myObject.doSomething
var remover = meld.before(myObject, 'doSomething', function() {
console.log(arguments.length);
});
myObject.doSomething(1, 2, 3); // Logs: "3"
remover.remove();
myObject.doSomething(1, 2, 3); // Nothing logged
NOTE: When adding advice to functions and constructors, meld leaves the original function or constructor unmodified, and returns a new function or constructor. Thus, there is no need to "remove" advice. Simply discard the new function or constructor and use the original.
Advice can be added directly to functions that don't have a context. In this case, an advised function is returned, and the original function is left untouched.
var doSomething = function(a, b) {
return a + b;
}
var originalDoSomething = doSomething;
doSomething(1, 2); // Doesn't log anything
// Add before advice directly to the original function.
// This returns an advised function, so it's important
// to use the return value.
doSomething = meld.before(doSomething, function() {
console.log(arguments.length);
});
// New function has before advice applied
doSomething(1, 2); // Logs: "2"
// Original function has not been modified
originalDoSomething(1, 2); // Doesn't log anything
IMPORTANT: As ES5 compliant Object.create()
must be available to advise constructors. If you're running in an environment that does not provide Object.create()
, consider using a shim, such as cujo.js's poly, or es5-shim
Advice can be added to constructor functions that are invoked with new
. Similarly to advising functions, advising a constructor returns an advised constructor function, and the original constructor is left untouched.
var Thing = function(name) {
this.name = name;
}
// Save the original constructor
var OriginalThing = Thing;
var t = new Thing('Bob');
console.log(t.name); // Logs 'Bob'
// Add after advice directly to the Thing constructor.
// This returns an advised constructor, so it's important
// to use the return value.
Thing = meld.after(Thing, function() {
// Note that `this` is the constructed object, and
// is an instanceof Thing, just as if this code
// were inside the original Thing constructor
this.name += 'by';
});
// New constructor has advice applied
t = new Thing('Bob');
console.log(t.name); // Logs 'Bobby'
// Original constructor has not been modified
t = new OriginalThing('Bob');
console.log(t.name); // Logs 'Bob'