App Isomorphic: la Single Page App parfaite ?

App Isomorphic: la Single Page App parfaite ?

Qu’est ce qu’une Single Page App (SPA) ?

« As rich and responsive as a desktop app but built with HTML5, CSS and Javascript »

Les SPA se répandent de plus en plus, et deviennent un choix « commun » lorsque l’on veut développer un Front riche (souvent câblé sur des API REST) que l’on souhaite :

Les Frameworks type AngularJs et EmberJs tiennent le haut du panier et ont largement fait leurs preuves, mais ils continuent à échouer sur deux sujets pourtant primordiaux dans beaucoup de cas :

La performance

Aujourd’hui, quand vous chargez une SPA, voici grossièrement ce qui se passe coté client :

Avoir ces quelques secondes à attendre avant de se retrouver dans un état fonctionnel est peut être acceptable pour un backoffice. Mais ça l’est beaucoup moins pour un front riche.Et ce temps aura tendance à augmenter fortement, parallèlement à l’enrichissement de votre application.

Si l’on se soucie un minimum des aspects de performances Web, c’est forcément dérangeant. Et d’un point de vue plus global, tout le monde sait aujourd’hui que la performance brute n’est pas le point fort de ces frameworks.

Le référencement

Autre sujet, qui peut être très problématique, si le site en question s’y prête. Ces applications vont fournir comme « source HTML » quelque chose de ce style (pour du Angular) :

{% highlight html %}

... {% endhighlight %}

Container qui servira à recevoir le HTML généré par votre appli JS une fois exécutée.

De base, Google (et autres moteurs/crawler) ne verra donc rien, tout votre contenu allant être injecté via JS dans votre balise ng-view. Excepté le fait qu’il parait que depuis des mois/années, Google commence à réellement crawler du JS … Si le site est important, cette supposition ne devrait pas suffire à vous convaincre, et vous avez raison.

Rassurez vous, à ce stade, des solutions existent pour fournir spécifiquement à Google une version correspondante aux snapshots HTML générés par vos applications. Ces solutions sont accessibles soit en mode SAAS (payante et hébérgé), soit en mode Open-Source à héberger vous même. Je pense notamment à Prerender.io qui fait plutôt bien le job, et vous propose d’indiquer aux moteurs que vous faites une application de type « Ajax » en respectant les recommandations de Google.

Prerender est composée de plusieurs briques : Un middleware applicatif (Rails, Node, Varnish, Nginx, etc selon votre infrastructure), qui va intercepter les moteurs et les renvoyer sur votre service de Prerender Un service de Prerender qui est une brique Node.js qui va lancer des HeadLess Browser (PhantomJS ou SlimerJs …) pour executer votre appli JS et renvoyer un snapshot HTML une fois le rendu JS terminé.

La solution permet à priori de faire le boulot, mais cela reste une gymnastique complexe, et beaucoup d’interrogations subsistent (pertinence, maintenance, stabilité, Page Rank, pondération vs sites classiques …)

La lumière au fond du tunnel ?

Vous l’avez donc compris, dans certains cas, les SPA basées sur des frameworks Js posent deux problèmes très gênants et difficilement résolvables. C’est là qu’entre en piste, une nouvelle façon de penser les SPA, grace à une librairie développée par Facebook : React.JS

React fait parler de lui car il commence à être utilisé massivement par des très gros acteurs Web, Facebook bien entendu pour ses composants Chat, ou son éditeur vidéo, Instagram pour l’intégralité du site, Yahoo Mail, Github avec l’IDE Atom, Khan Academy, NyTimes, Feed.ly

Au premier abord, React n’est qu’une librairie qu’on pourrait comparer à la partie Vue d’un Framework MVC (voir aux Directives d’Angular), mais il a la particularité d’être basé sur un Virtual DOM. Ce qui parait au départ simplement une bonne idée pour avoir des performances bien supérieures à celle d’un framework MVC basé sur le DOM, et éviter par exemple les Dirty checking du DOM (qui explique en partie le manque de perf d’Angular), permet aussi d’utiliser ces mêmes composants coté serveur !

C’est ce qu’on appelle l’approche « Isomorphic » .

Un composant React n’est finalement qu’un module CommonJs et peut donc aussi bien être utilisé coté browser sur le client, que coté server dans du Node.Js (ou IO.js devrais-je dire maintenant ?). L’idée de l’isomorphisme est aussi d’être capable de servir le premier rendu directement par le serveur. Exemple:

Et là, vous répondez de manière parfaite aux deux points problématiques. Google n’y verra que du feu, et pourra crawler votre site entièrement comme si il n’était composé que de fichiers statiques. La performance du premier rendu sera quasi imbattable, car ne nécessitant aucun JS !

Sur le papier, c’est juste le rêve ultime de tout développeur Front-end : tous les avantages d’une SPA sans les inconvénients !

Facebook propose aussi sur son Github, une solution pour ceux ayant déjà un applicatif dans un autre language (ici PHP) : Server side rendering

La solution parfaite ?

Presque. React n’est au final que la partie Vue de votre application, il va falloir encore organiser tout ça. C’est ici qu’entre en compte Flux, un pattern d’architecture unidirectionnel proposé aussi par Facebook, à priori plus scalable que ne l’est le pattern MVC.

Mais là encore, l’approche de Flux est plutôt prometteuse, alors quel est le problème ?

Si malgré ces points, vous souhaitez tester cette approche, je vous conseille de regarder du coté de Yahoo, qui après avoir annoncé la migration de Yahoo Mail de PHP/YUI vers React/Flux Isomorphic a aussi publié quelques packages Open-Source très intéressants, pouvant constituer une bonne base de départ pour un projet isomorphic :

Si vous souhaitez plus d’infos sur React et Flux, je vous conseille ces deux articles en anglais de @andrewray:

Ou ce tuto chez nos amis de Jolicode, pour faire un Gifomatic avec React et Flux

D’autres solutions existent aussi conservant la même approche, mais sur la base d’autres technos, notamment celle d’Airbnb: RendR, permettant d’utiliser du Backbone coté client et serveur.

Et pour finir, si ces sujets vous passionnent tout comme nous, restez à l’écoute ici, d’autres posts pourraient arriver à l’avenir ;-)