Skip to content

Commit

Permalink
added JWT page and fixed docs config
Browse files Browse the repository at this point in the history
  • Loading branch information
rocambille committed Jan 6, 2024
1 parent 6b1312d commit e05a78d
Show file tree
Hide file tree
Showing 2 changed files with 238 additions and 2 deletions.
236 changes: 236 additions & 0 deletions docs/JWT.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
[revoir l'épisode précédent](HASHING-PASSWORD)

## Sécuriser les requêtes

Lors de l'interaction avec un serveur, garantir que les requêtes proviennent d'utilisateurs authentifiés et autorisés est une étape cruciale pour assurer la sécurité de votre application.

Dans notre exemple, une vulnérabilité potentielle existe dans le fichier `backend/src/controllers/itemControllers.js`. Actuellement, la vérification de l'authentification se fonde sur la présence de `req.body.userId` :

```js
// Rappel : router.post("/items", itemControllers.add);

const add = async (req, res, next) => {
// Autorisé uniquement si authentifié... mais est-ce vraiment le cas ?
if (req.body.userId == null) {
res.sendStatus(401);
return;
}

// ...
};
```

Cette approche est vulnérable à des manipulations côté client.
Dans le code suivant (`frontend/src/pages/Home.jsx`), toute personne malveillante peut modifier la requête pour y injecter un ID de son choix :

```js
const response = await fetch(`${import.meta.env.VITE_BACKEND_URL}/api/items`, {
method: "post",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
title: titleRef.current.value,
userId: 42, // Le serveur pense que je suis le user 42
}),
});
```
Pour renforcer la sécurité, nous adopterons une approche basée sur l'utilisation de JSON Web Tokens (JWT) côté serveur.
L'idée est de générer un jeton JWT côté serveur après l'authentification réussie d'un utilisateur, puis d'exiger ce jeton dans les en-têtes de toutes les requêtes ultérieures.
Changez de branche dans votre projet avec la commande Git :
```bash
git switch jwt
```
Relancez le projet :
```bash
npm install
npm run dev
```
Après ces étapes, connectez vous sur [http://localhost:3000/login](http://localhost:3000/login) (adaptez le port au besoin).
Vous pouvez créer un nouvel utilisateur depuis la page [http://localhost:3000/register](http://localhost:3000/register) si vous avez oublié votre mot de passe 😉.
Rien n'a changé en apparence, mais vous pouvez voir quelques modifications avec le [diff GitHub](https://github.com/WildCodeSchool/workshop-js-auth/compare/hash-password...jwt). Nous allons regarder en détail quelques points en repartant du login.
## Login 2.0
Côté serveur, la modification que nous avons apporté est de générer un jeton JWT lors du processus de connexion, plutôt que de renvoyer simplement les informations de l'utilisateur.
Avant la modification dans `backend/src/controllers/authControllers.js` :
```js
const login = async (req, res, next) => {
// ... (votre code pour la validation de l'utilisateur)
res.json(user);
// ...
};
```

Après la modification :

```js
const jwt = require("jsonwebtoken");
const login = async (req, res, next) => {
// ... (votre code pour la validation de l'utilisateur)
const token = await jwt.sign(
{ sub: user.id, isAdmin: user.isAdmin },
process.env.APP_SECRET,
{
expiresIn: "1h",
}
);
res.json({
token,
user,
});
// ...
};
```

Avec cette modification, au lieu de simplement renvoyer les informations de l'utilisateur (`res.json(user)`), vous générez un jeton JWT en utilisant une clé secrète (`APP_SECRET`) et les données pertinentes.
Dans cet exemple, l'identifiant de l'utilisateur et s'il est administrateur.
Le jeton est ensuite renvoyé au client avec les informations de l'utilisateur.
Vous pouvez ajuster les données incluses dans le jeton selon vos besoins, mais le but est de le garder minimaliste.
Il sera envoyé avec chaque requête, et doit être le plus léger possible.
Vous devez y intégrer le minimum d'information qui seront utiles ensuite à votre serveur.
{: .alert-warning }

## Client 2.0

```js
const response = await fetch(`${import.meta.env.VITE_BACKEND_URL}/api/items`, {
method: "post",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${auth.token}`, // Inclusion du jeton JWT
},
body: JSON.stringify({
title: titleRef.current.value,
}),
});
```

Avec cette mise à jour côté client, chaque requête est désormais accompagnée du jeton JWT, renforçant ainsi la sécurité du processus d'authentification.
Le jeton ne peut être modifé sans la clé secrète pour le signer : le client ne peut plus usurper une identité sans casser l'authenticité du jeton.

## Authentication & Authorization

Pour sécuriser vos routes et garantir que seuls les utilisateurs authentifiés et autorisés accèdent à certaines fonctionnalités, vous pouvez créer des services d'authentification personnalisés.
```js
// Importation des services d'authentification pour les opérations de sécurité
const { hashPassword, verifyToken } = require("./services/auth");

