LOXODATA

Connexion à une instance PostgreSQL, deuxième article

2018-09-04   1099 mots, 6 minutes de lecture   Sébastien Lardière

Certificats de connexion

Ce deuxième article de la série présente une fonctionnalité sous-exploitée de PostgreSQL : l’authentification par certificat SSL.

Beaucoup d’administrateurs de bases de données et de développeurs, n’étant pas experts en sécurité, se contentent d’une authentification par mot de passe. Si la version 10 de PostgreSQL introduit l’algorithme SCRAM, améliorant l’authentification par mot de passe, le mot de passe reste stocké et connu par l’application, avec tous les problèmes que ça peut poser en matière de sécurité.

Quel que soit le mode d’authentification, il est possible de chiffrer tout échange entre le client et le serveur, ce qui limite les risques d’interception des données, à l’image de ce qu’est https pour http. L’échange chiffré des données n’est pas optionnel pour l’authentification basée sur des certificats.

Autorité de certification

La première étape consiste donc à créer une autorité de certification qui pourra délivrer des certificats décrivant l’identité du serveur ou des clients. Dans le contexte d’un échange un client et un serveur PostgreSQL, il n’est généralement pas nécessaire de recourir à une autorité de certification publique, car le client est très souvent connu du serveur à l’avance (et inversement), contrairement à un serveur Web.

Les commandes suivantes, effectuées en tant que root, vont créer une clé privée et le certificat signé :

openssl genrsa -des3 -out /etc/ssl/private/trustly-ca.key
chown root:root /etc/ssl/private/trustly-ca.key
chmod 400 /etc/ssl/private/trustly-ca.key

La clé privée n’est ici accessible qu’à l’utilisateur root.

openssl req -new -x509 -days 3650 \
  -subj '/C=FR/ST=BZH/L=Nantes/O=Loxodata/CN=trustly' \
  -key /etc/ssl/private/trustly-ca.key \
  -out /usr/local/share/ca-certificates/trustly-ca.crt

L’option -days permet d’indiquer la durée de validité du certificat ainsi signé. La valeur par défaut est 30 jours.

Le mot de passe demandé doit bien sûr être solide et secrètement conservé.

Cette clé et ce certificat peuvent être utilisés pour signer les clés de plusieurs serveurs et de plusieurs clients, de telle sorte qu’une organisation peut n’avoir qu’un seul jeu de fichier.

Ces fichiers n’ont pas besoin d’être distribués dans tous les systèmes concernés, mais seulement dans le ou les systèmes ou l’on signe les clés.

Dans cet article, le serveur et le client PostgreSQL fonctionnent dans le même système, pour simplifier la démonstration. Lors d’une utilisation réelle, il suffit de copier les fichiers dans le bon système pour créer les certificats et les fichiers racine (root.crt).

Certificats coté serveur

La connexion du client nécessitant une communication chiffrée, la première chose à faire est de créer une clé et un certificat pour le serveur :

su - postgres
mkdir ssl
cd ssl/
openssl genrsa -des3 -out server.key
openssl rsa -in server.key -out server.key

Le deuxième appel permet de créer une clé sans mot de passe, ce qui évite de devoir saisir ce mot de passe à chaque démarrage de l’instance PostgreSQL.

Demande

Ensuite, à partir de cette clé, on crée un fichier de demande de certificat :

openssl req -new -nodes -key server.key -out server.csr \
    -subj '/C=FR/ST=BZH/L=Nantes/O=Loxodata/CN=tolva'

L’option CN est le nom qui va effectivement être utilisé par le client lors de la connexion au serveur.

Il est possible d’examiner le contenu du fichier produit avec la commande :

openssl req -in server.csr -noout -text

Signature

Puis, la demande de certification est signée à partir de l’autorité de certification, effectuée en tant que root :

openssl req -x509 -key /etc/ssl/private/trustly-ca.key -days 360 \
    -in ~postgres/ssl/server.csr -out ~postgres/ssl/server.crt
chown postgres: ~postgres/ssl/server.crt

L’option -days permet d’indiquer la durée de validité du certificat ainsi signé. La valeur par défaut est 30 jours.

Il est possible d’examiner le contenu du fichier produit avec la commande :

openssl x509 -in ~postgres/ssl/server.crt -noout -text

Racine

Enfin, les certificats doivent être ajouté au fichier root.crt :

cat /usr/local/share/ca-certificates/trustly-ca.crt >> ~postgres/ssl/root.crt
chown postgres: ~postgres/ssl/root.crt
su - postgres
cat ~postgres/ssl/server.crt >> ~postgres/ssl/root.crt

