Skip to content
This repository has been archived by the owner on Apr 24, 2020. It is now read-only.

Latest commit

 

History

History
685 lines (489 loc) · 20.7 KB

README.md

File metadata and controls

685 lines (489 loc) · 20.7 KB

Compte-rendu du devoir final Frontend-Backend

Le projet est disponible à cette adresse:

luzin.hazegard.fr

Bilan du travail

  • Mise en place d'un repository sur Gitlab
  • Mise en place de Bootstrap et jQuery
  • Mise en place d'une base de données MySQL contenant les articles et les paragraphes
  • Mise en place d'une API REST permettant de gérer les articles et leurs paragraphes
  • Mise en place d'un front-office fournissant une interface à l'utilisateur et requêtant l'API

Choix effectués

Langages et librairies utilisées

Au niveau du Back-End, les choix se sont portés sur:

  • Apache comme serveur web
  • Php 7 afin d'apporter un typage, ermettant une meilleure robustesse du code
  • apidoc pour générer une documentation de l'api
  • MariaDB/MySQL comme SGBD

Au niveau du Front-End :

  • Bootstrap comme librairie pour la présentation
  • jQuery pour l'interaction

Structure générale

Nous avons décidé de nous diriger vers l'utilisation d'une API Restful afin d'une part, de découpler le développement du Back-End et celui du Front-End.

Ainsi, cela nous a permis de travailler dans deux dossiers complétements séparés, nous permettant d'éviter les régression de l'un lorsque l'on modifie la structure de l'autre.

Nous avons ainsi choisi de de réaliser notre Back-End en PHP 7, en utilisant le typage des arguments de fonctions, ainsi que le typage de retour de fonction, afin d'augmenter la robustesse du code produit.

Concernant le Front-End, nous utilisont la librairie jQuery pour réaliser les interactions avec l'utilisateur, ainsi que la librairie Bootstrap pour la présentation.

Le serveur sur lequel notre projet s'execute est apache, avec une configuration que nous détaillerons ultérieurement.

Présentation du Back-End

Le Framework

Nous avons décidé d'écrire notre propre Framework afin de réaliser notre API. En effet, cela nous permettait d'avoir une bone compréhension de son fonctionement, tout en étant léger car adapté à nos besoins.

Ainsi le routage s'effectue à l'aide de regex, qui nous permet de facilement faire évoluer nos routes, mais également de facilement récupérer les paramétres éventuels nécessaire au bon fonctionnement de l'API, par exemple, dans le cas de la route:

/api/v1/articles/5/paragraphs/1

Nous voulons récupérer le 5 associé à un ID d'article, ainsi que le 1, associé à un ID de paragraphe.

Cela est, grâce à l'utilisation de regex, facilement réalisable, par exemple en utilisant cette regex :

~^/articles/(\d+)/paragraphs/(\d+)/?$~

Ainsi, le Framework se compose de 3 classes :

  • Route.php
  • Router.php
  • RouterUtils.php

Route.php

La classe Route est un objet qui contient:

  • L'expression régulière qui doit correspondre
  • La méthode HTTP qui doit correspondre
  • La fonction de callback utilisé lorsque la Route est apelée

Router.php

La classe Router.php se charge de lister l'ensemble des routes de l'application. Elle est également appelée lors d'une nouvelle requête afin d'essayer de trouver une route qui correspond à la requête.

RouterUtils.php

Cette classe permet de réaliser divers traitements sur la requete:

  • Récupérer la partie intéressante de la route (retirer la partie /api/v1)
  • Récupérer leBody de la requête
  • Executer le callback de la route
  • Emettre la réponse au client

Utilisation

Configuration de la base de donnée

Une base de donnée MySQL ou MariaDB est nécessaire pour faire fonctionnet le projet. Afin d'initier le contenu, un utilisateur ainsi qu'une base sont nécessaire.

Ensuite, il suffit d'exécuter la commande suivante pour intialiser les tables:

source {/path/to/the/project}/Back/src/script.sql

Configuration d'apache