// ...

// Cette route est protégée
router.post("/items", verifyToken, itemControllers.add);
```
Dans cet extrait, le middleware `verifyToken` est utilisé pour s'assurer que seuls les utilisateurs authentifiés peuvent faire une requête POST à la route `/items`.
Le middleware `verifyToken` est responsable de la vérification de l'authenticité et de la validité du jeton d'authentification.
Dans `backend/src/services/auth.js` :
```js
const verifyToken = (req, res, next) => {
try {
// Vérifier la présence de l'en-tête "Authorization" dans la requête
const authorizationHeader = req.get("Authorization");

if (authorizationHeader == null) {
throw new Error("Authorization header is missing");
}

// Vérifier que l'en-tête a la forme "Bearer <token>"
const [type, token] = authorizationHeader.split(" ");

if (type !== "Bearer") {
throw new Error("Authorization header has not the 'Bearer' type");
}

// Vérifier la validité du token (son authenticité et sa date d'expiration)
// En cas de succès, le payload est extrait et décodé
req.auth = jwt.verify(token, process.env.APP_SECRET);

next();
} catch (err) {
console.error(err);

res.sendStatus(401);
}
};
```
Ce middleware `verifyToken` vérifie la présence de l'en-tête d'autorisation dans la requête, valide le format du jeton, et s'assure de l'authenticité et de la validité du jeton à l'aide de la clé secrète (`APP_SECRET`).
Si le jeton est valide, le "payload" est extrait et décodé, puis attaché à l'objet de requête (`req.auth`) pour une utilisation ultérieure :
```js
// Rappel : router.post("/items", itemControllers.add);

const add = async (req, res, next) => {
// Nous savons qu'un utilisateur est authentifié
// Autorisé uniquement si administrateur
if (req.auth.isAdmin === false) {
res.sendStatus(403);
return;
}

// ...
};
```
Dans `itemControllers.add`, la présence de l'objet `req.auth` (attaché par le middleware `verifyToken`) est utilisée pour autoriser uniquement les utilisateurs administrateurs à accéder à cette fonctionnalité.
## Mur d'authentification
Reprenons le fichier `backend/src/router.js` :
```js
// Importation des services d'authentification pour les opérations de sécurité
const { hashPassword, verifyToken } = require("./services/auth");

// ...

// Cette route est protégée
router.post("/items", verifyToken, itemControllers.add);
```
Dans cet exemple, le middleware `verifyToken` est utilisé sur une seule route.
Il existe un moyen simple de l'utiliser sur un ensemble de routes :
```js
// Importation des services d'authentification pour les opérations de sécurité
const { hashPassword, verifyToken } = require("./services/auth");

// Avant le mur
// Cette route n'est pas protégée
router.get("/items", itemControllers.browse);

// ...

router.use(verifyToken); // Mur d'authentification

// Après le mur
// Cette route est protégée
router.post("/items", itemControllers.add);
```
Dans cet exemple, le middleware `verifyToken` est utilisé comme un "mur d'authentification".
Avant ce mur, les routes sont accessibles sans authentification.
Après le mur, les routes sont protégées et nécessitent une authentification valide.
## Et le reste ?
Si vous souhaitez explorer les détails du code lié à la gestion des tokens JWT dans cette branche, les modifications apportées sont visibles dans le [diff GitHub](https://github.com/WildCodeSchool/workshop-js-auth/compare/hash-password).
4 changes: 2 additions & 2 deletions docs/_config.yml
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,8 @@ defaults:
layout: "default" # possible values: default or tic-tac

style: dev # possible values: dev, data, ciber, design or wild.
title: "Auth with Express"
description: "abc"
title: "Auth"
description: "Développer une authentification sécurisée"
# type: ''
# main_image: '' # http or relative path to image
show_clone: true
Expand Down

0 comments on commit e05a78d

Please sign in to comment.