Configuration de PostgreSQL

Les fichiers créés doivent être renseignés dans le fichier de configuration postgresql.conf, tout en activant le mode ssl si ça n’est pas déjà fait :

ssl = true
ssl_cert_file = '/var/lib/postgresql/ssl/server.crt'
ssl_key_file = '/var/lib/postgresql/ssl/server.key'
ssl_ca_file = '/var/lib/postgresql/ssl/root.crt'

Ces changements nécessitent un redémarrage de l’instance.

Ensuite, la règle d’accès autorisant la connexion depuis un certificat doit être ajoutée au fichier pg_hba.conf :

hostssl all slardiere 192.168.1.0/32 cert clientcert=1

Comme pour toute manipulation de ce fichier, il faut être attentif à l’ordre des lignes et à la hiérarchie induite des règles.

Certificats coté client

Le certificat coté client doit être placé dans le répertoire ~/.postgresql/. La clé privée ayant un mot de passe, celui-ci devra être saisi à chaque connexion, mais n’est jamais connu pas le serveur ni ne transite sur le réseau :

openssl genrsa -des3 -out ~/.postgresql/postgresql.key
openssl req -new -key ~/.postgresql/postgresql.key -out ~/.postgresql/postgresql.csr  \
        -subj '/C=FR/ST=BZH/L=Nantes/O=Loxodata/CN=slardiere'

L’option CN correspond au rôle de connexion connu par la base de données.

Signature

En tant que root, la demande de certification est signée :

openssl x509 -req -in /home/slardiere/.postgresql/postgresql.csr \
                  -CA /usr/local/share/ca-certificates/trustly-ca.crt \
                  -CAkey /etc/ssl/private/trustly-ca.key -days 360 \
                  -out /home/slardiere/.postgresql/postgresql.crt -CAcreateserial
chown postgres: ~postgres/ssl/postgresql.crt

L’option -days permet d’indiquer la durée de validité du certificat ainsi signé. La valeur par défaut est 30 jours.

Racine

Comme côté serveur, on ajoute les certificats au fichier root.crt, en tant que root :

cat /usr/local/share/ca-certificates/trustly-ca.crt >> ~slardiere/.postgresql/root.crt
chown slardiere: ~slardiere/.postgresql/root.crt

puis, en tant qu’utilisateur :

cat ~postgres/ssl/server.crt >> ~postgres/ssl/root.crt

Connexion

La connexion peut alors se faire, en précisant le mode ssl, parmi :

  • require : si le fichier root.crt est présent, vérifie que le certificat coté serveur vient bien de l’autorité de certification ;
  • verify-ca : vérifie que le certificat coté serveur vient bien de l’autorité de certification ;
  • verify-full : en plus que ce que fait verify-ca, et que le champ CN correspond au nom utilisé.
$ psql -h tolva "sslmode=verify-full"
Enter PEM pass phrase:
psql (10.5 (Debian 10.5-1.pgdg90+1))
SSL connection (protocol: TLSv1.2, cipher: ECDHE-RSA-AES256-GCM-SHA384, bits: 256, compression: off)
Type "help" for help.

tolva:5432 slardiere@postgres=>

Services

Il est possible de définir plusieurs connexions, en utilisant des fichiers différents que ci-dessus, avec les options de connexions : sslcert sslkey et sslrootcert. Pour cela, le plus simple est de définir un service dans le fichier ~/.pg_service.conf :

[tolva]
host=tolva
sslmode=verify-full
sslcert=/home/slardiere/.postgresql/postgresql.crt
sslkey=/home/slardiere/.postgresql/postgresql.key
sslrootcert=/home/slardiere/.postgresql/root.crt

puis d’utiliser le service :

$ psql service=tolva
Enter PEM pass phrase:
psql (10.5 (Debian 10.5-1.pgdg90+1))
SSL connection (protocol: TLSv1.2, cipher: ECDHE-RSA-AES256-GCM-SHA384, bits: 256, compression: off)
Type "help" for help.

tolva:5432 slardiere@postgres=>

Il est donc possible de créer autant de services que nécessaire, en indiquant les chemins vers les fichiers correspondants.

Conclusion

En maitrisant les processus de création des clés et des certificats, il n’est pas compliqué de mettre en place des connexions sécurisées, évitant ainsi de voir des mots de passe en clair dans des fichiers, tout en contrôlant la création des certificats pour son organisation.