JENKINS WORKFLOW
15 DÉCEMBRE 2015
JEAN DETOEUF
CONCEPTEUR-DÉVELOPPEUR
Passionné de nouvelles technologies
#jvm #docker #craftmanship #rpi #diy
SONDAGE
Intégration continue
Livraison continue
Déploiement continu
INTÉGRATION CONTINUE
JENKINS
ET BEAUCOUP D'AUTRES
POURQUOI JENKINS ?
CI le plus utilisé
Extensible
Stable
Open Source
Communauté
Livraisons régulières
Vieillissant par rapport à la concurrence ?
JOBS CLASSIQUES
COMMENT ENCHAÎNER
DU BUILD AU DEPLOY ?
Histoire de faire du déploiement continu
EN CHAÎNANT LES
JOBS
OUI, MAIS POUR
SUIVRE
L'ENCHAÎNEMENT ?
PIPELINE PLUGIN !
PROBLÈME
Complexe à mettre en oeuvre
On finit par avoir des jobs "étapes"
File d'exécution impossible à lire
Un job a des successeurs prédéterminés
Le paramétrage se fait dans Jenkins
Solution peu utilisable pour du déploiement continu
DÉPLOIEMENT CONTINU
Le but du déploiement continu est d'avoir une application
stable en production.
La chaine de livraison doit être automatisée au maximum
pour réduire l'intervention humaine entre une modification
de code et sa livraison en production.
L'erreur est humaine
QUEL INTERVALE DE TEMPS ENTRE
DEUX MEP ?
Ce n'est pas le but premier du déploiement continu.
Pouvoir livrer rapidement ne veut pas dire livrer
régulièrement.
GO DELIVERY
Un build est composé d'étapes
Chaque build peut être appliqué à un environnement
Fan-out-in (parallelisation des étapes)
GO DELIVERY
BONNES IDÉES
Parallèlisation des étapes
Environnements de déploiement
Visualisation du build et de l'étape en cours
Possibilité d'avoir plusieurs repositories en entrée
PROBLÈMES
Configuration gérée dans l'outil
Pas de possibilité de scripter le job
Difficulté pour gérer plusieurs versions nécessitant des
jobs différents
JENKINS WORKFLOW
PLUGIN
PLUGIN(S)
Ajout à une installation existante
Nouveau type de job
SCRIPTER LE JOB
ET METTRE LE SCRIPT AVEC LES
SOURCES
PIPELINE AS CODE !
SCRIPT STANDARD
Ecrit en shell
S'exécute sur un noeud (maître ou esclave)
JENKINSFILE
Fichier de build en Groovy
Du build au deploy dans un fichier compréhensible
Permet une orchestration plus avancée
Convention de nommage : Jenkinsfile
node {
git url: 'https://github.com/spring-projects/spring-petclinic.git'
def mvnHome = tool 'M3'
sh "${mvnHome}/bin/mvn -B verify"
}
JENKINSFILE
Le base de code déclare elle-même comment s'installer
Différentes versions (tags/branches/commits) peuvent
avoir des manières différentes de s'installer
node {
git url: 'https://github.com/spring-projects/spring-petclinic.git'
def mvnHome = tool 'M3'
sh "${mvnHome}/bin/mvn -B verify"
}
ACTIONS FOURNIES
PAR WORKFLOW
COMMANDE SHELL
sh 'ls /tmp'
bat 'dir C:tmp'
Pour reprendre un script standard, il suffit de reprendre
toutes les lignes du script comme ceci
ECHO (PRINT)
echo 'Compilation'
TRY-CATCH-FINALLY
try {
checkpoint('Before production')
} catch (NoSuchMethodError _) {
echo 'Checkpoint feature available in CloudBees Jenkins Enterprise.'
}
FOR-LOOPS
files.each{ file ->
sh "check.sh $file"
}
for num in 1..10
sh "echo $num"
}
OUTILS
def mvn(args) {
sh "${tool 'Maven 3.x'}/bin/mvn ${args}"
}
mvn 'clean install'
VALIDATION HUMAINE
input 'Est-ce que la page ${url} est correcte ?'
Si cette validation intervient avant le déploiement, on fait de
la livraison continue
CHOIX DU NOEUD
D'EXÉCUTION
node('unix && 64bit') {
sh 'make install'
}
PARAMÈTRES
EXPOSÉES SOUS FORME DE
VARIABLES D'ENVIRONNEMENT
Shell
$VAR
Groovy
env.VAR
PARALLÈLISATION
PARALLEL
for (int i = 0; i < splits.size(); i++) {
branches["split${i}"] = {
node('remote') {
sh 'rm -rf *'
sh "${tool 'M3'}/bin/mvn -B -Dmaven.test.failure.ignore test"
}
}
}
parallel branches
parallel prend une map en paramètre
la clé représente le nom de la branche
la valeur correspond au bloc de code à exécuter
PARALLEL
STAGE
Permet de séparer le phases du job
Permet de limiter le nombre d'exécutions parallèles
stage 'build'
sh "${tool 'M3'}/bin/mvn clean install"
stage concurrency: 1, name: 'deploy'
sh 'mv target/app.war /tmp/webapps/'
STAGE
stage 'build'
sh "${tool 'M3'}/bin/mvn clean install"
stage name: 'deploy', concurrency: 1
sh 'mv target/app.war /tmp'
Tant que le stage deploy n'est pas terminé, les exécutions
suivantes s'arrêtent avant deploy
Si plusieurs exécutions sont en attente de deploy, c'est la
dernière qui est retenue.
STAGE
CHARGEMENT D'UN FICHIER DE
SCRIPT
node{
def install = load 'install.groovy'
install.main()
}
Permet de diviser le script de build en plusieurs parties
SUIVI À L'IHM : OPEN SOURCE
Liens pour afficher les logs de chaque étape
SUIVI À L'IHM : CLOUDBEES EDITION
LONGUE INSTALLATION
Les builds de Workflow survivent à un redémarrage de
Jenkins
Le thread d'orchestration ne s'exécute que pour lancer la
prochaine tache
RÉSUMÉ
Une orchestration intégrée à la base de code
Un langage expressif (DSL workflow et Groovy)
Parallèlisation et distribution des tâches
Pipeline de déploiement continu
EST-CE QU'ON A LE TEMPS POUR
UNE DÉMO ?
MULTIBRANCH
Jenkins a uniquement besoin de connaître l'URL du
repository
MULTIBRANCH
Pour chaque branche à la racine de ce repository doit se
trouver le fichier Jenkinsfile
Les branches sont scrutées régulièrement et mises à jour et
lancées
On peut utiliser un hook pour lancer les builds
JENKINS SLAVES
Besoin d'installer des outils
Le bon JDK, la version de Maven qu'il me faut, gradle ? sbt ?
compilateur ?
DOCKER TO THE RESCUE !
Docker est la solution à tout de toutes manières
WORKFLOW DANS UNE IMAGE
DOCKER
docker.image('maven:3.3.3-jdk-8').inside {
sh 'mvn -B clean install'
}
Le répertoire courant dans docker est le workspace Jenkins
Montage du workspace dans un volume docker
Plus de problèmes d'outils, mis à part Docker qui doit être
installé sur les slaves
Le container est supprimé après son exécution
FAIRE DES IMAGES DOCKER DANS
JENKINS
DOCKER BUILD
node {
git scm // Récupérer le repository git en mode multibranch
def myEnv = docker.build 'my-environment:snapshot'
myEnv.push()
}
DOCKER BUILD
node {
git scm // Récupérer le repository git en mode multibranch
def myEnv = docker.build "my-environment:${env.BUILD_TAG}"
myEnv.push()
}
Utilisation du tag git pour pousser dans le registry
DOCKER POUR TESTER
node {
git scm
docker.image('mysql').withRun('-p 81:3126') {c ->
sh './test-with-local-db'
}
}
On veut exécuter le corps du withRun alors que l'image
mysql tourne
Le withRun prend optionnellement les paramètres du
docker run
Adapté aux tests d'intégration
DOCKER REGISTRY
node {
docker.withRegistry('https://docker.mycorp.com/', 'docker-login'){
git scm
docker.build('myapp').push('latest')
}
}
Possibilité d'utiliser un registry privé
On passe des credentials enregistrés dans Jenkins
DOCKER SERVER
docker.withServer('tcp://swarm.mycorp.com:2376', 'swarm-certs'){
docker.image('httpd').withRun('-p 8080:80') {c ->
sh "curl -i http://${hostIp(c)}:8080/"
}
}
def hostIp(container) {
sh "docker inspect -f {{.Node.Ip}} ${container.id} > hostIp"
readFile('hostIp').trim()
}
Possibilité de déléguer à un serveur docker (swarm ou autre)
Le client docker doit être installé
A utiliser si le slave ne peut pas supporter l'application à
tester
QUESTIONS ?
MERCI POUR VOTRE ÉCOUTE
Cette présentation :
@thebignet
thebignet
thebignet
talk-jenkins-workflow

Jenkins Workflow