Tutoriel pour apprendre à embarquer Jetty dans une application web Java

Image non disponible

Cet article explique pas à pas comment embarquer un conteneur Jetty dans sa propre application. Nul besoin d'utiliser Spring ou Scala.

Pour réagir au contenu de ce tutoriel, un espace de dialogue vous est proposé sur le forum 1 commentaire Donner une note à l'article (5).

Article lu   fois.

L'auteur

Site personnel

Liens sociaux

Viadeo Twitter Facebook Share on Google+   

I. Introduction

Une fois le développement d'une application web terminé, vient le moment (douloureux ou non) de son installation sur un serveur. En général, plusieurs prérequis sont nécessaires : JRE, serveur d'application, base de données… Aujourd'hui, Docker et/ou des outils comme Ansible et Puppet facilitent le provisionning du middleware. Néanmoins, il est possible de simplifier encore davantage cette phase d'installation. Des applications comme Sonar et Jenkins le font depuis des années : packager l'application avec son propre conteneur de Servlets et sa propre base de données. Afin de pouvoir déployer des applications les plus légères possible, les architectures microservices poussent dans ce sens. Et c'est d'ailleurs ce que proposent des frameworks comme Play Framework et Spring Boot. Ce dernier permet en effet de créer un JAR exécutable démarrant au choix un Tomcat ou un Jetty.

Pour distribuer votre application web, vous aurez le choix entre :

  1. Une archive ZIP contenant JAR, scripts shells et fichiers de configuration ;
  2. Un unique JAR auto-exécutable.

Le packaging est assuré par différents plugins Maven.

Disposer d'une JVM et le seul prérequis. Sachant qu'OpenJDK est installé sur la plupart des distributions Linux, ce n'est pas nécessairement une contrainte. Seule la version de Java devra être vérifiée avec soin.

II. Code source

Le code source utilisé pour illustrer ce billet provient du projet embedded-jetty-webapp hébergé sur GitHub. Pour des raisons de lisibilité, certaines parties ont été simplifiées.

Si vous souhaitez rendre autonome votre propre application, je vous conseille de vous inspirer directement du code disponible sur GitHub ( pom.xml maven et classes Java).

III. Dépendances Maven

Avant de pouvoir utiliser l'API de Jetty pour démarrer/arrêter un serveur, il faut tout d'abord tirer toutes les dépendances nécessaires au fonctionnement d'une application web. Voici la configuration Maven :

 
Sélectionnez
1.
2.
3.
4.
<properties>
    <version.javax-servlet>3.1.0</version.javax-servlet>
    <version.jetty>9.2.7.v20150116</version.jetty>
</properties>
 
Sélectionnez
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.
42.
43.
<!-- Jetty -->
<dependency>
    <groupId>org.eclipse.jetty</groupId>
    <artifactId>jetty-server</artifactId>
    <version>${version.jetty}</version>
</dependency>
<dependency>
    <groupId>org.eclipse.jetty</groupId>
    <artifactId>jetty-webapp</artifactId>
    <version>${version.jetty}</version>
</dependency>
<dependency>
    <groupId>org.eclipse.jetty</groupId>
    <artifactId>jetty-servlet</artifactId>
    <version>${version.jetty}</version>
</dependency>
<dependency>
    <groupId>org.eclipse.jetty</groupId>
    <artifactId>jetty-util</artifactId>
    <version>${version.jetty}</version>
</dependency>
<dependency>
    <groupId>org.eclipse.jetty</groupId>
    <artifactId>jetty-servlets</artifactId>
    <version>${version.jetty}</version>
</dependency>
<dependency>
    <groupId>org.eclipse.jetty</groupId>
    <artifactId>jetty-jsp</artifactId>
    <version>${version.jetty}</version>
</dependency>
<dependency>
    <groupId>org.eclipse.jetty</groupId>
    <artifactId>jetty-http</artifactId>
    <version>${version.jetty}</version>
</dependency>

<!--  Servlet API -->
<dependency>
    <groupId>javax.servlet</groupId>
    <artifactId>javax.servlet-api</artifactId>
    <version>${version.javax-servlet}</version>
</dependency>

Comme vous pouvez le constater, Jetty est particulièrement modulaire. Si vous utilisez JSP comme technologie de rendu, il faudra ajouter l'artefact jetty-jsp sous peine du message d'erreur « JSP support not configured ».

IV. Démarrer un Jetty

