I. Définition▲
Un interceptor est une méthode qui intercepte l'appel d'une méthode métier.
Les interceptors fonctionnent à la manière des filtres avec les servlets.
Ils sont automatiquement appelés, juste avant que les méthodes métiers du bean soient appelées.
Ils peuvent être appliqués à des méthodes métiers dans des session beans (stateful et stateless) et dans des message-driven beans.
On peut définir une méthode interceptor dans une classe bean ou dans une classe dédiée.
La signature d'une méthode interceptor est la suivante:
1.
2.
@javax.ejb.AroundInvoke
public Object [nomMethode](javax.ejb.InvocationContext ctx) throws java.lang.Exception
Une classe interceptor ou une classe bean ne peut avoir qu'une seule méthode @AroundInvoke.
II. Avantages▲
- un contrôle plus fin dans le déroulement d'une méthode.
- on peut de façon élégante analyser et manipuler des paramètres et autoriser ou interdire l'exécution d'une méthode métier.
- on arrive ainsi à gérer des fonctionnalités techniques (logging, traçage, sécurité) de façon transparente dans les méthodes métiers.
III. Exemples d'utilisation▲
- contrôle des permissions sur des méthodes
- modification des paramètres avant qu'ils soient passés au bean ( InvocationContext.getParameters() )
- modifier la valeur retourner par un bean
- catcher et gérer les exceptions des méthodes
- totalement interrompre l'appel
- etc
IV. Le pattern Chain of Responsibility (CoR)▲
Les interceptors sont une forme du pattern Chain of Responsibility.
Ce pattern concerne une chaîne d'objets qui sont liés à travers des références indirectes pour permettre à un objet de traiter une requête ou la passer à un autre objet dans la chaîne.
C'est ce qui se passe avec les interceptors dans les EJB :
la méthode proceed() provoque l'appel de la prochaine méthode interceptor dans la chaîne.
Quand la dernière méthode interceptor AroundInvoke est appelée, la méthode proceed() appelle la méthode métier du bean.
La flèche ci-dessous illustre les enchaînements dans le cas de plusieurs intercepteurs appliqués à un bean:
1.
2.
3.
4.
5.
6.
@Interceptors({Interceptor1.class, Interceptor2.class, Interceptor3.class})
public class MonBean {
…
public void methode() {
…}
}
V. L'interface InvocationContext▲
L'interface InvocationContext permet de propager un état à travers une chaîne d'interceptors.
view plainprint?
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
package javax.interceptor;
public interface InvocationContext {
public Object getBean();
public java.lang.reflect.Method getMethod();
public Object[] getParameters();
public void setParameters(Object[] params);
public java.util.Map getContextData();
public Object proceed() throws Exception;
}
Il faut toujours appeler la méthode proceed() dans l'appel de l'objet invocationContext. Sinon, l'invocation de la méthode ne continue pas.
VI. Exemple▲
Voici un exemple d'un mini projet qui définit une classe interceptor, ainsi qu'une méthode interceptor à l'intérieur du bean.
Les deux sont appelées mais selon un ordre:
les interceptors sont appelés dans l'ordre dans lequel ils sont spécifiés mais les interceptors définis dans des classes externes sont éxécutées AVANT un interceptor définit dans une classe bean.
Dans cet exemple, elles affichent des traces dans les fichiers logs du serveur d'application, ainsi que dans le fichier de trace log4J.
Si on choisit le serveur d'application JBoss, il faut aller voir dans : {jboss-install-path}\server\default\log\server.log
La business interface BonjourRemote.java :
1.
2.
3.
4.
5.
6.
7.
8.
9.
package org.developpez.tutorial.stateless.bean;
import javax.ejb.Remote;
@Remote
public interface BonjourRemote {
public void bonjour();
public String aurevoir();
}
Avec l'annotation @Interceptors, on indique quelles classes externes vont être utilisées en tant qu'interceptors.
Toutes les méthodes du bean sont interceptées.
L'annotation @AroundInvoke est utilisée pour définir une méthode en tant qu'interceptor.
La classe bean BonjourBean.java :
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
32.
33.
34.
35.
36.
37.
38.
39.
40.
41.
package org.developpez.tutorial.stateless.bean;
import javax.ejb.Stateless;
import javax.interceptor.AroundInvoke;
import javax.interceptor.Interceptors;
import javax.interceptor.InvocationContext;
import org.apache.log4j.Logger;
@Stateless
@Interceptors(org.developpez.tutorial.stateless.bean.TraceInterceptor.class)
public class BonjourBean implements BonjourRemote{
private static org.apache.log4j.Logger log = Logger.getLogger(BonjourBean.class);
public void bonjour()
{
log.info("Bonjour toi");
}
public String aurevoir()
{
return "au revoir";
}
@AroundInvoke
public Object myBeanInterceptor(InvocationContext ctx) throws Exception
{
if (ctx.getMethod().getName().equals("bonjour"))
{
log.info("*** bonjour!");
}
if (ctx.getMethod().getName().equals("aurevoir"))
{
log.info("*** Au revoir!");
}
return ctx.proceed();
}
}
La classe interceptor TraceInterceptor.java :
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
32.
package org.developpez.tutorial.stateless.bean;
import javax.interceptor.AroundInvoke;
import javax.interceptor.InvocationContext;
import org.apache.log4j.Logger;
public class TraceInterceptor {
private static org.apache.log4j.Logger log = Logger.getLogger(TraceInterceptor.class);
@AroundInvoke
public Object log(InvocationContext ctx) throws Exception
{
log.info("*** Interception par TraceInterceptor");
long start = System.currentTimeMillis();
try
{
return ctx.proceed();
}
catch(Exception e)
{
throw e;
}
finally
{
long time = System.currentTimeMillis() - start;
String method = ctx.getClass().getName() + "." + ctx.getMethod().getName() + "()";
log.info("*** TracingInterceptor : l'invocation de " + method + " a pris " + time + "ms");
}
}
}
La classe Client.java :
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
package org.developpez.tutorial.client;
import javax.naming.InitialContext;
import org.apache.log4j.Logger;
import org.developpez.tutorial.stateless.bean.BonjourRemote;
public class Client
{
private static org.apache.log4j.Logger log = Logger.getLogger(Client.class);
public static void main(String[] args) throws Exception
{
InitialContext ctx = new InitialContext();
BonjourRemote bonjourRemote = (BonjourRemote)ctx.lookup("BonjourBean/remote");
bonjourRemote.bonjour();
log.info(bonjourRemote.aurevoir());
System.out.println("Et voila !");
}
}
Le fichier jndi.properties :
1.
2.
3.
java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory
java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces
java.naming.provider.url=localhost:1099
Le fichier log4j.properties :
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
log4j.appender.R=org.apache.log4j.RollingFileAppender
log4j.appender.R.File=sortie.log
log4j.appender.R.layout=org.apache.log4j.PatternLayout
log4j.appender.R.layout.ConversionPattern=%p %d{hh:mm:ss} %t %c{1} -%m%n
log4j.appender.R.MaxBackupIndex=1
log4j.appender.R.MaxFileSize=100KB
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%5p [%t] (%F:%L) -%m%n
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.rootLogger=stdout,R
log4j.appender.R.Append=true
En sortie, le fichier sortie.log (spécifique log4J) :
1.
2.
3.
4.
DEBUG 09:22:02 main SecurityAssociation -Using ThreadLocal: false
DEBUG 09:22:02 main Client -invoke called, but our invoker is disconnected, discarding and fetching another fresh invoker for: InvokerLocator [socket:
DEBUG 09:22:02 main SocketClientInvoker -connect called for: org.jboss.remoting.transport.socket.SocketClientInvoker@158f9d3
INFO 09:22:02 main Client -au revoir
En sortie, le fichier server.log (spécifique JBoss) :
1.
2.
3.
4.
5.
6.
7.
8.
2007-12-20 21:22:02,953 DEBUG [org.jboss.remoting.transport.socket.ServerThread] beginning dorun
2007-12-20 21:22:02,953 INFO [org.developpez.tutorial.stateless.bean.TraceInterceptor] *** Interception par TraceInterceptor
2007-12-20 21:22:02,953 INFO [org.developpez.tutorial.stateless.bean.BonjourBean] *** bonjour!
2007-12-20 21:22:02,953 INFO [org.developpez.tutorial.stateless.bean.BonjourBean] Bonjour toi
2007-12-20 21:22:02,953 INFO [org.developpez.tutorial.stateless.bean.TraceInterceptor] *** TracingInterceptor : l'invocation de org.jboss.ejb3.interceptor.InvocationContextImpl.bonjour() a pris 0ms
2007-12-20 21:22:02,968 INFO [org.developpez.tutorial.stateless.bean.TraceInterceptor] *** Interception par TraceInterceptor
2007-12-20 21:22:02,968 INFO [org.developpez.tutorial.stateless.bean.BonjourBean] *** Au revoir!
2007-12-20 21:22:02,968 INFO [org.developpez.tutorial.stateless.bean.TraceInterceptor] *** TracingInterceptor : l'invocation de org.jboss.ejb3.interceptor.InvocationContextImpl.aurevoir() a pris 0ms
On voit bien ici que la méthode log(InvocationContext ctx) définit dans la classe interceptor est exécutée avant celle définit dans la classe bean, myBeanInterceptor(InvocationContext ctx).
Cette technique d'interception se rapproche de la programmation par aspect (AOP): le code métier est séparé du code d'interception.
Une méthode interceptor est appelée un advice, et la classe contenant un advice est appelée un aspect.
On peut également appliquer une méthode interceptor à une seule méthode d'un bean, en annotant cette dernière :
1.
2.
3.
4.
@Interceptors(MaClass.class)
public void uneMethodeDuBean(int param){
...
}
On peut également définir les interceptors de façon déclarative dans le fichier de déploiement ejb-jar.xml :
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
<session>
<ejb-name>Bjour</ejb-name>
<remote>org.developpez.tutorial.stateless.bean.BonjourRemote</remote>
<ejb-class>org.developpez.tutorial.stateless.bean.BonjourBean </ejb-class>
...
<interceptor>
<class> TraceInterceptor </class>
<method-name> log </method-name>
</interceptor>
<interceptor>
<method-name> myBeanInterceptor</method-name>
</interceptor>
</session>
VII. Liens▲
VIII. Remerciements▲
Je tiens à remercier Ricky81, OButterlin pour les conseils, remarques et relectures, ainsi que RideKick pour la correction orthographique.
Je remercie aussi www.developpez.com me permettant de publier cet article et Nono40 pour ses outils.