Le module rewrite est nécessaire :

a2enmod rewrite

Ajouter :

AllowOverride All

Afin d'activer le .htaccess

Configuration du serveur php

Renommer Back/src/config.example.php en config.php

Remplir les champs:

  • $host correspond à l'adresse où la base de donnée se situe (ex: localhost)
  • $dbname est le nom donné à la base préalablement créé
  • $username et $password informations de connexion à la base

Fonctionnement

La manière dont sont transmis les paramètres s'effectue de la manière suivante:

  1. La classe Router récupére les paramètres issues des parenthèses capturantes de l'expression régulière.

  2. Ils sont transmis dans un tableau et dans l'ordre d'apparition au sein de l'expression régulière.

  3. La fonction match() renvoie ainsi un tableau contenant comme premier argument le callback de la route, et comme deuxième argument, un tableau des identifiants récupérés par la regex.

  4. Au sein de executeRoute() de RouterUtils, le tableau [callback, params] et les données du Body sont transformé pour donner un tableau associatif contenant les paramètres et les données du Body.

En résumé, le callback reçoit un unique paramètre qui est un tableau associatif et qui contient;

  • Dans URL_PARAMS, un tableau contenant les paramètres récupérés dans l'url de la requete
  • Dans BODY_DATA, un tableau associatif correspondant au json envoyé dans la requête

Le callback se charge ainsi simplement d'extraire les valeurs des paramètres et des données du Body, et au besoin, d'affecter des valeurs par défaut dans le cas d'arguments facultatifs.

  1. Le Callback extrait les valeurs des paramètres des données du Body, et assigne des valeurs par défaut si besoin.
  2. Le controlleur, appelé par le callback, vérifie la validité des arguments extraits par le callback.
  3. Le modèle, appelé par le controlleur, construit la requête SQL adapté, avec la éventuels paramètres.
  4. DBAccess, appelé par le modèle, se charge d'executer la requête.
  5. Le controlleur reçoit la réponse de DBAccess, vérifie sa validité, et transforme le résultat en json.
  6. Le callback apelle la fonction response() de RouterUtils avec comme paramètre le résultat du contrôlleur pour l'émmettre au client.

Présentation du Front-End

Vue la nature du travail demandé, nous avons choisi pour le Front-End de faire une Single Page Application. Le Front-End ne comprend donc qu'une seule page qui affiche dynamiquement les informations demandées.

Affichage par défaut

À l'arrivée sur la page, l'utilisateur ne voit qu'un message de présentation et la barre de navigation (qui sera toujours présente), depuis laquelle il peut :

  • Réafficher ce "message de garde" en cliquant sur le titre de la page
  • Voir la liste des articles présents dans la base de données depuis le menu déroulant
  • Ajouter un article dans le formulaire prévu à cet effet

Affichage d'un article en mode vue

C'est l'affichage par défaut lorsqu'on sélectionne un article depuis le menu déroulant. Il affiche simplement le titre et les paragraphes concernant l'article. Il y a également un bouton pour passer en "mode édition".

Affichage d'un article en mode édition

C'est dans ce mode, moins adapté à la lecture, qu'on peut modifier les données concernant l'article en question. C'est aussi l'affichage par défaut de l'article après l'avoir créé.

  • Il y a deux boutons, l'un pour revenir en mode vue, l'autre pour supprimer totalement l'article.
  • On peut cliquer sur le titre ou sur un paragraphe pour en modifier le contenu. Il faut valider les changements en appuyant sur ENTRÉE, ou les annuler en appuyant sur ÉCHAP.
  • On peut cliquer sur le bouton + à la fin d'un paragraphe pour ajouter un nouveau paragraphe juste après. Le fonctionnement est similaire à l'édition.
  • On peut cliquer sur le bouton X qui apparaît au survol d'un paragraphe pour le supprimer.
  • On peut glisser-déposer les paragraphes pour les réarranger.