Manipuler l'API Jetty pour démarrer un conteneur de servlet depuis une classe Main ne présente pas de difficulté :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
public static void main(String[] args) throws Exception {

    Server server = new Server(8080);

    WebAppContext root = new WebAppContext();
    root.setContextPath("/");
    root.setDescriptor("webapp/WEB-INF/web.xml");
    URL webAppDir = Thread.currentThread().getContextClassLoader().getResource("webapp");
    if (webAppDir == null) {
        throw new RuntimeException("No webapp directory was found into the JAR file");
    }
    root.setResourceBase(webAppDir.toURI().toString());
    root.setParentLoaderPriority(true);

    server.setHandler(root);
    server.start();
}
2015-05-WAR-less-Java-web-application-with-Jetty.png

Une première subtilité réside dans l'utilisation du ClassLoader du thread courant. Sans quoi, en dehors d'un IDE, le répertoire webapp ne sera pas trouvé.

La seconde subtilité vient du fait que l'artefact construit est de type JAR et non un WAR. Bien qu'elle y ressemble, l'arborescence du projet n'est donc pas celle d'un WAR.

Le répertoire webapp ne se trouve pas dans le répertoire src/main/webapp, mais dans src/main/resources/webapp. Ainsi, lors de la construction du JAR, le répertoire webapp sera copié à la racine du JAR sans configuration maven particulière.

Dans notre exemple, l'application web utilise un descripteur de déploiement web.xml. Optionnel depuis Servlet 3.0, l'appel à la méthode setDescriptor est facultatif.

Enfin, le port HTTP utilisé dans notre exemple est le 8080. Ce dernier aurait pu être passé en paramètre du main() ou bien chargé depuis un fichier de configuration.

Lors de l'appel à la méthode start(), le conteneur Jetty démarre. L'application web est ensuite aussitôt démarrée. Il n'y a pas réellement de phase de déploiement.

V. Arrêter proprement Jetty

Pour arrêter le serveur, une solution peu recommandée est d'utiliser un kill -9 sur le PID du process Java. Les traitements en cours s'arrêtent brutalement et les ressources ne sont pas correctement libérées.

Une solution plus élégante est de demander au serveur Jetty de s'arrêter proprement. Le contexte de servlets est alors fermé par Jetty. Les listeners JEE implémentant l'interface ServletContextListener en sont notifiés.

Pour communiquer avec Jetty, une solution possible est d'utiliser un socket TCP. Je me suis grandement inspiré du code Java utilisé par le plugin Jetty pour maven.

Le principe est simple, un thread Monitor est démarré à la suite du serveur Jetty, et ceci dans la même JVM :

 
Sélectionnez
1.
2.
3.
4.
5.
server.start();

Monitor monitor = new Monitor(8090, new Server[] {server});
monitor.start();
server.join();

Ce thread démarre un SocketServer écoutant sur le port 8090. Il attend que l'instruction stop lui soit envoyée.

Pour davantage de détails, vous pouvez vous reporter à la méthode statique stop de la classe JettyServer ainsi qu'à la classe Monitor.

Une autre technique serait d'utiliser JMX pour communiquer avec Jetty. L'ajout du module jetty-jmx est alors nécessaire.

VI. Création du package

Comme je vous l'indiquais en introduction, je vous propose de packager votre application web de deux manières différentes.

VI-A. 1. Appassembler

Le plugin Appassembler pour maven permet de créer un répertoire target/appass__embler qu'il suffit de copier/coller pour installer l'application. Ce dernier contient trois sous-répertoires :

  1. bin : scripts start.sh, start.bat, stop.sh et stop.bat permettant de démarrer/arrêter la webapp. Ces scripts se chargent de trouver le JRE, sont compatibles avec cygwin et positionnent le classpath ;
  2. conf : facultatif, ce répertoire contient la configuration de l'application (fichiers properties ou YAML, logback.xml…) ;
  3. lib : tous les JAR nécessaires au fonctionnement de l'application.

