Apache Zookeeper qui se base sur la notion de quorum pour l'élection d'un master, les autres devenant standby master.
Dans les faits, les serveurs dont on dispose sont toujours hétérogènes !
On peut définir ses propres ressources ce qui est pratique dans le cas d'un cluster que l'on construit petit à petit et dont les machines sont donc de puissances et fonctionnalités hétérogènes (ex: on achète une machine avec un GPU car on a besoin de traiter des images).
Mesos est un projet qui va apporter une grande valeur ajoutée à votre cluster de machine pour l'exploiter au mieux. Des frameworks tels que Marathon ou Chronos de la Mesosphere enrichissent le framework pour aisément pouvoir construire son propre PaaS et gérer ses jobs. Les speakers ont également mis en avant l'importance du DevOps dans l'utilisation réussie de Mesos. En effet les développeurs sont les mieux placés pour savoir les ressources dont ont besoin leurs applications pour fonctionner.
En conclusion, pour réussir son déploiement Mesos :
Keep C.A.L.M.S ! (Culture Automation Lean Measurement Sharing)
Ce Quickie par Etienne Peiniau présentait l’outil Fig plutôt connu dans l’écosystème Docker, qui est devenu depuis peu Docker Compose suite au rachat par Docker. Il répond aux cas d'utilisation suivants :
Pour le configurer, on écrit un fichier docker-compose.yml dans son projet qui va assembler à la racine l'ensemble des services docker à bootstrapper.
Quelques commandes utiles (qui s'apparentent fortement à la ligne de commandes Docker) :
- docker-compose up : bootstraper l'ensemble des conteneurs
- docker-compose logs : voir tous les logs en sortie standard des conteneurs
- docker-compose ps : lister les conteneurs de cette composition
En conclusion, j'entendais souvent parler de Fig sans trop comprendre à quoi m'attendre. Après ce quickie, cet outil me semble à la fois indispensable et très simple à utiliser si on a déjà quelques bases en Docker.
Une présentation par l’équipe Groovy, dont Guillaume Laforge, et sponsorisé par JFrog, l’outil de management d’artéfacts utilisés par Groovy depuis peu. Le talk a permis de retracer l'histoire de Groovy et l'évolution de son processus de release pour finir par une petite apparté sur JFrog.
Groovy qui est un compilateur a des besoins spécifiques qui sortent du lifecycle classique de maven. Initialement le déploiement était effectué par Maven avec le plugin Jelly mais manquait de flexibilité. Il leur a été nécessaire de passer à Ant pour plus de souplesse avant de finalement utiliser Gradle, lui même basé sur Groovy.
Concernant l'intégration continue du projet, initialement celui-ci utilisait Codehaus qui :
Finalement Groovy est passé à Team City développé par Jetbrains. Ce dernier dispose d'une communauté qui maintient des agents de build spécifiques à différentes plateformes. Les besoins de Groovy sont très spécifiques et il ont une grosse combinatoire à tester pour le langage :
D'autre part il leur manquait une grande partie d'automatisation et de nombreux workflows ont été longtemps effectués manuellement (tagging, mise à jour de version, VCS, etc.) engendrant de nombreuses erreurs. Un autre problème se posait pour la maintenance des codes insérés dans la documentation qui finissaient par être obsolètes. Désormais, la documentation réalisée sous Ascii Doctor permet de tester tous les codes unitairement !
Finalement, Groovy utilise JFrog pour la gestion de ses artéfacts. Artifactory est l'équivalent d'un dépôt Nexus qui permet de gérer les différents snapshots générés à chaque commit. Il supporte Python, Debian, Docker, npm, maven, gradle, ivy, nuget, rpm, yum, etc. Bintray stocke les releases qui sont signées et déployées sur Bintray JCenter (synchronisé avec Maven Central). Le site web et la documentation peuvent être uploadés et mis à jour automatiquement.
Google déploie près de 20 milliards de conteneurs par semaine ! Docker, qui adresse les applications orientées services essentiellement, est une nouvelle solution qui apporte ses propres problèmes...
Docker combine LXC et UFS (Union FS), un système de fichiers qui présente différentes couches. Linux est requis (cf. Boot2Docker MAC/WIN) mais on s'approche du Build once, run anywhere et de nombreux dépôts officiels sont disponibles pour récupérer les images Docker
Conteneurs vs VM :
Les conteneurs sont isolés mais partagent le même OS (et librairies lorsque c'est possible). Les VM virtualisent au dessus d'un OS Hôte ou d'un Hyperviseur qui va initialiser un OS complet pour chaque VM.
DevOps :
Docker est un outil orienté développeur car il permet d'exécuter son app localement, de la configurer et d'exposer le minimum. Il est orienté opérationnel car il permet de masquer l'application et de laisser les opérationnels se concentrer sur les problématiques d'infra.
Kubernetes a été crée par Google il y a un peu plus d'un an et vient adresser la problématique de gestion de conteneurs sur plusieurs hôtes. Il peut être qualifié de container cluster orchestration. Il permet de configurer de multiples hôtes (nodes / minions) grâce notamment à une API et offre la possibilité d'agir plus ou moins sur l'emplacement des conteneurs spawnés. D'autre part, il propose un monitoring plus haut niveau que Docker.
Kubernetes est composé d'un master qui joue le rôle de serveur d'API et se base sur ETCD pour la synchronisation de configuration. Il s'articule autour de nodes qui se composent à minima de Docker, ETCD, SkyDNS et Kubelet (qui s'assure de l'état des pods). Enfin les pods sont un groupe de conteneurs collocalisés qui partagent le même network/namespace/ip et volume (host mounted, empty volumes, GCE data disks, ...). Pour chaque pod on configurera le nom de l'image Docker, les ports à exposer, les volumes à monter et un label (tag) si nécessaire.
De nombreux fournisseurs de conteneurs existent déjà :
Carlos Sanchez a cité pendant sa session plusieurs projets/plugins autour de Kubernetes et le support déjà assuré par de nombreux providers nous pousse à vouloir tester rapidement cet outil !
Un quickie de Nicolas Muller présentant la base de données InfluxDB, opensource par ErrPlane (MIT License) écrit en Go, taillée pour le stockage de séries temporelles. Une time serie est un regroupement d'événements qui se sont passés dans le temps :
En 2015, on attend d'un outil de collecte, stockage et visualisation d'événements qu'il soit simple à installer et administrer, adapté au BigData, qu'il supporte HTTP et JSON et qu'il soit bâti sur des API. Ce n'est pas encore vraiment le cas des outils existants :
InfluxDB répond à ces contraintes. Il est simple à installer et manager, n'a pas de dépendances externes, propose une interface HTTP et est scalable horizontalement. Il propose également l'échantillonnage de données par période de temps ce qui est une fonctionnalité souvent attendue.
InfluxDB est assez simple à appréhender. Il est basé sur le concept de Database équivalent à celui qu'on connait avec les SGBD classiques. Le concept de Time Series est une sorte de table pour le temps avec un sequence number et des colonnes. Une série temporelle est composée de points (ou événements) correspondant en quelque sorte aux lignes d'une base de données SQL. Enfin InfluxDB propose une langage SQL Like pour requêter des séries.
Nicolas Muller a évoqué un certain nombre de bonnes pratiques connues à ce jour.
InfluxDB est très adapté à des nombreuses séries avec peu de colonnes. Requêter sur des colonnes autre que le temps est possible mais entrainera de mauvaises perfs. Dans ces conditions, on crée autant de tables qu'il y a de valeur métier.
En terme de nommage, la règle à suivre est la suivante :
<tagName>.<tagValue>.serieName
Exemples v0.8 :
- arduino.uno.shield.ethernet.sensor.dht11.temperature
- arduino.uno.shield.ethernet.sensor.dht12.temperature
- arduino.uno.shield.wifi.sensor.dht22.humidity
- ...
En V0.9, les tags seront gérés directement par Influx :
{
name: <b>"temperature"</b>
tags: {
arduino: <b>"uno"</b>
shield: "ethernet"
sensor: "dht11"
}
}
A noter qu'une migration automatique en V0.9 d'InfluxDB sera possible si on a respecté la convention de nommage ci-dessus en V0.8 et inférieur. De plus, la nouvelle version de Grafana va supporter directement la V0.9.
InfluxDB semble être une techno prometteuse dans le domaine des séries temporelles omniprésentes de nos jours (IoT, monitoring, etc.). Le produit évolue rapidement et Grafana, l'un des incontournables de la DataViz, est associé à InfluxDB et OpenTSB qui permet des montées de versions synchronisées de ces outils.
Un quickie par Alexandre Victoor (SCGIB) présentant Apache Avro, un système de stockage initialement utilisé pour le stockage distribué dans Hadoop qui est maintenant un projet à part entière chez Apache.
Avro permet d’échanger du JSON (ou binaire ou autre) pour lequel on va pouvoir définir un schéma de référence pour ses données. Ainsi, à l’initialisation, client et serveur réalisent un HandCheck pour vérifier qu’ils sont en accord sur un schéma et commencer à échanger. Dans le cas contraire, la connexion est coupée.
Aux données échangées est associé un hash correspondant au schéma de données utilisé. Ceci permet de facilement faire évoluer un schéma d’échange de données dans Avro et de le diffuser petit à petit vers ses clients. Le schéma complet et donc stocké côté serveur et l’overhead à chaque échange est donc limité à quelques bits pour connaître l’identifiant du schéma.
Julien Viet, ingénieur sur le projet Vert.x, nous a fait une présentation de ce framework polyglotte dont on entend beaucoup parler depuis un certain temps. Vert.x est un projet sous la fondation Eclipse financé par RedHat, actuellement en version 2.1.x de production, la version 3 milestone 4 est en cours de développement. Java 8 obligatoire depuis la version 3 qui utilise beaucoup les expressions lambda.
Vert.x est une stack pour développer des applications pour la JVM inspiré par Erlang/OTP et Node.js pour les paradigmes asynchrones non bloquants. Il est polyglotte (Java, JS, Groovy, JRuby, Jython, Scala, ...), léger et embarquable adapté aux hautes performances (support du cluster / failover).
Le framework suit le modèle réactif (il implémente reactive-streams en version 3) éliminant ainsi les problèmes de concurrence (l'utilisation du même thread au cours de l'exécution est garantie). Vert.x est composé d'un core léger avec peu de dépendances et expose une API asynchrone.
Vert.x est fondé sur deux principaux composants :
- Verticle qui est déployé dans Vert.x quel que soit le langage et exécuté sur un thread unique. La communication avec lui se fait par messages. De manière générale on va, pour une application donnée, tenter de séparer les différentes briques en autant de Verticles.
- Le Bus d'événements est le système nerveux de Vert.x. Il force l'isolation des Verticles qui échangent des messages grâce à lui. L'échange est essentiellement Point à Point de données immuables dans le format souhaité (généralement JSON ou byte).
Enfin, basé sur Nashorn et SockJS, Vert.x est également utilisable dans le navigateur qui fourni la même API que pour le serveur. En conclusion Vert.x favorise l'écriture d'architectures en microservices par son modèle asynchrone et sa communication par event bus. Sa compatibilité multilangages (automatisée à partir de la version 3) en facilitera d'autant plus son adoption.
HyperLogLog est un algorithme qui consomme peu de mémoire pour gérer des compteurs sur un très grand nombre de données, moyennant une perte de précision de quelques pourcents. Pour cela, il faut calculer des probabilités basées sur les données rencontrés (hash a forte variabilité) en analysant le nombre de bits consécutifs à zéro. Il faut également faire des moyennes pondérées pour éviter que certaines valeurs occasionnelles faussent les résultats.
Paxos est un algorithme de consensus distribué. Dans les faits, un maître gère les conciliations et en cas de problème, un nouveau maître prend la main. Il est capable de terminer les travaux du premier. En cas de collision, un délai aléatoire est ajouté avant la reprise.
DuyHai DOAN a eu l’ambition de nous expliquer les algorithmes HyperLogLog et Paxos. Il s'est plutôt concentré sur les justifications sous-jacentes aux algorithmes que sur l'implémentation réelle ou l'utilisation. Mission accomplie !
Pour finir la journée du jeudi, une présentation de la base de données Firebase en action dans un jeu vidéo aux allures d’Ingress ! L'objectif de ce tools-in-action était de faire un jeu android pour capturer les gares avant que les autres ne le fasse.
Firebase est une BDD NoSQL orientée document qui offre un mécanisme de synchronisation de la donnée en temps réel. Elle se charge pour nous de l'abstraction de la couche transport (WebSocket, SEE, ...), des problématiques de coupures réseaux et des fallbacks pour support les anciennes plateformes.
En plus de se charger du problème de synchronisation, Firebase propose de nombreux SDK qui permet de profiter de ce service pour toutes ses applications (Android, Java, Angular, iOS, Ember, React, Node, etc.).
Pour Egress, GeoFire a été utilisé pour apporter le support de la localisation. De manière concrète, on va créer un objet Firebase mappé sur l'url de la base de données. L'authentification sera gérée par Firebase (ex : ref.authWithOAuthPopup), puis on définit des callbacks côté client :
En conclusion, avec cette petite démo fun, on voit rapidement que Firebase va plus loin qu'une base de données classique. Non seulement elle propose un mécanisme de synchronisation des différentes applications par événements mais aussi elle offre des fonctions pratiques comme l'authentification, où la localisation via la librairie GeoFire.
Jean-Philippe Bempel de ULLINK a partagé ses connaissances sur les différentes approches pour proposer des algorithmes sans verrou. Il a évoqué les implémentations proposées depuis Java 7 et les améliorations dans Java 8. Il a évoqué les algorithmes immuables CopyOnWriteArrayList<E> pour insister sur les impacts au niveau du GC.
La stratégie lock striping utilisée par l'implémentation Java 7 de ConcurrentHashMap<K,V> consistait à sharder les données en bloc pour y associer des locks. Cette approche n’est plus utilisée en Java 8.
Il a par ailleurs expliqué les différents type de barrières, gérées par le compilateur et/ou le processeur. Il a évoqué les différents types de verrous processeurs, en lecture et écriture. A noter qu'un processeur x86 n’a besoin que d’un lock en écriture, via des instructions assembleurs proposées par la classe Unsafe. Il a également rappelé que le passage en Kernel n’implique pas forcément un Context Switch. Cela dépend du traitement demandé. Il y a traversée en Ring 0 du processeur.
Finalement, il nous conseille de suivre le blog de Martin Thomson (Mechanical Sympathy) qui prône le rapprochement du code au plus près de la machine et du processeurs. Une conférence de très haut niveau, riche en information, simple dans les explications et technique comme on aime !
Ce tools-in-action proposait un retour d’expérience sur l'utilisation de HAProxy dans un test géant effectué pour Critéo.
En résumé, HAProxy tient très bien la charge tant qu’il n’est pas saturé en CPU. Une petite combinaison d’outils permet d’avoir de la haute disponibilité entre plusieurs instances HAProxy.
Dans le domaine de la finance de marché, les performances sont de l'ordre de la micro seconde. En effet, il faut profiter d'opportunités financières qui sont disputées par des dizaines de milliers de concurrents. Aux Etats Unis, les stock options représentent 1,1M de transactions par seconde.
Objectifs :
La conférence se construisait autour d'une application de finance de marché dont il fallait optimiser les performances. L'architecture initiale de l’application était la suivante :
- un thread unique qui va jouer le rôle d'event loop
- des threads qui interceptent les IO en amont et en aval
- la communication entre les deux est réalisée par une BlockingQueue
Pour se mettre en jambe, le talk a commencé par un petit quizz :
En matière de performance, l'une des obsessions est de constamment mesurer. Faire constamment du microbenchmarking avec JMH par exemple car une intuition d’optimisation peut se révéler fausse ou pire contre productive.
L’outil statistique est primordial dans les mesures de performance. Attention, utiliser la moyenne pour la latence d'un système n'est pas forcément une bonne solution car on n'a pas une distribution aléatoire de valeurs et la moyenne risque simplement d'être noyée dans la variation des valeurs. On utilise l'outil statistique percentile qui permet d'avoir la distribution de toutes les valeurs et de produire un histogramme (cf. HDRHistogram by Gil Tene pour réaliser des histogrammes avec impact minimal sur les performances).
Martin Thompson parle de "Mechanical Sympathy" pour réconcilier le code avec le matériel. En effet pour obtenir les meilleures performances il ne faut pas négliger la machine sur laquelle les instructions de notre programme vont être exécutées.
Pour avoir des performances optimales, on cherche à développer des algorithmes non bloquants, on veut ainsi au moins un thread en progression tout le temps. Plusieurs choix se posent sur le type de file à implémenter ConcurrentLinkedQueue, Queue SPSC ou encore Lamport Queue. Cette dernière utilise une file circulaire qui consiste en un tableau, un index d'écriture et un modulo. Ainsi on a un gain de débit de x4 entre BlockingQueue et Lamport Queue.
Notes sur volatile :
Le buffer (tableau) d'index sont stockés en "volatile". En Java 1.4 on a pas le comportement attendu car c'est le buffer qui est volatile pas la valeur à l'intérieur ! En Java 1.5, le volatile a plus d'utilité car il se base sur le principe d'happens before.
Exemple : si mon tableau a été écrit dans le buffer à un index 2, et qu'au moment de la lecture de l'index par mon consumer il vaut 3, celui-ci va quand même aller chercher la dernière valeur avec laquelle le thread en question à écrit dans le buffer, en l'occurrence 2.
D'autre part, le coeur d'un processeur a un Store Buffer qui permet au registre de limiter un peu les accès au cache L1. Le volatile a un coût lié au fait qu'à chaque accès, il va commiter le Store Buffer dans le cache L1 et vider ce buffer. On perd donc cette optimisation.
Il a été également évoqué l'utilisation d'un Chronicle Logback Appender basé sur un Memory Map File pour des raisons de performance, ou encore de surparallélisation et de context switching lié au changement de processeur d'exécution pour un thread.
En conclusion, un talk très abordable et très intéressant par deux ingénieurs de la SCGIB (Thierry Abaléa et Alexandre Victoor) concernant les enjeux d’une application haute performance en finance et les solutions qu’on peut avoir en Java. Des solutions rapides à mettre en oeuvre et efficaces et toujours la même obsession : mesurer !
Menée par Simon Baslé et Laurent Doguin, cette session était un Hands On Lab sur RxJava que j’ai trouvé très bien construit et où j’ai vraiment découvert et compris l’intérêt que peut avoir Rx dans son projet.
Le GitHub du HOL est fourni (et associé à un projet externe qui sert d’API défaillante https://github.com/simonbasle/practicalRxExternal) et chaque étape a une branche solution associée ce qui permet de comparer avec ce qu’on a fait.
Objectifs du HOL :
- Migrer une application Web composée de divers services, dont certains externes et utilisant des mécanismes de l’API java.util.concurrent vers une application RxJava
- Après la migration des premiers services, on commencera par faire compiler le code naïvement en faisant des appels bloquant dans les contrôleurs appelant ces services jusqu’à arriver à du RxJava full stack et un retour asynchrone avec le DeferredResult Spring.
- On se concentrera ensuite sur la réalisation du ExchangeRateService. Appel de deux API externes :
- Change Doge vers Dollar
- Change Monnaie A vers Monnaie B
L'API gratuite n'est pas fiable. Une API payante est plus fiable il faudrait l'utiliser seulement lorsque l'API gratuite ne marche pas. Etant donné l'API payante, il faudrait tracer les coûts.
Solution :
- Pour compter les coûts, compteur dans l'AdminService
- On wrappe dans un onErrorResumeNext
- On ajoutera un doOnNext pour le logging
ReactiveX est inspiré de Microsoft .NET (linq, rx) porté en Java par Netflix OSS. Il peut être intéressant d'utiliser RxJava dès lors que l'une des problématiques suivantes se présente : Asynchrone, Event Based, IO. Rx vient combler les limites de certains langages et offre des fonctions de composition, transformation, filtrage, gestion d'erreurs avancé, temporalité :
Ceci mène souvent à un code plutôt illisible, en tout cas peu autoportant.
Le principe est simple, généralement en Java on manipule les collections en pull (pattern Iterable / Iterator). En RxJava, on va plutôt faire du push de données (pattern Observable / Observer). La donnée devient un flux, poussé vers le consommateur.
Le framework offre un grand nombre d'opérateurs utiles en tout genre. En voici les principaux :
L'intérêt du paradigme réactif ne se fait sentir que s'il est déployé sur toute la stack de l'application. Il est pour cela nécessaire que notre framework REST permette de retourner des "promesses de réponse". DeferredResult est un object de Spring qui permet de retourner des réponses REST asynchrone.
Observable<Map<String, Object>> costObservable = adminService.costForMonth(year, month)
.map(cost -> {
Map<String, Object> json = new HashMap<>();
json.put("month", month + " " + year);
json.put("cost", cost);
json.put("currency", "USD");
json.put("currencySign", "@@ARTICLE_CONTENT@@quot;);
return json;
});
DeferredResult<Map<String, Object>> deferredResult = new DeferredResult<>();
// Souscription du DeferredResult à l'Observable
costObservable.subscribe(
deferredResult::setResult,
deferredResult::setErrorResult
);
return deferredResult;
Avec l’utilisation de DeferredResult, on est enfin complètement asynchrone puisqu'on n’attend pas le résultat de manière active. On ne fait pas de timeout, si le résultat n'arrive pas, on ne fait pas d'attente active, Spring pourra faire autre chose. A noter que DeferredResult est bien un objet Spring et n’a rien à voir avec RxJava. RxJava ne propose pas d’outil pour réaliser des retours asynchrones d’appel HTTP, c’est à la charge du framework utilisé (en l’occurrence Spring). A noter RatPack, une alternative à Spring pour les WebServices http://ratpack.io.
Un hands-on bien mené qui nous a bien montré les intérêts directs que l'on peut avoir à utiliser Rx (polyglotte, performance, fallbacks, etc.).
Ceylon est un langage de programmation sur la JVM à l’instar de Scala ou Groovy qui a passé le stade de la 1.0 il y a quelques mois maintenant. Cette présentation proposait de passer en revue une dizaine d’idiomes intéressants du langage et il est vrai que son système de typage est assez bluffant.
Ceylon dispose déjà de nombreux outils dans son écosystème et est compatible avec de nombreux contextes d'exécutions (Java SE, OSGi, Vert.x, Node.js, Browser, etc.).
Parmi les fonctionnalités caractéristiques :
Tout d'abord, la combinaison des types permet d'avoir des fonctions à plusieurs types de retour potentiel sans avoir besoin de passé par un super type.
File|Path|SyntaxError parsePath(String path) => ...
switch(result)
case(is File|Url)
...
Il est possible d'avoir des collections d'objets hétérogènes fortement typés où l'on pourra jouer avec le switch de Ceylon pour effectuer le traitement adéquat.
List<Integer|Float> list = ArrayList { 1, 1.0, 1 };
Null|Integer|Float elt = list[3];
Le type Tuple est géré avec une LinkedList qui ne lui donne aucune limite et est utilisé pour définir dynamiquement une liste de paramètres à passer à une fonction.
[Protocol,Path] protocolAndPath = parseUrl("...");
// équivalent à
Tuple<Protocol, Path> protocolAndPath = parseUrl("...");
[Integer, Integer, String] tuple = ...
function f(Integer x, Integer y, String string) => x+y
Callable<Integer, [Integer, Integer, String]> callable = f;
Integer integer = f(*tuple)
Gavin King présentait en tout 9 idiomes. La souplesse du système de typage est ce qui m'a le plus marqué dans ce langage et donne envie d'aller plus loin avec et de voir un peu plus ce qu'il permet dans le domaine fonctionnel.
Brian Goetz est venu nous présenter Jigsaw, le projet de modularisation du JDK dont on commence à désespérer la sortie reportée depuis le JDK 6 ! Jigsaw c'est la volonté de rendre la plateforme Java SE modulaire et de se débarasser du classpath et du rt.jar.
Motivations :
- Java a commencé comme une petite plateforme de ~2500 classes
- Désormais java a grandi mais il est toujours nécessaire d'avoir toute la plateforme même si on ne se sert que d'une partie (JRE = 55Mb)
- Les classes sun.* et *.internal.* sont des classes internes au JDK dont on ne devrait jamais a avoir utiliser directement. Dans les faits, il faut utiliser SecurityManager::checkPackageAccess dans chaque classe pour assurer l'isolation...
- Jar Hell : le problème apparaît du fait que le classpath n'a aucune information sur les choix a effectuer lors d'un possible conflit (lié à l'import de versions différentes du même jar par exemple)
La solution à ce problème de classpath, de sécurité est d'apporter un système de module (une cinquantaine) au JDK utilisable par le Java SE lui même mais aussi par les autres projets. Pour le problème de taille du JDK, la version 8 a commencé par apporter les profils compact :
Le projet Jigsaw a connu du tumulte. Le premier gros travail a été d'identifier les dépendances existantes entre les différents potentiels module du JDK. Désormais, le projet est séparé en 4 JEP :
La JEP 200 a proposé l'ajout d'une nouvelle méthode getModule() à Class permettant de connaître le module qui a chargé la classe.
$ java -XListModule => liste les modules chargés
jar:file:/path/to/jdk/jdk8/jre/lib/rt.jar|/java/lang/Class.class // Java 8-
jrt:/java.base/java/lang/Class.class // Java 9
Les modules peuvent dépendre les uns des autres mais il existe deux types de dépendances :
Ceci règle le problème d'exposition des packages internes sun.* et *.internal.*.
_La JSR 376 p_ropose un moyen pour les développeurs de définir leurs propres modules. Elle prévoie :
Finalement j’ai trouvé la présentation très intéressante et le projet à l’air d’avancer correctement !
Après avoir rappelé les évolutions du langage pour proposer les classes génériques, Brian Goetz nous parle des Value Object pour regrouper les attributs des classes en mémoire et dans les conteneurs, afin de réduire le travail du Garbage Collector et les cache-miss du processeur lors de la navigation dans les objets. Cela permettra d’avoir les mêmes performances que le C++ ou C#. C’est une bonne évolution pour les algorithmes à haute fréquence.
Présentation d’Emmanuel Bernard (Red Hat, Cast Codeurs) sur sa manière d’utiliser son outil de travail en automatisant au maximum son workflow, pour éviter le context switching durant son travail.
Il nous a notamment parlé de son environnement de commande TMux qui offre la possibilité de préparer plusieurs "univers" qui vont permettre de conserver différents contextes de travail. Il est également un grand friand de Vim qui offre un système de commandes efficace basé sur le pattern suivant :
[action][repetition multiplier]movement
- d2f. : supprimer 2 lignes et avancer jusqu'au prochain "."
- shift + v : mode visuel vim
- set col "n" : largeur du document
- gq : mettre la sélection trop large à la bonne largeur de document
- J : mettre la sélection trop courte à la bonne largeur de document
- 0 : début de la ligne
- W : mot suivant
- dW : supprimer le mot suivant
Il existe également des plugins IdeaVim (IntelliJ) ou Vrapper (Eclipse) qui permettent d'avoir un environnement Vim dans son IDE préféré.
Plusieurs automatismes quotidiennement utilisés par Emmanuel :
Lorsque je me connecte pour la première fois à un serveur, on me demande mon mot de passe et j'envoie automatiquement ma clé ssh en autorized_keys.
Lancer une pull request qui faire le merge et l'ouverture du navigateur à la bonne page automatiquement.
Utiliser remote pour lancer compilation et tests en background et lancer une notification desktop à la fin permettant de faire du context switching efficace sur ses réseaux sociaux !
Le leitmotiv de ce talk fût finalement de ne pas hésiter à automatiser un maximum de tâche que l'on répète ne serait-ce que quelques fois par semaine. Pour s'inspirer, une communauté existe autour des dotfiles qui permet de trouver plein d’exemples (cf. https://dotfiles.github.io) qui donne vraiment envie de s'y mettre ! De même pour Git ou l'utilisation d'alias peut-être un premier gain de temps mais l'écriture d'un plugin relativement simple peut rendre de grands services aussi.
Inspiré de Delivering Ebay's, ce Tools in Action par Jean-Louis Rigau consistait à présenter rapidement un use case d’utilisation d’Apache Mesos et un de ses framework, Marathon, pour rendre son Jenkins fortement scalable !
Objectifs du Tools in Action (TIA) :
- Simplifier la gestion de Jenkins en le conteneurisant avec Docker et le déployer dans Mesos via Marathon
- Utiliser les ressources de Mesos pour créer des slaves Jenkins à la demande
Jenkins est un serveur d'intégration continue, maintenu par Cloudbees, écrit en Java. Ceux qui l'utilisent savent qu'il est généralement difficile à maintenir et à mettre à jour !
Docker est un gestionnaire de conteneurs opensource, écrit en Go, créé et maintenu par Docker Inc. Il sert à la construction d'applications et au déploiement sur tout type d'environnement.
Mesos est un gestionnaire de cluster opensource par Apache, écrit en C++. Il permet de gérer les ressources de son Datacenter et de se créer son propre cloud en allouant intelligemment ses ressources selon les besoins de ses applications.
Enfin Marathon est un framework Mesos opensource par Mesosphere qui est écrit en Scala. Il se charge du déploiement et management de conteneurs basés sur Mesos et joue en quelque sorte le rôle de sur couche simplificatrice.
En bref, le schéma d'architecture qui nous a été présenté lors du tools-in-action. J'achète si cela peut permettre de retrouver confiance en son Jenkins !
L’équipe des furets ont fait le choix d’utiliser Git en mode feature branch. Malgré toutes les alertes sur cette approche, ils ont persisté et proposé des plugins à Git pour faciliter le merge multiple. L’idée est de sélectionner avant la mise en production, l’ensemble des branches à lier. Le résultat est placé dans un branche tampon sans historique. Les problèmes peuvent alors être résolus suivant plusieurs approches :
Avec cette approche, ils publient plusieurs fois par jours.
Devoxx c'est 200 conférences en 3 jours et des choix à chaque session difficiles à faire ! Ci-après vous trouverez quelques conférences en vrac, non moins intéressantes !