Perspectives d'amélioration

  • Mettre en place le HTTPS
  • Mettre en place de l'écriture collaborative : voir les gens en train de modifier ou de lire un article
  • Intégrer des sous-titres (en tant que paragraphe spécial) à l'article
  • Sécuriser l'API

Bilan du module

Nous avons dans l'ensemble apprécié le module, nous donnant un aperçu des technologies front/back et du type de structure auquel on peut être amené à être confronté.
Le gros plus a été le fait de faire participer les élèves autant que possible, en leur laissant le temps de réfléchir aux petits exercices ou via le biais de ce devoir final. On apprend en codant !
On pourra cependant peut-être regretter la présentation de certaines technologies ayant mal vieilli ou de certaines pratiques n'étant plus vraiment d'actualité.

APIDOC

Une version web de l'apidoc est disponible à cette adresse:

luzin.hazegard.fr/api/doc/

AllArticles

Insert a new Article

Back to top

POST /api/v1/articles

Parameter Parameters

Name Type Description
TITLE String

The Title of article to create

Success Response

Success-Response:

HTTP/1.1 201 OK
{
    "ID": 1,
    "TITLE" : "Lorem Ipsum"
}

Success 200

Name Type Description
ID Number

Id of the article created

TITLE String

Title of the article created

Request all articles

Back to top

GET /api/v1/articles

Success Response

Success-Response:

HTTP/1.1 200 OK
[
    {
        "ID": 1,
        "TITLE" : "Lorem Ipsum"
    },
    {
        "ID": 2,
        "TITLE" : "The game"
    },
    ...
]

Success 200

Name Type Description
ID Number

Id of the article

TITLE String

Title of the article

Request all articles with their paragraphs

Back to top

GET /api/v1/articles?paragraphs=true

Success Response

Success-Response:

HTTP/1.1 200 OK
[
    {
        "ID": 1,
        "TITLE" : "Lorem Ipsum",
        "CONTENT": [
                {
                    "ID": 1,
                    "CONTENT": "Lorem ipsum dolor sit amet.",
                    "POSITION": 1,
                    "ARTICLE_ID": 1
                },
                {
                    "ID": 2,
                    "CONTENT": "Ut enim ad minim veniam.",
                    "POSITION": 2,
                    "ARTICLE_ID": 1
                },
                ...
    },
    {
        "ID": 2,
        "TITLE" : "The game",
        "CONTENT" : [
                {
                    "ID": 3,
                    "CONTENT": "Perdu !",
                    "POSITION": 1,
                    "ARTICLE_ID": 2
                },
                ...
    },
    ...
]

Success 200

Name Type Description
ID Number

Id of the article

TITLE String

Title of the article

CONTENT Object[]

List of paragraphs

CONTENT.ID Number

Id of the article patched

CONTENT.TITLE String

Title of the article patched

CONTENT.POSITION Number

The position of the paragraph in the article

CONTENT.ARTICLE_ID Number

The Id of the article associated to the paragraph

AllParagraphs

Request a paragraph

Back to top

GET /api/v1/paragraphs/:id

Success Response

Success-Response:

HTTP/1.1 200 OK
{
        "ID": 1,
        "CONTENT": "Lorem ipsum dolor sit amet.",
        "POSITION": 1,
        "ARTICLE_ID": 1
}

Success 200

Name Type Description
ID Number

Id of the paragraph

CONTENT String

Content of the article

POSITION Number

The position of the paragraph in the article

ARTICLE_ID Number

The Id of the article associated to the paragraphe

Article

Delete an article

Back to top

DELETE /api/v1/articles/:id

Success Response

201 Success-Response:

HTTP/1.1 201 OK
{
    "Response" : "Successfully deleted article with ID <code>ID</code>",
}

Success 200

Name Type Description
ID Number

Id of the deleted article

Error 4xx

Name Type Description
ArticleNotFound

No article with the ID ID found

Error Response

Error-Response:

HTTP/1.1 404 Not Found
    {
        "Error": "No article with the ID <code>ID</code> found"
    }

request Article information

Back to top

GET /api/v1/articles/:id

Success Response

Success-Response:

HTTP/1.1 201 OK
{
    "ID": 1,
    "TITLE" : "Lorem Ipsum"
}

Success 200

Name Type Description
ID Number

Id of the article

TITLE String

Title of the article

Error 4xx

Name Type Description
ArticleNotFound

No article with the ID ID found

Modify an article

Back to top

PATCH /api/v1/articles/:id

Parameter Parameters

Name Type Description
TITLE String

The Title of article to patch

Success Response

Success-Response:

HTTP/1.1 201 OK
{
    "ID": 1,
    "TITLE" : "Lorem Ipsum"
}

Success 200

Name Type Description
ID Number

Id of the article patched

TITLE String

Title of the article patched

Paragraph

Add a new paragraph in the article

Back to top

POST /api/v1/articles/:idA/paragraphs

Parameter Parameters

Name Type Description
CONTENT String

Content of the paragraph patched

POSITION Number

Optional position of the paragraph in the article (if empty, the paragraph is added at the end of the article)

ARTICLE_ID Number

Id of the article associated to the paragraph

Success Response

Success-Response:

HTTP/1.1 201 OK
    [
        {
            "ID": 2,
            "CONTENT": "Ut enim ad minim veniam.",
            "POSITION": 2,
            "ARTICLE_ID": 1
        },
        ...
    ]

Success 200

Name Type Description
ID Number

Id of the article

TITLE String

Title of the article

POSITION Number

The position of the paragraph in the article

ARTICLE_ID Number

The Id of the article associated to the paragraph

Delete a paragraph

Back to top

DELETE /api/v1/paragraphs/:id

Success Response

201 Success-Response:

HTTP/1.1 201 OK
{
    "Response" : "Successfully deleted paragraph with ID <code>ID</code>",
}

Success 200

Name Type Description
ID Number

Id of the deleted paragraph

Error 4xx

Name Type Description
ArticleNotFound

No paragraph with the ID ID found

Error Response

Error-Response:

HTTP/1.1 404 Not Found
    {
        "Error": "No paragraph with the ID <code>ID</code> found"
    }

Get the pos-ian paragraph of the article

Back to top

GET /api/v1/articles/:idA/paragraphs/:pos

Success Response

Success-Response:

HTTP/1.1 201 OK
    [
        {
            "ID": 2,
            "CONTENT": "Ut enim ad minim veniam.",
            "POSITION": 2,
            "ARTICLE_ID": 1
        },
        ...
    ]

Success 200

Name Type Description
ID Number

Id of the article patched

TITLE String

Title of the article patched

POSITION Number

The position of the paragraph in the article

ARTICLE_ID Number

The Id of the article associated to the paragraph

Modify a paragraph

Back to top

PATCH /api/v1/paragraphs/:id

Parameter Parameters

Name Type Description
ID Number

Optional Id of the paragraph patched

CONTENT String

Optional Content of the paragraph patched

POSITION Number

Optional position of the paragraph in the article

ARTICLE_ID Number

Optional Id of the article associated to the paragraph

Success Response

Success-Response:

HTTP/1.1 201 OK
    {
        "ID": 2,
        "CONTENT": "Ut enim ad minim veniam.",
        "POSITION": 2,
        "ARTICLE_ID": 1
    }

Success 200

Name Type Description
ID Number

Id of the article patched

TITLE String

Title of the article patched

POSITION Number

The position of the paragraph in the article

ARTICLE_ID Number

The Id of the article associated to the paragraph

Paragraphs

Request all paragraphs of the article

Back to top

GET /api/v1/articles/:idA/paragraphs

Success Response

Success-Response:

HTTP/1.1 201 OK
    [
        {
            "ID": 2,
            "CONTENT": "Ut enim ad minim veniam.",
            "POSITION": 2,
            "ARTICLE_ID": 1
        },
        ...
    ]

Success 200

Name Type Description
ID Number

Id of the paragraph

CONTENT String

Content of the paragraph

POSITION Number

The position of the paragraph in the article

ARTICLE_ID Number

The Id of the article associated to the paragraph