Activé par défaut, le profile maven appassembler regroupe la configuration nécessaire :

 
Sélectionnez
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.
42.
43.
44.
45.
46.
47.
48.
49.
50.
51.
52.
53.
<profile>
    <id>appassembler</id>
    <activation>
        <activeByDefault>true</activeByDefault>
    </activation>
    <build>
        <plugins>
            <!-- Generate both Windows and Linux bash shell execution scripts -->
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>appassembler-maven-plugin</artifactId>
                <version>${version.plugin.appassembler-maven-plugin}</version>
                <executions>
                    <execution>
                        <goals>
                            <goal>assemble</goal>
                        </goals>
                    </execution>
                </executions>
                <configuration>
                    <repositoryLayout>flat</repositoryLayout>
                    <useWildcardClassPath>true</useWildcardClassPath>
                    <!-- Set the target configuration directory to be used in the bin scripts -->
                    <configurationDirectory>conf</configurationDirectory>
                    <!-- Copy the contents from "/src/main/config" to the target
                         configuration directory in the assembled application -->
                    <copyConfigurationDirectory>true</copyConfigurationDirectory>
                    <!-- Include the target configuration directory in the beginning of
                         the classpath declaration in the bin scripts -->                    
                    <includeConfigurationDirectoryInClasspath>true</includeConfigurationDirectoryInClasspath>
                    <!-- Extra JVM arguments that will be included in the bin scripts -->
                    <extraJvmArguments>-Xmx1024m</extraJvmArguments>

                    <programs>
                        <program>
                            <id>start</id>
                            <mainClass>com.javaetmoi.jetty.JettyServer</mainClass>
                            <name>start</name>
                        </program>
                        <program>
                            <id>stop</id>
                            <mainClass>com.javaetmoi.jetty.Stop</mainClass>
                            <name>stop</name>
                        </program>
                    </programs>
                    <binFileExtensions>
                        <unix>.sh</unix>
                    </binFileExtensions>
                </configuration>
            </plugin>
        </plugins>
    </build>
</profile>

Voici les commandes à exécuter pour tester ce type de packaging :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
git clone git://github.com/arey/embedded-jetty-webapp.git
cd embedded-jetty-webapp
mvn clean install
target/appassembler/bin/start.sh &
curl http://localhost:8080/HelloWorld
target/appassembler/bin/stop.sh

VI-B. 2. Assembly

L'une des fonctionnalités offertes par le plugin Assembly pour Maven est de rassembler tous les JAR d'une application en un seul gros JAR couramment suffixé par jar-with-dependencies (exemple : jetty-webapp-1.0.0-SNAPSHOT-jar-with-dependencies.jar). Afin de rendre ce JAR auto-exécutable, sa class main doit être spécifiée dans son manifeste.

Voici la configuration du profile maven flatjar :

 
Sélectionnez
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.
<profile>
    <id>fatjar</id>
    <build>
        <plugins>
            <plugin>
                <artifactId>maven-assembly-plugin</artifactId>
                <version>${version.plugin.maven-assembly-plugin}</version>
                <configuration>
                    <descriptorRefs>
                        <descriptorRef>jar-with-dependencies</descriptorRef>
                    </descriptorRefs>
                    <archive>
                        <manifest>
                            <mainClass>com.javaetmoi.jetty.JettyServer</mainClass>
                        </manifest>
                    </archive>
                </configuration>
                <executions>
                    <execution>
                        <id>make-assembly</id>
                        <phase>package</phase>
                        <goals>
                            <goal>single</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
</profile>

Voici les commandes à exécuter pour tester ce type de packaging :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
git clone git://github.com/arey/embedded-jetty-webapp.git
cd embedded-jetty-webapp
mvn clean install -Pflatjar
java -jar target/jetty-webapp-1.0.0-SNAPSHOT-jar-with-dependencies.jar &
curl http://localhost:8080/HelloWorld
java -cp target/jetty-webapp-1.0.0-SNAPSHOT-jar-with-dependencies.jar com.javaetmoi.jetty.Stop

VII. Conclusion

Par cet article, j'espère vous avoir convaincu de la facilité d'embarquer Jetty dans n'importe quelle application web. Tomcat s'intègre d'une manière similaire.

Avec cette approche, la mise à jour de Jetty ne nécessite qu'une simple montée de version de Jetty dans le pom.xml.

Autre atout : l'exécution d'un Jetty au démarrage de son application est profitable lors du développement. En effet, il n'est plus nécessaire d'installer et/ou d'utiliser le moindre plugin dans son IDE. L'application web est démarrée par un simple Run ou Debug sur la classe Main.

Références :

VIII. Remerciements

Cet article a été publié avec l'aimable autorisation d'Antoine Rey. L'article original (Embarquer Jetty dans une web app) a été rédigé par Antoine Rey.

Nous tenons à remercier Claude Leloup pour la relecture orthographique attentive de cet article et Mickael Baron pour la mise au gabarit.

Vous avez aimé ce tutoriel ? Alors partagez-le en cliquant sur les boutons suivants : Viadeo Twitter Facebook Share on Google+   

  

Copyright © 2016 Antoine Rey. Aucune reproduction, même partielle, ne peut être faite de ce site et de l'ensemble de son contenu : textes, documents, images, etc. sans l'autorisation expresse de l'auteur. Sinon vous encourez selon la loi jusqu'à trois ans de prison et jusqu'à 300 000 € de dommages et intérêts.