wiki/HowtoTomcat.md

406 lines
12 KiB
Markdown
Raw Permalink Normal View History

2017-02-20 17:06:36 +01:00
---
2017-02-23 04:45:21 +01:00
categories: web java
2017-02-20 17:06:36 +01:00
title: Howto Tomcat
...
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
* Documentation : <http://tomcat.apache.org/tomcat-8.0-doc/>
* Rôle Ansible : <https://forge.evolix.org/projects/ansible-roles/repository/revisions/stable/show/tomcat-instance>
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
[Tomcat](http://tomcat.apache.org/) est un serveur d'applications Java : il rend accessible des pages JSP (moteur _Jasper_) et des servlets Java (moteur _Catalina_) via le protocole [HTTP](HowtoHTTP) (connecteur _Coyote_) ou [AJP (Apache JServ Protocol)](https://fr.wikipedia.org/wiki/Protocole_Apache_JServ). On l'utilise en général avec [Apache](HowtoApache) en complément.
2017-02-20 17:06:36 +01:00
## Installation
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
~~~
# apt install openjdk-7-jdk tomcat8 tomcat8-user
# /usr/share/tomcat8/bin/version.sh
Using CATALINA_BASE: /usr/share/tomcat8
Using CATALINA_HOME: /usr/share/tomcat8
Using CATALINA_TMPDIR: /usr/share/tomcat8/temp
Using JRE_HOME: /usr
Using CLASSPATH: /usr/share/tomcat8/bin/bootstrap.jar:/usr/share/tomcat8/bin/tomcat-juli.jar
Server version: Apache Tomcat/8.0.14 (Debian)
Server built: Feb 13 2017 09:37:48
Server number: 8.0.14.0
OS Name: Linux
OS Version: 3.16.0-4-amd64
Architecture: amd64
JVM Version: 1.7.0_121-b00
JVM Vendor: Oracle Corporation
~~~
> *Note* : on peut également installer Tomcat 7 avec les paquets _tomcat7*_
## Configuration
La configuration système de Tomcat, et notamment les paramètres de la JVM, se trouve dans le fichier `/etc/default/tomcat8` que l'on adapte ainsi :
~~~
# Rare sont les cas ou l'on peut activer le Java security manager
TOMCAT6_SECURITY=no
# Reglages memoire de base (a adapter selon votre quantite de RAM, dans l'exemple on a une seule instance et 4 Go de RAM)
JAVA_OPTS="${JAVA_OPTS} -Xms2g -Xmx2g -XX:NewSize=512m -XX:MaxPermSize=256m -Xss256k"
2017-02-23 04:45:21 +01:00
# Reglages du garbage collector
JAVA_OPTS="${JAVA_OPTS} -XX:+UseConcMarkSweepGC -XX:+CMSIncrementalMode -XX:CMSInitiatingOccupancyFraction=80"
# Compression des adresses 64bits sur 32bits pour gagner en memoire libre
JAVA_OPTS="${JAVA_OPTS} -XX:+UseCompressedOops"
# file limits pour eviter les "too many open files"
ulimit -n 8192
~~~
La configuration des options de Tomcat se trouve dans le répertoire `/etc/tomcat8/` notamment dans le fichier `/etc/tomcat8/server.xml` :
~~~{.xml}
<?xml version='1.0' encoding='utf-8'?>
<Server port="8005" shutdown="SHUTDOWN">
<Listener className="org.apache.catalina.startup.VersionLoggerListener" />
<Listener className="org.apache.catalina.core.JreMemoryLeakPreventionListener" />
<Listener className="org.apache.catalina.mbeans.GlobalResourcesLifecycleListener" />
<Listener className="org.apache.catalina.core.ThreadLocalLeakPreventionListener" />
<GlobalNamingResources>
<Resource name="UserDatabase" auth="Container"
type="org.apache.catalina.UserDatabase" description="User database that can be updated and saved"
factory="org.apache.catalina.users.MemoryUserDatabaseFactory" pathname="conf/tomcat-users.xml" />
</GlobalNamingResources>
<Service name="Catalina">
<Connector port="8080" protocol="HTTP/1.1" connectionTimeout="20000" URIEncoding="UTF-8" redirectPort="8443" />
<Engine name="Catalina" defaultHost="localhost">
<Realm className="org.apache.catalina.realm.LockOutRealm">
<Realm className="org.apache.catalina.realm.UserDatabaseRealm" resourceName="UserDatabase"/>
</Realm>
<Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="true">
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="localhost_access_log" suffix=".txt" pattern="%h %l %u %t &quot;%r&quot; %s %b" />
</Host>
</Engine>
</Service>
</Server>
~~~
On peut ainsi désactiver le déploiement automatique (à chaud) en modifiant :
~~~{.xml}
<Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="false">
~~~
Ou encore changer les paramètres du connecteur HTTP (port TCP utilisé, _timeout_ de la connexion) via :
~~~{.xml}
<Connector port="8765" protocol="HTTP/1.1" connectionTimeout="5000" URIEncoding="UTF-8" redirectPort="8443" />
~~~
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
## Arborescence
2017-02-21 14:42:46 +01:00
2017-02-23 04:45:21 +01:00
Les fichiers de _Tomcat_ sont dispersés, voici les répertoires importants :
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
* CATALINA_BASE : ${catalina.base} = /var/lib/tomcat8
* CATALINA_HOME : ${catalina.home} = /usr/share/tomcat8
Le répertoire CATALINA_BASE contient :
~~~
/var/lib/tomcat8/
├── conf -> /etc/tomcat8
├── lib
├── logs -> /var/log/tomcat8
├── policy
│   └── catalina.policy
├── webapps (répertoire où l'on peut déposer les servlets)
│   └── ROOT
│   ├── index.html
│   └── META-INF
│   └── context.xml
└── work -> /var/cache/tomcat8
~~~
Un élément important concerne les classes Java qui peuvent être déposées à différents endroits.
L'ordre de chargement de ces classes (_Class Loader_) se fait ainsi :
* /usr/share/tomcat6/lib/ : classes communes à toutes les servlets
* /var/lib/tomcat8/webapps/foo/WEB-INF/lib/ : classes spécifiques à la servlet foo
## Administration
### Manager Tomcat
On peut installer le _Manager Tomcat_, une interface web permettant notamment déployer des servlets Java :
~~~
# apt install tomcat8-admin
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
Il faut ensuite ajouter un utilisateur au rôle _manager_, dans le fichier `/etc/tomcat8/tomcat-users.xml` :
~~~{.xml}
<tomcat-users>
[...]
<role rolename="manager"/>
<user username="foo" password="PASSWORD" roles="manager"/>
</tomcat-users>
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
L'accès au _Manager Tomcat_ se fait sur http://127.0.0.1:8080/manager/html
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
## Instances Tomcat
On peut faire tourner plusieurs instances Tomcat complètement indépendantes, avec leurs propres réglages et en permettant des arrêts/redémarrages d'une instance sans impacter les autres, le tout en mutualisant les binaires.
2022-01-26 16:08:15 +01:00
### Installation et configuration d'une instance
2017-02-23 04:45:21 +01:00
Pour cela, on s'appuye sur [systemd](HowtoSystemd) pour lequel il faut installer la bibliothèque PAM :
2017-02-21 14:42:46 +01:00
~~~
# apt install libpam-systemd
~~~
2017-02-23 04:45:21 +01:00
Et l'on crée une unité systemd `/etc/systemd/user/tomcat.service` :
2016-12-29 11:25:39 +01:00
2017-02-20 17:21:00 +01:00
~~~{.ini}
2016-12-29 11:25:39 +01:00
[Unit]
2017-02-21 14:42:46 +01:00
Description=Tomcat %u.
2016-12-29 11:25:39 +01:00
After=network.target
[Service]
2017-02-20 17:06:36 +01:00
WorkingDirectory=%h
Environment="CATALINA_BASE=%h"
EnvironmentFile=%h/conf/env
2016-12-29 11:25:39 +01:00
UMask=0002
ExecStart=/usr/share/tomcat8/bin/catalina.sh run
2017-02-21 14:42:46 +01:00
# Pour Tomcat 7
#ExecStart=/usr/share/tomcat7/bin/catalina.sh run
SyslogIdentifier=tomcat@%u
2016-12-29 11:25:39 +01:00
[Install]
WantedBy=default.target
~~~
2017-02-23 04:45:21 +01:00
On peut ensuite créer une instance nommée _foo_ avec un utilisateur dédié :
2016-12-29 11:25:39 +01:00
~~~
2017-02-20 17:06:36 +01:00
# mkdir -p /srv/tomcat
2017-02-23 04:45:21 +01:00
# tomcat8-instance-create /srv/tomcat/foo
# useradd -d /srv/tomcat/foo foo
# chown -R app:app /srv/tomcat/foo
# chmod -R u=rwX,g=rX,o= /srv/tomcat/foo
# chmod -R g+ws /srv/tomcat/foo
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
Créer les variables d'environnement dans `/srv/tomcat/foo/conf/env` :
2016-12-29 11:25:39 +01:00
2017-02-20 17:21:00 +01:00
~~~{.bash}
2017-02-20 17:06:36 +01:00
# Memory allocation options.
# Xmx Max memory allocated to instance.
# Xms Allocated memory at startup.
# XX:MaxPermSize Memory allocated to internal objects.
2017-02-20 17:21:00 +01:00
JAVA_HOME="/usr/lib/jvm/java-1.7.0-openjdk-amd64"
JAVA_OPTS="${JAVA_OPTS} -server -Xms2g -Xmx2g -XX:NewSize=512m -XX:MaxPermSize=256m -Xss256k"
2017-02-23 04:45:21 +01:00
JAVA_OPTS="${JAVA_OPTS} -XX:+UseConcMarkSweepGC -XX:+CMSIncrementalMode -XX:CMSInitiatingOccupancyFraction=80"
JAVA_OPTS="${JAVA_OPTS} -XX:+UseCompressedOops"
JAVA_OPTS="${JAVA_OPTS} -XX:+UseParNewGC -XX:+CMSPermGenSweepingEnabled -XX:+CMSClassUnloadingEnabled -Xverify:none"
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
Il reste ensuite à ajuster les ports HTTP et SHUTDOWN dans `/srv/tomcat/app/conf/server.xml` :
2016-12-29 11:25:39 +01:00
2017-02-20 17:21:00 +01:00
~~~{.xml}
<Server port="SHUTDOWN_PORT" shutdown="SHUTDOWN">
<Connector port="HTTP_PORT" protocol="HTTP/1.1" connectionTimeout="20000" URIEncoding="UTF-8" redirectPort="8443"/>
~~~
2017-02-23 04:45:21 +01:00
Enfin, il est nécessaire d'activer la conservation de la session de l'utilisateur :
2016-12-29 11:25:39 +01:00
~~~
2017-02-20 17:06:36 +01:00
# loginctl enable-linger foo
2016-12-29 11:25:39 +01:00
~~~
2022-01-26 16:08:15 +01:00
### Gestion d'une instance
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
Activation au démarrage de l'instance :
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
# su - foo
$ systemctl --user enable tomcat
2017-02-20 17:06:36 +01:00
~~~
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
Démarrage de l'instance :
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
~~~
# su - foo
$ systemctl --user start tomcat
~~~
2017-02-20 17:21:00 +01:00
2017-02-23 04:45:21 +01:00
Arrêt de l'instance :
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
# su - foo
$ systemctl --user stop tomcat
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
Redémarrage de l'instance :
2017-02-20 17:21:00 +01:00
~~~
2017-02-23 04:45:21 +01:00
# su - foo
$ systemctl --user restart tomcat
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
Afficher le statut de l'instance :
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
# su - foo
$ systemctl --user status -l tomcat
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
## Configuration avec Apache
Nous conseillons d'utiliser [Apache](HowtoApache) comme reverse-proxy HTTP/HTTPS devant _Tomcat_.
Pour cela on conseille d'utiliser le module *proxy_http* avec le connecteur HTTP de _Tomcat_.
On peut également utiliser le module mod-jk avec le connecteur AJP de _Tomcat_.
### avec proxy_http
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
# a2enmod proxy_http
2017-02-20 17:21:00 +01:00
~~~
2017-02-23 04:45:21 +01:00
Voici un VirtualHost type :
2017-02-21 18:02:08 +01:00
2017-02-23 04:45:21 +01:00
~~~{.apache}
<VirtualHost *:80>
ServerName www.example.com
<IfModule mod_proxy_http.c>
ProxyPass / http://127.0.0.1:8080/
ProxyPassReverse / http://127.0.0.1:8080/
<Proxy http://127.0.0.1:8080/>
Require all granted
2017-02-23 04:45:21 +01:00
</Proxy>
</IfModule>
</VirtualHost>
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
### avec mod-jk
~~~
# apt install libapache2-mod-jk
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
Il faut ensuite s'assurer d'avoir activer le connecteur AJP dans le fichier `server.xml` :
2017-02-21 18:02:08 +01:00
2017-02-23 04:45:21 +01:00
~~~{.xml}
<Connector port="8009" protocol="AJP/1.3" redirectPort="8443" />
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
On édite `/etc/libapache2-mod-jk/workers.properties` pour lister les instances Tomcat concernées :
2017-02-21 18:02:08 +01:00
~~~
2017-02-23 04:45:21 +01:00
workers.tomcat_home=/usr/share/tomcat8
workers.java_home=/usr/lib/jvm/default-java
worker.list=ajp13_worker
worker.ajp13_worker.port=8009
worker.ajp13_worker.host=localhost
worker.ajp13_worker.type=ajp13
~~~
Voici un VirtualHost type :
~~~{.apache}
<VirtualHost *:80>
ServerName www.example.com
2017-11-07 10:16:56 +01:00
# Tout rediriger vers l'instance :
2017-02-23 04:45:21 +01:00
JkMount /* ajp13_worker
2017-11-07 10:16:56 +01:00
# Ne pas rediriger une destination spécifique :
JkUnMount /favicon.ico ajp13_worker
JkUnMount /documents/* ajp13_worker
2017-02-23 04:45:21 +01:00
</VirtualHost>
~~~
> *Note* : les options pour _libapache2-mod-jk_ sont configurables dans le fichier `/etc/apache2/mods-available/jk.conf`
2017-02-21 18:02:08 +01:00
2017-02-23 11:10:35 +01:00
## Monitoring
### Nagios
2017-02-23 14:15:45 +01:00
Vérification simple du port HTTP :
2017-02-23 11:10:35 +01:00
~~~
2017-02-23 14:15:45 +01:00
$ /usr/lib/nagios/plugins/check_http -H 127.0.0.1 -p 8080
~~~
Vérification simple du port AJP :
~~~
$ /usr/lib/nagios/plugins/check_tcp -H 127.0.0.1 -p 8009
2017-02-23 11:10:35 +01:00
~~~
2017-02-20 17:21:00 +01:00
2017-02-23 04:45:21 +01:00
## FAQ
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
### access_log de Tomcat
_Tomcat_ peut générer des journaux d'accès similaires à ceux d'Apache. On configure cela via le fichier `server.xml` :
2017-01-03 11:20:35 +01:00
2017-02-20 17:06:36 +01:00
~~~{.xml}
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="localhost_access_log." suffix=".txt" pattern="common" resolveHosts="false"/>
2016-12-29 11:25:39 +01:00
~~~
2017-02-20 17:06:36 +01:00
### Connecteur HTTP
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
Pour activer la compression GZIP de certains types de fichiers, ajouter les options suivantes dans le connecteur HTTP via le fichier `server.xml` :
2017-01-03 11:20:35 +01:00
2016-12-29 11:25:39 +01:00
~~~
2017-02-20 17:06:36 +01:00
compression="on" compressionMinSize="2048" compressableMimeType="text/html,text/xml,text/plain,text/javascript,application/javascript"
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
### Drivers JDBC
2016-12-29 11:25:39 +01:00
2017-02-23 04:45:21 +01:00
Pour installer le driver JDBC pour MySQL :
2017-01-03 11:20:35 +01:00
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
# aptitude install libmysql-java
# cd /usr/share/tomcat6/lib/
# ln -s ../../java/mysql-connector-java.jar mysql.jar
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
Pour installer le driver JDBC pour PostgreSQL :
2017-01-03 11:20:35 +01:00
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
# aptitude install libpg-java
# cd /usr/share/tomcat6/lib/
# ln -s ../../java/mysql-connector-java.jar mysql.jar
2016-12-29 11:25:39 +01:00
~~~
2017-02-23 04:45:21 +01:00
### Qu'est-ce que c'est JSVC ? Pourquoi l'utiliser ?
JSVC est un outil permettant d'uniformiser le lancement de "daemon" en JAVA pour tous les systèmes, notamment sous Windows. Sous Linux, il est parfois utilisé, mais son utilisation reste déconseillée vu qu'il existe d'autres outils pour faire cela *et* que son utilisation ne permet pas une gestion via JMX. Sous Debian, il a été utilisé sous Debian Lenny, mais il n'est plus utilisé dans les versions suivantes.
### Tomcat n'écoute pas en IPv4, pourquoi ? Comment le corriger ?
Les versions récentes de Java préfèrent l'IPv6 si elle est présente. Le problème c'est que dans ce cas, il peut arriver que tomcat n'écoute pas en IPv4. Il faut alors modifier les propriétés par défaut en ajoutant ceci aux JAVA_OPTS :
~~~
-Djava.net.preferIPv4Stack=true
~~~
### Problème taille upload
Le vhost doit forcer l'envoi du header Content-Length, il ne le fait pas toujours si le contenu est trop large. Voir https://httpd.apache.org/docs/current/mod/mod_proxy.html#request-bodies pour plus de détails.
~~~
SetEnv proxy-sendcl
~~~