CERT-FR

Score inclusion / TO DO Il faut pouvoir comparer le input des select avant de generer le ichier texte

// ==UserScript==
// @name          CERT-FR
// @namespace     CERT-FR
// @version       3.12.1
// @description   Score inclusion / TO DO Il faut pouvoir comparer le input des select avant de generer le ichier texte
// @author        SRI/DELANOY
// @icon          
// @match         https://cert.ssi.gouv.fr/avis/*
// @match         https://cert.ssi.gouv.fr/alerte/*
// @match         https://www.cert.ssi.gouv.fr/avis/*
// @match         https://www.cert.ssi.gouv.fr/alerte/*
// @match         https://www.cert.ssi.gouv.fr/actualite/*
// @connect       cyberwatch.internet.np
// @connect       cvedetails.com
// @connect       vuldb.com
// @connect       api.sourceclear.com
// @connect       cve.circl.lu
// @connect       nvd.nist.gov
// @connect       msrc.microsoft.com
// @connect       pastebin.com
// @connect       feedly.com
// @connect       gitlab-ce.internet.np
// @connect       rocketchat.internet.np
// @require       https://code.jquery.com/jquery-3.1.0.min.js
// @grant         GM.xmlHttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_listValues
// @grant        GM_deleteValue
// @grant        GM_registerMenuCommand
// ==/UserScript==

(function() {
    'use strict';
    var scriptVersion = "3.07.13"; // doit être modifiée manuellement en cas de changement majeur du contenu ; si ce n'est que fonctionnel alors inutile de le changer
    /* 3.07.01* : ajout des métadonnées du site NVD (date de publication, date de modification NVD et source de la publication)
	   3.07.13* : ajout d'une mention "(rejected)" si la CVE avait été rejetée par NVD
	   3.07.14 : modification de l'info de publication NVD : "NA (<date du jour>)" par "inconnu de nvd en date du <date du jour>"
	   3.07.15 : choix de NVD meme si le stockage est plus elevé (en cas de rabaissement)
	   3.08.01 : ajout du stochage depuis PasteBin
	   3.10.01 : ajout du stockage depuis https://gitlab-ce.internet.np/vuln/cert-fr/-/tree/main/
	   3.10.05 : début de factorisation de code en MVC
                 ; ajout de controle supplémentaires sur les erreurs dans la page
                 ; ajout d'option de debug comme remplacer la liste des cve
       3.10.06 : ajout de boutons pour modifier les CVE, titre, une option pour ne pas sauvegarder
       3.11.00 : ajout de la fonction d'aide
       3.11.05 : ajout d'une fonction pour ajouter un UL pour les references CVE en cas de manque de ce dernier
       3.11.06 : patch liens vuldb de chaque CVE en bas de page
       3.11.07 : ajout d'un highlight sur la note cliqué depuis le tableau CSV
       3.11.09 : suppression du footer et amélioration de l'affichage (évolution possible, un bouton pour afficher le footer ?
	*/ // Notes de versions />

    // https://greasyfork.org/scripts/372105-cert-fr/code/CERT-FR.user.js

    /* TO DO :
          activer l'animation pendant le tri / améliorer le tri
          revoir le nomage du titre PAste surtout en le mettant à jour, actuellement il a un prefixe constant
	*/ // TO DO />
    /* < PROBLEMES
		Probleme quand on switch de score v2 a v3 depuis le tableau des CVE, l'image V2 passe à grise à un moment donnée
		https://www.cert.ssi.gouv.fr/avis/CERTFR-2022-AVI-881/ => probleme en basculant de stockage a sans stockage
		https://www.cert.ssi.gouv.fr/avis/CERTFR-2022-AVI-937/ => ne prends pas le stockage
		https://www.cert.ssi.gouv.fr/avis/CERTFR-2022-AVI-1019/ le stockage est superieur à la nouvelle mesure CVE-2022-3602
		https://www.cert.ssi.gouv.fr/avis/CERTFR-2023-AVI-0258/ Problème recupération vecteur  ? désactiver le stockage pour les tests
		QUAND ALE, le nom du fichier de stockage ne change pas, revoir le nom peut être
		    nom precedent : du 2022-556 au 2023-0093 ; agrandissement de la partie ID (5 caracteres) 2023-ALE-00015_2023-AVI-00093 ; et comparer les titres avec le max AVI et le MAX ALE
		https://www.cert.ssi.gouv.fr/avis/CERTFR-2023-AVI-0120/ :
		    Le score NVD a évolué, mais un ancien NVD etait superieur à celui présent dans le stockage, ajouter une priorité sur les nouveaux NVD
    */ // PROBLEMES />

    // NOUVELLE FACTORISATION #####################################
    /* LEGENDE
        prefixe :
           - get_XXX = retourne l'objet XXX
           - is_XXX = retourne un bouleen satisfaisant la condition XXX
           - DOM_action_XXX = action sur la page (DOM) concernant XXX
           - ---_XXX = action multiple --- permet d'exprimer simplement l'action sur XXX (clean_listCVE -> nettoye la liste mais également supprime les objets DOM relatif)
    */
    // GENERIQUE
        // DOM
            function get_list_DOM_items_of_a_class(className, itemTag){
                var obj_DOM_parent, list_DOM_parent;
                try{
                    obj_DOM_parent = $("."+className+" "+itemTag); // retourne object{0:DOM,..., n:DOM}
                    list_DOM_parent = Object.values(obj_DOM_parent); // recupère la liste des valeur en tant que tableau (pour appliquer filter)
                }catch (error) {
                    debug("[get_list_DOM_items_of_a_class] ERROR : "+error, "error");
                }
                return list_DOM_parent;
            } // OK
    // FIN GENERIQUE

    /*
    Modèle : cette partie gère ce qu'on appelle la logique métier de votre site. Elle comprend notamment la gestion des données qui sont stockées,
    mais aussi tout le code qui prend des décisions autour de ces données. Son objectif est de fournir une interface d'action la plus simple possible au contrôleur.
    On y trouve donc entre autres des algorithmes complexes et des requêtes SQL.
    */ // MODELE
        // CVE
            function get_regEx_CVE(){return /((CVE)\-([0-9]{4})\-[0-9]*)/g;}//OK
            function get_regEx_lienCVE(){return /(http[s]{0,1}:\/\/(.*))/g;}//OK

        // Page
            function get_regEx_title_referenceCVE(){return /documentation/ig;}//OK
            function is_title_referenceCVE(DOM_item){ // retourne vrai si l'objet en entrée est le titre des references CVE
                var isOK = false;
                if(typeof DOM_item == "object"){
                    if("innerText" in DOM_item){
                        var regEx = get_regEx_title_referenceCVE();
                        isOK = DOM_item.innerText.match(regEx) != null
                    }
                }
                return isOK;
            }//OK

            function get_regEx_txtCVE_CERTFR(){return /(.*?)( CVE )((CVE)\-([0-9]{4})\-[0-9]*)/g;}//OK
            function is_referenceCVE_CERTFR(DOM_item){ // retourne vrai si l'objet en entrée  est l'objet referencant une CVE
                var isOK = false;
                if(typeof DOM_item == "object"){
                    if("innerText" in DOM_item){
                        var regEx = get_regEx_txtCVE_CERTFR();
                        isOK = DOM_item.innerText.match(regEx) != null
                    }
                }
                return isOK;
            }//OK
            function get_text_referenceCVE(DOM_item){ // retourne le text correspondant au noeud text
                let result = document.evaluate(
                    'text()[1]', // xpathExpression ; "text()[matches(., '.*')]" ne fonctionne pas
                    DOM_item, // contextNode
                    null, // namespaceResolver
                    XPathResult.STRING_TYPE, // resultType
                    null //result
                ).stringValue;
                return result;
            } // OK mais pourrait être amélioré au niveau de la regex à la recherche du bon textnode
            function get_CVE_fromItem(DOM_item, attribute="innerText", defaultValue=""){ // extrait la référence CVE depuis l'objet référencant une CVE
                var CVE = defaultValue;
                var matchCVE
                if(typeof DOM_item == "object" && DOM_item!= null){
                    if(DOM_item.hasAttribute(attribute) || attribute in DOM_item){
                        matchCVE = DOM_item[attribute].match(get_regEx_CVE());
                        if(matchCVE){
                            CVE = matchCVE[0];
                        }
                    }
                }
                return CVE;
            }//OK
            function getNextItem_as_(DOM_item, searchFor="a"){
                searchFor=searchFor.toLowerCase();
                let nextSibling = DOM_item.nextElementSibling;
                if(nextSibling!=null){// a un voisin
                    let siblingType = nextSibling.nodeName.toLowerCase();
                    if(siblingType!=searchFor){ // on passe au suivant
                        nextSibling = getNextItem_as_(nextSibling, searchFor)
                    }
                }
                return nextSibling;
            }
            function get_a_CVE_fromItem(DOM_item){ // extrait la référence CVE depuis l'objet référencant une CVE
                var DOM_a_CVE = null;
                var matchCVE
                if(typeof DOM_item == "object" && DOM_item!= null){
                    DOM_a_CVE = DOM_item.querySelector('a');
                    if(DOM_a_CVE==null){ // on cherche le suivant dans ses siblings
                        DOM_a_CVE=getNextItem_as_(DOM_item, "a");
                    }
                }
                return DOM_a_CVE;
            }//OK
            function get_list_DOM_li(){return get_list_DOM_items_of_a_class("article-content", "li");} // OK
            function fix_DOMul_nextSibling(DOMul_expected){
                var DOMul, sibling;


                switch(DOMul_expected.nodeName.toLowerCase()){
                    case 'ul': // all good
                        DOMul = DOMul_expected;
                        break;
                    case 'li': // pas un UL
                        // création de l'ul
                        DOMul = document.createElement('ul');
                        // clone de l'actuel li
                        sibling = DOMul_expected;
                        // remplacement de l'objet attendu ul par un ul
                        DOMul_expected.replaceWith(DOMul);
                        // ajout de l'élément dans le ul
                        DOMul.appendChild(sibling);
                        while(DOMul.nextElementSibling){
                            if(DOMul_expected.nodeName.toLowerCase() === "li"){
                                sibling = DOMul.nextElementSibling;
                                DOMul.appendChild(DOMul_expected);
                                DOMul_expected = sibling;
                            }else{
                                break;
                            }
                        }
                        break;
                }

                return DOMul;
            }
            function get_ul_CVE(){
                var list_DOM_parent, list_DOM_itemsTitleCVE = [];
                var last_DOM_titleCVE, DOM_UL_CVES=null;
                try{
                    list_DOM_parent = get_list_DOM_items_of_a_class("article-content", "h2");
                    list_DOM_itemsTitleCVE = list_DOM_parent.filter(is_title_referenceCVE) // récupère la liste des objets li qui sont une référence à une CVE par le CERTFR
                    if(list_DOM_itemsTitleCVE.length>0){
                        last_DOM_titleCVE = list_DOM_itemsTitleCVE[list_DOM_itemsTitleCVE.length -1];
                        if("nextElementSibling" in last_DOM_titleCVE){
                            DOM_UL_CVES = last_DOM_titleCVE.nextElementSibling
                            DOM_UL_CVES = fix_DOMul_nextSibling(DOM_UL_CVES);
                            //console.log("### UL obtenu : ",DOM_UL_CVES)
                        }
                    }
                }catch (error) {
                    debug("[get_ul_CVE] ERROR : "+error, "error");
                }
                return DOM_UL_CVES;
            } // OK
            function get_list_DOM_CVEs_fromPage(){ // retourne la liste des objets DOM qui sont des references à une CVE
                var list_DOM_parent, list_DOM_itemsCVE = []; // sortie
                try{
                    list_DOM_parent = get_list_DOM_li();
                    list_DOM_itemsCVE = list_DOM_parent.filter(is_referenceCVE_CERTFR) // récupère la liste des objets li qui sont une référence à une CVE par le CERTFR
                }catch (error) {
                    debug("[get_list_DOM_CVEs_fromPage] ERROR : "+error, "error");
                }
                return list_DOM_itemsCVE;
            }//OK
            function get_list_string_CVEs_from_list_DOM_CVEs(list_DOM_itemsCVE){ // retourne la liste des CVE depuis la liste des DOM
                var listCVEs = [];
                for(var i=0;i<list_DOM_itemsCVE.length;i++){
                    listCVEs.push(get_CVE_fromItem(list_DOM_itemsCVE[i]));
                }
                return listCVEs;
            } // OK

            function make_OG_CVEitem(CVE, DOM_ul= get_ul_CVE()){ // OG pour originale
                if(DOM_ul){
                    if(typeof DOM_ul == "object"){
                        var li = document.createElement( 'li' );
                        li.appendChild(document.createTextNode("Référence CVE "+CVE));
                        li.appendChild(document.createElement("br"));
                        var a = document.createElement("a");
                        a.href = "https://www.cve.org/CVERecord?id="+CVE;
                        a.innerText = a.href;
                        li.appendChild(a);
                        DOM_ul.appendChild(li);
                    }
                }
            } // OK (entre la vue et la donnée
            function replace_CVEitem(DOM_itemCVE_old, DOM_ul= get_ul_CVE()){
                if(DOM_ul){
                    var CVE = get_CVE_fromItem(DOM_itemCVE_old);
                    var children = DOM_itemCVE_old.childNodes
                    var child, newItem, url_cyberwatch;

                    // recupération du texte de reference
                    var textReference = get_text_referenceCVE(DOM_itemCVE_old);

                    // creation du nouveau LI
                    var DOM_itemCVE_new = document.createElement('li');
                    // creation du texte de reference
                    child = document.createElement('p');
                    child.id = "li_CVE_"+CVE;
                    child.innerText = textReference;

                    DOM_itemCVE_new.appendChild(child);

                    // ajout du lien originel
                    child = get_a_CVE_fromItem(DOM_itemCVE_old);//.querySelector('a');
                    DOM_itemCVE_new.appendChild(child);

                    // ajout d'un br custom
                    child = document.createElement('br');//DOM_itemCVE_old.querySelector('br');
                    child.classList.add("avoid_from_selection_but_for_html_export")
                        DOM_itemCVE_new.appendChild(child);

                    // ajout du lien cyberwatch
                    url_cyberwatch = "https://cyberwatch.internet.np/cve_announcements/"+CVE;
                    child = document.createElement('a');
                    child.href = url_cyberwatch;
                    child.target = "_blank";
                    child.classList.add("only_for_html_export");
                    child.innerText = url_cyberwatch;
                    DOM_itemCVE_new.appendChild(child);
                    DOM_ul.replaceChild(DOM_itemCVE_new, DOM_itemCVE_old);
                }
            }


        // storage

        // online Storage

        // APIs
            function searchVulDB(event){
                // fonction qui lance la recherhce sur vuldb
                //console.log(event)
                try{
                    var cveToSearch = event.target.getAttribute('_toSearch');
                    console.log("[searchVulDB] : ",cveToSearch)
                    // recupération de l'objet cible
                    switch (event.which) {
                        case 1: // Left
                        case 2: // Middle
                            $('#'+cveToSearch+'_vuldb').children(":first").removeAttr('style');
                            $('#'+cveToSearch+'_vuldb').children(":first").css("margin","0 0 3px 20px");
                            //$(document).delegate("#"+CVE+"_vuldb","click",function(){
                            updateClipboard(cveToSearch);
                            $('#search_vuldb_'+cveToSearch).submit();
                            //});
                            break;
                            break;
                        case 3: // Right
                            $('#'+cveToSearch+'_vuldb').children(":first").css("filter","invert(1)");
                            break;
                    }
                }catch(error){
                    console.error("[searchVulDB] ERROR :", error.message);
                }
            }

    /*
    Vue : cette partie se concentre sur l'affichage. Elle ne fait presque aucun calcul et se contente de récupérer des variables pour savoir ce qu'elle doit afficher.
    On y trouve essentiellement du code HTML mais aussi quelques boucles et conditions PHP très simples, pour afficher par exemple une liste de messages.
    */ // VUE
            function DOM_remove_object(DOM_item){DOM_item.remove();}//OK
            function get_DOMid_debugPage(){return "cmb_log_page";}//OK en devenir select_debugPage
            function add_option_in_select(selectID, optionValue, optionText){
                try{
                    add_log_to_CMB_PAGE(optionText);
                }catch (error){
                    debug("[add_option_in_select] ERROR : "+error, "error");
                }
            }//WIP

            // DEMO / AIDE / TUTO / INTRO
					// ************ <BLOC INTRO/DEMO/AIDE copié
						// FONCTION GENERIQUES OK
							// GUI SIMPLE
								function show(DOMobject){DOMobject.style.display="block";} // v1.0
								function hide(DOMobject){DOMobject.style.display="none";} // v1.0
								function getTheHigerZindex_of_the_document(){ // fonction qui retourne le zIndex max de la page (et modifie les auto dans la valeur max pour figer la cascade // v1.0
									// declaration des variables
										var zIndex = 0;
										var lst_DOMobject_with_zIndex_auto = []
										var zIndex_temp, css_obj, i;

										var all = document.querySelectorAll("*");

									// pour chaque element de la feuille, on recherche le zindex max ; et on garde dans une liste ceux en auto
										for(i=0 ; i<all.length ; i++){
											try{
												css_obj = getComputedStyle(all[i]);
												zIndex_temp =css_obj.getPropertyValue("z-index");

												if(zIndex_temp){
													if( zIndex_temp == "auto"){ // cas particulier, lister ceux à auto, puis leur donner la valeur max à la fin
														lst_DOMobject_with_zIndex_auto.push(all[i]);
													}else{
														zIndex = zIndex<zIndex_temp ? zIndex_temp : zIndex;
													}
												}
											}catch{
											}
										}
                                    // remise des auto à la valeur max actuelle, sera équivalent à un auto en terme de rendu mais permettra l'aide d'etre au dessus temporairement
										for(i=0;i<lst_DOMobject_with_zIndex_auto.length;i++){
											lst_DOMobject_with_zIndex_auto[i].style.zIndex = zIndex;
										}
									// SORTIE
										return zIndex;
								}


						// FONCTIONS PRIVEES
							// GUI style
								function set_style_popup(DOMobject){
									DOMobject.style.border= "5px solid green";
									DOMobject.style.position= "fixed";
									DOMobject.style.zIndex= "1";
									DOMobject.style.left= "0";
									DOMobject.style.top= "0";
									DOMobject.style.width= "100%";
									DOMobject.style.height= "100%";
									DOMobject.style.overflow= "auto";
									DOMobject.style.backgroundColor= "rgba(0, 0, 0, 0.6)";
									DOMobject.style.display= "none";
								}
								function set_style_popupContent(DOMobject){
									DOMobject.style.border= "5px solid yellow";
									DOMobject.style.backgroundColor= "white";
									DOMobject.style.padding= "20px";
									DOMobject.style.width= "auto";
									DOMobject.style.height= "auto";
									DOMobject.style.fontWeight= "bolder";
								}
							// GUI POPUP MESSAGE
								function set_popupBoxMessage(zIndexTop){ // cree la div autour du message // v1.0
									var DOMBoxMessage = document.createElement("div");
									set_style_popup(DOMBoxMessage)
										DOMBoxMessage.id="myPopup";
										DOMBoxMessage.style.zIndex = zIndexTop
										DOMBoxMessage.style.position = "absolute"

									// positionnement
										var body = document.querySelector("body");
										body.insertBefore(DOMBoxMessage, body.firstChild)

									return DOMBoxMessage
								} // v1.0
								function placePopupMessageAroundTarget(DOMBoxMessage, DOMhighlight){ // place le corps du message au meilleur endroit // v1.0
									// recupération du message
										var DOMmessage = DOMBoxMessage.firstChild;
									// récupération de la taille de la fenetre
										var w = window.innerWidth;
										var h = window.innerHeight;
										//console.log("#1 windows :", w," x ", h);

									// récupération de la taille de la cible
										var rectTarget = DOMhighlight.getBoundingClientRect();
										//console.log("#2 rectTarget", rectTarget)

									// récupération de la taille du message
										show(DOMBoxMessage);
										var rectMessage = DOMmessage.getBoundingClientRect();
										//console.log("#3 rectMessage", rectMessage)
										//console.log(DOMBoxMessage.style.display)
										hide(DOMBoxMessage); // on cache l'objet après avoir récupéré ses dimensions
										//console.log(DOMBoxMessage.style.display)

									// définition de la position idéale
										var rightSpace = w - rectTarget.right;
										var topSpace = rectTarget.top;
										var leftSpace = rectTarget.left;
										var bottomSpace = h - rectTarget.bottom;

										var leftMessage = rectMessage.left;
										var topMessage = rectMessage.top;

										if(rightSpace>=rectMessage.width){ // recherche si de la place à droite de l'element
											//console.log("droite");
											leftMessage = rectTarget.right;
											topMessage = rectTarget.top;
										}else if(topSpace>=rectMessage.height){ // recherche de la place dessus l'element
											//console.log("dessus");
											leftMessage = 0;
											topMessage = rectTarget.top-rectMessage.height;
										}else if(leftSpace>=rectMessage.width){ // recherche de la place à gauche de l'element
											//console.log("gauche");
											leftMessage = (rectTarget.left-rectMessage.width);
											topMessage = rectTarget.top;
										}else if(bottomSpace>=rectMessage.height){ // recherche de la place en dessous l'element
											//console.log("dessous");
											leftMessage = 0;
											topMessage = rectTarget.top + rectTarget.height;
										}else{// affichage en plein centre mais déplacable et redimensionnable // WIP
										}
										DOMmessage.style.left = Math.round(leftMessage) +"px";
										DOMmessage.style.top = Math.round(topMessage) +"px";
									} // v1.0


								function closePopUp(event, DOMhighlight, DOMBoxMessage){ // fonction qui supprime l'objet popup message et arrete le surlignement d'un objet// v1.0
									if (event.target == DOMBoxMessage && event.target != DOMBoxMessage.firstChild) { // en dehors du popup message
										cleanHighLight(DOMhighlight);
										DOMBoxMessage.remove();
									}
								} // v1.0
								function clickAtParent(child){ // fonction qui click sur le parent du message (autour du popup message) ce qui pousse à la fermeture // v1.0
									child.parentNode.click();
								} // v1.0

							// GUI HIGHLIGHT
								function makeHighLight(DOMtarget){ // fonction qui met en lumière la cible // v1.0
									var DOMhighlight = document.createElement("div");
									DOMhighlight.style = DOMtarget.style
									DOMhighlight.style.boxSizing="border-box";
										DOMhighlight.style.padding = "10px";
										DOMhighlight.style.height = "max-content";
										DOMhighlight.style.width = "max-content";
										DOMhighlight.style.backgroundColor = "white";
										DOMhighlight.style.border = "3px dotted red"; // = "border:1px solid black;border-spacing:10px;margin: 20px auto";

									DOMtarget.replaceWith(DOMhighlight);
									DOMhighlight.appendChild(DOMtarget)
									return DOMhighlight;
								} // v1.0

								function cleanHighLight(DOMtarget){ // fonction qui récupère le highlight et reverse la mise en lumière // v1.0
									var DOMoriginalTarget = DOMtarget.firstChild;
									DOMtarget.replaceWith(DOMoriginalTarget);
								} // v1.0


						// FONCTIONS PUBLIQUES
							function closePopUpFromInnerMessage(DOMmessage){if(DOMmessage){clickAtParent(DOMmessage);}} // v1.0

							function messageFor(DOMmessage, DOMtarget=document.querySelector("body")){ // fonction qui génère le popup contenant le message et highlight une cible // v1.0
								var zIndexTop = getTheHigerZindex_of_the_document()+1;
								// Highlight la cible
									var DOMhighlight = makeHighLight(DOMtarget);
								// cadre du message
									var DOMBoxMessage = set_popupBoxMessage(zIndexTop);
										DOMBoxMessage.appendChild(DOMmessage);

								// modificaiton de la cible
									DOMhighlight.style.zIndex = zIndexTop+1;
									DOMhighlight.style.position = "absolute"

								// pre modification du message
									DOMmessage.style.position = 'absolute';

								// Positionnement du message si pas précisé
									placePopupMessageAroundTarget(DOMBoxMessage, DOMhighlight);

								// affichage du message
									//console.log(DOMBoxMessage.style.display)
									DOMBoxMessage.classList.add("popup", "show");
									show(DOMBoxMessage);
									//console.log(DOMBoxMessage.style.display)

								// ajout des fonction de fermeture # https://weekendprojects.dev/posts/how-to-pass-parameter-to-a-addeventlistener-function/
									window.addEventListener("click", (event) => closePopUp(event, DOMhighlight, DOMBoxMessage));

							}

							// fonction de creation du corps du message
								function makePopUpMessage(title, message, precedentFctToCall = false, nextFctToCall = false){ // fonction qui permet la creation d'un message générique // v1.0
									// declaration des variables
										var child, DOMmessage, btnTd_gauche, btnTd_droite;

									// creation du cadre du message
										DOMmessage = document.createElement("div");
											DOMmessage.id = "messageDiv";
											set_style_popupContent(DOMmessage)

									// bouton fermer (x)
										child = document.createElement('span');
											child.innerText = "X"
											child.style.textAlign = "right";
											child.style.float = "right";
											child.style.borderRadius = "10px";
											child.style.padding = "5px";
											child.style.border = "2px solid black";
											child.style.color = "darkred";
											child.style.backgroundColor = "lightgrey";
											child.addEventListener("click",(event) => closePopUpFromInnerMessage(DOMmessage));
										DOMmessage.appendChild(child)

									// titre
										child = document.createElement("h1");
											child.innerText = title;
											child.style.color = "green";
										DOMmessage.appendChild(child)

									// corps du message
										child = document.createElement("p");
											child.innerText = message;
										DOMmessage.appendChild(child)

									// creation des div avec les boutons
										btnTd_gauche = document.createElement("div");
											btnTd_gauche.style.textAlign = "left";
											btnTd_gauche.style.float = "left";
											btnTd_gauche.style.border = "2px solid red";
										btnTd_droite = document.createElement("div");
											btnTd_droite.style.textAlign = "right";
											btnTd_droite.style.float = "right";
											btnTd_droite.style.border = "2px solid blue";
										DOMmessage.appendChild(btnTd_gauche);
										DOMmessage.appendChild(btnTd_droite);

									// ajout des boutons
										if(precedentFctToCall){
											child = document.createElement('a');
												child.href="#";
												child.target=""
												child.innerText = "Précedent"
												child.addEventListener("click",(event) => precedentFctToCall(DOMmessage)); // permet de fermet ce message pour afficher le precedent
											btnTd_gauche.appendChild(child)
										}

										if(nextFctToCall){
											child = document.createElement('a');
												child.href="#";
												child.target=""
												child.innerText = "Suivant"
												child.addEventListener("click",(event) => nextFctToCall(DOMmessage)); // permet de fermet ce message pour afficher le suivant
											btnTd_droite.appendChild(child)
										}
									// sortie
										return DOMmessage;
								} // v1.0

								function makePopUpMessage_lst_dict(title, message, list_dictBoutons = [{txt:"txt bouton", fct:false}]){ // fonction qui permet la creation d'un message générique XX fct du dict peut être faux, sinon cela doit être le nom d'une fonction qui prend en paramètre facultatif l'objet DOM à l'origine de ce popup message // v1.1
									// declaration des variables
										var child, DOMmessage, i, box_btn_div, btn_txt, btn_fct, btn_div;

									// creation du cadre du message
										DOMmessage = document.createElement("div");
											DOMmessage.id = "messageDiv";
											set_style_popupContent(DOMmessage);

									// bouton fermer (x)
										child = document.createElement('span');
											child.innerText = "X"
											child.style.textAlign = "right";
											child.style.float = "right";
											child.style.borderRadius = "10px";
											child.style.padding = "5px";
											child.style.border = "2px solid black";
											child.style.color = "darkred";
											child.style.backgroundColor = "lightgrey";
											child.addEventListener("click",(event) => closePopUpFromInnerMessage(DOMmessage));
										DOMmessage.appendChild(child)

									// titre
										child = document.createElement("h1");
											child.innerText = title;
											child.style.color = "green";
										DOMmessage.appendChild(child)

									// corps du message
										child = document.createElement("div");
											child.innerHTML = message;
										DOMmessage.appendChild(child)

									// creation de la div contenant les div des liens
										box_btn_div = document.createElement("div");
											box_btn_div.style.textAlign = "justify";
											box_btn_div.style.textJustify = "distribute-all-lines";
											//box_btn_div.style.flexWrap = "wrap";
											//box_btn_div.style.display = "inline-flex";

									// ajout des boutons
										for(i=0;i<list_dictBoutons.length;i++){
											btn_txt = list_dictBoutons[i].txt
											btn_fct = list_dictBoutons[i].fct


											// creation de la div contenant le lien (pour mieux le placer)
												btn_div = document.createElement("div");
													btn_div.style.border = "2px solid red"; // pour debug
													//btn_div.with = "auto"; // pour debug
													btn_div.with = "calc(100% / "+i+")";
													btn_div.style.verticalAlign = "top";
													btn_div.style.display = "inline-block";
													btn_div.style.display = "inline";
													btn_div.style.zoom = 1;

												// on regarde à quelle position de la liste on se trouve
													if(i==0){ // c'est le premier element
														//btn_div.style.textAlign = "left";
														//btn_div.style.float = "left";
													}else if(i == list_dictBoutons.length -1){ // c'est le dernier element
														//btn_div.style.textAlign = "right";
														//btn_div.style.float = "right";
													}else{ // case au centre
														//btn_div.style.textAlign = "justify";
														//btn_div.style.display="inline-block";
														//btn_div.style.display="flex";
														//btn_div.style.textAlign = "left";
														//btn_div.style.float = "left";
														//btn_div.style.justifyContent="space-around";
														//btn_div.style.justifyContent="space-between";
													}

											if(btn_fct){ // une fonction est à apellé

												// on s'assure qu'il y a du texte
													btn_txt = btn_txt ? btn_txt : 'fonction_'+i;

												// création du bouton
													child = document.createElement('a'); // lien sans cible
														child.href="#";
														child.target=""
														child.innerText = btn_txt
													// ajout d'un event de click
														child.addEventListener("click",(event) => precedentFctToCall(DOMmessage)); // permet de fermet ce message pour afficher le precedent

												// on ajoute le bouton à la div
													btn_div.appendChild(child);
											}else{ // pas de fonction appelée, donc sert à bloquer un espace
											}

											box_btn_div.appendChild(btn_div);
											DOMmessage.appendChild(box_btn_div);
										}
										// creation de la SPAN qui les relier
											child = document.createElement("span");
												child.style.width = "100%"
												child.style.display = "inline-block";
												child.style.display = "inline";
												child.style.fontSize = 0;
												child.style.lineHeight = 0;

											btn_div.appendChild(child);

									// sortie
										return DOMmessage;
								} // v1.1

					// ************ BLOC INTRO/DEMO/AIDE copié>

                function makePopUpMessage_squelette_v1_0(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("myButton");
                    var title = "";
                    var message = "";
                    var precedentFctToCall = false;
                    var nextFctToCall = false;//makePopUpMessage_2;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("tableCVEs");
                    var title = "Liste des CVE";
                    var message = "Dans ce tableau se trouve les CVE trouvés dans l'article,\nIl existe plusieurs interactions depuis ce tableau";
                    var precedentFctToCall = false;
                    var nextFctToCall = makePopUpMessage_listCVE_ths;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_ths(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("tableCVEs_THs");
                    var title = "Tri";
                    var message = "Les entêtes fournissent un tri en cliquant dessus";
                    var precedentFctToCall = makePopUpMessage_listCVE;
                    var nextFctToCall = makePopUpMessage_listCVE_CVE;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }

                function get_DOM_td(DOMtable, row=0,col=0, rowType="tr", colType="td"){
                    var i, cpt, trs, tr, tds, td, DOMoutput = DOMtable;
                    trs = DOMtable.querySelectorAll(rowType);
                    //console.log(trs)
                    if(row<trs.length){
                        tr = trs[row];
                        cpt = 0;
                        //console.log(tr)
                        for(i=0;i<tr.childNodes.length;i++){
                            td = tr.childNodes[i];
                            //console.log("? ",td);
                            if(td.nodeName == colType.toUpperCase()){
                                if(cpt==col){
                                    DOMoutput = td;
                                    break;
                                }
                                cpt++;
                            }
                        }
                    }
                    return DOMoutput ;
                }

                function makePopUpMessage_listCVE_CVE(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOM_table = document.getElementById("tableCVEs");
                    var DOMtarget = get_DOM_td(DOM_table,1,0, "tr", "td");
                    if(DOMtarget == DOM_table){// pas de CVE, on prend l'entete th
                        //console.log("### indentique ###");
                        DOMtarget = get_DOM_td(DOM_table,0,0, "tr", "th");
                        //console.log(DOMtarget);
                    }
                    var title = "CVE";
                    var message = "La cellule CVE (pour chaque CVE) contient des informations cachées obtenues depuis NVD\nEn cliquant dessus cela copie dans le presse-papier le nom de la CVE\n\n1/100";
                    var precedentFctToCall = makePopUpMessage_listCVE_ths;
                    var nextFctToCall = makePopUpMessage_listCVE_v2;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_v2(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOM_table = document.getElementById("tableCVEs");
                    var DOMtarget = get_DOM_td(DOM_table,1,2);
                    if(DOMtarget == DOM_table){// pas de CVE, on prend l'entete th
                        DOMtarget = get_DOM_td(DOM_table,0,2, "tr", "th");
                    }
                    var title = "Le score v2";
                    var message = "La cellule v2 (pour chaque CVE) contient le score trouvé depuis NVD\nEn cliquant dessus cela selectionne la CVE dans le cartouche et applique le score et meta données relatives";
                    var precedentFctToCall = makePopUpMessage_listCVE_CVE;
                    var nextFctToCall = makePopUpMessage_listCVE_v3;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_v3(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOM_table = document.getElementById("tableCVEs");
                    var DOMtarget = get_DOM_td(DOM_table,1,3);
                    if(DOMtarget == DOM_table){// pas de CVE, on prend l'entete th
                        DOMtarget = get_DOM_td(DOM_table,0,3, "tr", "th");
                    }
                    var title = "Le score v3";
                    var message = "La cellule v3 (pour chaque CVE) contient le score trouvé depuis NVD\nEn cliquant dessus cela selectionne la CVE dans le cartouche et applique le score et meta données relatives";
                    var precedentFctToCall = makePopUpMessage_listCVE_v2;
                    var nextFctToCall = makePopUpMessage_listCVE_v3_CNA;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_v3_CNA(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOM_table = document.getElementById("tableCVEs");
                    var DOMtarget = get_DOM_td(DOM_table,1,4);
                    if(DOMtarget == DOM_table){// pas de CVE, on prend l'entete th
                        DOMtarget = get_DOM_td(DOM_table,0,4, "tr", "th");
                    }
                    var title = "Le score v3 CNA";
                    var message = "La cellule v3 CNA (pour chaque CVE) contient le score trouvé depuis NVD mais fournis par l'éditeur\nEn cliquant dessus cela selectionne la CVE dans le cartouche et applique le score et meta données relatives";
                    var precedentFctToCall = makePopUpMessage_listCVE_v3;
                    var nextFctToCall = makePopUpMessage_listCVE_lien_NVD;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_lien_NVD(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOM_table = document.getElementById("tableCVEs");
                    var DOMtarget = get_DOM_td(DOM_table,1,5);
                    if(DOMtarget == DOM_table){// pas de CVE, on prend l'entete th
                        DOMtarget = get_DOM_td(DOM_table,0,5, "tr", "th");
                    }
                    var title = "Le lien NVD";
                    var message = "Le lien NVD ouvre un nouvel onglet vers la page NVD de la CVE";
                    var precedentFctToCall = makePopUpMessage_listCVE_v3_CNA;
                    var nextFctToCall = makePopUpMessage_listCVE_lien_google;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_lien_google(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOM_table = document.getElementById("tableCVEs");
                    var DOMtarget = get_DOM_td(DOM_table,1,6);
                    if(DOMtarget == DOM_table){// pas de CVE, on prend l'entete th
                        DOMtarget = get_DOM_td(DOM_table,0,5, "tr", "th");
                    }
                    var title = "Le lien Google";
                    var message = "Le lien Google ouvre un nouvel onglet vers la page Google qui recherche 'cvss' 'CVE'";
                    var precedentFctToCall = makePopUpMessage_listCVE_lien_NVD;
                    var nextFctToCall = makePopUpMessage_listCVE_lien_CW;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_lien_CW(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOM_table = document.getElementById("tableCVEs");
                    var DOMtarget = get_DOM_td(DOM_table,1,7);
                    if(DOMtarget == DOM_table){// pas de CVE, on prend l'entete th
                        DOMtarget = get_DOM_td(DOM_table,0,5, "tr", "th");
                    }
                    var title = "Le lien Cyberwatch";
                    var message = "Le lien Cyberwatch ouvre un nouvel onglet vers la page Cyberwatch de la CVE";
                    var precedentFctToCall = makePopUpMessage_listCVE_lien_google;
                    var nextFctToCall = makePopUpMessage_listCVE_div_input;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }

                function makePopUpMessage_listCVE_div_input(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("div_input");
                    var title = "La zone input";
                    var message = "Cette zone sert à modifier le cartouche de criticité";
                    var precedentFctToCall = makePopUpMessage_listCVE_lien_CW;
                    var nextFctToCall = makePopUpMessage_listCVE_input_text_cvss;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_input_text_cvss(DOMprevious=false){ // ajouter la surbrillance sur les case à cocher : DIV + call fonction + call clean @ close
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("input_text_cvss");
                    var title = "Saisie du vecteur";
                    var message = "Cette barre de saisie permet de convertir le vecteur CVSS saisi en un score calculé pour le cartouche \n\nATTENTION aux verrouillages (les cases à cocher plus bas)";
                    var precedentFctToCall = makePopUpMessage_listCVE_div_input;
                    var nextFctToCall = makePopUpMessage_listCVE_boxToGMstorage_label;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_listCVE_boxToGMstorage_label(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("divBoxToGMstorage");
                    var title = "Case à cocher 'Stocker (local + Git)'";
                    var message = "Cette case permet de tester le script sans affecter le stockage.\nLaisser cochée pour le traitement courant.\n\nPI : En décochant la case, au téléchargement du fichier, vous ne modifierez pas le stockage";
                    var precedentFctToCall = makePopUpMessage_listCVE_input_text_cvss;
                    var nextFctToCall = makePopUpMessage_cartouche_table;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_cartouche_table(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("cartouche_table");
                    var title = "Cartouche de criticité";
                    var message = "Dans ce tableau se trouvent les CVE avec le score v2 et v3 les plus élevés,\nIl y a qu'un seul champ éditable\n, les autres sont obtenus en cliquant sur la cellule du score de la CVE choisie et en modifiant le vecteur";
                    var precedentFctToCall = makePopUpMessage_listCVE_boxToGMstorage_label;
                    var nextFctToCall = makePopUpMessage_refCVE_vecteur_v2_source;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_refCVE_vecteur_v2_source(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("refCVE_vecteur_v2_source");
                    var title = "Saisie de l'origine du score";
                    var message = "Dans cette barre de saisie, vous pouvez ajouter un commentaire quant à l'origine comme l'URL du bulletin de l'éditeur ou d'autres infos";
                    var precedentFctToCall = makePopUpMessage_cartouche_table;
                    var nextFctToCall = makePopUpMessage_link_vecteur_v2;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_link_vecteur_v2(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("link_vecteur_v2");
                    var title = "Lien vecteur";
                    var message = "En cliquant sur ce lien, vous appliquerez le vecteur dans la zone de modification";
                    var precedentFctToCall = makePopUpMessage_refCVE_vecteur_v2_source;
                    var nextFctToCall = makePopUpMessage_btn_add_RLof_RCc;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_btn_add_RLof_RCc(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("btn_add_RLof_RCc");
                    var title = "Bouton appliquer score temporel par défaut";
                    var message = "Ce bouton applique aux deux scores les attributs temporels les plus courants : confirmé par l'éditeur, et correction officielle";
                    var precedentFctToCall = makePopUpMessage_link_vecteur_v2;
                    var nextFctToCall = makePopUpMessage_html_gen;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }
                function makePopUpMessage_html_gen(DOMprevious=false){
                    closePopUpFromInnerMessage(DOMprevious)
                    var DOMtarget = document.getElementById("html_gen");
                    var title = "Bouton de téléchargement HTML";
                    var message = "Ce bouton télécharge le bulletin dûment rempli au format HTML pour la publication sur notre portail";
                    var precedentFctToCall = makePopUpMessage_btn_add_RLof_RCc;
                    var nextFctToCall = false;
                    var DOMmessage = makePopUpMessage(title, message, precedentFctToCall, nextFctToCall)
                    messageFor(DOMmessage, DOMtarget);
                }

                function defaultTableBorder(){ return "1px solid black";}
                function higlightCellBorder(DOMitem){ DOMitem.style.border="2px solid red";}

                function resetTableBorderStyle(DOMitem){
                    switch(DOMitem.nodeName){
                        case "TABLE":
                        case "TBODY":
                        case "TR":
                        case "TH":
                        case "TD":
                            DOMitem.style.border=defaultTableBorder();
                            break;
                    }
                }

                function resetBorderStyle(DOMitem, cleanOnly=""){
                    try{
                        var id= DOMitem.id;
                        var matchingResult = DOMitem.id.match(cleanOnly);
                        if(Array.isArray(matchingResult)){
                            if(matchingResult[0] == cleanOnly){
                                resetTableBorderStyle(DOMitem);
                            }
                        }
                        if(DOMitem.children.length>0){
                            for(var i=0 ; i<DOMitem.children.length;i++){
                                resetBorderStyle(DOMitem.children[i], cleanOnly);
                            }
                        }
                    }catch(error){
                        console.error("[resetBorderStyle] ERROR : cleanOnly=",cleanOnly," ; id=",id, error.message);
                    }
                }

                function afficherAide(){ // fonction qui permet l'affichage d'une aide sur les différents elements
                    makePopUpMessage_listCVE();
                }
GM_registerMenuCommand("AIDE", afficherAide); // WIP

    /*
    Contrôleur : cette partie gère les échanges avec l'utilisateur. C'est en quelque sorte l'intermédiaire entre l'utilisateur, le modèle et la vue.
    Le contrôleur va recevoir des requêtes de l'utilisateur. Pour chacune, il va demander au modèle d'effectuer certaines actions (lire des articles de blog depuis une base de données,
    supprimer un commentaire) et de lui renvoyer les résultats (la liste des articles, si la suppression est réussie). Puis il va adapter ce résultat et le donner à la vue.
    Enfin, il va renvoyer la nouvelle page HTML, générée par la vue, à l'utilisateur.
    */ // CONTROLLEUR
        // DEBOGAGE
            /* debugLevelList
                "silent":"aucun message de debug",
                "info":"n'affiche au mieux que les messages dans la console",
                "custom":"respecte les choix propres à chaque appel"
            */
            var debugLevel = "custom";
            function update_debug_listOfWhere_from_actual_status(listOfWhere){ // adapte le listOfWhere en fonction du niveau de debugage choisi
                switch(debugLevel){
                    case "silent":
                        listOfWhere=[];
                        break;
                    case "info" :
                        listOfWhere = ["console"];
                        break;
                    default:
                        break;
                }
                return listOfWhere;
            } // OK
            function debug(message, listOfWhere=["console"]){ // WIP
                if(typeof listOfWhere == "string"){listOfWhere = [listOfWhere];} // transformation en liste de string
                listOfWhere = update_debug_listOfWhere_from_actual_status(listOfWhere);
                for(var i=0;i<listOfWhere.length;i++){
                    switch(listOfWhere[i]){
                        case "console":
                            console.log(message);
                            break;
                        case "error":
                            console.error(message);
                            break;
                        case get_DOMid_debugPage():
                            add_option_in_select(get_DOMid_debugPage(), "", message);
                            break;
                        default:
                            break;
                    }
                }
            } // WIP

            /*
                prompt : demande à l'utilisateur que faire (keep_CVE, keep_link ou keep_both)
                keep_CVE : applique la référence au lien
                keep_link : recupere la reference du texte du lien et modifie le label
                keep_href : recupere la reference du lien et modifie le label
                keep_all : uniformise en suppriment l'element et en creant les nouvelles entrees conservant les CVE
                delete : supprime l'élément
            */
            var fixBadLinks_CVE_default_behavior = "prompt" ;
            var fixBadLinks_CVE_default_proposition = "keep_all" ;
            function promptUserFixCVEs(CVE, linkCVE, hrefCVE, defaultValue=""){
                var message = "Une référence est en erreur, la CVE et son lien ne correspondent pas :"+"\n"+
                    "  - la référence : "+CVE+"\n"+
                    "  - le lien : "+linkCVE+"\n"+
                    "  - le texte du lien : "+hrefCVE+"\n"+
                    "\n"+
                    "LES CHOIX :"+"\n"+
                    "  - keep_CVE : applique la référence au lien"+"\n"+
                    "  - keep_link : recupere la reference du texte du lien et modifie le label"+"\n"+
                    "  - keep_href : recupere la reference du lien et modifie le label"+"\n"+
                    "  - keep_all : uniformise en suppriment l'element et en creant les nouvelles entrees conservant les CVE"+"\n"+
                    "  - delete : supprime l'élément"+"\n"+
                    "\n"+
                    "ANNULER pour arreter le script";
                let answer = prompt(message, defaultValue);
                if(typeof answer== "string"){answer = answer.trim();}
                return answer;
            } // OK

        // Production custom des CVEs : permet de lister les CVE qui apparaitront dans l'article en lieu et place de ceux existant
            var is_replaceCVEs = false;
            var newCVEs = ['CVE-2023-5346'] ; //['CVE-2022-47966','CVE-2023-4516','CVE-2023-123456','CVE-2023-123457']; // pour test : ['CVE-2023-4516','CVE-2023-4516','CVE-2023-123456','CVE-2023-123457']

            function replaceCVEs(list_CVEs){
                var i;
                var list_DOM_CVES = get_list_DOM_CVEs_fromPage();
                for(i =0;i<list_DOM_CVES.length;i++){DOM_remove_object(list_DOM_CVES[i]);}
                var DOM_ul = get_ul_CVE();
                for(i =0;i<list_CVEs.length;i++){make_OG_CVEitem(list_CVEs[i], DOM_ul);}

            }//OK
            function promptNewCves(){
                var message = "Veuillez listez vos CVE (separés par des espaces)";
                let answer = prompt(message, "");
                if(typeof answer== "string"){
                    replaceCVEs(answer.split(" "));
                    prepare_CVElistFromCurrentPage();
                }

            }
            if(is_replaceCVEs){replaceCVEs(newCVEs);}
     GM_registerMenuCommand("set New CVEs ( WIP )", promptNewCves); // WIP

        // RECUPERATION DES CVE
            function fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE, action=""){ // fonction qui agit sur les erreurs de CVE (pas le bon lien avec la reference de la CVE ; devrait être inutilisée
                var DOM_ul, retourFix=true;
                debug("[fix_CVE_issue] CVE='"+CVE+"', linkCVE='"+linkCVE+"' hrefCVE='"+hrefCVE+"' action="+action, ["console", get_DOMid_debugPage()]);
                switch(action){
                    case "prompt":
                        action = promptUserFixCVEs(CVE, linkCVE, hrefCVE, fixBadLinks_CVE_default_proposition);
                        if(action){
                            retourFix = fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE, action);
                        }else{ // l'utilisateur n'a rien saisi ou a fait annuler, on arrete la chaine
                            debug("[fix_CVE_issue] CVE='"+CVE+"', linkCVE='"+linkCVE+"' hrefCVE='"+hrefCVE+"' action=ANNULER", ["console", get_DOMid_debugPage()]);
                            retourFix = false;
                        }
                        break;
                    case "keep_CVE" :
                        DOM_ul = get_ul_CVE();
                        make_OG_CVEitem(CVE, DOM_ul);
                        retourFix = fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE,"delete");
                        break;
                    case "keep_link" :
                        DOM_ul = get_ul_CVE();
                        make_OG_CVEitem(linkCVE, DOM_ul);
                        retourFix = fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE,"delete");
                        break;
                    case "keep_href" :
                        DOM_ul = get_ul_CVE();
                        make_OG_CVEitem(hrefCVE, DOM_ul);
                        retourFix = fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE,"delete");
                        break;
                    case "keep_all" :
                        DOM_ul = get_ul_CVE();
                        make_OG_CVEitem(CVE, DOM_ul);
                        make_OG_CVEitem(linkCVE, DOM_ul);
                        make_OG_CVEitem(hrefCVE, DOM_ul);
                        retourFix = fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE,"delete");
                        break;
                    case "delete" :
                        DOM_remove_object(DOM_itemCVE)
                        break;
                    default :
                        retourFix = fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE, fixBadLinks_CVE_default_behavior);
                        break;
                }
                return retourFix;
            } // OK

            function check_and_correct_list_DOM_itemsCVE(list_DOM_itemsCVE){
                var i ,DOM_itemCVE, CVE, DOM_a_CVE, linkCVE, hrefCVE, retourFix;
                for(i=0;i<list_DOM_itemsCVE.length;i++){
                    DOM_itemCVE = list_DOM_itemsCVE[i];
                    CVE = get_CVE_fromItem(DOM_itemCVE);
                    DOM_a_CVE = get_a_CVE_fromItem(DOM_itemCVE);
                    linkCVE = get_CVE_fromItem(DOM_a_CVE, "innerText", "Pas de texte de lien ?");
                    hrefCVE = get_CVE_fromItem(DOM_a_CVE, "href", "Pas de lien ?");
                    if(CVE != linkCVE || CVE != hrefCVE || hrefCVE != linkCVE){
                        retourFix = fix_CVE_issue(DOM_itemCVE, CVE, linkCVE, hrefCVE);
                        if(! retourFix){break;} // on s'arrete
                    }
                }
            } // OK
            function suppression_doublon_listCVE(list_DOM_itemsCVE, removeDOM_if_double = true){
                var list_DOM_itemsCVE_unique = [];
                var list_CVE = [];
                var DOM_itemCVE, CVE;

                for(var i=0;i<list_DOM_itemsCVE.length;i++){
                    DOM_itemCVE = list_DOM_itemsCVE[i];
                    CVE = get_CVE_fromItem(DOM_itemCVE);
                    if(list_CVE.includes(CVE)){ // doublon
                        debug("la CVE '"+CVE+"' est en doublon", ['error', get_DOMid_debugPage()]);
                        if(removeDOM_if_double){DOM_remove_object(DOM_itemCVE);}
                    }else{
                        list_CVE.push(CVE);
                        list_DOM_itemsCVE_unique.push(DOM_itemCVE);
                    }
                }
                return list_DOM_itemsCVE_unique;
            } // OK

            function clean_page_listCVE(removeDOM_if_double = true){ // nettoie la liste des CVE et supprimes les doublons de la page
                var list_DOM_itemsCVE = get_list_DOM_CVEs_fromPage();
                // VERIFICATION DES ERREURS ENTRE LE LIEN ET L'ID
                    check_and_correct_list_DOM_itemsCVE(list_DOM_itemsCVE);

                list_DOM_itemsCVE = get_list_DOM_CVEs_fromPage();
                // suppression des doublons
                    list_DOM_itemsCVE = suppression_doublon_listCVE(list_DOM_itemsCVE, removeDOM_if_double);

                return list_DOM_itemsCVE;
            } // OK

            function change_page_listCVE(list_DOM_itemsCVE){ // change tous les anciens element CVE par la nouvelle forme
                var DOM_itemCVE;
                for(var i=0;i<list_DOM_itemsCVE.length;i++){
                    DOM_itemCVE = list_DOM_itemsCVE[i];
                    replace_CVEitem(DOM_itemCVE);
                }
            }

            function add_links_cvss_foreach_itemCVE(list_DOM_itemsCVE){
                var DOM_itemCVE, childDiv, child, subChild, subSubChild;
                var CVE;
                for(var i=0;i<list_DOM_itemsCVE.length;i++){
                    DOM_itemCVE = list_DOM_itemsCVE[i];
                    CVE = get_CVE_fromItem(DOM_itemCVE);


                    // création de la div avec les différents ajouts de score
                    childDiv = document.createElement('div');
                    childDiv.id = CVE;
                    childDiv.classList.add("avoid_from_selection");// avoid from selection est utilisé dans get_article_to_string pour cacher les div et ainsi pas les enregistrer dans le fichier texte généré
                    DOM_itemCVE.appendChild(childDiv);

                    childDiv.style.display='inline-flex';
                    childDiv.style.position='relative';
                    childDiv.style.left='36%';

                    // GOOGLE
                    child = document.createElement('a');
                    child.href = "https://www.google.com/search?q=%22"+CVE+"%22";
                    child.target = "_blank";
                    child.innerText = "Google:"+CVE;
                    childDiv.appendChild(child);
                    childDiv.appendChild(document.createTextNode("\u00A0"));

                    // CVEDETAILS
                    child = document.createElement('a');
                    child.href = "https://www.cvedetails.com/cve-details.php?cve_id="+CVE;
                    child.target = "_blank";
                    child.innerText = "??";
                    child.id = CVE+"_cveDetails"
                    child.style.backgroundcolor = "black"
                    childDiv.appendChild(child);
                    childDiv.appendChild(document.createTextNode("\u00A0")); // .createTextNode("&nbsp;"));

                    // NVD
                    child = document.createElement('span');
                    child.id = CVE+"_NVD"
                    child.style.marginleft = "20px"
                    childDiv.appendChild(child);

                    // VULDB
                    child = document.createElement('span');
                    child.id = CVE+"_vuldb"
                    child.style.cursor="pointer";

                    subChild = document.createElement('img');
                    subChild.src = ""
                    subChild.width="22";
                    subChild.height="22";
                    subChild.style.margin="0 0 3px 20px";
                    subChild.setAttribute('_toSearch',CVE);
                    subChild.addEventListener("mousedown", searchVulDB);
                    child.appendChild(subChild);

                    subChild = document.createElement('form');
                    subChild.id = "search_vuldb_"+CVE;
                    subChild.action="https://vuldb.com/?search";
                    subChild.method="post";
                    subChild.target="_blank";

                    subSubChild =document.createElement('input');
                    subSubChild.name="search";
                    subSubChild.type="hidden";
                    subSubChild.value = CVE;

                    subChild.appendChild(subSubChild);
                    child.appendChild(subChild);
                    childDiv.appendChild(child);

                    // FEEDLY
                    child = document.createElement('span');
                    child.id = CVE+"_feedly"
                    child.style.marginleft = "20px"

                    subChild = document.createElement('a');
                    subChild.href = "https://feedly.com/cve/"+CVE;
                    subChild.target = "_blank";

                    subSubChild = document.createElement('img');
                    subSubChild.src = "https://feedly.com/favicon.ico"
                    subSubChild.width="22";
                    subSubChild.height="22";


                    subChild.appendChild(subSubChild);
                    child.appendChild(subChild);
                    childDiv.appendChild(child);

                }
            }

            function get_CVElistFromCurrentPage(nettoyage = true, addHelperLinks = true){ // nettoie eventuellement ; ajoute eventuellement les liens ; ajoute les ID et extensions
                var list_DOM_itemsCVE;
                // NETTOYAGE
                    if(nettoyage){// nettoyage de la liste
                        list_DOM_itemsCVE = clean_page_listCVE()
                    }else{// RECUPERATION BRUTE
                        list_DOM_itemsCVE = get_list_DOM_CVEs_fromPage();
                    }

                // Ajout des modifications UTILES
                    change_page_listCVE(list_DOM_itemsCVE);
                list_DOM_itemsCVE = get_list_DOM_CVEs_fromPage();

                // Ajout des aide au score
                    add_links_cvss_foreach_itemCVE(list_DOM_itemsCVE)

                return list_DOM_itemsCVE
            } // WIP

            function prepare_CVElistFromCurrentPage(){
                var list_DOM_itemsCVE = get_CVElistFromCurrentPage() // récupération des CVE + nettoyage
                var list_CVEs = get_list_string_CVEs_from_list_DOM_CVEs(list_DOM_itemsCVE); // recupération de la liste
                return list_CVEs;
            } // WIP

// envoi rocket chat
    var rocketChatToken={
        "token":"QgjTG8yxBRQJ5GIY4atls7nJtyg8fPv8qODBrtF8HVE",
        "userID" : "iTPeKb8vYEZviMKaq"
    }
    var payload = {
        "msg": "method",
        "method": "sendMessage",
        "id": "423",
        "params": [
            {
                "_id": "8gMsLe9A7pZjo2D2iB",
                "rid": "64a1f373376181965ab77f54",
                "msg": "Hello World!"
            }
        ]
    };
    function sendRocketChat(){

        // console.log(payload)

        // Instantiating easyHTTP
        const http = new easyHTTP;
        var url = "https://rocketchat.internet.np/hooks/65169ce7ea4299ea9bb82153/sAeJzQuvB92bX6TFpuXanQLQCzMPQ3bnDwcdLNBTSavdv5Ji" ; // "https://rocketchat.internet.np/api/v1/chat.postMessage"
        var HEADER = {
            //"X-Auth-Token":rocketChatToken.token,
            //"X-User-Id":rocketChatToken.userID,
            "Content-type":"application/json"
        }
        var DATA = {
            "channel":"#Vulns",
            "text": "This is a test!"
        }
        var functionErrPost=function(err, post){if(err) {console.log("err:"+err);} else {console.log("post:"+post);}}


        //return new Promise((resolve, reject) => {
            GM.xmlHttpRequest({
                method: "POST",
                url: url,
                headers: HEADER,
                data:DATA,
                onload: function(response) {
                    console.log(response);
                    //resolve(response);
                },
                onerror: function(error) {
                    console.log(error.message);
                    //reject("sendRocketChat : ERROR : "+error.message);
                }
            });
        //});
        // requete depuis la page du certfr
        /*http.post(
            url,
            HEADER,
            DATA,
            functionErrPost
        );*/

    }
    GM_registerMenuCommand("to rocketchat", sendRocketChat); // WIP


    // FIN FACTORISATION ##########################################


    //if("DEBUG PARAMETERS"){
        // Contient des fonctions ou parametres utile aux développeurs
    if("Looking for CVE"){
        var WatchForDEBUG_CVE = "" //"CVE-2023-1671";
        var WatchForDEBUG_AVI = "" //"CERTFR-2023-AVI-0283";
        }
    //}
    //if("FONCTIONS"){// < FONCTIONS
        //if("FONCTIONS : GENERIQUES"){

            function intTostringFixedSize(i,size=5){
                return ("00000" + i).slice(-1*size)
            }//intTostringFixedSize

            function get_indent(nb_indent){
                var output="";
                var indent_text="\t";
                nb_indent = (nb_indent<0?0:nb_indent);
                for(var i =0 ; i<nb_indent;i++){
                    output+=indent_text;
                }
                return output;
            }

            // EASY HTTP : https://www.geeksforgeeks.org/how-to-make-put-request-using-xmlhttprequest-by-making-custom-http-library/
            function easyHTTP() {
                // Initializing new XMLHttpRequest method.
                this.http = new XMLHttpRequest();
            }

            // Make an HTTP PUT Request
            easyHTTP.prototype.put = function(url, dict_header, data, callback) {

                // Open an object (POST, PATH, ASYNC-TRUE/FALSE)
                this.http.open('PUT', url, true);

                // Set header
                var header_keys = Object.keys(dict_header)
                for(var i=0;i<header_keys.length;i++){
                    this.http.setRequestHeader(header_keys[i], dict_header[header_keys[i]]);
                }


                // Assigning this to self to have
                // scope of this into the function onload
                let self = this;

                // When response is ready
                this.http.onload = function() {

                    // Callback function (Error, response text)
                    callback(null, self.http.responseText);
                }

                // Since the data is an object so
                // we need to stringify it
                this.http.send(JSON.stringify(data));
            }
            // Make an HTTP POST Request
            easyHTTP.prototype.post = function(url, dict_header, data, callback) {

                // Open an object (POST, PATH, ASYNC-TRUE/FALSE)
                this.http.open('POST', url, true);

                // Set header
                var header_keys = Object.keys(dict_header)
                for(var i=0;i<header_keys.length;i++){
                    this.http.setRequestHeader(header_keys[i], dict_header[header_keys[i]]);
                }


                // Assigning this to self to have
                // scope of this into the function onload
                let self = this;

                // When response is ready
                this.http.onload = function() {

                    // Callback function (Error, response text)
                    callback(null, self.http.responseText);
                }

                // Since the data is an object so
                // we need to stringify it
                this.http.send(JSON.stringify(data));
            }
            // FIN EASY HTTP

            function updateClipboard(newClip) {
                navigator.clipboard.writeText(newClip).then(function() {
                    /* le presse-papier est correctement paramétré */
                    console.log ("clipboard ok");
                    add_log_to_CMB("[updateClipboard] ok", 0);
                }, function() {
                    /* l'écriture dans le presse-papier a échoué */
                    console.log ("clipboard ok");
                    add_log_to_CMB("[updateClipboard] ok", 0);
                });
            }

            function dateUStoFR(USdateSTR){
                return ((new Date(USdateSTR)).toLocaleDateString("fr"));
            }

        //} // FONCTIONS : GENERIQUES
        //if("FONCTIONS : STOCKAGE"){
            function resetGMstorage(){//ok
                var listKey = GM_listValues();
                for(var i=0;i<listKey.length;i++){
                    GM_deleteValue(listKey[i]);
                }
            } // resetGMstorage
            function replaceGMstorage(content){//ok
                //console.log("replaceGMstorage : taille content ="+content.length);
                //console.log(content);

                var regexKeyVal = new RegExp('(?<=")(.*?)(": ")(.*?)(?=")',"g");
                var regexKey = new RegExp('(?<=")(.*?)(?=(": "))',"g");
                var regexVal = new RegExp('(?<=(": "))(.*?)(?=")',"g");
                var listKey = content.match(regexKey);
                var listval = content.match(regexVal);
                //console.log("replaceGMstorage : listKey");
                //console.log(listKey);
                //console.log("replaceGMstorage : listval");
                //console.log(listval);
                if(listKey.length == listval.length && listval.length>0 ){
                    resetGMstorage();
                    for(var i=0;i<listKey.length;i++){
                        GM_setValue(listKey[i],listval[i]);
                    }
                }
            }//replaceGMstorage
            function extractGMstorage(){//ok
                var listKey = GM_listValues();
                var content = "";
                for(var i=0;i<listKey.length;i++){
                    //console.log("extractGMstorage ; i="+i+" ; key="+listKey[i]);
                    content = content +'\n"'+listKey[i]+'": "'+GM_getValue(listKey[i])+'",';
                }
                content ="{" + content.substring(0,content.length -1) + "\n}";
                return content
            }//extractGMstorage

            function getHigherBulletin_from_GMstorage(typeBulletin="AVI"){//ok
                var listKey = GM_listValues();
                var regexYear = new RegExp("(?<=(CERTFR-))([0-9]+)(?=-"+typeBulletin+")","i");
                var regexID = new RegExp("(?<=(-["+typeBulletin+"]+-))([0-9]+)","i");// avant, comprends les ALE : new RegExp("(?<=(-[a-zA-Z]+-))([0-9]+)","i");
                var listIdMacth,listYearMacth;
                var compareValYear=0;
                var compareValID=0;
                var content, key;
                for(var i=0;i<listKey.length;i++){
                    key=listKey[i];
                    listYearMacth = key.match(regexYear);
                    listIdMacth = key.match(regexID);
                    if(listYearMacth && listIdMacth){
                        if(listYearMacth.length>0 && listIdMacth.length>0){
                            //console.log("compare : "+compareValYear+"<"+parseInt(listYearMacth[0]) +"&&"+ compareValID+"<"+parseInt(listIdMacth[0]))
                            if( (compareValYear<=parseInt(listYearMacth[0])) && (compareValID<=parseInt(listIdMacth[0]))){
                                compareValYear = parseInt(listYearMacth[0]);
                                compareValID = parseInt(listIdMacth[0]);
                                content=compareValYear+"-"+typeBulletin+"-"+intTostringFixedSize(compareValID,5);
                                //console.log(content);
                            }
                        }
                    }
                }
                return content;
            }//getHigherBulletin_from_GMstorage

            function getVecteurV2fromStorageAVI(avi){
                var vecteur="";
                if(GM_getValue(avi+".vecteurV2")) {vecteur=GM_getValue(avi+".vecteurV2");}
                return vecteur ;
            }
            function getVecteurV3fromStorageAVI(avi){
                var vecteur="";
                if(GM_getValue(avi+".vecteurV3")) {vecteur=GM_getValue(avi+".vecteurV3");}
                return vecteur ;
            }
            function getSourceV2fromStorageAVI(avi){
                var vecteur="";
                if(GM_getValue(avi+".vecteurV2Source")) {vecteur=GM_getValue(avi+".vecteurV2Source");}
                return vecteur ;
            }
            function getSourceV3fromStorageAVI(avi){
                var vecteur="";
                if(GM_getValue(avi+".vecteurV3Source")) {vecteur=GM_getValue(avi+".vecteurV3Source");}
                return vecteur ;
            }
            function getCVSS_fromStorage(cveID, field=""){
                var avi = "";
                var currentKey="";
                var currentValue="";
                if(cveID==WatchForDEBUG_CVE) { console.log("AVI : ###########################");}
                var allValues=GM_listValues();
                if(cveID==WatchForDEBUG_CVE) { console.log(allValues);}
                for(var i=(allValues.length)-1;i>=0;i--){
                    currentKey=allValues[i];
                    if(currentKey.includes(field) && GM_getValue(currentKey)) {
                        currentValue = GM_getValue(currentKey);
                        if(cveID==WatchForDEBUG_CVE && (currentKey.includes(WatchForDEBUG_AVI) || WatchForDEBUG_AVI =="")) { console.log(currentKey+" : "+GM_getValue(currentKey)+" vs "+cveID);}
                        if(currentValue == cveID){
                            avi=currentKey.split(".")[0];
                            if(cveID==WatchForDEBUG_CVE) { console.log("currentValue="+currentValue+" => currentKey="+currentKey);}
                            break;
                        }
                    }
                }
                //console.log("trouvé : "+avi);
                return (avi);

            }


            // FONCTION GIT AS STORAGE
            async function updateStorage(){
                var gitTitle = await getGitStorageTitle();
                var localTitle = "";
                var updateFromRemote = false;
                if(!gitTitle){
                    console.log("updateStorage ; ERREUR : Fichier introuvable");
                    add_log_to_CMB("[updateStorage] ERREUR : Fichier introuvable", 0);
                    return false;
                }
                if(GM_getValue("PASTEBIN_PasteName_regex") && GM_getValue("PASTEBIN_PasteName_partALE") && GM_getValue("PASTEBIN_PasteName_partAVI")) { // récupération du titre local
                    // comparaison avec le stockage
                    if(GM_getValue("PASTEBIN_PasteName")) {
                        if(gitTitle == GM_getValue("PASTEBIN_PasteName")){
                            // dejà à jour, on ne fait rien
                            console.log("updateStorage : Le fichier de stockage est déja à jour");
                            add_log_to_CMB("[updateStorage] Le fichier de stockage est déja à jour", 0);
                        }else{// comparaison du plus recent
                            //var pasteName_TO_regex = new RegExp(GM_getValue("PASTEBIN_PasteName_TO_regex"), 'i');
                            var pasteName_partALE = new RegExp(GM_getValue("PASTEBIN_PasteName_partALE"), 'i');
                            var pasteName_partAVI = new RegExp(GM_getValue("PASTEBIN_PasteName_partAVI"), 'i');
                            //var matches_Paste = gitTitle.match(pasteName_TO_regex);
                            var matches_PasteALE = gitTitle.match(pasteName_partALE);
                            var matches_PasteAVI = gitTitle.match(pasteName_partAVI);
                            //var matches_Local = (GM_getValue("PASTEBIN_PasteName")).match(pasteName_TO_regex);
                            var matches_Local_PasteALE = (GM_getValue("PASTEBIN_PasteName")).match(pasteName_partALE);
                            var matches_Local_PasteAVI = (GM_getValue("PASTEBIN_PasteName")).match(pasteName_partAVI);
                            if(matches_PasteALE&&matches_PasteAVI&&matches_Local_PasteALE&&matches_Local_PasteAVI){ //2023-0069
                                var compareALE_gt_stockage = compareTitles(matches_PasteALE,matches_Local_PasteALE);
                                //var compareALE_eq_stockage = compareTitles(matches_PasteALE,matches_Local_PasteALE,true) && !compareALE_gt_stockage;
                                var compareAVI_gt_stockage = compareTitles(matches_PasteAVI,matches_Local_PasteAVI);
                                //var compareAVI_eq_stockage = compareTitles(matches_PasteAVI,matches_Local_PasteAVI,true) && !compareAVI_gt_stockage;
                                var paste_gt_stockage = (compareALE_gt_stockage || compareAVI_gt_stockage)
                                if(paste_gt_stockage){ // le fichier distant est plus à jour
                                    //if(gitTitle>matches_Local[0]){ // le fichier distant est plus à jour
                                    //console.log("updateStorage #DEBUG : "+matches_Paste[0]+">"+matches_Local[0]);
                                    updateFromRemote = true;
                                }else{
                                    console.log("updateStorage : Le fichier de stockage est plus avancé que le fichier pastebin");
                                    add_log_to_CMB("[updateStorage] Le fichier de stockage est plus avancé que le fichier pastebin", 0);
                                }
                            }else{
                                console.log("updateStorage ; ERREUR : match gitTitle ou pastname local");
                                add_log_to_CMB("[updateStorage] ERREUR : match gitTitle ou pastname local", 0);
                            }
                        }
                    }else{
                        console.log("updateStorage ; ERREUR : clé PASTEBIN_PasteName");
                        add_log_to_CMB("[updateStorage] ERREUR : clé PASTEBIN_PasteName", 0);
                        return false;
                    }
                }else{ // les valeurs n'existent pas dans le stockage actuel, on propose la mise à jour à l'utilisateur
                    updateFromRemote = true;
                }
                if(updateFromRemote){
                    console.log("updateStorage : Mise à jour du fichier de stockage ("+gitTitle+")");
                    add_log_to_CMB("[updateStorage] Mise à jour du fichier de stockage("+gitTitle+")", 0);
                    var content = await getGitStorageContent();
                    replaceGMstorage(content);

                }
            }

        //} // FONCTIONS : STOCKAGE
        //if("FONCTIONS : STOCKAGE ONLINE"){

            // FONCTION FICHIERS ONLINE
            function responseTextToXML(responseText){
                var parser = new DOMParser();
                var xmlDoc = parser.parseFromString(responseText,"text/xml");
                try{
                    if((xmlDoc.children[0]).localName == "parsererror"){
                        //console.log("responseTextToXML : ajout body");
                        xmlDoc = parser.parseFromString("<body>"+responseText+"</body>","text/xml");
                    }
                }catch(e){
                    console.log("responseTextToXML : erreur ajout body");
                    add_log_to_CMB("[responseTextToXML] erreur ajout body", 0);
                    console.log(e);
                    add_log_to_CMB("[responseTextToXML]"+e, 0);
                    return null;
                }
                return xmlDoc;
                //console.log("responseTextToXML : responseText");
                //console.log(responseText);
                //console.log("responseTextToXML : xmlDoc");
                //console.log(xmlDoc);
            }

            function pasteBin_API(dict_data={},methode="POST"){
                var outputVal;
                // Parametres secondaire
                var api_paste_private = '1'; // 0=public 1=unlisted 2=private
                // parametres elementaires
                if(GM_getValue("PASTEBIN_api_dev_key")) { // your api_developer_key
                    dict_data["api_dev_key"] = GM_getValue("PASTEBIN_api_dev_key");
                }else{
                    console.log("pasteBin_API ; ERREUR : clé PASTEBIN_api_dev_key");
                    add_log_to_CMB("[pasteBin_API] ERREUR : clé PASTEBIN_api_dev_key", 0);
                    return false;
                }
                if(GM_getValue("PASTEBIN_api_user_key")) { // if an invalid or expired api_user_key is used, an error will spawn. If no api_user_key is used, a guest paste will be created
                    dict_data["api_user_key"] = GM_getValue("PASTEBIN_api_user_key");
                }else{
                    console.log("pasteBin_API ; ERREUR : clé PASTEBIN_api_user_key");
                    add_log_to_CMB("[pasteBin_API] ERREUR : clé PASTEBIN_api_user_key", 0);
                    return false;
                }
                if(GM_getValue("PASTEBIN_url")) {
                    var url = GM_getValue("PASTEBIN_url");
                }else{
                    console.log("pasteBin_API ; ERREUR : clé PASTEBIN_url");
                    add_log_to_CMB("[pasteBin_API] ERREUR : clé PASTEBIN_url", 0);
                    return false;
                }

                var data = "";

                for (const [key, value] of Object.entries(dict_data)) {
                    data = data+String(key)+'='+String(value)+'&';
                }

                //console.log("pasteBin_API : data");
                //console.log(data);

                return new Promise((resolve, reject) => {
                    GM.xmlHttpRequest({
                        method: methode,
                        url: url,
                        data: data,
                        headers: {
                            "Content-Type": "application/x-www-form-urlencoded"
                        },
                        onload: function(response) {
                            resolve(response);
                        },
                        onerror: function(error) {
                            reject(error.message);
                        }
                    });
                });
            }

            async function pasteBin_list(){
                var dict_data={api_option:"list"};
                var response = await pasteBin_API(dict_data);
                var status = response.status
                var responseText = response.responseText;

                //console.log("pasteBin_API : response");
                //console.log(response);
                //console.log("pasteBin_API : status");
                //console.log(status);
                //console.log("pasteBin_list : responseText");
                //console.log(responseText);

                var responseXML = responseTextToXML(responseText);
                return responseXML;
            }

            function get_XML_item_fromMatch_recursive(XML, regex, output="node",testOnKey="", parentNode=null){
                // output:
                //   node : retourne le noeud courant
                //   parentNode : retourne le noeud parent
                //   key : le contenu du match
                //   match : retourne true false si il y a match
                //
                // testOnKey : la clé pour tester la regex
                var reponse = null;
                if(XML.children.length>0){
                    parentNode = XML;
                    for(var i=0;i<XML.children.length;i++){
                        reponse = get_XML_item_fromMatch_recursive(XML.children[i], regex, output, testOnKey, parentNode);
                        if(reponse){break;}
                    }
                }else{
                    //console.log("get_XML_item_fromMatch_recursive : test "+XML.localName+" avec la clé "+testOnKey);
                    //console.log(XML);
                    try{
                        var toTest=XML[testOnKey]; //(regexTestContent? XML.textContent : XML.localName )
                        var matches = toTest.match(regex);
                        //console.log(matches);
                        if(matches){
                            switch(output){
                                case "node":
                                    reponse= XML;
                                    break;
                                case "parentNode":
                                    reponse= parentNode;
                                    break;
                                case "key":
                                    reponse=matches[0];
                                    break;
                                default: // match
                                    reponse = true;
                                    break;
                            }
                        }
                    }catch(e){
                    }
                }
                return reponse;
            }
            function compareTitles(paste,local,ouEgal=false){
                if(paste && local){
                    paste = (paste.length>0?paste[0]:"");
                    local = (local.length>0?local[0]:"");
                }else{paste="";local="";}
                var year = new RegExp("[0-9]+(?=(\-[A-Z]))", 'i');
                var id = new RegExp("(?<=([A-Z]\-))[0-9]+", 'i');

                var year_Paste = parseInt((paste.match(year))[0]);
                var year_local = parseInt((local.match(year))[0]);
                var id_Paste = parseInt((paste.match(id))[0]);
                var id_local = parseInt((local.match(id))[0]);
                var yearPaste_ge_yearLocal = year_Paste>=year_local;
                var retour_ge = yearPaste_ge_yearLocal && id_Paste>=id_local;
                var retour_gt = yearPaste_ge_yearLocal && id_Paste>id_local;

                return (ouEgal && retour_ge) || (!ouEgal && retour_gt);

            }

            async function pasteBin_getStoragePasteTitle(output="title"){ // output : pasteTitle, api_paste_key, succes?
                var paste=null;
                var pasteTitle,api_paste_key;
                var result = await pasteBin_list();
                //console.log("pasteBin_getStoragePasteTitle : result");
                //console.log(result);

                if(GM_getValue("PASTEBIN_PasteName_regex") && GM_getValue("PASTEBIN_PasteName_partALE") && GM_getValue("PASTEBIN_PasteName_partAVI")) {
                    // recherche du titre du paste courant et de sa clé de paste
                    var pasteName_regex = new RegExp(GM_getValue("PASTEBIN_PasteName_regex"), 'i');
                    paste = get_XML_item_fromMatch_recursive(result, pasteName_regex, "parentNode", "textContent");
                    //console.log("pasteBin_getStoragePasteTitle : paste ##################");
                    //console.log(paste);
                    if(!paste){
                        console.log("pasteBin_getStoragePasteTitle ; ERREUR : paste : pas de paste de stockage");
                        add_log_to_CMB("[pasteBin_getStoragePasteTitle] ERREUR : paste : pas de paste de stockage", 0);
                        return false;
                    }

                    pasteTitle = get_XML_item_fromMatch_recursive(paste, pasteName_regex, "key", "textContent");
                    //console.log("pasteBin_getStoragePasteTitle : pasteTitle ##################");
                    //console.log(pasteTitle);
                    update_paste_table("pasteBin_getStoragePasteTitle",pasteTitle,"",pasteTitle);
                    if(!pasteTitle){
                        update_paste_table("pasteBin_getStoragePasteTitle",pasteTitle,"",pasteTitle);
                        console.log("pasteBin_getStoragePasteTitle ; ERREUR : pasteTitle");
                        add_log_to_CMB("[pasteBin_getStoragePasteTitle] ERREUR : pasteTitle", 0);
                        return false;
                    }
                    var paste_key = get_XML_item_fromMatch_recursive(paste, new RegExp("paste_key","i"), "node", "localName");
                    if(paste_key){
                        api_paste_key = paste_key.textContent;
                        update_paste_table("pasteBin_getStoragePasteTitle",pasteTitle,api_paste_key,pasteTitle);
                    }else{
                        update_paste_table("pasteBin_getStoragePasteTitle",pasteTitle,"",pasteTitle);
                        console.log("pasteBin_getStoragePasteTitle ; ERREUR : paste_key ##################");
                        add_log_to_CMB("[pasteBin_getStoragePasteTitle] ERREUR : paste_key ##################", 0);
                        return false;
                    }
                    // comparaison avec le stockage
                    if(GM_getValue("PASTEBIN_PasteName")) {
                        if(pasteTitle == GM_getValue("PASTEBIN_PasteName")){
                            // dejà à jour, on ne fait rien
                            console.log("pasteBin_getStoragePasteTitle : Le fichier de stockage est déja à jour");
                            add_log_to_CMB("[pasteBin_getStoragePasteTitle] Le fichier de stockage est déja à jour", 0);
                        }else{// comparaison du plus recent
                            //var pasteName_TO_regex = new RegExp(GM_getValue("PASTEBIN_PasteName_TO_regex"), 'i');
                            var pasteName_partALE = new RegExp(GM_getValue("PASTEBIN_PasteName_partALE"), 'i');
                            var pasteName_partAVI = new RegExp(GM_getValue("PASTEBIN_PasteName_partAVI"), 'i');
                            //var matches_Paste = pasteTitle.match(pasteName_TO_regex);
                            var matches_PasteALE = pasteTitle.match(pasteName_partALE);
                            var matches_PasteAVI = pasteTitle.match(pasteName_partAVI);
                            //var matches_Local = (GM_getValue("PASTEBIN_PasteName")).match(pasteName_TO_regex);
                            var matches_Local_PasteALE = (GM_getValue("PASTEBIN_PasteName")).match(pasteName_partALE);
                            var matches_Local_PasteAVI = (GM_getValue("PASTEBIN_PasteName")).match(pasteName_partAVI);
                            if(matches_PasteALE&&matches_PasteAVI&&matches_Local_PasteALE&&matches_Local_PasteAVI){ //2023-0069
                                var compareALE_gt_stockage = compareTitles(matches_PasteALE,matches_Local_PasteALE);
                                //var compareALE_eq_stockage = compareTitles(matches_PasteALE,matches_Local_PasteALE,true) && !compareALE_gt_stockage;
                                var compareAVI_gt_stockage = compareTitles(matches_PasteAVI,matches_Local_PasteAVI);
                                //var compareAVI_eq_stockage = compareTitles(matches_PasteAVI,matches_Local_PasteAVI,true) && !compareAVI_gt_stockage;
                                var paste_gt_stockage = (compareALE_gt_stockage || compareAVI_gt_stockage)
                                if(paste_gt_stockage){ // le fichier distant est plus à jour
                                    //if(pasteTitle>matches_Local[0]){ // le fichier distant est plus à jour
                                    //console.log("pasteBin_getStoragePasteTitle #DEBUG : "+matches_Paste[0]+">"+matches_Local[0]);
                                    console.log("pasteBin_getStoragePasteTitle : Mise à jour du fichier de stockage ("+pasteTitle+")");
                                    add_log_to_CMB("[pasteBin_getStoragePasteTitle] Mise à jour du fichier de stockage("+pasteTitle+")", 0);
                                    var content = await pasteBin_getContent(api_paste_key,pasteTitle);
                                    replaceGMstorage(content);
                                }else{
                                    console.log("pasteBin_getStoragePasteTitle : Le fichier de stockage est plus avancé que le fichier pastebin");
                                    add_log_to_CMB("[pasteBin_getStoragePasteTitle] Le fichier de stockage est plus avancé que le fichier pastebin", 0);
                                }
                            }else{
                                console.log("pasteBin_getStoragePasteTitle ; ERREUR : match pasteTitle ou pastname local");
                                add_log_to_CMB("[pasteBin_getStoragePasteTitle] ERREUR : match pasteTitle ou pastname local", 0);
                            }
                        }
                    }else{
                        console.log("pasteBin_getStoragePasteTitle ; ERREUR : clé PASTEBIN_PasteName");
                        add_log_to_CMB("[pasteBin_getStoragePasteTitle] ERREUR : clé PASTEBIN_PasteName", 0);
                        return false;
                    }
                }else{
                    console.log("pasteBin_getStoragePasteTitle ; ERREUR : clé PASTEBIN_PasteName_regex ou PASTEBIN_PasteName_partALE ou PASTEBIN_PasteName_partAVI");
                    add_log_to_CMB("[pasteBin_getStoragePasteTitle] ERREUR : clé PASTEBIN_PasteName_regex ou PASTEBIN_PasteName_partALE ou PASTEBIN_PasteName_partAVI", 0);
                    return false;
                }
                switch (output){
                    case "title":
                        return pasteTitle;
                        break;
                    case "api_paste_key":
                        return api_paste_key;
                        break;
                    default:
                        return true;
                }
            }

            async function pasteBin_getContent(api_paste_key,pasteTitle=""){
                var dict_data={api_option:"show_paste", api_paste_key:api_paste_key};
                var response = await pasteBin_API(dict_data);
                var status = response.status
                var responseText = response.responseText;

                update_paste_table("pasteBin_getContent",status,api_paste_key,(pasteTitle?pasteTitle:""));
                //console.log("pasteBin_API : response");
                //console.log(response);
                //console.log("pasteBin_API : status");
                //console.log(status);
                //console.log("pasteBin_getContent : responseText");
                //console.log(responseText);
                return responseText;
            }

            async function pasteBin_delete(api_paste_key){
                if(!api_paste_key){
                    console.log("pasteBin_delete ; pas d'api_paste_key a utiliser, pas de suppression ; cle : "+api_paste_key);
                    add_log_to_CMB("[pasteBin_delete] pas d'api_paste_key a utiliser, pas de suppression ; cle : "+api_paste_key, 0);
                    return null;
                }
                var dict_data={api_option:"delete", api_paste_key:api_paste_key};
                var response = await pasteBin_API(dict_data);
                var status = response.status
                var responseText = response.responseText;

                update_paste_table("pasteBin_delete",status,api_paste_key);
                //console.log("pasteBin_delete : response");
                //console.log(response);
                //console.log("pasteBin_delete : status");
                //console.log(status);
                //console.log("pasteBin_delete : responseText");
                //console.log(responseText);
                return responseText;
            }

            async function pasteBin_add(title, content,output="responseText"){
                var dict_data={
                    api_option:"paste",
                    api_paste_name:title,
                    api_paste_code:encodeURIComponent(content),
                    api_paste_private:2 // public = 0, unlisted = 1, private = 2
                };
                var response = await pasteBin_API(dict_data);
                var status = response.status
                var responseText = response.responseText;

                //console.log("pasteBin_add : response");
                //console.log(response);
                //console.log("pasteBin_add : status");
                //console.log(status);
                //console.log("pasteBin_add : responseText");
                //console.log(responseText);

                update_paste_table("pasteBin_add",status,responseText,title);
                if(output=="status"){
                    return status;
                }else{
                    return responseText;
                }
            }

            function makeNewTitle(){ // ne prends pas en compte l'ancien titre pour le remplacer par le nouveau
                // var newTitle = "du 2022-556 au "+getHigherBulletin_from_GMstorage("AVI");
                var newTitle = getHigherBulletin_from_GMstorage("ALE")+"_"+getHigherBulletin_from_GMstorage("AVI");
                console.log("makeNewTitle : newtitle : "+newTitle);
                add_log_to_CMB("[makeNewTitle] newtitle : "+newTitle, 0);
                GM_setValue("PASTEBIN_PasteName",newTitle);
                return newTitle;
            }

            // FIN FONCTION FICHIERS ONLINE
            function allowToStore(){ // fonction qui autorise ou non à mettre à jour le stockage distant
                return getGUIinputValue("checkbox_stocker", true);
            }
            function getGitStorageAPIToken(){
                var apiKey = GM_getValue("GITLAB_token");
                if(apiKey){
                    console.log("all good : "+apiKey);
                }else{
                    apiKey = prompt("Veuillez saisir le token qui peut lire et ecrire dans le repo", "allez sur : https://gitlab-ce.internet.np/vuln/cert-fr/-/blob/main/README.md");
                    GM_setValue("GITLAB_token", apiKey);
                }
                return apiKey;
            }
            async function setGitStorageContent(newTitle, content, functionErrPost=function(err, post){if(err) {console.log("err:"+err);} else {console.log("post:"+post);}} , serverURL="https://gitlab-ce.internet.np", project="vuln/cert-fr", file="stockage.txt"){
                var acces_token = getGitStorageAPIToken();
                var project_id = await getGitStorageProjectId(serverURL, project);
                var url = serverURL+"/api/v4/projects/"+project_id+"/repository/files/"+file;

                // Instantiating easyHTTP
                const http = new easyHTTP;

                // Data that we need to update
                var DATA={
                    branch: "main",
                    commit_message: newTitle,
                    content : content
                };
                var HEADER={
                    "Content-Type": "application/json",
                    'PRIVATE-TOKEN': acces_token
                };

                // Put prototype method(url, data,
                // response text)
                var actualTitle = await getGitStorageTitle(serverURL, project, file);
                console.log("actualTitle : "+actualTitle);
                if(actualTitle){ // mise à jour
                    http.put(
                        url,
                        HEADER,
                        DATA,
                        functionErrPost
                    );
                }else{ // création
                    http.post(
                        url,
                        HEADER,
                        DATA,
                        functionErrPost
                    );
                }
            }

            function getGitStorageTitle(serverURL="https://gitlab-ce.internet.np", project="vuln/cert-fr", file="stockage.txt"){// last commit
                var url = serverURL+"/"+project+"/-/blob/main/"+file;
                var title = "";
                return new Promise((resolve, reject) => {
                    GM.xmlHttpRequest({
                        method: "GET",
                        url: url,
                        responseType: "document",
                        onload: function(response) {
                            // recherche du titre
                            var dom = response.response.body
                            try{
                                title = dom.getElementsByClassName("item-title")[0].innerText; // last commit
                            }catch(error){}
                            resolve(title);
                        },
                        onerror: function(error) {
                            reject("getGitStorageTitle : ERROR : "+error.message);
                        }
                    });
                });
            }

            function getGitStorageContent(serverURL="https://gitlab-ce.internet.np", project="vuln/cert-fr", file="stockage.txt"){// last commit
                var url = serverURL+"/"+project+"/-/raw/main/"+file;
                // a completer ou utiliser l'API
                return new Promise((resolve, reject) => {
                    GM.xmlHttpRequest({
                        method: "GET",
                        url: url,
                        onload: function(response) {
                            // récupération du contenu
                            var content = response.responseText
                            resolve(content);
                        },
                        onerror: function(error) {
                            reject("getGitStorageContent : ERROR : "+error.message);
                        }
                    });
                });
            }

            function get_projetcID_git_fromID(domObj, id="project_id"){ // le DOM doit correspondre a la page principale du projet
                var project_id = ""
                try{
                    project_id = domObj.querySelector("#"+id).value
                }catch(error){
                    console.error("[get_projetcID_git_fromID] ERROR : ", error.message,"pour",domObj, "#"+id)
                }
                return project_id
            }
            function get_projetcID_git_bodyParameters(domObj, parameter="data-project-id"){ // le DOM doit correspondre a la page principale du projet
                var project_id = ""
                try{
                    project_id = domObj[parameter]
                }catch(error){
                    console.error("[get_projetcID_git_bodyParameters] ERROR : ", error.message)
                }
                return project_id
            }
            function get_projetcID_git_bodyAttributes(domObj, attribute="data-project-id"){ // le DOM doit correspondre a la page principale du projet
                var project_id = ""
                try{
                    project_id = domObj.getAttribute(attribute);
                }catch(error){
                    console.error("[get_projetcID_git_bodyAttributes] ERROR : ", error.message)
                }
                return project_id
            }

            function get_projetcID_git(domObj){ // le DOM doit correspondre a la page principale du projet
                console.log("[get_projetcID_git] récupération du project ID");
                var lsr_function_to_try=[get_projetcID_git_fromID, get_projetcID_git_bodyParameters, get_projetcID_git_bodyAttributes];
                var project_id = "";
                var fctToCall;
                for(var i=0;i<lsr_function_to_try.length;i++){
                    fctToCall = lsr_function_to_try[i];
                    console.log("[get_projetcID_git] essai sur :", fctToCall);
                    project_id = fctToCall(domObj);
                    if(project_id!="" && project_id!=undefined){
                        break;
                    }
                }
                console.log("[get_projetcID_git] project ID=",project_id,".");
                return project_id;
            }

            function getGitStorageProjectId(serverURL="https://gitlab-ce.internet.np", project="vuln/cert-fr"){// last commit
                var url = serverURL+"/"+project+"/-/tree/main/";
                console.log("getGitStorageProjectId",url)
                // a completer ou utiliser l'API
                return new Promise((resolve, reject) => {
                    GM.xmlHttpRequest({
                        method: "GET",
                        url: url,
                        responseType: "document",
                        onload: function(response) {
                            // recherche du project id
                            var dom = response.response.body
                            var project_id = get_projetcID_git(dom);
                            resolve(project_id);
                        },
                        onerror: function(error) {
                            reject("getGitStorageProjectId : ERROR : "+error.message);
                        }
                    });
                });
            }


            // FIN FONCTION GIT AS STORAGE

        //}
        //if("FONCTIONS : CVSS"){
            // < FONCTIONS\CVSS\officielles
            var vectorKeysOrdered_v2 = ['AV','AC','Au','C','I','A','E','RL','RC'];
            var vectorKeysOrdered_v3 = ['AV','AC','PR','UI','S','C','I','A','E','RL','RC'];

            var CVSS31 = {};
            garnir_CVSS31();
            function garnir_CVSS31() {

                CVSS31.CVSSVersionIdentifier = "CVSS:3.1";
                CVSS31.exploitabilityCoefficient = 8.22;
                CVSS31.scopeCoefficient = 1.08;

                // A regular expression to validate that a CVSS 3.1 vector string is well formed. It checks metrics and metric
                // values. It does not check that a metric is specified more than once and it does not check that all base
                // metrics are present. These checks need to be performed separately.

                CVSS31.vectorStringRegex_31 = /^CVSS:3\.1\/((AV:[NALP]|AC:[LH]|PR:[UNLH]|UI:[NR]|S:[UC]|[CIA]:[NLH]|E:[XUPFH]|RL:[XOTWU]|RC:[XURC]|[CIA]R:[XLMH]|MAV:[XNALP]|MAC:[XLH]|MPR:[XUNLH]|MUI:[XNR]|MS:[XUC]|M[CIA]:[XNLH])\/)*(AV:[NALP]|AC:[LH]|PR:[UNLH]|UI:[NR]|S:[UC]|[CIA]:[NLH]|E:[XUPFH]|RL:[XOTWU]|RC:[XURC]|[CIA]R:[XLMH]|MAV:[XNALP]|MAC:[XLH]|MPR:[XUNLH]|MUI:[XNR]|MS:[XUC]|M[CIA]:[XNLH])$/;


                // Associative arrays mapping each metric value to the constant defined in the CVSS scoring formula in the CVSS v3.1
                // specification.

                CVSS31.Weight = {
                    AV:   { N: 0.85,  A: 0.62,  L: 0.55,  P: 0.2},
                    AC:   { H: 0.44,  L: 0.77},
                    PR:   { U:       {N: 0.85,  L: 0.62,  H: 0.27},         // These values are used if Scope is Unchanged
                           C:       {N: 0.85,  L: 0.68,  H: 0.5}},         // These values are used if Scope is Changed
                    UI:   { N: 0.85,  R: 0.62},
                    S:    { U: 6.42,  C: 7.52},                             // Note: not defined as constants in specification
                    CIA:  { N: 0,     L: 0.22,  H: 0.56},                   // C, I and A have the same weights

                    E:    { X: 1,     U: 0.91,  P: 0.94,  F: 0.97,  H: 1},
                    RL:   { X: 1,     O: 0.95,  T: 0.96,  W: 0.97,  U: 1},
                    RC:   { X: 1,     U: 0.92,  R: 0.96,  C: 1},

                    CIAR: { X: 1,     L: 0.5,   M: 1,     H: 1.5}           // CR, IR and AR have the same weights
                };


                // Severity rating bands, as defined in the CVSS v3.1 specification.

                CVSS31.severityRatings  = [ { name: "None",     bottom: 0.0, top:  0.0},
                                           { name: "Low",      bottom: 0.1, top:  3.9},
                                           { name: "Medium",   bottom: 4.0, top:  6.9},
                                           { name: "High",     bottom: 7.0, top:  8.9},
                                           { name: "Critical", bottom: 9.0, top: 10.0} ];

                /* ** CVSS31.roundUp1 **
						 *
						 * Rounds up its parameter to 1 decimal place and returns the result.
						 *
						 * Standard JavaScript errors thrown when arithmetic operations are performed on non-numbers will be returned if the
						 * given input is not a number.
						 *
						 * Implementation note: Tiny representation errors in floating point numbers makes rounding complex. For example,
						 * consider calculating Math.ceil((1-0.58)*100) by hand. It can be simplified to Math.ceil(0.42*100), then
						 * Math.ceil(42), and finally 42. Most JavaScript implementations give 43. The problem is that, on many systems,
						 * 1-0.58 = 0.42000000000000004, and the tiny error is enough to push ceil up to the next integer. The implementation
						 * below avoids such problems by performing the rounding using integers. The input is first multiplied by 100,000
						 * and rounded to the nearest integer to consider 6 decimal places of accuracy, so 0.000001 results in 0.0, but
						 * 0.000009 results in 0.1.
						 *
						 * A more elegant solution may be possible, but the following gives answers consistent with results from an arbitrary
						 * precision library.
						 */
                CVSS31.roundUp1 = function Roundup (input) {
                    var int_input = Math.round(input * 100000);

                    if (int_input % 10000 === 0) {
                        return int_input / 100000;
                    } else {
                        return (Math.floor(int_input / 10000) + 1) / 10;
                    }
                };


                /* ** CVSS31.severityRating **
						 *
						 * Given a CVSS score, returns the name of the severity rating as defined in the CVSS standard.
						 * The input needs to be a number between 0.0 to 10.0, to one decimal place of precision.
						 *
						 * The following error values may be returned instead of a severity rating name:
						 *   NaN (JavaScript "Not a Number") - if the input is not a number.
						 *   undefined - if the input is a number that is not within the range of any defined severity rating.
						 */
                CVSS31.severityRating = function (score) {
                    var severityRatingLength = CVSS31.severityRatings.length;

                    var validatedScore = Number(score);

                    if (isNaN(validatedScore)) {
                        return validatedScore;
                    }

                    for (var i = 0; i < severityRatingLength; i++) {
                        if (score >= CVSS31.severityRatings[i].bottom && score <= CVSS31.severityRatings[i].top) {
                            return CVSS31.severityRatings[i].name;
                        }
                    }

                    return undefined;
                };

            } //garnir_CVSS31

            // FONCTIONS\CVSS\officielles />

            // < FONCTIONS\CVSS\internes
            let big_data_select_v2_v3=[
                {select_id: "AccessVectorVar", short: "AV", mandatory: true, correspondances: {"L":"L", "A":"A", "N":"N", "P":"P"}},
                {select_id: "AccessComplexityVar", short: "AC", mandatory: true, correspondances: {"H":"H", "M":"M", "L":"L"}},
                {select_id: "AuthenticationVar", short: "Au", mandatory: true, correspondances: {"M":"M", "S":"S", "N":"N"}},
                {select_id: "ConfImpactVar", short: "C", mandatory: true, correspondances: {"N":"N", "P":"P", "L":"P", "C":"C", "H":"C"}},
                {select_id: "IntegImpactVar", short: "I", mandatory: true, correspondances: {"N":"N", "P":"P", "L":"P", "C":"C", "H":"C"}},
                {select_id: "AvailImpactVar", short: "A", mandatory: true, correspondances: {"N":"N", "P":"P", "L":"P", "C":"C", "H":"C"}},
                {select_id: "ExploitabilityVar", short: "E", mandatory: false, correspondances: {"ND":"ND", "X":"ND", "U":"U", "POC":"POC", "P":"POC", "F":"F", "H":"H"}},
                {select_id: "RemediationLevelVar", short: "RL", mandatory: false, correspondances: {"ND":"ND", "X":"ND","OF":"OF", "O":"OF", "TF":"TF", "T":"TF", "W":"W", "U":"U"}},
                {select_id: "ReportConfidenceVar", short: "RC", mandatory: false, correspondances: {"ND":"ND", "X":"ND","UC":"UC", "U":"UC", "UR":"UR", "R":"UR", "C":"C"}},
                {select_id: "PrivilegeRequiredVar", short: "PR", mandatory: true, correspondances: {"N":"N", "L":"L", "H":"H"}},
                {select_id: "UserInteractionVar", short: "UI", mandatory: true, correspondances: {"N":"N", "R":"R"}},
                {select_id: "ScopeVar", short: "S", mandatory: true, correspondances: {"U":"U", "C":"C"}}

            ]; //big_data_select_v2_v3

            function getScroreV2(dict_vecteur){
                var key,i;
                //console.log("update_v2 : lock_v2 = "+document.getElementById('lock_v2').checked);

                var val_temp, accessvector, accesscomplexity, accessauthentication, confimpact, integimpact, availimpact, exploitability, remediationlevel, reportconfidence;
                var impact, exploit, fimpact=0, basescore=undefined, temporalscore;
                var vecteur = "";
                var svg_id = "sous-titre-chart_v2";
                var svg_img_id = "img_svg_v2";
                var png_img_id = "img_png_v2";
                var sous_tritre_val = 0;

                switch(dict_vecteur["AV"]){
                    case "L":
                        val_temp = 0.395;
                        break;
                    case "A":
                        val_temp = 0.646;
                        break;
                    case "N":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                accessvector = val_temp;
                switch(dict_vecteur["AC"]){
                    case "H":
                        val_temp = 0.35;
                        break;
                    case "M":
                        val_temp = 0.61;
                        break;
                    case "L":
                        val_temp = 0.71;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                accesscomplexity = val_temp;
                switch(dict_vecteur["Au"]){
                    case "M":
                        val_temp = 0.45;
                        break;
                    case "S":
                        val_temp = 0.56;
                        break;
                    case "N":
                        val_temp = 0.704;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                accessauthentication = val_temp;
                switch(dict_vecteur["C"]){
                    case "N":
                        val_temp = 0;
                        break;
                    case "P":
                        val_temp = 0.275;
                        break;
                    case "C":
                        val_temp = 0.66;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                confimpact = val_temp;
                switch(dict_vecteur["I"]){
                    case "N":
                        val_temp = 0;
                        break;
                    case "P":
                        val_temp = 0.275;
                        break;
                    case "C":
                        val_temp = 0.66;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                integimpact = val_temp;
                switch(dict_vecteur["A"]){
                    case "N":
                        val_temp = 0;
                        break;
                    case "P":
                        val_temp = 0.275;
                        break;
                    case "C":
                        val_temp = 0.66;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                availimpact = val_temp;

                switch(dict_vecteur["E"]){
                    case "U":
                        val_temp = 0.85;
                        break;
                    case "POC":
                        val_temp = 0.9;
                        break;
                    case "F":
                        val_temp = 0.95;
                        break;
                    case "H":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = 1;
                        break;
                }
                exploitability = val_temp;
                switch(dict_vecteur["RL"]){
                    case "OF":
                        val_temp = 0.87;
                        break;
                    case "TF":
                        val_temp = 0.9;
                        break;
                    case "W":
                        val_temp = 0.95;
                        break;
                    case "U":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = 1;
                        break;
                }
                remediationlevel = val_temp;
                switch(dict_vecteur["RC"]){
                    case "UC":
                        val_temp = 0.9;
                        break;
                    case "UR":
                        val_temp = 0.95;
                        break;
                    case "C":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = 1;
                        break;
                }
                reportconfidence = val_temp;

                // Calcul du score de base
                if((confimpact != -1)&&(integimpact != -1)&&(availimpact != -1)&&
                   (accessvector != -1)&&(accesscomplexity != -1)&&(accessauthentication != -1)) {

                    impact = 10.41 * ( 1 - ( 1 - confimpact ) * ( 1 - integimpact ) * ( 1 - availimpact ));
                    exploit = 20 * accessvector * accesscomplexity * accessauthentication;
                    if (impact != 0) {
                        fimpact = 1.176;
                    }
                    basescore = ( ( ( 0.6 * impact ) + ( 0.4 * exploit ) - 1.5 ) * fimpact );
                    basescore = Math.round( basescore * 10 ) / 10;
                    // Calcul du score temporel
                    temporalscore = basescore * exploitability * remediationlevel * reportconfidence;
                    temporalscore = Math.round( temporalscore * 10 ) / 10;
                }

                if(!(typeof basescore === "undefined")){
                    //for (var key in dict_vecteur) { //ordre alphabetique
                    for (i=0;i<vectorKeysOrdered_v2.length;i++) {
                        key = vectorKeysOrdered_v2[i];
                        vecteur =vecteur + key+":"+dict_vecteur[key]+"/";
                    }
                    vecteur = vecteur.substring(-1);



                    if(basescore>=9) {
                        sous_tritre_val = 4;
                    }else if(basescore>=7) {
                        sous_tritre_val = 3;
                    }else if(basescore>=3) {
                        sous_tritre_val = 2;
                    }else if(basescore>=0) {
                        sous_tritre_val = 1;
                    }

                }else{
                    basescore = undefined;
                    temporalscore = undefined;
                    vecteur = undefined;
                    sous_tritre_val=0;
                }

                //console.log("getScroreV2 : lock_v2 = "+document.getElementById('lock_v2').checked);
                if(!document.getElementById('lock_v2').checked){
                    //change_color(sous_tritre_val,svg_id);
                    set_img_png_src(sous_tritre_val,png_img_id);
                    //svg_to_img(svg_id, svg_img_id); //creer l'image svg
                }
                return basescore;
            } // getScroreV2

            function getScroreV3(dict_vecteur){
                var key,i;

                var vecteur = "";
                var svg_id = "sous-titre-chart_v3";
                var svg_img_id = "img_svg_v3";
                var png_img_id = "img_png_v3";
                var sous_tritre_val = 0;

                var temporalScore = undefined, baseScore = undefined;

                // variables contenant les valeurs
                var AV = dict_vecteur["AV"]     || "";
                var AC = dict_vecteur["AC"]     || "";
                var PR = dict_vecteur["PR"]     || "";
                var UI = dict_vecteur["UI"]     || "";
                var S  = dict_vecteur["S"]      || "";
                var C  = dict_vecteur["C"]      || "";
                var I  = dict_vecteur["I"]      || "";
                var A  = dict_vecteur["A"]      || "";

                var E =   dict_vecteur["E"]     || "X";
                var RL =  dict_vecteur["RL"]    || "X";
                var RC =  dict_vecteur["RC"]    || "X";
                //console.log("update_v3 1 ");
                //console.log([AV,AC, PR, UI, S, C, I, A, E, RL, RC]);
                // console.log(dict_vecteur);
                // console.log(A+"/"+AC+"/"+AV+"/"+C+"/"+E+"/"+I+"/"+PR+"/"+RC+"/"+RL+"/"+S+"/"+UI);

                if (!([AV,AC, PR, UI, S, C, I, A, E, RL, RC].includes('')) ) {
                    // GATHER WEIGHTS FOR ALL METRICS
                    var metricWeightAV  = CVSS31.Weight.AV    [AV];
                    var metricWeightAC  = CVSS31.Weight.AC    [AC];
                    var metricWeightPR  = CVSS31.Weight.PR    [S][PR];  // PR depends on the value of Scope (S).
                    var metricWeightUI  = CVSS31.Weight.UI    [UI];
                    var metricWeightS   = CVSS31.Weight.S     [S];
                    var metricWeightC   = CVSS31.Weight.CIA   [C];
                    var metricWeightI   = CVSS31.Weight.CIA   [I];
                    var metricWeightA   = CVSS31.Weight.CIA   [A];

                    var metricWeightE   = CVSS31.Weight.E     [E];
                    var metricWeightRL  = CVSS31.Weight.RL    [RL];
                    var metricWeightRC  = CVSS31.Weight.RC    [RC];


                    // CALCULATE THE CVSS BASE SCORE
                    var iss; /* Impact Sub-Score */
                    var impact;
                    var exploitability;

                    iss = (1 - ((1 - metricWeightC) * (1 - metricWeightI) * (1 - metricWeightA)));

                    if (S === 'U') {
                        impact = metricWeightS * iss;
                    } else {
                        impact = metricWeightS * (iss - 0.029) - 3.25 * Math.pow(iss - 0.02, 15);
                    }

                    exploitability = CVSS31.exploitabilityCoefficient * metricWeightAV * metricWeightAC * metricWeightPR * metricWeightUI;

                    if (impact <= 0) {
                        baseScore = 0;
                    } else {
                        if (S === 'U') {
                            baseScore = CVSS31.roundUp1(Math.min((exploitability + impact), 10));
                        } else {
                            baseScore = CVSS31.roundUp1(Math.min(CVSS31.scopeCoefficient * (exploitability + impact), 10));
                        }
                    }

                    // CALCULATE THE CVSS TEMPORAL SCORE
                    temporalScore = CVSS31.roundUp1(baseScore * metricWeightE * metricWeightRL * metricWeightRC);
                }


                if(!(typeof baseScore === "undefined")){
                    //for (var key in dict_vecteur) { //ordre alphabetique
                    for (i=0;i<vectorKeysOrdered_v3.length;i++) {
                        key = vectorKeysOrdered_v3[i];
                        vecteur =vecteur + key+":"+dict_vecteur[key]+"/";
                    }
                    vecteur = vecteur.substring(-1);


                    if(baseScore>=9) {
                        sous_tritre_val = 4;
                    }else if(baseScore>=7) {
                        sous_tritre_val = 3;
                    }else if(baseScore>=3) {
                        sous_tritre_val = 2;
                    }else if(baseScore>=0) {
                        sous_tritre_val = 1;
                    }

                }else{
                    baseScore = undefined;
                    temporalScore = undefined;
                    vecteur = undefined;
                    sous_tritre_val = 0;
                }



                //console.log("getScroreV3 : lock_v3 = "+document.getElementById('lock_v3').checked);
                if(!document.getElementById('lock_v3').checked){
                    //change_color(sous_tritre_val,svg_id);
                    set_img_png_src(sous_tritre_val,png_img_id);
                    //svg_to_img(svg_id, svg_img_id); //creer l'image svg
                }

                return baseScore;
            } // getScroreV3

            function update_v3(dict_vecteur){
                var key,i;
                if(document.getElementById('lock_v3').checked) {return 0;}

                var vecteur = "";
                var svg_id = "sous-titre-chart_v3";
                var svg_img_id = "img_svg_v3";
                var png_img_id = "img_png_v3";
                var sous_tritre_val = 0;

                var temporalScore = undefined, baseScore = undefined;

                // variables contenant les valeurs
                var AV = dict_vecteur["AV"]     || "";
                var AC = dict_vecteur["AC"]     || "";
                var PR = dict_vecteur["PR"]     || "";
                var UI = dict_vecteur["UI"]     || "";
                var S  = dict_vecteur["S"]      || "";
                var C  = dict_vecteur["C"]      || "";
                var I  = dict_vecteur["I"]      || "";
                var A  = dict_vecteur["A"]      || "";

                var E =   dict_vecteur["E"]     || "X";
                var RL =  dict_vecteur["RL"]    || "X";
                var RC =  dict_vecteur["RC"]    || "X";
                //console.log("update_v3 1 ");
                //console.log([AV,AC, PR, UI, S, C, I, A, E, RL, RC]);

                if (!([AV,AC, PR, UI, S, C, I, A, E, RL, RC].includes(''))) {
                    // GATHER WEIGHTS FOR ALL METRICS
                    var metricWeightAV  = CVSS31.Weight.AV    [AV];
                    var metricWeightAC  = CVSS31.Weight.AC    [AC];
                    var metricWeightPR  = CVSS31.Weight.PR    [S][PR];  // PR depends on the value of Scope (S).
                    var metricWeightUI  = CVSS31.Weight.UI    [UI];
                    var metricWeightS   = CVSS31.Weight.S     [S];
                    var metricWeightC   = CVSS31.Weight.CIA   [C];
                    var metricWeightI   = CVSS31.Weight.CIA   [I];
                    var metricWeightA   = CVSS31.Weight.CIA   [A];

                    var metricWeightE   = CVSS31.Weight.E     [E];
                    var metricWeightRL  = CVSS31.Weight.RL    [RL];
                    var metricWeightRC  = CVSS31.Weight.RC    [RC];


                    // CALCULATE THE CVSS BASE SCORE
                    var iss; /* Impact Sub-Score */
                    var impact;
                    var exploitability;

                    iss = (1 - ((1 - metricWeightC) * (1 - metricWeightI) * (1 - metricWeightA)));

                    if (S === 'U') {
                        impact = metricWeightS * iss;
                    } else {
                        impact = metricWeightS * (iss - 0.029) - 3.25 * Math.pow(iss - 0.02, 15);
                    }

                    exploitability = CVSS31.exploitabilityCoefficient * metricWeightAV * metricWeightAC * metricWeightPR * metricWeightUI;

                    if (impact <= 0) {
                        baseScore = 0;
                    } else {
                        if (S === 'U') {
                            baseScore = CVSS31.roundUp1(Math.min((exploitability + impact), 10));
                        } else {
                            baseScore = CVSS31.roundUp1(Math.min(CVSS31.scopeCoefficient * (exploitability + impact), 10));
                        }
                    }

                    // CALCULATE THE CVSS TEMPORAL SCORE
                    temporalScore = CVSS31.roundUp1(baseScore * metricWeightE * metricWeightRL * metricWeightRC);
                }

                if(!(typeof baseScore === "undefined")){
                    //for (var key in dict_vecteur) { //ordre alphabetique
                    for (i=0;i<vectorKeysOrdered_v3.length;i++) {
                        key = vectorKeysOrdered_v3[i];
                        vecteur =vecteur + key+":"+dict_vecteur[key]+"/";
                    }
                    vecteur = vecteur.substring(-1);


                    if(baseScore>=9) {
                        sous_tritre_val = 4;
                    }else if(baseScore>=7) {
                        sous_tritre_val = 3;
                    }else if(baseScore>=3) {
                        sous_tritre_val = 2;
                    }else if(baseScore>=0) {
                        sous_tritre_val = 1;
                    }

                }else{
                    baseScore = undefined;
                    temporalScore = undefined;
                    vecteur = undefined;
                    sous_tritre_val = 0;
                }
                document.getElementById("display_score_base_v3").innerText = baseScore;
                document.getElementById("display_score_temp_v3").innerText = temporalScore;
                document.getElementById("link_vecteur_v3").innerText = vecteur;

                set_img_png_src(sous_tritre_val,png_img_id);

                // modifier l'image png

            } // update_v3

            function update_v2(dict_vecteur){
                var key,i;
                //console.log("update_v2 : lock_v2 = "+document.getElementById('lock_v2').checked);
                if(document.getElementById('lock_v2').checked) {return 0;}

                var val_temp, accessvector, accesscomplexity, accessauthentication, confimpact, integimpact, availimpact, exploitability, remediationlevel, reportconfidence;
                var impact, exploit, fimpact=0, basescore=undefined, temporalscore;
                var vecteur = "";
                var svg_id = "sous-titre-chart_v2";
                var svg_img_id = "img_svg_v2";
                var png_img_id = "img_png_v2";
                var sous_tritre_val = 0;

                switch(dict_vecteur["AV"]){
                    case "L":
                        val_temp = 0.395;
                        break;
                    case "A":
                        val_temp = 0.646;
                        break;
                    case "N":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                accessvector = val_temp;
                switch(dict_vecteur["AC"]){
                    case "H":
                        val_temp = 0.35;
                        break;
                    case "M":
                        val_temp = 0.61;
                        break;
                    case "L":
                        val_temp = 0.71;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                accesscomplexity = val_temp;
                switch(dict_vecteur["Au"]){
                    case "M":
                        val_temp = 0.45;
                        break;
                    case "S":
                        val_temp = 0.56;
                        break;
                    case "N":
                        val_temp = 0.704;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                accessauthentication = val_temp;
                switch(dict_vecteur["C"]){
                    case "N":
                        val_temp = 0;
                        break;
                    case "P":
                        val_temp = 0.275;
                        break;
                    case "C":
                        val_temp = 0.66;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                confimpact = val_temp;
                switch(dict_vecteur["I"]){
                    case "N":
                        val_temp = 0;
                        break;
                    case "P":
                        val_temp = 0.275;
                        break;
                    case "C":
                        val_temp = 0.66;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                integimpact = val_temp;
                switch(dict_vecteur["A"]){
                    case "N":
                        val_temp = 0;
                        break;
                    case "P":
                        val_temp = 0.275;
                        break;
                    case "C":
                        val_temp = 0.66;
                        break;
                    default:
                        val_temp = -1;
                        break;
                }
                availimpact = val_temp;

                switch(dict_vecteur["E"]){
                    case "U":
                        val_temp = 0.85;
                        break;
                    case "POC":
                        val_temp = 0.9;
                        break;
                    case "F":
                        val_temp = 0.95;
                        break;
                    case "H":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = 1;
                        break;
                }
                exploitability = val_temp;
                switch(dict_vecteur["RL"]){
                    case "OF":
                        val_temp = 0.87;
                        break;
                    case "TF":
                        val_temp = 0.9;
                        break;
                    case "W":
                        val_temp = 0.95;
                        break;
                    case "U":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = 1;
                        break;
                }
                remediationlevel = val_temp;
                switch(dict_vecteur["RC"]){
                    case "UC":
                        val_temp = 0.9;
                        break;
                    case "UR":
                        val_temp = 0.95;
                        break;
                    case "C":
                        val_temp = 1;
                        break;
                    default:
                        val_temp = 1;
                        break;
                }
                reportconfidence = val_temp;

                // Calcul du score de base
                if((confimpact != -1)&&(integimpact != -1)&&(availimpact != -1)&&
                   (accessvector != -1)&&(accesscomplexity != -1)&&(accessauthentication != -1)) {

                    impact = 10.41 * ( 1 - ( 1 - confimpact ) * ( 1 - integimpact ) * ( 1 - availimpact ));
                    exploit = 20 * accessvector * accesscomplexity * accessauthentication;
                    if (impact != 0) {
                        fimpact = 1.176;
                    }
                    basescore = ( ( ( 0.6 * impact ) + ( 0.4 * exploit ) - 1.5 ) * fimpact );
                    basescore = Math.round( basescore * 10 ) / 10;
                    // Calcul du score temporel
                    temporalscore = basescore * exploitability * remediationlevel * reportconfidence;
                    temporalscore = Math.round( temporalscore * 10 ) / 10;
                }

                if(!(typeof basescore === "undefined")){
                    //for (var key in dict_vecteur) { //ordre alphabetique
                    for (i=0;i<vectorKeysOrdered_v2.length;i++) {
                        key = vectorKeysOrdered_v2[i];
                        vecteur =vecteur + key+":"+dict_vecteur[key]+"/";
                    }
                    vecteur = vecteur.substring(-1);



                    if(basescore>=9) {
                        sous_tritre_val = 4;
                    }else if(basescore>=7) {
                        sous_tritre_val = 3;
                    }else if(basescore>=3) {
                        sous_tritre_val = 2;
                    }else if(basescore>=0) {
                        sous_tritre_val = 1;
                    }

                }else{
                    basescore = undefined;
                    temporalscore = undefined;
                    vecteur = undefined;
                    sous_tritre_val = 0;
                }
                document.getElementById("display_score_base_v2").innerText = basescore;
                document.getElementById("display_score_temp_v2").innerText = temporalscore;
                document.getElementById("link_vecteur_v2").innerText = vecteur;

                set_img_png_src(sous_tritre_val,png_img_id);

                //console.log("update_v2 : "+vecteur);
                // modifier l'image png
            } // update_v2

            function vectorString_To_VectorList(str){
                var vecteur = update_options(str, true);
                // console.log("vectorV2String_To_VectorV2List => input : "+str);
                // console.log(vecteur);
                return vecteur;
            } // vectorString_To_VectorList

            function computeCVSS(){
                //console.log("computeCVSS");
                var vector_v2={}, vector_v3={}, value_v2_v3, value_splitted_v2_v3, value_split_short_val, value_short_val;
                for(var i=0;i<big_data_select_v2_v3.length;i++){
                    value_v2_v3 = document.getElementById(big_data_select_v2_v3[i]["select_id"]).value; // ex v2=A:/v3=A:N
                    value_splitted_v2_v3 = value_v2_v3.split('/'); // ex v2=A: et v3=A:N
                    if(Array.isArray(value_splitted_v2_v3)) {
                        if(value_splitted_v2_v3.length>0) {
                            value_split_short_val = value_splitted_v2_v3[0].split("="); // v2 et A:
                            if(Array.isArray(value_split_short_val)) {
                                if(value_split_short_val.length>1) {
                                    value_short_val = value_split_short_val[1].split(":"); // A et ""
                                    if(Array.isArray(value_short_val)) {
                                        if(value_short_val.length>1) {
                                            vector_v2[value_short_val[0]] = value_short_val[1];
                                        }
                                    }
                                }
                            }
                            value_split_short_val = value_splitted_v2_v3[1].split("="); // v3 et A:N
                            if(Array.isArray(value_split_short_val)) {
                                if(value_split_short_val.length>1) {
                                    value_short_val = value_split_short_val[1].split(":"); // A et N
                                    if(Array.isArray(value_short_val)) {
                                        if(value_short_val.length>1) {
                                            vector_v3[value_short_val[0]] = value_short_val[1];
                                        }
                                    }
                                }
                            }
                        }
                    }
                }

                // console.log(vector_v2);
                //console.log("v3");
                //console.log(vector_v3);
                update_v2(vector_v2);
                update_v3(vector_v3);
            } // computeCVSS

            function reset_cvss_options() {
                var i, select_id;
                //console.log("size array ("+(big_data_select_v2_v3.length)+")");
                for(i=0 ; i<(big_data_select_v2_v3.length);i++) {
                    select_id = (big_data_select_v2_v3[i])['select_id'];
                    document.getElementById(select_id).getElementsByTagName('option')[0].selected='selected';
                    //console.log("reset ID ("+select_id+")");
                }

                computeCVSS();
            } // reset_cvss_options

            let array_exclude_char=['(',')','[',']','{','}',/s/g,'>','-','_','<',' ']; // sera remplacé par le delimiteur";

            function update_options(input="", getVecteur=false){
                //console.log("update_options(input="+input+", getVecteur="+getVecteur+")");
                if (! (typeof input === 'string') || input === ''){
                    input = document.getElementById('input_text_cvss').value;
                }
                //console.log("update_options : "+input);
                //console.log("update_options : "+document.getElementById('lock_v2').checked);

                var delim = document.getElementById('input_text_cvss_delimiter').value;
                var op = document.getElementById('input_text_cvss_operateur').value;
                var i,j, key,val,entry, select_id, option, select_obj, options;
                var vecteur = {};

                if(! getVecteur) {reset_cvss_options();}

                // nettoyage de l'input
                //console.log ("input avant : "+input);
                for(i=0;i<array_exclude_char.length;i++){
                    //if(delim !==array_exclude_char[i] && op !==array_exclude_char[i]) {input=input.replaceAll(array_exclude_char[i],delim);}
                    if(delim !==array_exclude_char[i] && op !==array_exclude_char[i]) {
                        // replaceAll !!! AV : A => AV/:/A
                        while(input != input.replace(array_exclude_char[i],delim)) {
                            input=input.replace(array_exclude_char[i],delim);
                        }
                    }
                }
                //console.log ("input apres : "+input);

                const A1_cvss_entries = input.split(delim);
                for (i=0;i<A1_cvss_entries.length;i++) {

                    //console.log ("test : "+A1_cvss_entries[i]);
                    entry = A1_cvss_entries[i];
                    if (entry.split(op).length == 2) {
                        key = entry.split(op)[0];
                        val = entry.split(op)[1];


                        // is the key OK then get option value
                        for(j=0 ; j<(big_data_select_v2_v3.length);j++) {
                            //console.log(key+" => "+val);
                            if((big_data_select_v2_v3[j])['short'] === key) {
                                //console.log("apres "+key+" trouvé val => "+val+" j="+j);
                                vecteur[key] = val;

                                if(! getVecteur) {
                                    select_id = (big_data_select_v2_v3[j])['select_id'];
                                    val = ((big_data_select_v2_v3[j])['correspondances'])[val];
                                    select_obj = document.getElementById(select_id);
                                    options = Array.from(select_obj.options);
                                    option = options.find(item => item.text === val);
                                    option.selected = true;
                                }

                            }
                        }
                    }
                }
                if(! getVecteur){
                    computeCVSS();
                }else{
                    return vecteur;
                }


            } // update_options

            function scoreString_to_float(str, defaultValue=-1){
                var output=defaultValue;
                try {
                    str=(str==""?defaultValue:str);
                    output = +(str);
                    if(!output && output!=0){
                        output=defaultValue;
                    }
                }catch{

                }

                return output;
            } // scoreString_to_float

            function getStorageScore_if_ActualScoreIsUnkown(cveID, actualScore, storageType="CVEv2") {
                var fctName = "getStorageScore_if_ActualScoreIsUnkown"
                var maxScore = actualScore;
                var avi, vecteurFromStorage, scoreFromStorage
                if(cveID==WatchForDEBUG_CVE){console.log("["+fctName+"]"+" cveID="+cveID+", actualScore="+actualScore+", storageType="+storageType);}
                if(actualScore<=0){ // pas de score V2 pour cette CVE (NA ou vide) ; on vérifie si il existe une version dans le stockage
                    avi = getCVSS_fromStorage(cveID, storageType);
                    if(cveID==WatchForDEBUG_CVE){console.log("["+fctName+"]"+" cveID="+cveID+", actualScore="+actualScore+", storageType="+storageType+"==>"+avi);}
                    if(avi) {
                        if(storageType=="CVEv2") {
                            vecteurFromStorage = getVecteurV2fromStorageAVI(avi);
                            scoreFromStorage = getScroreV2(vectorString_To_VectorList(vecteurFromStorage));
                        } else {
                            vecteurFromStorage = getVecteurV3fromStorageAVI(avi);
                            scoreFromStorage = getScroreV3(vectorString_To_VectorList(vecteurFromStorage));
                        }

                        maxScore = scoreFromStorage;
                    }
                }

                return maxScore;
            } // getStorageScore_if_ActualScoreIsUnkown
        //}
        //if("FONCTIONS : API CVE"){
            // FONCTIONS\CVSS\internes />

            // < FONCTIONS\CVSS\externes
            // < TO DO
            function getCyberwatchScore(cve_id) {
                var url = "https://cyberwatch.internet.np/cve_announcements/"+cve_id;
                return 0;
                // a completer ou utiliser l'API
                GM.xmlHttpRequest({
                    method: "GET",
                    url: url,
                    onload: function(response) {
                        //console.log("getCyberwatchScore onload");
                        //console.log(response.responseText);
                    }
                });
            } // getCyberwatchScore
            // TO DO />

            function get_MS_score(cve_id) {
                var url = "https://msrc.microsoft.com/update-guide/vulnerability/"+cve_id+"";
                // verifier les flux xhr
                GM.XMLHttpRequest.responseXML({
                    method: "GET",
                    url: url,
                    onload: function(response) {
                        var el = document.createElement( 'html' );
                        el.innerHTML = response.responseText;
                        var score = el.getElementsByClassName('root-204');
                        //console.log(response.responseText);
                    }
                });
            } // get_MS_score

            function get_vuldb(id) {
                GM.xmlHttpRequest({
                    method: "POST",
                    url: "https://vuldb.com/?search",
                    data: "search="+id,
                    headers: {
                        "Content-Type": "application/x-www-form-urlencoded"
                    },
                    onload: function(response) {
                        var vul_html = response.responseText;
                        var res = vul_html.match(/VulDB \[.{1,3}\]/m);
                        if (res != null) {
                            $('#'+id+'_vuldb').before('<span style="padding-left: 30px;">'+res+'(V3)</span>');
                        }
                    }
                });
            } // get_vuldb

            function get_sourceclear(id) {
                var url_sourceclear = "https://api.sourceclear.com/catalog/search?q="+id+"%20type%3Avulnerability";
                GM.xmlHttpRequest({
                    method: "GET",
                    url: url_sourceclear,
                    onload: function(response) {
                        var veracode = JSON.parse(response.responseText);
                        if (veracode['metadata']['hits'] == 0) {
                            document.getElementById(id+'_sourceclear').style.display = "none";
                        }
                    }
                });
            } // get_sourceclear

            function get_circl(id) {
                var url_circl = "https://cve.circl.lu/cve/"+id;
                GM.xmlHttpRequest({
                    method: "GET",
                    url: url_circl,
                    onload: function(response) {
                        var circl = response.responseText;
                        // if (circl.includes("This CVE does not exist")) {
                        if (circl.indexOf('This CVE does not exist') !== -1) {
                            document.getElementById(id+'_circl').style.display = "none";
                        }
                    }
                });
            } // get_circl

            function get_NVD(id, intervalAnimation="") {
                var url_NVD = "https://nvd.nist.gov/vuln/detail/"+id;
                // declaration de variables NVD
                var class_baseScores = 'severityDetail';
                var class_NVD_vector_v3 = 'tooltipCvss3NistMetrics';
                var class_NVD_vector_v2 = 'tooltipCvss2NistMetrics';
                var class_CNA_vector_v3 = 'tooltipCvss3CnaMetrics';

                var tempBS, tempBS_NVD, tempBS_CNA;

                var nvd_return = new Map();
                nvd_return.set("score",0);
                nvd_return.set("cartouche","");

                nvd_return.set("v2_BaseScore","");
                nvd_return.set("v2_Vector","");
                nvd_return.set("v3_BaseScore","");
                nvd_return.set("v3_Vector","");
                nvd_return.set("v3_CNA_BaseScore","");
                nvd_return.set("v3_CNA_Vector","");

                GM.xmlHttpRequest({
                    method: "GET",
                    url: url_NVD,
                    onload: function(response) {
                        console.log(id+" : "+response.status);
                        if(response.status==403) {
                            // erreur de chargement, on relance
                            console.log("erreur 403 pour : "+id);
                            add_log_to_CMB("[get_NVD]erreur 403 pour : "+id, 0);
                            compteur403 = compteur403+1;
                            if(compteur403>=nombre403max){
                                if(
                                    confirm(
                                    "[get_NVD]("+id+")"+"\n\n"+
                                    "Le nombre d'erreur de chargement NVD (403) a atteint le seuil de "+nombre403max+"\n"+
                                    "La dernière erreur est apparue sur la CVE "+id+"\n"+
                                    "Retournez sur NVD pour verifier l'état :"+"\n"+
                                    "\thttps://nvd.nist.gov/vuln/detail/"+id+"\n"+
                                    "Si vous cliquez sur OK, la page rechargera,"+"\n"+
                                    "Si vous cliquez sur annuler, vous reinitialiserez le compteur d'erreur et continuerez les requêtes"+"\n"+
                                    "\tActuellement "+compteurCVEloaded+"/"+get_nbCVEsFromTableHeader()+"\n"+
                                    "\nIdéalement attendez 1 minute pour que NVD vous redonne l'accès"
                                )
                                    //confirm("trop d'erreur de chargement NVD,\nrecharger la page ?\n(\n\tsi vous rechargez, attendez un peu \n\tou retourner sur NVD pour verifier l'état :\n\thttps://nvd.nist.gov/vuln/detail/"+id+"\n)")
                                ){
                                    location.reload();
                                }else{
                                    compteur403=0;
                                }
                            }else{
                                get_NVD(id);
                            }
                            return false;
                        }

                        var elem = document.createElement( 'html' );
                        var spanItem, NVDpublishDateDOM, NVDmodifiedDateDOM, NVDsourceDOM, NVDwarningDOM
                        elem.innerHTML = response.responseText;
                        // récupération des infos de la publication NVD
                        var NVDpublishDate = "inconnu de NVD en date du "+(new Date()).toLocaleDateString("fr");
                        var NVDmodifiedDate = "inconnu de NVD en date du "+(new Date()).toLocaleDateString("fr");
                        var NVDsource = "inconnu de NVD en date du "+(new Date()).toLocaleDateString("fr");
                        var NVDwarning = "";
                        // console.log(elem);
                        try {
                            spanItem = elem.querySelectorAll('[data-testid]');
                            NVDpublishDateDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-published-on");
                            NVDmodifiedDateDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-last-modified-on");
                            NVDsourceDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-current-description-source");
                            NVDwarningDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-warning-status-name");
                            if(NVDpublishDateDOM){NVDpublishDate = dateUStoFR(NVDpublishDateDOM.innerText);}
                            if(NVDmodifiedDateDOM){NVDmodifiedDate = dateUStoFR(NVDmodifiedDateDOM.innerText);}
                            if(NVDsourceDOM){NVDsource = NVDsourceDOM.innerText;}
                            if(NVDwarningDOM){NVDwarning = NVDwarningDOM.innerText;}
                        } catch{

                        }

                        // récupération des scores
                        var nvd_DOM = elem.getElementsByClassName(class_baseScores);
                        var nvd=[];
                        for(var i=0;i<nvd_DOM.length;i++) {
                            tempBS = (nvd_DOM[i]).innerText.trim();
                            if(tempBS === 'N/A') {
                                nvd.push('N/A');
                            }else{
                                tempBS = tempBS.split(' ')[0];
                                tempBS = parseInt(tempBS.replace(/\./g, ''),10)/10;
                                nvd.push(tempBS);
                            }
                        }
                        // récupération des cartouches
                        var cartouche_v2 = elem.getElementsByClassName(class_NVD_vector_v2);
                        if (cartouche_v2.length==0) {cartouche_v2="";}else{cartouche_v2=(cartouche_v2[0]).innerText;}
                        var cartouche_v3 = elem.getElementsByClassName(class_NVD_vector_v3);
                        if (cartouche_v3.length==0) {cartouche_v3="";}else{cartouche_v3=(cartouche_v3[0]).innerText;}
                        var cartouche_v3_CNA = elem.getElementsByClassName(class_CNA_vector_v3);
                        if (cartouche_v3_CNA.length==0) {cartouche_v3_CNA="";}else{cartouche_v3_CNA=(cartouche_v3_CNA[0]).innerText;}

                        switch(nvd.length) {
                            case 1:
                                tempBS = nvd[0];
                                // récupération de l'élément
                                if(cartouche_v2){
                                    nvd_return.set("v2_BaseScore",tempBS);
                                    nvd_return.set("v2_Vector",cartouche_v2);
                                }else if(cartouche_v3){
                                    nvd_return.set("v3_BaseScore",tempBS);
                                    nvd_return.set("v3_Vector",cartouche_v3);
                                }else if(cartouche_v3_CNA){
                                    nvd_return.set("v3_CNA_BaseScore",tempBS);
                                    nvd_return.set("v3_CNA_Vector",cartouche_v3_CNA);
                                }else{// pas de cartouche mais NVD à 1
                                    document.getElementById("tableCVEs_Td_"+id+"_CVE").innerText = document.getElementById("tableCVEs_Td_"+id+"_CVE").innerText + " (anomalie NVD)";
                                }
                                break;
                            case 2:
                                // récupération du score v2
                                tempBS = nvd[(nvd.length)-1];
                                nvd_return.set("v2_BaseScore",tempBS);
                                nvd_return.set("v2_Vector",cartouche_v2);
                                // récupération du score v3
                                tempBS = nvd[0];
                                if(cartouche_v3) {
                                    nvd_return.set("v3_BaseScore",tempBS);
                                    nvd_return.set("v3_Vector",cartouche_v3);
                                }else{
                                    nvd_return.set("v3_CNA_BaseScore",tempBS);
                                    nvd_return.set("v3_CNA_Vector",cartouche_v3_CNA);
                                }
                                break;

                            case 3:
                                // récupération du score v2
                                tempBS = nvd[2];
                                nvd_return.set("v2_BaseScore",tempBS);
                                nvd_return.set("v2_Vector",cartouche_v2);

                                // récupération du score v3
                                tempBS_NVD = nvd[0];
                                nvd_return.set("v3_BaseScore",tempBS_NVD);
                                nvd_return.set("v3_Vector",cartouche_v3);
                                tempBS_CNA = nvd[1];
                                nvd_return.set("v3_CNA_BaseScore",tempBS_CNA);
                                nvd_return.set("v3_CNA_Vector",cartouche_v3_CNA);

                                break;
                        }

                        if (nvd_DOM.length == 0) {
                            nvd_DOM = 'ØØ';
                        } else {
                            nvd_DOM = nvd_DOM.item(1).innerText.trim(); // item(0) == CVSS V3
                            nvd_DOM = nvd_DOM.split(' ')[0];
                            nvd_return.set("score",parseInt(nvd_DOM.replace(/\./g, ''),10)/10); // remplace le texte x.y en xy puis divise par 10 pour recuperer la version numérique
                        }
                        var score_NVD = '<a href=https://nvd.nist.gov/vuln/detail/'+id+' target="_blank">'+nvd_DOM+'</a>';
                        try{
                            document.getElementById(id+"_NVD").innerHTML = score_NVD;
                        }catch(error){
                            console.log("Erreur : chargement de "+id+"_NVD");
                        }

                        // remplissage du tableau
                        var prefixe="tableCVEs_Td_"+id


                        document.getElementById(prefixe+"_v2_score").innerText = nvd_return.get("v2_BaseScore");
                        document.getElementById(prefixe+"_v2_vector").value = nvd_return.get("v2_Vector");
                        document.getElementById(prefixe+"_v3_score").innerText = nvd_return.get("v3_BaseScore");
                        document.getElementById(prefixe+"_v3_vector").value = nvd_return.get("v3_Vector");
                        document.getElementById(prefixe+"_v3_CNA_score").innerText = nvd_return.get("v3_CNA_BaseScore");
                        document.getElementById(prefixe+"_v3_CNA_vector").value = nvd_return.get("v3_CNA_Vector");

                        document.getElementById(prefixe+"_NVDpublishDate").value = NVDpublishDate;
                        document.getElementById(prefixe+"_NVDmodifiedDate").value = NVDmodifiedDate;
                        document.getElementById(prefixe+"_NVDsource").value = NVDsource;
                        document.getElementById(prefixe+"_Editeur").title = NVDsource;
                        document.getElementById(prefixe+"_Editeur").innerText = NVDsource.substring(0,20);


                        // barre les rejected
                        if(NVDwarning=="Rejected") {
                            // la CVE est rejetée, on a barre
                            document.getElementById("tableCVEs_Tr_"+id).style.textDecoration  = "line-through"
                            document.getElementById("tableCVEs_Tr_"+id).title=NVDwarning;
                            try{
                                if(document.getElementById("li_CVE_"+id)) {
                                    console.log("Rejected : li_CVE_"+id)
                                    add_log_to_CMB("[get_NVD] Rejected : li_CVE_"+id, 0);
                                    console.log(document.getElementById("li_CVE_"+id))
                                    document.getElementById("li_CVE_"+id).innerText += " (REJECTED)"
                                }
                            } catch (error){
                                console.log("Erreur ; pas de li");
                            }
                        }else if(NVDwarning) {
                            console.log("WARNING [get_NVD] "+id+" => NVD WARNING = "+NVDwarning);
                            add_log_to_CMB("WARNING [get_NVD] "+id+" => NVD WARNING = "+NVDwarning, 0);
                        }

                        // DEBUG
                        //console.log("[get_NVD] "+id);
                        //console.log("[get_NVD] "+id+" _v2_score = "+nvd_return.get("v2_BaseScore")+" ["+nvd_return.get("v2_Vector")+"] => "+document.getElementById("tableCVEs_Td_"+id+"_v2_score").innerText+" ["+document.getElementById("tableCVEs_Td_"+id+"_v2_vector").value+"]");
                        //console.log("[get_NVD] "+id+" _v3_score = "+nvd_return.get("v3_BaseScore")+" ["+nvd_return.get("v3_Vector")+"] => "+document.getElementById("tableCVEs_Td_"+id+"_v3_score").innerText+" ["+document.getElementById("tableCVEs_Td_"+id+"_v3_vector").value+"]");
                        //console.log("[get_NVD] "+id+" _v3_CNA_score = "+nvd_return.get("v3_CNA_BaseScore")+" ["+nvd_return.get("v3_CNA_Vector")+"] => "+document.getElementById("tableCVEs_Td_"+id+"_v3_CNA_score").innerText+" ["+document.getElementById("tableCVEs_Td_"+id+"_v3_CNA_vector").value+"]");

                        // colorisation
                        document.getElementById("tableCVEs_Td_"+id+"_CVE").style.color = "black";
                        if(nvd_return.get("v2_BaseScore")=== "N/A" || nvd_return.get("v2_BaseScore") === ""){
                            // Highlight les CVE sans résultats
                            document.getElementById("tableCVEs_Td_"+id+"_CVE").style.color = "red";
                            document.getElementById("tableCVEs_Td_"+id+"_CVE").style.fontWeight = "bold";
                        }

                        // colorisation en orange de l'entete du tableau des que fini
                        compteurCVEloaded = compteurCVEloaded+1;
                        console.log("compteurCVEloaded",compteurCVEloaded);
                        if(compteurCVEloaded>= compteurCVEtoLoad){
                            document.getElementById("tableCVEs_THs").style.color = "orange";
                            if(intervalAnimation){
                                getMaxScroreFromListCVE(intervalAnimation);
                                document.getElementById("pProcessMessage").innerText = "Chargement des scores terminé, Selection du plus elevé"
                            }
                        }
                    },
                    onerror: function(error) {
                        console.log("get_NVD(error id="+id+")");
                        add_log_to_CMB("[get_NVD] (error id="+id+")", 0);
                    }
                });
            } // get_NVD

            function feedlyGetUpdateDate(publishDate, updateSpan){
                var publishDateObject = new Date(publishDate);
                var updateDateObject = new Date(publishDate);
                var number = parseInt(/\d+/g.exec(updateSpan)) ;
                var typeOfSpan = /\D+/g.exec(updateSpan) ;

                if(typeOfSpan == "d"){updateDateObject = publishDateObject.setDate(publishDateObject.getDate()-number);}
                if(typeOfSpan == "mo"){updateDateObject = publishDateObject.setMonth(publishDateObject.getMonth()-number);}

                var returnDateStr = (new Date(updateDateObject)).toLocaleDateString("en-US");
                return returnDateStr;
            } // feedlyGetUpdateDate

            function get_feedly(id) {
                var url_feedly = "https://feedly.com/cve/"+id;
                return new Promise(resolve => {
                    GM.xmlHttpRequest({
                        method: "GET",
                        url: url_feedly,
                        onload: function(response) {
                            var feedly = response.responseText;
                            // if (feedly.includes("This CVE does not exist")) {
                            if (feedly.indexOf('The CVE description is not yet public') !== -1) {
                                document.getElementById(id+'_feedly').style.display = "none";
                            }else{// il y a un résultats on le parse
                                var elem = document.createElement( 'html' );
                                var ht_CVE={
                                    from:url_feedly,
                                    CVEid:id
                                }
                                elem.innerHTML = response.responseText;
                                var i;
                                // DATES
                                var divDates, divDatesTxt, divDatesTxtMatchAll, publishDate, updateDate, updateSpan;
                                var datesExtractRegEx = /(?<=(: ))(.*?)(?= )/g;

                                try { // parse dates
                                    divDates = elem.getElementsByClassName("CVEHeader_date__ztRD1");
                                    if(divDates.length > 0){
                                        divDatesTxt = divDates[0].innerText
                                        divDatesTxtMatchAll = [...divDatesTxt.matchAll(datesExtractRegEx)];
                                        if(divDatesTxtMatchAll.length>0){
                                            if(divDatesTxtMatchAll[0].length>0){publishDate=divDatesTxtMatchAll[0][0];}
                                            if(divDatesTxtMatchAll.length>1){
                                                if(divDatesTxtMatchAll[1].length>0){
                                                    updateSpan=(divDatesTxtMatchAll[1])[0];
                                                    updateDate = feedlyGetUpdateDate(publishDate, updateSpan);
                                                }
                                            }
                                            // console.log("publish='"+publishDate+"' ; updated='"+updateDate+"'");
                                            ht_CVE.publishDate=publishDate;
                                            ht_CVE.updateDate=updateDate;
                                        }
                                    }
                                } catch{}
                                // SCORE
                                var divScore, divScoreTxt, score;
                                var scroreExtractRegEx = /[0-9]+\.?[0-9]?/g;

                                try { // parse Score
                                    divScore = elem.getElementsByClassName("CVECategories_field__W6K_4");
                                    for(i=0;i<divScore.length;i++){
                                        divScoreTxt = divScore[i].innerText;
                                        if(divScoreTxt.includes("CVSS")){
                                            score = parseFloat(scroreExtractRegEx.exec(divScoreTxt)) ;
                                            break;
                                        }
                                    }
                                    // console.log("Score="+score);
                                    ht_CVE.score=score;
                                } catch{}
                                // VECTEUR
                                var divVecteur, divVecteurTxt, vecteur;
                                var vecteurExtractRegEx = /[0-9]+\.?[0-9]?/g;

                                try { // parse vecteur
                                    divVecteur = elem.getElementsByClassName("Text_size--smallish__t7GHl");
                                    for(i=0;i<divScore.length;i++){
                                        divVecteurTxt = divVecteur[i].innerText;
                                        if(divVecteurTxt.includes("CVSS")){
                                            vecteur = divVecteurTxt ;
                                            break;
                                        }
                                    }
                                    // console.log("vecteur="+vecteur);
                                    ht_CVE.vecteur=vecteur;
                                } catch{}


                                //console.log(ht_CVE);
                                if(ht_CVE.score){add_log_to_CMB_FEEDLY(ht_CVE.CVEid+"="+ht_CVE.score);}
                                resolve(ht_CVE);
                            }
                        }
                    });
                }); // promise
            } // retourne {"cvss"= ; "vecteur"= ; "publishDate"= ; "updateDate"= } // get_feedly


            function get_cveDetails(id) {
                var url_cveD = "https://www.cvedetails.com/cve-details.php?cve_id="+id;
                GM.xmlHttpRequest({
                    method: "GET",
                    url: url_cveD,
                    onload: function(response) {
                        var el = document.createElement( 'html' );
                        el.innerHTML = response.responseText;
                        var score = el.getElementsByClassName('cvssbox');
                        if (score.length == 0) {
                            var score_item = 'ØØ';
                            var color = 'rgb(242, 242, 242)';
                        } else {
                            color = score.item(0).style.backgroundColor;
                            score_item = score.item(0).innerText;
                        }
                        //get_MS_score(id);
                        document.getElementById(id+'_cveDetails').innerText = score_item;
                        document.getElementById(id+'_cveDetails').style="background-color:"+color;

                        var list_cwe = el.getElementsByClassName('details');
                        $(list_cwe).find('a').each(function() {
                            if (this.href.indexOf("/cwe.html") > -1) {
                                var cwe = this.href.split("/");
                                if (CWEs.includes(cwe[4])) {
                                    // négation !
                                } else {
                                    CWEs.push(cwe[4]);
                                }
                            }
                        });
                    }
                });
            } // get_cveDetails

            function get_nbCVEsFromTableHeader(){
                var id="tableCVEs_THs_CVE";
                var header, nb_str, nb_int=-1;
                try{
                    header = document.getElementById(id).innerText
                    nb_str=/(?<=(\())(.*?)(?=(\)))/.exec(header);
                    nb_int = parseInt(nb_str);
                }catch(error){}
                return nb_int;
            }

            var compteurCVEloaded = 0;
            var compteurCVEtoLoad = 0;
            var compteur403 = 0;
            var nombre403max = 50;
            async function loadScore(id, intervalAnimation="") {
                var url_NVD = "https://nvd.nist.gov/vuln/detail/"+id;
                // declaration de variables NVD
                var class_baseScores = 'severityDetail';
                var class_NVD_vector_v3 = 'tooltipCvss3NistMetrics';
                var class_NVD_vector_v2 = 'tooltipCvss2NistMetrics';
                var class_CNA_vector_v3 = 'tooltipCvss3CnaMetrics';

                var tempBS, tempBS_NVD, tempBS_CNA;

                var nvd_return = new Map();
                nvd_return.set("score",0);
                nvd_return.set("cartouche","");

                nvd_return.set("v2_BaseScore","");
                nvd_return.set("v2_Vector","");
                nvd_return.set("v3_BaseScore","");
                nvd_return.set("v3_Vector","");
                nvd_return.set("v3_CNA_BaseScore","");
                nvd_return.set("v3_CNA_Vector","");

                get_feedly(id); // a faire evoluer, await n'est pas la solution
                GM.xmlHttpRequest({
                    method: "GET",
                    url: url_NVD,
                    onload: function(response) {
                        console.log(id+" : "+response.status);
                        if(response.status==403) {
                            // erreur de chargement, on relance
                            console.log("erreur 403 pour : "+id);
                            add_log_to_CMB("[loadScore] erreur 403 pour : "+id, 0);
                            compteur403 = compteur403+1;
                            if(compteur403>=nombre403max){
                                if(confirm(
                                    "[loadScore]("+id+")"+"\n\n"+
                                    "Le nombre d'erreur de chargement NVD (403) a atteint le seuil de "+nombre403max+"\n"+
                                    "La dernière erreur est apparue sur la CVE "+id+"\n"+
                                    "Retournez sur NVD pour verifier l'état :"+"\n"+
                                    "\thttps://nvd.nist.gov/vuln/detail/"+id+"\n"+
                                    "Si vous cliquez sur OK, la page rechargera,"+"\n"+
                                    "Si vous cliquez sur annuler, vous reinitialiserez le compteur d'erreur et continuerez les requêtes"+"\n"+
                                    "\tActuellement "+compteurCVEloaded+"/"+get_nbCVEsFromTableHeader()+"\n"+
                                    "\nIdéalement attendez 1 minute pour que NVD vous redonne l'accès"
                                )
                                  ){location.reload();
                                }else{
                                    compteur403=0;
                                }
                            }else{
                                get_NVD(id);
                            }
                            return false;
                        }

                        var elem = document.createElement( 'html' );
                        var spanItem, NVDpublishDateDOM, NVDmodifiedDateDOM, NVDsourceDOM, NVDwarningDOM
                        elem.innerHTML = response.responseText;
                        // récupération des infos de la publication NVD
                        var NVDpublishDate = "inconnu de NVD en date du "+(new Date()).toLocaleDateString("fr");
                        var NVDmodifiedDate = "inconnu de NVD en date du "+(new Date()).toLocaleDateString("fr");
                        var NVDsource = "inconnu de NVD en date du "+(new Date()).toLocaleDateString("fr");
                        var NVDwarning = "";
                        // console.log(elem);
                        try {
                            spanItem = elem.querySelectorAll('[data-testid]');
                            NVDpublishDateDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-published-on");
                            NVDmodifiedDateDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-last-modified-on");
                            NVDsourceDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-current-description-source");
                            NVDwarningDOM = getDOMitemFromitems_withASpecificAttribValue(spanItem, "data-testid", "vuln-warning-status-name");
                            if(NVDpublishDateDOM){NVDpublishDate = dateUStoFR(NVDpublishDateDOM.innerText);}
                            if(NVDmodifiedDateDOM){NVDmodifiedDate = dateUStoFR(NVDmodifiedDateDOM.innerText);}
                            if(NVDsourceDOM){NVDsource = NVDsourceDOM.innerText;}
                            if(NVDwarningDOM){NVDwarning = NVDwarningDOM.innerText;}
                        } catch{

                        }

                        // récupération des scores
                        var nvd_DOM = elem.getElementsByClassName(class_baseScores);
                        var nvd=[];
                        for(var i=0;i<nvd_DOM.length;i++) {
                            tempBS = (nvd_DOM[i]).innerText.trim();
                            if(tempBS === 'N/A') {
                                nvd.push('N/A');
                            }else{
                                tempBS = tempBS.split(' ')[0];
                                tempBS = parseInt(tempBS.replace(/\./g, ''),10)/10;
                                nvd.push(tempBS);
                            }
                        }
                        // récupération des cartouches
                        var cartouche_v2 = elem.getElementsByClassName(class_NVD_vector_v2);
                        if (cartouche_v2.length==0) {cartouche_v2="";}else{cartouche_v2=(cartouche_v2[0]).innerText;}
                        var cartouche_v3 = elem.getElementsByClassName(class_NVD_vector_v3);
                        if (cartouche_v3.length==0) {cartouche_v3="";}else{cartouche_v3=(cartouche_v3[0]).innerText;}
                        var cartouche_v3_CNA = elem.getElementsByClassName(class_CNA_vector_v3);
                        if (cartouche_v3_CNA.length==0) {cartouche_v3_CNA="";}else{cartouche_v3_CNA=(cartouche_v3_CNA[0]).innerText;}

                        switch(nvd.length) {
                            case 1:
                                tempBS = nvd[0];
                                // récupération de l'élément
                                if(cartouche_v2){
                                    nvd_return.set("v2_BaseScore",tempBS);
                                    nvd_return.set("v2_Vector",cartouche_v2);
                                }else if(cartouche_v3){
                                    nvd_return.set("v3_BaseScore",tempBS);
                                    nvd_return.set("v3_Vector",cartouche_v3);
                                }else if(cartouche_v3_CNA){
                                    nvd_return.set("v3_CNA_BaseScore",tempBS);
                                    nvd_return.set("v3_CNA_Vector",cartouche_v3_CNA);
                                }else{// pas de cartouche mais NVD à 1
                                    document.getElementById("tableCVEs_Td_"+id+"_CVE").innerText = document.getElementById("tableCVEs_Td_"+id+"_CVE").innerText + " (anomalie NVD)";
                                }
                                break;
                            case 2:
                                // récupération du score v2
                                tempBS = nvd[(nvd.length)-1];
                                nvd_return.set("v2_BaseScore",tempBS);
                                nvd_return.set("v2_Vector",cartouche_v2);
                                // récupération du score v3
                                tempBS = nvd[0];
                                if(cartouche_v3) {
                                    nvd_return.set("v3_BaseScore",tempBS);
                                    nvd_return.set("v3_Vector",cartouche_v3);
                                }else{
                                    nvd_return.set("v3_CNA_BaseScore",tempBS);
                                    nvd_return.set("v3_CNA_Vector",cartouche_v3_CNA);
                                }
                                break;

                            case 3:
                                // récupération du score v2
                                tempBS = nvd[2];
                                nvd_return.set("v2_BaseScore",tempBS);
                                nvd_return.set("v2_Vector",cartouche_v2);

                                // récupération du score v3
                                tempBS_NVD = nvd[0];
                                nvd_return.set("v3_BaseScore",tempBS_NVD);
                                nvd_return.set("v3_Vector",cartouche_v3);
                                tempBS_CNA = nvd[1];
                                nvd_return.set("v3_CNA_BaseScore",tempBS_CNA);
                                nvd_return.set("v3_CNA_Vector",cartouche_v3_CNA);

                                break;
                        }

                        if (nvd_DOM.length == 0) {
                            nvd_DOM = 'ØØ';
                        } else {
                            nvd_DOM = nvd_DOM.item(1).innerText.trim(); // item(0) == CVSS V3
                            nvd_DOM = nvd_DOM.split(' ')[0];
                            nvd_return.set("score",parseInt(nvd_DOM.replace(/\./g, ''),10)/10); // remplace le texte x.y en xy puis divise par 10 pour recuperer la version numérique
                        }
                        var score_NVD = '<a href=https://nvd.nist.gov/vuln/detail/'+id+' target="_blank">'+nvd_DOM+'</a>';
                        try{
                            document.getElementById(id+"_NVD").innerHTML = score_NVD;
                        }catch(error){
                            console.log("Erreur : chargement de "+id+"_NVD");
                        }

                        // remplissage du tableau
                        var prefixe="tableCVEs_Td_"+id


                        document.getElementById(prefixe+"_v2_score").innerText = nvd_return.get("v2_BaseScore");
                        document.getElementById(prefixe+"_v2_vector").value = nvd_return.get("v2_Vector");
                        document.getElementById(prefixe+"_v3_score").innerText = nvd_return.get("v3_BaseScore");
                        document.getElementById(prefixe+"_v3_vector").value = nvd_return.get("v3_Vector");
                        document.getElementById(prefixe+"_v3_CNA_score").innerText = nvd_return.get("v3_CNA_BaseScore");
                        document.getElementById(prefixe+"_v3_CNA_vector").value = nvd_return.get("v3_CNA_Vector");

                        document.getElementById(prefixe+"_NVDpublishDate").value = NVDpublishDate;
                        document.getElementById(prefixe+"_NVDmodifiedDate").value = NVDmodifiedDate;
                        document.getElementById(prefixe+"_NVDsource").value = NVDsource;
                        document.getElementById(prefixe+"_Editeur").title = NVDsource;
                        document.getElementById(prefixe+"_Editeur").innerText = NVDsource.substring(0,20);


                        // barre les rejected
                        if(NVDwarning=="Rejected") {
                            // la CVE est rejetée, on a barre
                            document.getElementById("tableCVEs_Tr_"+id).style.textDecoration  = "line-through"
                            document.getElementById("tableCVEs_Tr_"+id).title=NVDwarning;
                            try{
                                if(document.getElementById("li_CVE_"+id)) {
                                    console.log("Rejected : li_CVE_"+id)
                                    add_log_to_CMB("[loadScore]Rejected : li_CVE_"+id, 0);
                                    console.log(document.getElementById("li_CVE_"+id))
                                    document.getElementById("li_CVE_"+id).innerText += " (REJECTED)"
                                }
                            } catch (error){
                                console.log("Erreur ; pas de li");
                            }
                        }else if(NVDwarning) {
                            console.log("WARNING [get_NVD] "+id+" => NVD WARNING = "+NVDwarning);
                            add_log_to_CMB("WARNING [get_NVD] "+id+" => NVD WARNING = "+NVDwarning, 0);
                        }

                        // DEBUG
                        //console.log("[get_NVD] "+id);
                        //console.log("[get_NVD] "+id+" _v2_score = "+nvd_return.get("v2_BaseScore")+" ["+nvd_return.get("v2_Vector")+"] => "+document.getElementById("tableCVEs_Td_"+id+"_v2_score").innerText+" ["+document.getElementById("tableCVEs_Td_"+id+"_v2_vector").value+"]");
                        //console.log("[get_NVD] "+id+" _v3_score = "+nvd_return.get("v3_BaseScore")+" ["+nvd_return.get("v3_Vector")+"] => "+document.getElementById("tableCVEs_Td_"+id+"_v3_score").innerText+" ["+document.getElementById("tableCVEs_Td_"+id+"_v3_vector").value+"]");
                        //console.log("[get_NVD] "+id+" _v3_CNA_score = "+nvd_return.get("v3_CNA_BaseScore")+" ["+nvd_return.get("v3_CNA_Vector")+"] => "+document.getElementById("tableCVEs_Td_"+id+"_v3_CNA_score").innerText+" ["+document.getElementById("tableCVEs_Td_"+id+"_v3_CNA_vector").value+"]");

                        // colorisation
                        document.getElementById("tableCVEs_Td_"+id+"_CVE").style.color = "black";
                        if(nvd_return.get("v2_BaseScore")=== "N/A" || nvd_return.get("v2_BaseScore") === ""){
                            // Highlight les CVE sans résultats
                            document.getElementById("tableCVEs_Td_"+id+"_CVE").style.color = "red";
                            document.getElementById("tableCVEs_Td_"+id+"_CVE").style.fontWeight = "bold";
                        }

                        // colorisation en orange de l'entete du tableau des que fini
                        compteurCVEloaded = compteurCVEloaded+1;
                        if(compteurCVEloaded>= compteurCVEtoLoad){
                            document.getElementById("tableCVEs_THs").style.color = "orange";
                            if(intervalAnimation){
                                getMaxScroreFromListCVE(intervalAnimation);
                                document.getElementById("pProcessMessage").innerText = "Chargement des scores terminé, Selection du plus elevé"
                            }
                        }
                    },
                    onerror: function(error) {
                        console.log("get_NVD(error id="+id+")");
                        add_log_to_CMB("[get_NVD](error id="+id+")", 0);
                    }
                });
            } // loadScore
            // FONCTIONS\CVSS\externes />
            // FONCTIONS\CVSS />
        //}
        //if("FONCTION : GUI"){
            function removeIfNotAvalable(url, DOM_object){
                GM.xmlHttpRequest({
                    method: "GET",
                    url: url,
                    onload: function(response) {
                        //console.log(url+" : "+response.status);
                        try{

                            switch(response.status){
                                case 200:
                                    // ne rien faire
                                    break;
                                default:
                                    DOM_object.remove()
                            }
                        }catch(error){
                            console.log("[removeIfNotAvalable] ERREUR",error);
                        }
                    },
                    onerror: function(error) {
                        console.log("getURLresponseCode(error pour "+url);
                        add_log_to_CMB("[colorURLavailability]getURLresponseCode(error pour "+url, 0);
                    }
                });
            } // colorURLavailability
            function colorURLavailability(url, objectID, c200style="backgroundcolor:green;", c404style="backgroundcolor:darkgreen;text-decoration : line-through;", c400style="color:red;text-decoration : line-through;", c500style="color:orange;"){
                GM.xmlHttpRequest({
                    method: "GET",
                    url: url,
                    onload: function(response) {
                        //console.log(url+" : "+response.status);
                        try{
                            document.getElementById(objectID).setAttribute("title","response Code : "+response.status);
                            switch(response.status){
                                case 200:
                                    document.getElementById(objectID).style = c200style;
                                    break;
                                case 404:
                                    document.getElementById(objectID).style = c404style;
                                    break;
                                case 400:
                                    document.getElementById(objectID).style = c400style;
                                    break;
                                case 500:
                                    document.getElementById(objectID).style = c500style;
                                    break;
                            }
                        }catch(error){
                            console.log("[colorURLavailability] : Erreur colorisation lien");
                        }
                    },
                    onerror: function(error) {
                        console.log("getURLresponseCode(error pour "+url);
                        add_log_to_CMB("[colorURLavailability]getURLresponseCode(error pour "+url, 0);
                    }
                });
            } // colorURLavailability

            function getDOMitemFromitems_withASpecificAttribValue(items, attribute, value) {
                var itemOut
                for(var i=0;i<items.length;i++){
                    if((items[i]).getAttribute(attribute) == value) {
                        itemOut = items[i];
                        break;
                    }
                }
                return itemOut;
            }

            // FONCTIONS\graphique + alteration de page
            var chart_color_list_of_map;
            initialisation_svg_variables();
            function initialisation_svg_variables(){
                var chart_contour_grey = "#9c9c9c";
                var chart_inside_left_grey = "#efefef";
                var chart_inside_center_grey = "#f6f6f6";
                var chart_inside_right_grey = "#c9c9c9";

                var chart_contour_green = "#007f00";
                var chart_inside_left_green = "#b9eca3";
                var chart_inside_center_green = "#55d337";
                var chart_inside_right_green = "#00c000";

                var chart_contour_yellow = "#b69f0f";
                var chart_inside_left_yellow = "#fce700";
                var chart_inside_center_yellow = "#dac800";
                var chart_inside_right_yellow = "#c4b400";

                var chart_contour_orange = "#b47700";
                var chart_inside_left_orange = "#ffe6b4";
                var chart_inside_center_orange = "#ffc656";
                var chart_inside_right_orange = "#ffb012";

                var chart_contour_red = "#9a0808";
                var chart_inside_left_red = "#f7c5c5";
                var chart_inside_center_red = "#ee7c7c";
                var chart_inside_right_red = "#e94545";

                chart_color_list_of_map = [new Map(),new Map(),new Map(),new Map(),new Map()];
                // 0 : grey/undifened
                chart_color_list_of_map[0].set("chart_contour",chart_contour_grey);
                chart_color_list_of_map[0].set("chart_inside_left",chart_inside_left_grey);
                chart_color_list_of_map[0].set("chart_inside_center",chart_inside_center_grey);
                chart_color_list_of_map[0].set("chart_inside_right",chart_inside_right_grey);
                // 1 : green
                chart_color_list_of_map[1].set("chart_contour",chart_contour_green);
                chart_color_list_of_map[1].set("chart_inside_left",chart_inside_left_green);
                chart_color_list_of_map[1].set("chart_inside_center",chart_inside_center_green);
                chart_color_list_of_map[1].set("chart_inside_right",chart_inside_right_green);
                // 2 : yellow
                chart_color_list_of_map[2].set("chart_contour",chart_contour_yellow);
                chart_color_list_of_map[2].set("chart_inside_left",chart_inside_left_yellow);
                chart_color_list_of_map[2].set("chart_inside_center",chart_inside_center_yellow);
                chart_color_list_of_map[2].set("chart_inside_right",chart_inside_right_yellow);
                // 3 : orange
                chart_color_list_of_map[3].set("chart_contour",chart_contour_orange);
                chart_color_list_of_map[3].set("chart_inside_left",chart_inside_left_orange);
                chart_color_list_of_map[3].set("chart_inside_center",chart_inside_center_orange);
                chart_color_list_of_map[3].set("chart_inside_right",chart_inside_right_orange);
                // 4 : red
                chart_color_list_of_map[4].set("chart_contour",chart_contour_red);
                chart_color_list_of_map[4].set("chart_inside_left",chart_inside_left_red);
                chart_color_list_of_map[4].set("chart_inside_center",chart_inside_center_red);
                chart_color_list_of_map[4].set("chart_inside_right",chart_inside_right_red);
            }

            function set_img_png_src(index_color_set=0,img_id){ // dans à la source la valeur base64 de l'image cible
                var liste_img = [
                    "iVBORw0KGgoAAAANSUhEUgAAABEAAAAOCAYAAADJ7fe0AAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAFVSURBVDhPnVPLisJAECzH5KBoIh78gJzFS07ehJCzCOsX+TNqELwJ+R4RokEhIj5i4m61iiS4rm5B09MznZqerk7h8gO8idPphNFohMPhAF3XwU+DIMDHJJPJBK1WC5ZlyR5jJatfcD6fEYYhlsulGNckms/n4lmNUup1JVEUYTweXxMLCmmaYL1eo9lsot1uo1KpXM9v+S/R6XTQ73+h1+uhVqvddh/IkLB83n633W6HOI6xWCxQLBZRKpXE55Eh2Ww28DwP0+lUbDabCVGSJLeM58iQsD2apsF1XXS7XTiOI837C4qa340dp6cS5XIZ1Wr1afl5qOFwKFrTfN8XkuPxKFW9C3kOu8/SbdsWOT+F2u/3cnuj0UC9Xv8fCWXdbrciJVXgM+gZ8ywfE2maSkwjCoPB4MIB4vRxk03l2jAMSeYPxvlgzCpXq5V40zRFuTAM8Q28m9+YaHoqhgAAAABJRU5ErkJggg=="
                    , "iVBORw0KGgoAAAANSUhEUgAAABEAAAAOCAYAAAG+6sciAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAAC4cAAAuHAZNAh1MAAACxSURBVChTnVHRFYMwCGSrOkg7T/9cxC5Rv7qDazhDYiAcJqSp2nuPl3A5DkQyPCnuF0tGvlQMA4SQZWJkCfFQhBAiR5UIcV8GS4ToIjUQQdWIm3AUj7voC9mKeoBzD/P6yi4eTHK0ArWc10kF+dRXorcS/mws/SmV9KGm0hwuQ3eE+L0rADtDpCJMcN5EivJ3ocjnqiyQSAggwnJ6+alxj3Idd4qP5SZ/4D8TP8nloLgBBWq+PZle8HYAAAAASUVORK5CYII="
                    , "iVBORw0KGgoAAAANSUhEUgAAABEAAAAOCAYAAAG+6sciAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAAC4cAAAuHAZNAh1MAAACySURBVChTlZIxDoMwDEV9g3JkbtKZHbFX6lD1CFVHJA7AWsnkW/lp0iQleZIBO/a3sSJkuV40fATnNg2aRgADFoydEIwxDaIec97P4Ku8HqIwYhkl0AAkjdAEFh/ynVSgZzGpBpW9m7Ovo6l494tFHVkCJXmQJXw20fsseYI9HdUEHlQTeuGOyN9dEf4gDUWcADSJlIpORRCMaRIpjYsF4q5giU0iLMIdZtGv3z1Jvw16ANc99saVhXigAAAAAElFTkSuQmCC"
                    , "iVBORw0KGgoAAAANSUhEUgAAABEAAAAOCAYAAAG+6sciAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAAC4cAAAuHAZNAh1MAAACxSURBVChTlVLBDcIwDMwMDIfEJPyYiUffPMsmLIGQm3Nzlu2moT3plNi+nJ0ohZjuRWxjwetRNyEDMKFJH1jSQz0I+X0FXIP3TZRIyHwVIxU9oAEEoRGagFqsDkHkA/QLa3bKoHMLt5DP1B8YSWUW0FILz8tWEApeECy7gl7BC86Cb0QO34rgBUk1aRMcNrFD+CH1CodMVNQEFOkh/Ko9k91xc+fRJJbMnf7Fo0nOs8gCgczit7YoWnoAAAAASUVORK5CYII="
                    , "iVBORw0KGgoAAAANSUhEUgAAABEAAAAOCAYAAAG+6sciAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAAC4cAAAuHAZNAh1MAAAC1SURBVChTlVHBDYQwDOPZxRgKCTEIz9uFYfhwz/sgAu7VoSRQwFJo47hxaCuiD0F0o8kHmwMDkIhkniiZI/YglnmWb9P8ianrhFFNbSt5RMUZYADBwQgmiLyoInxgHMnN62xNfTzYOaUev2GQsa69AJYIJ2BLFpzAFlRgWzrBVUEFb8G7YxTviuAPMnAIE+BFMMWjJnpoeyEesnmS7gBpx7XObpLLcQvObhIlC053uZvkfQRZAbjJlL0mcKqVAAAAAElFTkSuQmCC"
                ];
                var liste_img_portail = [
                    "images/VULNERABILITES/gris.png"
                    ,"images/VULNERABILITES/vert.png"
                    ,"images/VULNERABILITES/jaune.png"
                    ,"images/VULNERABILITES/orange.png"
                    ,"images/VULNERABILITES/rouge.png"
                ];
                var prefixe = "data:image/png;base64,";
                var index = (index_color_set>=liste_img.length || index_color_set<0 ? 0 : index_color_set);
                document.getElementById(img_id).src = prefixe+liste_img[index];
                document.getElementById(img_id).setAttribute("src_portail",liste_img_portail[index]);
            }

            function change_color(index_color_set=0,svg_id) {
                // declaration de variables
                var items, index, item, max_strips_count = 0 ;
                var color_set = chart_color_list_of_map[(index_color_set>=0 && index_color_set<chart_color_list_of_map.length ? index_color_set : 0 )];

                //console.log(color_set);
                color_set.forEach((color_value, class_name)=>{
                    items = document.getElementById(svg_id).getElementsByClassName(class_name);
                    //console.log(class_name+" : "+color_value + " ; "+index_color_set);
                    //console.log(contours);
                    max_strips_count = (index_color_set>=items.length || index_color_set<=0 ? items.length : index_color_set);
                    for (index = 0; index < max_strips_count; index++) {
                        item = items[index];
                        if((class_name.toLowerCase()).includes("contour")){
                            //console.log(item.getAttribute('stroke') + " couleur devient : "+color_value);
                            item.setAttribute('stroke', color_value);
                        }else{
                            //console.log(item.getAttribute('fill') + " couleur devient : "+color_value);
                            item.setAttribute('fill', color_value);
                        }
                        //console.log(item);
                    }
                    // blanchir le reste et continuer avec le contour
                    for (index = max_strips_count; index < items.length; index++) {
                        item = items[index];
                        if((class_name.toLowerCase()).includes("contour")){
                            item.setAttribute('stroke', color_value);
                        }else{
                            item.setAttribute('fill', "white");
                        }
                        //console.log(item);
                    }
                },index_color_set)
            }

            function svg_to_img(svg_id, img_id){
                //console.log("cration de l'image");
                var svgElement = document.getElementById(svg_id);

                // Create your own image
                //var img = document.createElement('img');
                var img = document.getElementById(img_id);
                // Serialize the svg to string
                var svgString = new XMLSerializer().serializeToString(svgElement);
                //console.log("======================== av svgString");
                //console.log(svgString);

                // Remove any characters outside the Latin1 range
                var decoded = unescape(encodeURIComponent(svgString));
                //console.log("======================== av decoded");
                //console.log(decoded);

                // Now we can use btoa to convert the svg to base64
                var base64 = btoa(decoded);
                //console.log("taille base64 : "+base64.length);
                //console.log(base64);

                var imgSource = `data:image/svg+xml;base64,${btoa(decoded)}`;

                img.setAttribute('src', imgSource);
                //document.getElementById(svg_id).after(img);

            }

    function getParentElementFrom(DOM_object){
        let DOM_parent = null;
        try{ // parentElement
            DOM_parent = DOM_object.parentElement;
        }catch{}
        return DOM_parent;
    }
    function getParentNodeFrom(DOM_object){
        let DOM_parent = null;
        try{ // parentElement
            DOM_parent = DOM_object.parentNode;
        }catch{}
        return DOM_parent;
    }
    function getParent(DOM_object){
        // déclaration des variables
        let DOM_parent=null;
        // test ELEMENT
        DOM_parent = getParentElementFrom(DOM_object)
        // test node
        if(DOM_parent==null){DOM_parent = getParentNodeFrom(DOM_object)}
        return DOM_parent;
    }
    function getFirstParent_as(DOM_object, search="div"){ // retourne le premier contenant de type search
        let DOM_parent=getParent(DOM_object);
        if(DOM_parent==null){return DOM_parent;}

        let DOM_parent_type = DOM_parent.nodeName.toLowerCase();
        search = search.toLowerCase();
        if(search == DOM_parent_type){
            return DOM_parent;
        }else{
            return getFirstParent_as(DOM_parent, search);
        }
    }

            function getLastItemOfClass(className, debug=false){
                let lastItem = null;
                if(className!=""){ // il y a une classe à tester
                    let classes = document.getElementsByClassName(className);
                    if(classes.length>0){
                        let toolbox = classes[classes.length-1] // on recupère le dernier element de la classe
                        let ul = toolbox.lastElementChild;
                        if(ul){
                            lastItem = ul.lastElementChild;
                            if(debug){
                                console.log("lastItem = ");
                                console.log(lastItem);
                            }
                        }
                    }
                }
                return lastItem;
            }

            function getClassesForToolBox(index=0){
                let classeName="";
                let classesName=["toolbox", "row-toolbox"];
                if(index<classesName.length && index>=0){
                    classeName=classesName[index];
                }
                return classeName;
            }
            function getLastItemInTheToolBox(indexListClassName=0, debug=false){
                let className=getClassesForToolBox(indexListClassName);
                let lastItem = document.querySelector('body');
                if(className!=""){ // il y a une classe à tester
                    lastItem = getLastItemOfClass(className, debug);
                    if(lastItem==null){// pas d'elements trouvés
                        return getLastItemInTheToolBox(indexListClassName+1, debug);
                    }
                }
                return lastItem;
            }
            function make_download_txt(){
                var doc_title = window.location.href;
                doc_title = doc_title.split("/");
                doc_title = doc_title[4];
                document.title = doc_title;
                var newLi = document.createElement('li');
                newLi.class = "link-txt";
                newLi.id = "link-txt-btn";
                var newA = document.createElement('a');
                newA.id = "txt_gen";
                newA.title = "Télécharger au format TXT";
                var newI = document.createElement('i');
                newI.setAttribute("class","fa fa-fw fa-clipboard");
                //newI.class = "fa fa-fw fa-clipboard";
                newI.setAttribute('aria-hidden',"true");
                newA.appendChild(newI);
                newLi.appendChild(newA);


                (getLastItemInTheToolBox()).before(newLi);
            }

    function evalOnAVI(actualAVI, toevaluate="+0"){
        let splitted=actualAVI.split("-");
        let suffix=splitted.slice(0, splitted.length-1).join("-");
        let AVI_ID=splitted[splitted.length-1]
        let zeros=""
        while(AVI_ID[0]=="0"){
            zeros=zeros+"0";
            AVI_ID=AVI_ID.substring(1);
            if(AVI_ID.length<=0){break;} // on quitte la boucle avant de creer une erreur
        }
        let tailleOldNumber=AVI_ID.length;
        let newNumber = eval(`${AVI_ID}${toevaluate}`)
        let toReturn="";
        if(newNumber==0){ // TODO il faudra chercher le précedent de l'année passée
        }else{
            let tailleNewNumber = newNumber.toString().length
            console.log(tailleNewNumber,'?>',tailleOldNumber)
            if(tailleNewNumber>tailleOldNumber){ // il faut retirer un zero
                zeros=zeros.substring(1)
            }else if(tailleNewNumber<tailleOldNumber){// il faut ajouter un zero
                zeros=zeros+"0"
            }
            toReturn = suffix+"-"+zeros+newNumber
        }
        return toReturn
    }
    function getNextAVI(actualAVI){
        return evalOnAVI(actualAVI,"+1");
    } // a ameliorer en testant aux passages des années
    function getPreviousAVI(actualAVI){
        return evalOnAVI(actualAVI,"-1");
    }

            function make_download_html(){
                var itemID = "link-html-btn"
                var doc_title = window.location.href;
                doc_title = doc_title.split("/");
                let suffixLink = doc_title.slice(0, doc_title.length-2).join("/");
                doc_title = doc_title[4];
                document.title = doc_title;


                // recupération du dernier element
                let lastToolBoxButton = getLastItemInTheToolBox();
                let firtParentNodeName="";
                let firstParent=lastToolBoxButton;
                while(firtParentNodeName!="div" && firtParentNodeName!="ul" && firtParentNodeName!="ol" && firtParentNodeName!="body" && firtParentNodeName!=null){
                    firstParent = getParent(firstParent);
                    firtParentNodeName = firstParent.nodeName.toLowerCase()
                }
                // ajout des classes des siblings
                    let lst_classes = firstParent.classList
                // normalement notre parent est un div ou un UL, worst scenario c'est body ou html (pour arreter la boucle)
                let itemToAppend_a = null;
                let itemToAppend_a_next = null;
                let itemToAppend_a_previous = null;
                if(firtParentNodeName == "ul" || firtParentNodeName == "ol"){// il s'agit d'une liste de li
                    itemToAppend_a = document.createElement('li');
                    itemToAppend_a_next = document.createElement('li');
                    itemToAppend_a_previous = document.createElement('li');
                }else if(firtParentNodeName=="div"){

                    itemToAppend_a = document.createElement('div');
                    itemToAppend_a_next = document.createElement('div');
                    itemToAppend_a_previous = document.createElement('div');
                    lastToolBoxButton=getParent(lastToolBoxButton);
                }
                itemToAppend_a.classList.add("link-html");
                for(let i=0;i<lst_classes.length;i++){
                    itemToAppend_a.classList.add(lst_classes[i]);
                    itemToAppend_a_next.classList.add(lst_classes[i]);
                    itemToAppend_a_previous.classList.add(lst_classes[i]);
                }
                itemToAppend_a.id = itemID;

                // ajout precedent
                let prec = document.createElement('a');
                prec.innerText = "<="
                prec.style.cursor="pointer"
                prec.href=suffixLink+"/"+getPreviousAVI(doc_title)
                itemToAppend_a_previous.appendChild(prec);

                var newA = document.createElement('a');
                newA.id = "html_gen";
                newA.title = "Télécharger au format html";
                newA.style.cursor="pointer"
                var newImg = document.createElement('img');
                newImg.src = " ";
                newImg.width = 18
                newImg.heigh=21
                newA.appendChild(newImg);
                itemToAppend_a.appendChild(newA);

                // ajout suivant
                let next = document.createElement('a');
                next.innerText = "=>"
                next.style.cursor="pointer"
                next.href=suffixLink+"/"+getNextAVI(doc_title)
                itemToAppend_a_next.appendChild(next);


                lastToolBoxButton.before(itemToAppend_a);
                itemToAppend_a.before(itemToAppend_a_previous)
                itemToAppend_a.after(itemToAppend_a_next)
                // suppression des boutons si la page desirée n'existe pas
                removeIfNotAvalable(prec.href, itemToAppend_a_previous)
                removeIfNotAvalable(next.href, itemToAppend_a_next)
                return itemID;
            }

            function make_html_from_list_of_dict(item_before=document.querySelector('body'), list_input, debug=false) {
                var item,dict_item, value, key, parent_id, has_parent, i, j, classes, attributes, DOMparent;
                try{
                // fonctions principales : parcourir le tableau d'objet à creer
                    for(i=0 ; i<list_input.length ; i++) {
                        dict_item = list_input[i];
                        item = document.createElement(dict_item["object"]);
                        has_parent = false;
                        for(key in dict_item){
                            value = dict_item[key];
                            //console.log (key + " : "+value);
                            switch(key){
                                case "strokewidth" :
                                    key = "stroke-width";
                                case "x" :
                                case "y" :
                                case "stroke" :
                                case "fill" :
                                case "viewBox":
                                case "width":
                                case "height":
                                    item.setAttribute(key,value);
                                    break;
                                case "class" :
                                    classes = value.split(" ");
                                    for(j=0;j<classes.length;j++){
                                        item.classList.add(classes[j]);
                                    }
                                    break;
                                case "standaloneAttributes":
                                    attributes = value.split(" ");
                                    for(j=0;j<attributes.length;j++){
                                        item[attributes[j]] = true;
                                    }
                                    break;
                                case "value":
                                    item.value = value;
                                    break;
                                case "parent_id":
                                    parent_id = value;
                                    has_parent = true;
                                    //console.log("le parent : "+value);console.log(parent_id);
                                    break;
                                case "object":
                                    break;
                                case "appendChildText" :
                                    item.appendChild(document.createTextNode(value));
                                    break;
                                default :
                                    item.setAttribute(key,value);
                                    //item[key] = value;
                                    break;
                            }
                        }
                        //console.log(item);
                        if(debug) {
                            //console.log("object : "+dict_item["object"]+" ; parent : "+parent_id+" ; id : "+dict_item["id"]);
                            console.log("before : ")
                            console.log(item_before)
                            console.log("look : "+parent_id+" for ");
                            console.log(item);
                        }
                        if(has_parent){
                            if(debug) {
                                console.log("has parent : ")
                                console.log(document.getElementById(parent_id))
                                console.log("...")
                            }
                            DOMparent = document.getElementById(parent_id);
                            DOMparent.appendChild(item);
                        } else {
                            $(item_before).after(item);
                        }
                        item_before = item;
                    }
                }catch(error){
                    console.error("[make_html_from_list_of_dict] ERROR : ",error.message);
                }

            }
            function onlyUnique(value, index, self) { // fonction à utiliser depuis un filtre de tableau pour recuperer que les elements uniques
                // var unique = tableauAunifier.filter(onlyUnique);
                return self.indexOf(value) === index;
            }

            function getGUIinputValue(searchFor="", defaultValue=""){
                var returnValue = defaultValue;
                try{
                    switch(searchFor){
                        case "checkbox_stocker":
                            returnValue = document.getElementById("boxToGMstorage").checked;
                            break;
                        default:
                            break;
                    }
                }catch (error){
                    console.log("[getGUIinputValue] ERROR :"+error.message);
                    add_log_to_CMB("[getGUIinputValue] ERROR :"+error.message);
                }
                return returnValue;
            }

            var cveMaxScoreV2 = "";
            var cveMaxScoreV3 = "";
            var cveMaxScoreV3_type = "";
            var cveMaxScoreV3_NVD = "";
            var cveMaxScoreV3_CNA = "";

            var colorV2 = "rgb(250, 171, 239)";
            var colorV3_NVD = "rgb(124, 124, 226)" ;
            var colorV3_CNA = "rgb(103, 218, 232)" ;
            var colorMixedV2V3_NVD = "rgb(187, 148, 233)" ;
            var colorMixedV2V3_CNA = "rgb(177, 195, 236)" ;
            async function getMaxScroreFromListCVE(intervalAnimation, debug=false){
                // fonction qui récupère les scores v2 et v3 les plus elevés parmis le tableau des CVE
                // les id des tds sont composés comme suit :
                // CVE : tableCVEs_Td_<CVE id>_CVE
                // v2 : tableCVEs_Td_<CVE id>_v2\tableCVEs_Td_<CVE id>_v2_score
                // 		tableCVEs_Td_<CVE id>_v2\tableCVEs_Td_<CVE id>_v2_vecteur
                // v3 : tableCVEs_Td_<CVE id>_v2\tableCVEs_Td_<CVE id>_v3_score
                // 		tableCVEs_Td_<CVE id>_v2\tableCVEs_Td_<CVE id>_v3_vecteur
                // v3 CNA : tableCVEs_Td_<CVE id>_v3\tableCVEs_Td_<CVE id>_v3_CNA_score
                // 			tableCVEs_Td_<CVE id>_v3\tableCVEs_Td_<CVE id>_v3_CNA_vecteur

                // initialisation de variables
                var fctName = "getMaxScroreFromListCVE";
                var td_prefixe = "tableCVEs_Td_";
                var row, cveID, td_cve, td_v2, score_v2, td_v3, score_v3, td_v3_CNA, score_v3_CNA;
                var cveMaxScoreV2_score, cveMaxScoreV3_score, cveMaxScoreV3_CNA_score;
                var v2Source = "";
                var v3Source = "";
                var aviV2, aviV3, vecteurFromStorage, scoreFromStorage;
                var dictToDisplay=0;


                // Parcours du tableau
                //console.log("["+fctName+"]"+"\n########################\n");
                //console.log("["+fctName+"]"+" Parcours du tableau");
                var table = document.getElementById("tableCVEs");
                var trs = table.querySelectorAll("tr");
                for(var i=1;i<trs.length;i++){
                    // console.log(trs[i]);
                    row=trs[i].id;
                    // console.log(row);
                    // TR example : tableCVEs_Tr_CVE-2022-31143
                    cveID = row.substring(13); //
                    // TD example  : tableCVEs_Td_CVE-2022-31143_CVE
                    td_cve = document.getElementById(td_prefixe+cveID+"_CVE");
                    td_v2 = document.getElementById(td_prefixe+cveID+"_v2");
                    score_v2 = scoreString_to_float(td_v2.innerText) ;
                    td_v3 = document.getElementById(td_prefixe+cveID+"_v3");
                    score_v3 = scoreString_to_float(td_v3.innerText);
                    td_v3_CNA = document.getElementById(td_prefixe+cveID+"_v3_CNA");
                    score_v3_CNA = scoreString_to_float(td_v3_CNA.innerText);


                    if(cveID==WatchForDEBUG_CVE){
                        console.log("["+fctName+"]"+" "+cveID);
                        console.log("["+fctName+"]"+" score_v2="+score_v2+" VS CVE v2 max : "+cveMaxScoreV2+" ("+cveMaxScoreV2_score+")");
                        console.log("["+fctName+"]"+" score_v3="+score_v3+" VS CVE v3 max : "+cveMaxScoreV3_NVD+" ("+cveMaxScoreV3_score+")");
                        console.log("["+fctName+"]"+" score_v3_CNA="+score_v3_CNA+" VS CVE v3 CNA max : "+cveMaxScoreV3_CNA+" ("+cveMaxScoreV3_CNA_score+")");

                    }
                    // comparaison v2
                    if(!cveMaxScoreV2){
                        // c'est la premiere valeur
                        cveMaxScoreV2 = cveID;
                        cveMaxScoreV2_score = score_v2;
                    }else{
                        // cveMaxScoreV2_score = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV2+"_v2").innerText);
                        // cas particulier : La premiere CVE etait choisie automatiquement mais sans valeurs, mais les suivantes aussi hors leur scrore proviendrai du stockage
                        score_v2 = getStorageScore_if_ActualScoreIsUnkown(cveID, score_v2, "CVEv2");
                        //console.log("["+fctName+"]"+" ap stockage : score_v2="+score_v2+" param("+cveID+","+score_v2+",CVEv2)");

                        cveMaxScoreV2_score = getStorageScore_if_ActualScoreIsUnkown(cveMaxScoreV2, cveMaxScoreV2_score, "CVEv2");
                        //console.log("["+fctName+"]"+" ap stockage : cveMaxScoreV2_score="+cveMaxScoreV2_score+" param("+cveMaxScoreV2+","+cveMaxScoreV2_score+",CVEv2)");

                        if(score_v2>cveMaxScoreV2_score){
                            //console.log("["+fctName+"]"+" ap stockage : cveMaxScoreV2_score="+cveMaxScoreV2_score+"   score_v2>cveMaxScoreV2_score");
                            cveMaxScoreV2 = cveID;
                            cveMaxScoreV2_score = score_v2;
                        }
                    }
                    // comparaison v3
                    if(!cveMaxScoreV3_NVD){
                        // c'est la premiere valeur
                        cveMaxScoreV3_NVD = cveID;
                        cveMaxScoreV3_score = score_v3;
                    }else{
                        // cveMaxScoreV3_score = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_NVD+"_v3").innerText);
                        // cas particulier : La premiere CVE etait choisie automatiquement mais sans valeurs, mais les suivantes aussi hors leur scrore proviendrai du stockage
                        score_v3 = getStorageScore_if_ActualScoreIsUnkown(cveID, score_v3, "CVEv3");
                        //console.log("["+fctName+"]"+" ap stockage : score_v3="+score_v3+" param("+cveID+","+score_v3+",CVEv3)");

                        cveMaxScoreV3_score = getStorageScore_if_ActualScoreIsUnkown(cveMaxScoreV3_NVD, cveMaxScoreV3_score, "CVEv3");
                        //console.log("["+fctName+"]"+" ap stockage : cveMaxScoreV3_score="+cveMaxScoreV3_score+" param("+cveMaxScoreV3_NVD+","+cveMaxScoreV3_score+",CVEv3)");

                        if(score_v3>cveMaxScoreV3_score){
                            //console.log("["+fctName+"]"+" ap stockage : cveMaxScoreV3_score="+cveMaxScoreV3_score+"   score_v3>cveMaxScoreV3_score");
                            cveMaxScoreV3_NVD = cveID;
                            cveMaxScoreV3_score = score_v3;
                        }
                    }
                    // comparaison v3_CNA
                    if(!cveMaxScoreV3_CNA){
                        // c'est la premiere valeur
                        cveMaxScoreV3_CNA = cveID;
                        cveMaxScoreV3_CNA_score = score_v3_CNA;
                    }else{
                        // cveMaxScoreV3_CNA_score = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_CNA+"_v3_CNA").innerText);
                        // cas particulier : La premiere CVE etait choisie automatiquement mais sans valeurs, mais les suivantes aussi hors leur scrore proviendrai du stockage
                        //score_v3_CNA = getStorageScore_if_ActualScoreIsUnkown(cveID, score_v3_CNA, "CVEv3");
                        //console.log("["+fctName+"]"+" ap stockage : score_v3_CNA="+score_v3_CNA+" param("+cveID+","+score_v3_CNA+",CVEv3)");

                        //cveMaxScoreV3_CNA_score = getStorageScore_if_ActualScoreIsUnkown(cveMaxScoreV3_CNA, cveMaxScoreV3_CNA_score, "CVEv3");
                        //console.log("["+fctName+"]"+" ap stockage : cveMaxScoreV3_CNA_score="+cveMaxScoreV3_CNA_score+" param("+cveMaxScoreV3_CNA+","+cveMaxScoreV3_CNA_score+",CVEv3)");

                        if(score_v3_CNA>cveMaxScoreV3_CNA_score){
                            //console.log("["+fctName+"]"+" ap stockage : cveMaxScoreV2_score="+cveMaxScoreV2_score+"   score_v3_CNA>cveMaxScoreV3_CNA_score");
                            cveMaxScoreV3_CNA = cveID;
                            cveMaxScoreV3_CNA_score = score_v3_CNA;
                        }
                    }
                }

                // unification v2, v3 et v3_CNA si possible
                if(debug){
                    //console.log("["+fctName+"]"+"\n########################\n");
                    //console.log("["+fctName+"]"+" Résultat AVANT unification");
                    // console.log("avant unification : "+cveMaxScoreV2+" ; "+cveMaxScoreV3_NVD+" ; "+cveMaxScoreV3_CNA);
                }
                cveMaxScoreV2_score = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV2+"_v2").innerText);
                cveMaxScoreV3_score = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_NVD+"_v3").innerText);
                cveMaxScoreV3_CNA_score = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_CNA+"_v3_CNA").innerText);

                if(debug){
                    //console.log("["+fctName+"]"+" cveMaxScoreV2="+cveMaxScoreV2+" ("+cveMaxScoreV2_score+")");
                    //console.log("["+fctName+"]"+" cveMaxScoreV3_NVD="+cveMaxScoreV3_NVD+" ("+cveMaxScoreV3_score+")");
                    //console.log("["+fctName+"]"+" cveMaxScoreV3_CNA="+cveMaxScoreV3_CNA+" ("+cveMaxScoreV3_CNA_score+")");
                    // console.log("v2 = "+cveMaxScoreV2+"("+cveMaxScoreV2_score+") ; v3 = "+cveMaxScoreV3_NVD+"("+cveMaxScoreV3_score+") ; v3_CNA = "+cveMaxScoreV3_CNA+"("+cveMaxScoreV3_CNA_score+")");
                    console.log("av garnissage dictToDisplay");
                    console.log(dictToDisplay);
                }
                // pour des raisons de simplification de debuggage, le stockage ne sera ajouté que si le score n'est pas connu autrement
                // from cveMaxScoreV2
                var score_v2_from_cveMaxScoreV2 = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV2+"_v2").innerText);
                var score_v3n_from_cveMaxScoreV2 = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV2+"_v3").innerText);
                var score_v3c_from_cveMaxScoreV2 = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV2+"_v3_CNA").innerText);
                var score_max_between_v3n_and_v3c_from_cveMaxScoreV2 = Math.max(score_v3n_from_cveMaxScoreV2, score_v3c_from_cveMaxScoreV2);
                // from cveMaxScoreV3 NVD
                var score_v2_from_cveMaxScoreV3n = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_NVD+"_v2").innerText);
                var score_v3n_from_cveMaxScoreV3n = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_NVD+"_v3").innerText);
                var score_v3c_from_cveMaxScoreV3n = scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_NVD+"_v3_CNA").innerText);
                var score_max_between_v3n_and_v3c_from_cveMaxScoreV3n = Math.max(score_v3n_from_cveMaxScoreV3n, score_v3c_from_cveMaxScoreV3n);
                dictToDisplay = {	"from cveMaxScoreV2":{
                    "CVE":cveMaxScoreV2,
                    "v2":score_v2_from_cveMaxScoreV2,
                    "v2 S":getStorageScore_if_ActualScoreIsUnkown(cveMaxScoreV2, score_v2_from_cveMaxScoreV2, "CVEv2"),
                    "v3 N":score_v3n_from_cveMaxScoreV2,
                    "v3 C":score_v3c_from_cveMaxScoreV2,
                    "v3 S":getStorageScore_if_ActualScoreIsUnkown(cveMaxScoreV2, score_max_between_v3n_and_v3c_from_cveMaxScoreV2, "CVEv3")
                },"from cveMaxScoreV3 NVD":{
                    "CVE":cveMaxScoreV3_NVD,
                    "v2":score_v2_from_cveMaxScoreV3n,
                    "v2 S":getStorageScore_if_ActualScoreIsUnkown(cveMaxScoreV3_NVD, score_v2_from_cveMaxScoreV3n, "CVEv2"),
                    "v3 N":score_v3n_from_cveMaxScoreV3n,
                    "v3 C":score_v3c_from_cveMaxScoreV3n,
                    "v3 S":getStorageScore_if_ActualScoreIsUnkown(cveMaxScoreV3_NVD, score_max_between_v3n_and_v3c_from_cveMaxScoreV3n, "CVEv3")
                },"from cveMaxScoreV3 CNA":{
                    "CVE":cveMaxScoreV3_CNA,
                    "v2":scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_CNA+"_v2").innerText),
                    "v2 S":-1,
                    "v3 N":scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_CNA+"_v3").innerText),
                    "v3 C":scoreString_to_float(document.getElementById(td_prefixe+cveMaxScoreV3_CNA+"_v3_CNA").innerText),
                    "v3 S":-1
                },"cveMaxScoreV2":{
                    "CVE":"",
                    "Vecteur":"",
                    "Score":-1,
                    "source":""
                },"cveMaxScoreV3":{
                    "CVE":"",
                    "Vecteur":"",
                    "Score":"",
                    "type":"",
                    "source":""
                }
                                };
                if(debug){
                    console.log("av dictToDisplay");
                    console.log(dictToDisplay);
                    console.log("ap dictToDisplay");
                    //console.log("["+fctName+"]"+" Unification via le dict");
                    //console.log("["+fctName+"]"+" -- Get max V2");
                }
                var listToCompare = ["from cveMaxScoreV2","from cveMaxScoreV3 NVD","from cveMaxScoreV3 CNA"];
                var egalités_v2=[]
                var oldScore_v2 = -2, currentScore;
                var old_Titre_v2="",currentTitre;

                var versionCom = "v2";
                var Origines=[""," S"];
                var oldOrigine_v2 = "", currentOrigine;// "S" ou ""
                for(i=0;i<listToCompare.length;i++){
                    currentTitre = listToCompare[i];
                    for(var j=0;j<listToCompare.length;j++){
                        currentOrigine = versionCom+Origines[j];
                        currentScore = dictToDisplay[currentTitre][currentOrigine];
                        if(currentScore>oldScore_v2){
                            oldScore_v2 = currentScore;
                            old_Titre_v2 = currentTitre;
                            oldOrigine_v2 = currentOrigine;
                            egalités_v2=[currentTitre]
                        }else if(currentScore==oldScore_v2){
                            egalités_v2.push(currentTitre)
                        }
                    }
                }

                //console.log("["+fctName+"]"+" -- Get max V3");
                var egalités_v3=[];
                var oldScore_v3 = -2;
                var old_Titre_v3="";

                versionCom = "v3";
                Origines=[" N", " C", " S"];
                var oldOrigine_v3 = ""
                for(i=0;i<listToCompare.length;i++){
                    currentTitre = listToCompare[i];
                    //console.log("# currentTitre="+currentTitre);
                    for(j=0;j<listToCompare.length;j++){
                        currentOrigine = versionCom+Origines[j];
                        //console.log("## currentOrigine="+currentOrigine);
                        currentScore = dictToDisplay[currentTitre][currentOrigine];
                        //console.log("### compare : currentScore>oldScore_v3 => "+currentScore+">"+oldScore_v3);
                        if(currentScore>oldScore_v3){
                            //console.log("#### currentScore>oldScore_v3");
                            oldScore_v3 = currentScore;
                            old_Titre_v3 = currentTitre;
                            oldOrigine_v3 = currentOrigine;
                            egalités_v3=[currentTitre]
                        }else if(currentScore==oldScore_v3){
                            egalités_v3.push(currentTitre)
                        }
                    }
                }

                egalités_v2 = egalités_v2.filter(onlyUnique)
                egalités_v3 = egalités_v3.filter(onlyUnique)


                //console.log("["+fctName+"]"+" -- Found v2");
                //console.log("["+fctName+"]"+" -- old_Titre_v2="+old_Titre_v2+", oldOrigine_v2="+oldOrigine_v2+", oldScore_v2="+oldScore_v2);
                //console.log(egalités_v2);
                //console.log("["+fctName+"]"+" -- Found v3");
                //console.log("["+fctName+"]"+" -- old_Titre_v3="+old_Titre_v3+", oldOrigine_v3="+oldOrigine_v3+", oldScore_v3="+oldScore_v3);
                //console.log(egalités_v3);

                //console.log("["+fctName+"]"+" -- application des résultats");
                var intersected_Arrays_egalite = egalités_v2.filter(value => egalités_v3.includes(value));
                var score, scoreS, scoreC, scoreN, CVEid, avi;
                //console.log("intersected_Arrays_egalite");
                //console.log(intersected_Arrays_egalite);
                //console.log("egalités_v2");
                //console.log(egalités_v2);
                //console.log("egalités_v3");
                //console.log(egalités_v3);
                if(intersected_Arrays_egalite.length==0){ // pas d'intersection, on prends le premier item de chaque tableau comme valeur de référence
                    if(egalités_v2.length>0){ // il y a au moins un element v2
                        currentTitre = egalités_v2[0]
                        CVEid = dictToDisplay[currentTitre]["CVE"];
                        dictToDisplay["cveMaxScoreV2"]["CVE"] = CVEid;
                        score = dictToDisplay[currentTitre]["v2"];
                        scoreS = dictToDisplay[currentTitre]["v2 S"];


                        if(scoreS>score){
                            avi = getCVSS_fromStorage(CVEid, "CVEv2");
                            dictToDisplay["cveMaxScoreV2"]["source"]=getSourceV2fromStorageAVI(avi);
                            dictToDisplay["cveMaxScoreV2"]["Vecteur"]=getVecteurV2fromStorageAVI(avi);
                            dictToDisplay["cveMaxScoreV2"]["Score"]=scoreS;
                            (document.getElementById('tableCVEs_Td_'+CVEid+'_v2_vector')).value = dictToDisplay["cveMaxScoreV2"]["Vecteur"];
                        }else{
                            dictToDisplay["cveMaxScoreV2"]["source"]="";
                            dictToDisplay["cveMaxScoreV2"]["Vecteur"]=(document.getElementById(td_prefixe+CVEid+"_v2_vector")).value;
                            dictToDisplay["cveMaxScoreV2"]["Score"]=score;
                        }
                    }
                    if(egalités_v3.length>0){ // il y a au moins un element v2
                        currentTitre = egalités_v3[0]
                        CVEid = dictToDisplay[currentTitre]["CVE"];
                        dictToDisplay["cveMaxScoreV3"]["CVE"] = CVEid;
                        if(currentTitre.includes("CNA")){
                            score = dictToDisplay[currentTitre]["v3 C"];
                            dictToDisplay["cveMaxScoreV3"]["type"]="_CNA";
                        }else{
                            score = dictToDisplay[currentTitre]["v3 N"];
                        }
                        scoreS = dictToDisplay[currentTitre]["v3 S"];
                        if(scoreS>score){
                            avi = getCVSS_fromStorage(CVEid, "CVEv3");
                            dictToDisplay["cveMaxScoreV3"]["source"]=getSourceV3fromStorageAVI(avi);
                            dictToDisplay["cveMaxScoreV3"]["Vecteur"]=getVecteurV3fromStorageAVI(avi);
                            dictToDisplay["cveMaxScoreV3"]["Score"]=scoreS;
                            (document.getElementById('tableCVEs_Td_'+CVEid+'_v3'+dictToDisplay["cveMaxScoreV3"]["type"]+'_vector')).value = dictToDisplay["cveMaxScoreV3"]["Vecteur"];
                        }else{
                            dictToDisplay["cveMaxScoreV3"]["source"]="";
                            dictToDisplay["cveMaxScoreV3"]["Vecteur"]=(document.getElementById(td_prefixe+CVEid+"_v3_vector")).value;
                            dictToDisplay["cveMaxScoreV3"]["Score"]=score;
                        }
                    }
                }else { // il y a au moins une référence commune, on prends la premiere de ce tableau comme référence
                    currentTitre = intersected_Arrays_egalite[0];
                    CVEid = dictToDisplay[currentTitre]["CVE"];
                    // v2
                    dictToDisplay["cveMaxScoreV2"]["CVE"] = CVEid;
                    score = dictToDisplay[currentTitre]["v2"];
                    scoreS = dictToDisplay[currentTitre]["v2 S"];
                    if(scoreS>score){
                        avi = getCVSS_fromStorage(CVEid, "CVEv2");
                        dictToDisplay["cveMaxScoreV2"]["source"]=getSourceV2fromStorageAVI(avi);
                        dictToDisplay["cveMaxScoreV2"]["Vecteur"]=getVecteurV2fromStorageAVI(avi);
                        dictToDisplay["cveMaxScoreV2"]["Score"]=scoreS;
                        (document.getElementById('tableCVEs_Td_'+CVEid+'_v2_vector')).value = dictToDisplay["cveMaxScoreV2"]["Vecteur"];
                    }else{
                        dictToDisplay["cveMaxScoreV2"]["source"]="";
                        dictToDisplay["cveMaxScoreV2"]["Vecteur"]=(document.getElementById(td_prefixe+CVEid+"_v2_vector")).value;
                        dictToDisplay["cveMaxScoreV2"]["Score"]=score;
                    }
                    // v3
                    dictToDisplay["cveMaxScoreV3"]["CVE"] = CVEid;
                    scoreN = dictToDisplay[currentTitre]["v3 N"];
                    scoreC = dictToDisplay[currentTitre]["v3 C"];

                    if(scoreC > scoreN){
                        score = dictToDisplay[currentTitre]["v3 C"];
                        dictToDisplay["cveMaxScoreV3"]["type"]="_CNA";
                    }else{
                        score = dictToDisplay[currentTitre]["v3 N"];
                    }
                    scoreS = dictToDisplay[currentTitre]["v3 S"];
                    if(scoreS>score){
                        avi = getCVSS_fromStorage(CVEid, "CVEv3");
                        dictToDisplay["cveMaxScoreV3"]["source"]=getSourceV3fromStorageAVI(avi);
                        dictToDisplay["cveMaxScoreV3"]["Vecteur"]=getVecteurV3fromStorageAVI(avi);
                        dictToDisplay["cveMaxScoreV3"]["Score"]=scoreS;
                        (document.getElementById('tableCVEs_Td_'+CVEid+'_v3'+dictToDisplay["cveMaxScoreV3"]["type"]+'_vector')).value = dictToDisplay["cveMaxScoreV3"]["Vecteur"];
                    }else{
                        dictToDisplay["cveMaxScoreV3"]["source"]="";
                        dictToDisplay["cveMaxScoreV3"]["Vecteur"]=(document.getElementById(td_prefixe+CVEid+"_v3_vector")).value;
                        dictToDisplay["cveMaxScoreV3"]["Score"]=score;
                    }
                }

                //console.log("["+fctName+"]"+" apres Unification via le dict");
                cveMaxScoreV2 = dictToDisplay["cveMaxScoreV2"]["CVE"];
                cveMaxScoreV3 = dictToDisplay["cveMaxScoreV3"]["CVE"];
                cveMaxScoreV3_type = dictToDisplay["cveMaxScoreV3"]["type"];
                v2Source = dictToDisplay["cveMaxScoreV2"]["source"];
                v3Source = dictToDisplay["cveMaxScoreV3"]["source"];
                //console.log(dictToDisplay);


                // colorisation de la CVE la plus critique
                //console.log("["+fctName+"]"+" colorisation");
                document.getElementById("tableCVEs_Td_"+cveMaxScoreV2+"_v2").style.backgroundColor = colorV2;
                document.getElementById("tableCVEs_Td_"+cveMaxScoreV2+"_CVE").style.backgroundColor = colorV2;
                if(cveMaxScoreV3_type){
                    document.getElementById("tableCVEs_Td_"+cveMaxScoreV3+"_v3"+cveMaxScoreV3_type).style.backgroundColor = colorV3_CNA;
                    if(cveMaxScoreV2==cveMaxScoreV3){
                        document.getElementById("tableCVEs_Td_"+cveMaxScoreV2+"_CVE").style.backgroundColor = colorMixedV2V3_CNA;
                    }else{
                        document.getElementById("tableCVEs_Td_"+cveMaxScoreV3+"_CVE").style.backgroundColor = colorV3_CNA;
                    }
                }else{
                    document.getElementById("tableCVEs_Td_"+cveMaxScoreV3+"_v3"+cveMaxScoreV3_type).style.backgroundColor = colorV3_NVD;
                    if(cveMaxScoreV2==cveMaxScoreV3){
                        document.getElementById("tableCVEs_Td_"+cveMaxScoreV2+"_CVE").style.backgroundColor = colorMixedV2V3_NVD;
                    }else{
                        document.getElementById("tableCVEs_Td_"+cveMaxScoreV3+"_CVE").style.backgroundColor = colorV3_NVD;
                    }
                }

                // application de la selection
                //console.log("["+fctName+"]"+" application de la selection, sourceV2="+v2Source+", sourceV3="+v3Source);
                // on applique la modification du cartouche que si la case from stockage est désactivée
                var IDcheckBoxVersion = 'v2boxFromGMstorage';
                if(document.getElementById(IDcheckBoxVersion)){
                    if(v2Source) {
                        document.getElementById(IDcheckBoxVersion).checked = true;
                    } else {
                        document.getElementById(IDcheckBoxVersion).checked = false;
                    }
                }
                set_v2_cartouche(cveMaxScoreV2,v2Source);

                IDcheckBoxVersion = 'v3boxFromGMstorage';
                if(document.getElementById(IDcheckBoxVersion)){
                    if(v3Source) {
                        document.getElementById(IDcheckBoxVersion).checked = true;
                    } else {
                        document.getElementById(IDcheckBoxVersion).checked = false;
                    }
                }
                set_v3_cartouche(cveMaxScoreV3,cveMaxScoreV3_type,v3Source);

                // colorisation en vert de l'entete du tableau des que fini
                //console.log("["+fctName+"]"+" fin");
                document.getElementById("tableCVEs_THs").style.color = "green";
                if(intervalAnimation){
                    //console.log("arret de l'interval de chargement");
                    // var voidVar = pasteBin_getStoragePasteTitle(); // DEBUT PASTEBIN COLLECT
                    clearInterval(intervalAnimation);
                }
                document.getElementById("pProcessMessage").innerText = "Selection du plus elevé terminé"
            }

            function make_input_cartouche_out(item_before=document.querySelector('body')){
                // variables graphiques
                var font_size = "14px" ;
                var border_size = "0" ;
                var indentation_td = '25px';

                // autres variables
                var list_input = [
                    {object : "div", id :'div_input', style : "font-size :"+font_size},
                    {object : "label", parent_id : "div_input", appendChildText: "ENTREE Vecteur v2 ou v3 :"},
                    {object : "input", parent_id : "div_input", type : "text", id : 'input_text_cvss', size : 70 , value : ''},

                    {object : "label", parent_id : "div_input", appendChildText: "Delimiteur entre entrées"},
                    {object : "input", parent_id : "div_input", type : "text", id : 'input_text_cvss_delimiter', size : 3, value : '/'},

                    {object : "label", parent_id : "div_input", appendChildText: "Delimiteur entrée-valeur"},
                    {object : "input", parent_id : "div_input", type : "text", id : 'input_text_cvss_operateur', size : 3, value : ':'},

                    {object : "table", parent_id : "div_input", id : "cvss_table", border : border_size},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr1"},
                    {object : "td", parent_id : "cvss_table_tr1", colSpan:"7", appendChildText: "Métrique de base"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr2"},
                    {object : "td", parent_id : "cvss_table_tr2"},
                    {object : "td", parent_id : "cvss_table_tr2", colSpan :'7', appendChildText: "Exploitabilité"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr3"},
                    {object : "td", parent_id : "cvss_table_tr3", width: indentation_td},
                    {object : "td", parent_id : "cvss_table_tr3", width: indentation_td},
                    {object : "td", parent_id : "cvss_table_tr3", appendChildText: "Vecteur d'attaque (AV)"},
                    {object : "td", parent_id : "cvss_table_tr3", appendChildText: "Complexité d'attaque (AC)", style :"padding-right:10px"},
                    {object : "td", parent_id : "cvss_table_tr3", appendChildText: "Authentification (Au)"},
                    {object : "td", parent_id : "cvss_table_tr3", appendChildText: "Privilèges requis (PR)", style :"padding-right:10px"},
                    {object : "td", parent_id : "cvss_table_tr3", appendChildText: "Interaction utilisateur (UI)"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr4"},
                    {object : "td", parent_id : "cvss_table_tr4", width :'25px'},
                    {object : "td", parent_id : "cvss_table_tr4", width :'25px'},
                    {object : "td", parent_id : "cvss_table_tr4", id : "cvss_table_tr4_td1"},
                    {object : "select", parent_id : "cvss_table_tr4_td1", id : 'AccessVectorVar', width :'100%'},
                    {object : "option", parent_id : "AccessVectorVar", value :'v2=AV:/v3=AV:', selected:'selected', label:'?',appendChildText:"?"},
                    {object : "option", parent_id : "AccessVectorVar", value :'v2=AV:L/v3=AV:L', label:'Local',appendChildText:"L"},
                    {object : "option", parent_id : "AccessVectorVar", value :'v2=AV:A/v3=AV:A', label:'Réseau contigue',appendChildText:"A"},
                    {object : "option", parent_id : "AccessVectorVar", value :'v2=AV:N/v3=AV:N', label:'Réseau',appendChildText:"N"},
                    {object : "option", parent_id : "AccessVectorVar", value :'v2=AV:L/v3=AV:P', label:'Physique',appendChildText:"P"},

                    {object : "td", parent_id : "cvss_table_tr4", id : "cvss_table_tr4_td2"},
                    {object : "select", parent_id : "cvss_table_tr4_td2", id : 'AccessComplexityVar'},
                    {object : "option", parent_id : "AccessComplexityVar", value :'v2=AC:/v3=AC:', selected: 'selected', label: '?',appendChildText:"?"},
                    {object : "option", parent_id : "AccessComplexityVar", value :'v2=AC:H/v3=AC:H', label: 'Haute',appendChildText:"H"},
                    {object : "option", parent_id : "AccessComplexityVar", value :'v2=AC:M/v3=AC:H', label: 'Moyenne/haute',appendChildText:"M"},
                    {object : "option", parent_id : "AccessComplexityVar", value :'v2=AC:M/v3=AC:L', label: 'Moyenne/faible',appendChildText:"M"},
                    {object : "option", parent_id : "AccessComplexityVar", value :'v2=AC:L/v3=AC:L', label: 'Faible',appendChildText:"L"},

                    {object : "td", parent_id : "cvss_table_tr4", id : "cvss_table_tr4_td3"},
                    {object : "select", parent_id : "cvss_table_tr4_td3", id : 'AuthenticationVar'},
                    {object : "option", parent_id : "AuthenticationVar", value :'v2=Au:/v3: ', selected: 'selected', label: '?',appendChildText:"?"},
                    {object : "option", parent_id : "AuthenticationVar", value :'v2=Au:M/v3: ', label: 'Nécessite plusieurs instances',appendChildText:"M"},
                    {object : "option", parent_id : "AuthenticationVar", value :'v2=Au:S/v3: ', label: 'Nécessite une seule instance',appendChildText:"S"},
                    {object : "option", parent_id : "AuthenticationVar", value :'v2=Au:N/v3: ', label: 'Aucune',appendChildText:"N"},

                    {object : "td", parent_id : "cvss_table_tr4", id : "cvss_table_tr4_td4"},
                    {object : "select", parent_id : "cvss_table_tr4_td4", id : 'PrivilegeRequiredVar'},
                    {object : "option", parent_id : "PrivilegeRequiredVar", value :'v2=/v3=PR:', selected: 'selected', label: '?',appendChildText:"?"},
                    {object : "option", parent_id : "PrivilegeRequiredVar", value :'v2=/v3=PR:N', label: 'Aucun',appendChildText:"N"},
                    {object : "option", parent_id : "PrivilegeRequiredVar", value :'v2=/v3=PR:L', label: 'Bas',appendChildText:"L"},
                    {object : "option", parent_id : "PrivilegeRequiredVar", value :'v2=/v3=PR:H', label: 'Hauts',appendChildText:"H"},

                    {object : "td", parent_id : "cvss_table_tr4", id : "cvss_table_tr4_td5"},
                    {object : "select", parent_id : "cvss_table_tr4_td5", id : 'UserInteractionVar'},
                    {object : "option", parent_id : "UserInteractionVar", value :'v2=/v3=UI:', selected: 'selected', label: '?',appendChildText:"?"},
                    {object : "option", parent_id : "UserInteractionVar", value :'v2=/v3=UI:N', label: 'Aucune',appendChildText:"N"},
                    {object : "option", parent_id : "UserInteractionVar", value :'v2=/v3=UI:R', label: 'Requise',appendChildText:"R"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr5"},
                    {object : "td", parent_id : "cvss_table_tr5"},
                    {object : "td", parent_id : "cvss_table_tr5", colSpan :'7', appendChildText: "Impact"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr6"},
                    {object : "td", parent_id : "cvss_table_tr6"},
                    {object : "td", parent_id : "cvss_table_tr6"},
                    {object : "td", parent_id : "cvss_table_tr6", appendChildText: "Portée (S)"},
                    {object : "td", parent_id : "cvss_table_tr6", appendChildText: "Confidentialité (C)"},
                    {object : "td", parent_id : "cvss_table_tr6", appendChildText: "Intégrité (I)"},
                    {object : "td", parent_id : "cvss_table_tr6", appendChildText: "Disponibilité (A)"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr7"},
                    {object : "td", parent_id : "cvss_table_tr7"},
                    {object : "td", parent_id : "cvss_table_tr7"},
                    {object : "td", parent_id : "cvss_table_tr7", id : "cvss_table_tr7_td1"},
                    {object : "select", parent_id : "cvss_table_tr7_td1", id : 'ScopeVar'},
                    {object : "option", parent_id : "ScopeVar", value :'v2=/v3=S:', selected: 'selected', label: '?', appendChildText:"?"},
                    {object : "option", parent_id : "ScopeVar", value :'v2=/v3=S:U', label: 'Pas de débordement', appendChildText:"U"},
                    {object : "option", parent_id : "ScopeVar", value :'v2=/v3=S:C', label: 'Débordement', appendChildText:"C"},

                    {object : "td", parent_id : "cvss_table_tr7", id : "cvss_table_tr7_td2"},
                    {object : "select", parent_id : "cvss_table_tr7_td2", id : 'ConfImpactVar'},
                    {object : "option", parent_id : "ConfImpactVar", value :'v2=C:/v3=C:', selected: 'selected', label: '?', appendChildText:"ND"},
                    {object : "option", parent_id : "ConfImpactVar", value :'v2=C:N/v3=C:N', label: 'Aucun', appendChildText:"N"},
                    {object : "option", parent_id : "ConfImpactVar", value :'v2=C:P/v3=C:L', label: 'Partiel/bas', appendChildText:"P"},
                    {object : "option", parent_id : "ConfImpactVar", value :'v2=C:C/v3=C:H', label: 'Complet/haut', appendChildText:"C"},

                    {object : "td", parent_id : "cvss_table_tr7", id : "cvss_table_tr7_td3"},
                    {object : "select", parent_id : "cvss_table_tr7_td3", id : 'IntegImpactVar'},
                    {object : "option", parent_id : "IntegImpactVar", value :'v2=I:/v3=I:', selected: 'selected', label: '?', appendChildText:"ND"},
                    {object : "option", parent_id : "IntegImpactVar", value :'v2=I:N/v3=I:N', label: 'Aucun', appendChildText:"N"},
                    {object : "option", parent_id : "IntegImpactVar", value :'v2=I:P/v3=I:L', label: 'Partiel/bas', appendChildText:"P"},
                    {object : "option", parent_id : "IntegImpactVar", value :'v2=I:C/v3=I:H', label: 'Complet/haut', appendChildText:"C"},

                    {object : "td", parent_id : "cvss_table_tr7", id : "cvss_table_tr7_td4"},
                    {object : "select", parent_id : "cvss_table_tr7_td4", id : 'AvailImpactVar'},
                    {object : "option", parent_id : "AvailImpactVar", value :'v2=A:/v3=A:', selected: 'selected', label: '?', appendChildText:"?"},
                    {object : "option", parent_id : "AvailImpactVar", value :'v2=A:N/v3=A:N', label: 'Aucun', appendChildText:"N"},
                    {object : "option", parent_id : "AvailImpactVar", value :'v2=A:P/v3=A:L', label: 'Partiel/bas', appendChildText:"P"},
                    {object : "option", parent_id : "AvailImpactVar", value :'v2=A:C/v3=A:H', label: 'Complet/haut', appendChildText:"C"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr8"},
                    {object : "td", parent_id : "cvss_table_tr8", colSpan:"7", appendChildText: "Score Temporel"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr9"},
                    {object : "td", parent_id : "cvss_table_tr9"},
                    {object : "td", parent_id : "cvss_table_tr9"},
                    {object : "td", parent_id : "cvss_table_tr9", appendChildText: "Exploitabilité (E)"},
                    {object : "td", parent_id : "cvss_table_tr9", appendChildText: "Remediation (RL)"},
                    {object : "td", parent_id : "cvss_table_tr9", appendChildText: "Confiance (RC)"},

                    {object : "tr", parent_id : "cvss_table", id : "cvss_table_tr10"},
                    {object : "td", parent_id : "cvss_table_tr10"},
                    {object : "td", parent_id : "cvss_table_tr10"},
                    {object : "td", parent_id : "cvss_table_tr10", id : "cvss_table_tr10_td1"},
                    {object : "select", parent_id : "cvss_table_tr10_td1", id : 'ExploitabilityVar'},
                    {object : "option", parent_id : "ExploitabilityVar", value :'v2=E:ND/v3=E:X', selected: 'selected', label: 'Non défini', appendChildText:"ND"},
                    {object : "option", parent_id : "ExploitabilityVar", value :'v2=E:U/v3=E:U', label: 'Non prouvé', appendChildText:"U"},
                    {object : "option", parent_id : "ExploitabilityVar", value :'v2=E:POC/v3=E:P', label: 'Programme de démonstration (PoC)', appendChildText:"POC"},
                    {object : "option", parent_id : "ExploitabilityVar", value :'v2=E:F/v3=E:F', label: 'Fonctionnel', appendChildText:"F"},
                    {object : "option", parent_id : "ExploitabilityVar", value :'v2=E:H/v3=E:H', label: 'Haut', appendChildText:"H"},

                    {object : "td", parent_id : "cvss_table_tr10", id : "cvss_table_tr10_td2"},
                    {object : "select", parent_id : "cvss_table_tr10_td2", id : 'RemediationLevelVar'},
                    {object : "option", parent_id : "RemediationLevelVar", value :'v2=RL:ND/v3=RL:X', selected: 'selected', label: 'Non défini', appendChildText:"ND"},
                    {object : "option", parent_id : "RemediationLevelVar", value :'v2=RL:OF/v3=RL:O', label: 'Correctif officiel', appendChildText:"OF"},
                    {object : "option", parent_id : "RemediationLevelVar", value :'v2=RL:TF/v3=RL:T', label: 'Correctif provisoire', appendChildText:"TF"},
                    {object : "option", parent_id : "RemediationLevelVar", value :'v2=RL:W/v3=RL:W', label: 'Palliatif', appendChildText:"W"},
                    {object : "option", parent_id : "RemediationLevelVar", value :'v2=RL:U/v3=RL:U', label: 'Non disponible', appendChildText:"U"},


                    {object : "td", parent_id : "cvss_table_tr10", id : "cvss_table_tr10_td3"},
                    {object : "select", parent_id : "cvss_table_tr10_td3", id : 'ReportConfidenceVar'},
                    {object : "option", parent_id : "ReportConfidenceVar", value :'v2=RC:ND/v3=RC:X', label: 'Non défini', appendChildText:"ND"},
                    {object : "option", parent_id : "ReportConfidenceVar", value :'v2=RC:UC/v3=RC:U', label: 'Non confirmé', appendChildText:"UC"},
                    {object : "option", parent_id : "ReportConfidenceVar", value :'v2=RC:UR/v3=RC:R', label: 'Présumé', appendChildText:"UR"},
                    {object : "option", parent_id : "ReportConfidenceVar", value :'v2=RC:C/v3=RC:C', selected: 'selected', label: 'Confirmé', appendChildText:"C"},

                    //{object : "td", parent_id : "cvss_table_tr10", id : "cvss_table_tr10_td4"},
                    //{object : "input", parent_id : "cvss_table_tr10_td4", type : "button", id : 'btn_add_RLof_RCc', size : 3, value : 'Ajouter RL:OF + RC:C'},


                    {object : "br", parent_id : "div_input"},
                    {object : "div", parent_id : "div_input", id: "divBoxOptions", style:"float:left;"},
                    {object : "input", type : "checkbox", parent_id : "divBoxOptions", id: "lock_v2"},
                    {object : "label", parent_id : "divBoxOptions", for : "lock_v2", appendChildText: "Verrouiller v2", id: "lock_v2_from_label", style :"padding-right:10px"},
                    {object : "input", type : "checkbox", parent_id : "divBoxOptions", id: "lock_v3"},
                    {object : "label", parent_id : "divBoxOptions", for : "lock_v3", appendChildText: "Verrouiller v3", id: "lock_v3_from_label", style :"padding-right:10px"},

                    {object : "input", type : "checkbox", parent_id : "divBoxOptions", id: "v2boxFromGMstorage"},
                    {object : "label", parent_id : "divBoxOptions", for : "v2boxFromGMstorage", appendChildText: "V2 from storage", id: "v2boxFromGMstorage_label", style :"padding-right:10px"},
                    {object : "input", type : "checkbox", parent_id : "divBoxOptions", id: "v3boxFromGMstorage"},
                    {object : "label", parent_id : "divBoxOptions", for : "v3boxFromGMstorage", appendChildText: "V3 from storage", id: "v3boxFromGMstorage_label", style :"padding-right:10px"},
                    {object : "div", parent_id : "div_input", id: "divBoxToGMstorage", style:"float:left;"},
                    {object : "input", type : "checkbox", parent_id : "divBoxToGMstorage", id: "boxToGMstorage", standaloneAttributes: 'checked'},
                    {object : "label", parent_id : "divBoxToGMstorage", for : "boxToGMstorage", appendChildText: "Stocker (local + Git)", id: "boxToGMstorage_label", style :"padding-right:10px" }

                ];

                make_html_from_list_of_dict(item_before, list_input);
            }
            function make_top_section(item_before=document.querySelector('body')){ // ajoute une table avec l'etat pastebin et un combobox de log
                // variables graphiques
                var font_size = "14px" ;
                var border_size = "1" ;
                var indentation_td = '25px';
                var td_style = "padding-right:10px";


                // autres variables
                var list_input = [
                    // pasteBin table
                    {object : "div", id :'div_top_section', style : "font-size :"+font_size},
                    {object : "div", id :'div_macroInfo', parent_id : "div_top_section", style : "font-size :"+font_size},
                    {object : "table", parent_id : "div_macroInfo", id : "pastebin_table", border : border_size},
                    {object : "tr", parent_id : "pastebin_table", id : "pastebin_table_tr1"}, // header
                    {object : "td", parent_id : "pastebin_table_tr1", appendChildText: "Pastes", style : td_style},//, colSpan:"7"
                    {object : "td", parent_id : "pastebin_table_tr1", appendChildText: "Status", style : td_style},
                    {object : "td", parent_id : "pastebin_table_tr1", appendChildText: "état", style : td_style},
                    {object : "tr", parent_id : "pastebin_table", id : "pastebin_table_tr2"}, // lien avant
                    {object : "td", parent_id : "pastebin_table_tr2", appendChildText: "", id : "pastebin_table_tr2_td_lien", style : td_style},
                    {object : "a", parent_id : "pastebin_table_tr2_td_lien", appendChildText: "", id : "pastebin_lien_avant", href: "", target: "_blank"},
                    {object : "td", parent_id : "pastebin_table_tr2", appendChildText: "", id : "pastebin_table_tr2_td_status", style : td_style},
                    {object : "td", parent_id : "pastebin_table_tr2", appendChildText: "", id : "pastebin_table_tr2_td_etat", style : td_style},
                    {object : "tr", parent_id : "pastebin_table", id : "pastebin_table_tr3"}, // lien ares
                    {object : "td", parent_id : "pastebin_table_tr3", appendChildText: "", id : "pastebin_table_tr3_td_lien", style : td_style},
                    {object : "a", parent_id : "pastebin_table_tr3_td_lien", appendChildText: "", id : "pastebin_lien_apres", href: "", target: "_blank"},
                    {object : "td", parent_id : "pastebin_table_tr3", appendChildText: "", id : "pastebin_table_tr3_td_status", style : td_style},
                    {object : "td", parent_id : "pastebin_table_tr3", appendChildText: "", id : "pastebin_table_tr3_td_etat", style : td_style},


                    // combobox log
                    {object : "div", id :'div_LOGS', parent_id : "div_top_section", style : "font-size :"+font_size},
                    {object : "label", parent_id : "div_LOGS", appendChildText: "logs :"},
                    {object : "select", parent_id : "div_LOGS", id : 'cmb_log', size : 1, style : "width:50em;"},
                    {object : "div", id :'div_LOGS_page', parent_id : "div_top_section", style : "font-size :"+font_size},
                    {object : "label", parent_id : "div_LOGS_page", appendChildText: "Erreurs Page :"},
                    {object : "select", parent_id : "div_LOGS_page", id : 'cmb_log_page', size : 1, style : "width:50em;"},
                    {object : "div", id :'div_LOGS_feedly', parent_id : "div_top_section", style : "font-size :"+font_size},
                    {object : "label", parent_id : "div_LOGS_feedly", appendChildText: "Logs feedly :"},
                    {object : "select", parent_id : "div_LOGS_feedly", id : 'cmb_log_feedly', size : 1, style : "width:50em;"}
                ];

                make_html_from_list_of_dict(item_before, list_input);
            }
            function add_log_to_CMB_x(cmbID, txt, indent, data_value=null){// ajoute une option dans le CMB comme un log et selection l'option // ressemble un peu a un spoiler mais avec le dernier spoil
                var txtIndent=get_indent(indent);
                // list all options values
                let DOM_cmb_log = document.getElementById(cmbID);
                var newOptionValue = DOM_cmb_log.children.length;

                let newOption = new Option(txtIndent+newOptionValue+" : "+txt,newOptionValue);
                DOM_cmb_log.add(newOption,undefined);
                DOM_cmb_log.value=newOptionValue;
                if(data_value){}//ajout de data-value et du onchange
            }
            function add_log_to_CMB(txt, indent=0){// ajoute une option dans le CMB comme un log et selection l'option // ressemble un peu a un spoiler mais avec le dernier spoil
                var id_cmb_log = "cmb_log";
                add_log_to_CMB_x(id_cmb_log, txt, indent);
            }
            function add_log_to_CMB_PAGE(txt, indent=0){// ajoute une option dans le CMB comme un log et selection l'option // ressemble un peu a un spoiler mais avec le dernier spoil
                var id_cmb_log = "cmb_log_page";
                add_log_to_CMB_x(id_cmb_log, txt, indent);
            }
            function add_log_to_CMB_FEEDLY(txt, indent=0){// ajoute une option dans le CMB comme un log et selection l'option // ressemble un peu a un spoiler mais avec le dernier spoil
                var id_cmb_log = "cmb_log_feedly";
                add_log_to_CMB_x(id_cmb_log, txt, indent);
            }
            function update_paste_table(from, status, api_paste_key,texteLien=""){ // met à jour la table celon une logique unique
                add_log_to_CMB("[update_paste_table] "+from+" / "+status+" / "+api_paste_key);
                var target="", etat="", lien="";
                var pasteBinURL_prefixe = "https://pastebin.com/"

                // determinaison du scénario:
                switch(from){
                    case "pasteBin_add": //("pasteBin_add",status,responseText)
                        // uniquement au moment du telechargement (apres avoir supprimé); envoi
                        target = "apres";
                        // récupération du lien
                        if(status==200){ // ajout OK
                            etat = "ajouté";
                            lien = api_paste_key;
                        }else{
                            etat = "erreur ajout, faites F12";
                            lien = pasteBinURL_prefixe+api_paste_key;
                        }
                        break;
                    case "pasteBin_delete": //("pasteBin_delete",status,api_paste_key)
                        // uniquement au au moment de remplacer le paste bin apres le telechargement avant d'ajouter
                        target = "avant";
                        if(status==200){ // suppression OK
                            status = 404;
                            etat = "supprimé";
                            texteLien = document.getElementById("pastebin_lien_avant").innerText
                        }else{
                            etat = "erreur suppression, faites F12";
                        }
                        lien = pasteBinURL_prefixe+api_paste_key;
                        break;
                    case "pasteBin_getContent": // ("pasteBin_getContent",status,api_paste_key)
                        // au moment de la mise à jour du stockage
                        target = "avant";
                        if(status==200){ // ajout OK
                            etat = "mise à jour dans le stockage";
                        }else{
                            etat = "erreur récupération du contenu, faites F12";
                        }
                        lien = pasteBinURL_prefixe+api_paste_key;
                        break;
                    case "pasteBin_getStoragePasteTitle": // ("pasteBin_getStoragePasteTitle",title,api_paste_key)
                        // deux scenario : avant la suppression pour recuperer la cle et apres le chargement des CVE
                        target = "avant";
                        if(status){
                            if(api_paste_key){
                                etat = status;
                                status=200;
                            }else{
                                etat = "paste ("+status+") mais pas de clé";
                                status="home";
                                api_paste_key="";
                            }
                        }else{
                            status="home";
                            etat = "pas de paste";
                            api_paste_key="";
                        }
                        lien = pasteBinURL_prefixe+api_paste_key;
                        break;
                }
                // id utilisés
                var id_link, id_status, id_etat;
                switch(target){
                    case "avant":
                        id_link = "pastebin_lien_avant";
                        id_status = "pastebin_table_tr2_td_status";
                        id_etat = "pastebin_table_tr2_td_etat";
                        break;
                    default:
                        id_link = "pastebin_lien_apres";
                        id_status = "pastebin_table_tr3_td_status";
                        id_etat = "pastebin_table_tr3_td_etat";
                        break;
                }
                document.getElementById(id_link).innerText = (texteLien?texteLien:api_paste_key);
                document.getElementById(id_link).href = lien;
                document.getElementById(id_status).innerText = status;
                document.getElementById(id_etat).innerText = etat;

            }

            function add_functions_onchange(){
                var items = document.querySelectorAll('select');

                for(var i=0;i<items.length;i++) {
                    //items[i].setAttribute("onchange", update_output); // inside the code
                    items[i].addEventListener('change', computeCVSS,false);
                }
                items = document.querySelectorAll('input');
                for(i=0;i<items.length;i++) {
                    if(items[i].type==="text"){
                        items[i].addEventListener('change', update_options);
                        items[i].addEventListener('keyup', update_options);
                        //items[i].setAttribute("onchange", update_options());//update_options lancera update_output // inside the code
                    }
                }
            }

            function add_functions_onclick(){
                document.getElementById('link_vecteur_v2').setAttribute('onclick',"document.getElementById('input_text_cvss').value = this.innerText;document.getElementById('lock_v3').checked = true;document.getElementById('lock_v2').checked = false;");
                document.getElementById('link_vecteur_v3').setAttribute('onclick',"document.getElementById('input_text_cvss').value = this.innerText;document.getElementById('lock_v2').checked = true;document.getElementById('lock_v3').checked = false;");

                document.getElementById('link_vecteur_v2').addEventListener('click',update_options);
                document.getElementById('link_vecteur_v3').addEventListener('click',update_options);

                document.getElementById('btn_add_RLof_RCc').addEventListener('click',addDefaultTempo);

            }

            function selectOption(select_id, optionText){
                var select_obj = document.getElementById(select_id);
                var options = Array.from(select_obj.options);
                var option = options.find(item => item.text === optionText);
                option.selected = true;
            }

            function addDefaultTempo(){
                addDefaultTempo_version('link_vecteur_v2');
                addDefaultTempo_version('link_vecteur_v3');
            }

            function addDefaultTempo_version(idLink){
                document.getElementById(idLink).click();

                var idSelect = 'RemediationLevelVar';
                var optionText = 'OF';
                selectOption(idSelect, optionText);

                idSelect = 'ReportConfidenceVar';
                optionText = 'C';
                selectOption(idSelect, optionText);

                computeCVSS();
            }

            function set_v2_cartouche(cveID, source=""){
                var TargetVersion="_v2";
                // console.log("cliqued ("+cveID+") : " +document.getElementById('tableCVEs_Td_'+cveID+TargetVersion+'_vector').value );
                document.getElementById('input_text_cvss').value = document.getElementById('tableCVEs_Td_'+cveID+TargetVersion+'_vector').value;
                document.getElementById('refCVE_vecteur'+TargetVersion).value = cveID;
                document.getElementById('refCVE_vecteur'+TargetVersion+'_source').value = (source ? source : "NVD auto");
                document.getElementById('lock_v2').checked = false;
                document.getElementById('lock_v3').checked = true;

                var lstMetaNVD=["NVDpublishDate","NVDmodifiedDate","NVDsource"];
                var metaValue, metaID;
                for(var i=0 ; i<lstMetaNVD.length;i++) {
                    metaValue = "";
                    metaID = 'tableCVEs_Td_'+cveID+'_'+lstMetaNVD[i];
                    if(document.getElementById(metaID)) {
                        metaValue = document.getElementById(metaID).value;
                    }
                    document.getElementById('refCVE'+TargetVersion+'_'+lstMetaNVD[i]).value = metaValue;
                    //console.log("application de '"+metaValue+"' a "+'refCVE'+TargetVersion+'_'+lstMetaNVD[i]);
                }

                update_options();
            }
            function set_v3_cartouche(cveID,CNA="", source=""){
                var TargetVersion="_v3";
                // console.log("cliqued ("+cveID+") : " +document.getElementById('tableCVEs_Td_'+cveID+TargetVersion+CNA+'_vector').value );
                document.getElementById('input_text_cvss').value = document.getElementById('tableCVEs_Td_'+cveID+TargetVersion+CNA+'_vector').value;
                document.getElementById('refCVE_vecteur'+TargetVersion).value = cveID;
                document.getElementById('refCVE_vecteur'+TargetVersion+'_source').value = (source ? source : "NVD "+(CNA ? "(CNA) " : '')+"auto");
                document.getElementById('lock_v2').checked = true;
                document.getElementById('lock_v3').checked = false;

                var lstMetaNVD=["NVDpublishDate","NVDmodifiedDate","NVDsource"];
                var metaValue, metaID;
                for(var i=0 ; i<lstMetaNVD.length;i++) {
                    metaValue = "";
                    metaID = 'tableCVEs_Td_'+cveID+'_'+lstMetaNVD[i];
                    if(document.getElementById(metaID)) {
                        metaValue = document.getElementById(metaID).value;
                    }
                    document.getElementById('refCVE'+TargetVersion+'_'+lstMetaNVD[i]).value = metaValue;
                    //console.log("application de '"+metaValue+"' a "+'refCVE'+TargetVersion+'_'+lstMetaNVD[i]);
                }

                update_options();
            }

            function set_cartouche_FromCVEtd(tdDOM) { //cveID, versionValue = 2, isNVD=true){
                // declaration des variables
                //console.log("set_cartouche_FromCVEtd 1");
                const cveID = (/CVE-[0-9]+-[0-9]+/).exec(tdDOM.id);
                const versionValue = (/v[0-9]/).exec(tdDOM.id); // v2 ou v3
                const isTD_v2 = (/v2$/).exec(tdDOM.id) ;
                const isTD_v3_NVD = (/v3$/).exec(tdDOM.id) ;
                const isTD_v3_CNA = (/v3_CNA/).exec(tdDOM.id) ;
                //console.log("set_cartouche_FromCVEtd 2");
                // générique
                var source = "" ;
                var scoreIsFromStorage = false;
                var IDcheckBoxVersion = versionValue+'boxFromGMstorage' ;
                //console.log("set_cartouche_FromCVEtd 3");

                // Remise à zero du style du table pour "selectionner" la bonne cellule
                    var cleanOnly ="";
                    var DOMtable = null;
                    if(isTD_v2){ // on a cliqué sur un v2 ; on conserve les v3
                        cleanOnly = "v2";
                    }else{ // on a cliqué sur un v3 ; on conserve le v2
                        cleanOnly = "v3";
                    }

                    try{
                        var cpt = 0;
                        DOMtable = tdDOM;
                        while(DOMtable.nodeName != "TABLE" && cpt<5){
                            DOMtable = DOMtable.parentNode;
                            cpt++;
                        }
                    }catch(error){
                        console.error("[set_cartouche_FromCVEtd] ERROR:",error.message);
                    }
                    resetBorderStyle(DOMtable, cleanOnly);
                // mise en lumiere de la cellule
                    higlightCellBorder(tdDOM);


                // cellule
                var prefixe_IDCell = 'tableCVEs_Td_'+cveID+'_'+versionValue;
                var IDvectorCell = prefixe_IDCell+'_vector';
                var vectorFromCell = "";
                var scoreFromVector = -1 ;
                var scoreFromCell = scoreString_to_float(tdDOM.innerText);
                //console.log("set_cartouche_FromCVEtd 4");

                var vectorFromCell_NVD ="" ;
                var vectorFromCell_CNA ="" ;
                var scoreFromCell_NVD =-1 ;
                var scoreFromCell_CNA =-1 ;
                var scoreFromVector_NVD =-1 ;
                var scoreFromVector_CNA =-1 ;
                //console.log("set_cartouche_FromCVEtd 5");
                // stockage
                var IDscoreCNAcell ;
                // initialisation
                try {
                    vectorFromCell = document.getElementById(IDvectorCell).value ;
                    scoreFromVector = getScroreV2(vectorString_To_VectorList(vectorFromCell));
                }catch{}
                //console.log("set_cartouche_FromCVEtd 6");


                // il faut determiner si la valeur de la cellule provient du stockage
                if(isTD_v3_NVD || isTD_v3_CNA) {
                    try{vectorFromCell_NVD = (
                        isTD_v3_NVD?scoreFromCell:document.getElementById(prefixe_IDCell+"_vector").innerText);
                        scoreFromVector_NVD = getScroreV3(vectorString_To_VectorList(vectorFromCell_NVD));
                       }catch{}
                    try{
                        vectorFromCell_CNA = (isTD_v3_CNA?scoreFromCell:document.getElementById(prefixe_IDCell+"_CNA_vector").innerText);
                        scoreFromVector_CNA = getScroreV3(vectorString_To_VectorList(vectorFromCell_CNA));
                    }catch{}
                    try{scoreFromCell_NVD = (isTD_v3_NVD?scoreFromCell:document.getElementById(prefixe_IDCell+"_score").innerText);}catch{}
                    try{scoreFromCell_CNA = (isTD_v3_CNA?scoreFromCell:document.getElementById(prefixe_IDCell+"_CNA_score").innerText);}catch{}

                    scoreIsFromStorage = scoreFromVector_NVD>scoreFromCell_NVD && scoreFromVector_CNA>scoreFromCell_CNA;
                }else{
                    scoreIsFromStorage=scoreFromVector>scoreFromCell;
                }

                //console.log("set_cartouche_FromCVEtd 7");
                try{document.getElementById(IDcheckBoxVersion).checked = scoreIsFromStorage;}catch{}

                //console.log("set_cartouche_FromCVEtd 8");

                // DEBUG
                //console.log("clicked "+cveID+" "+versionValue+(isTD_v3_CNA?"_CNA":"")+" : "+scoreFromCell+" / "+scoreFromVector);

                // application du score et vecteur
                if(isTD_v3_NVD || isTD_v3_CNA) {
                    if(scoreIsFromStorage){source = getSourceV3fromStorageAVI(getCVSS_fromStorage(cveID, "CVEv3")) ;}
                    set_v3_cartouche(cveID, (isTD_v3_CNA?"_CNA":""), source);
                }else{
                    if(scoreIsFromStorage){source = getSourceV2fromStorageAVI(getCVSS_fromStorage(cveID, "CVEv2")) ;}
                    set_v2_cartouche(cveID, source);
                }
                //console.log("set_cartouche_FromCVEtd ");
            }

            function printEphemeralMessage(event, message, delay = 1000){
                var DOMmessage=document.createElement("div")
                DOMmessage.style.border = "1px dashed blue";
                DOMmessage.style.position = "absolute";
                DOMmessage.style.left = event.pageX+5+"px";
                DOMmessage.style.top = event.pageY+5+"px";
                DOMmessage.style.background= "white";

                var txt = document.createElement("p");
                txt.innerHTML = message;
                DOMmessage.appendChild(txt);

                document.querySelector('body').appendChild(DOMmessage);

                setTimeout(function(){DOMmessage.remove();}, delay);

            }
            function copyCVEname(event){
                var DOMitem = event.target
                var cveID = (/CVE-[0-9]+-[0-9]+/).exec(DOMitem.id);

                updateClipboard(cveID);
                var messageHTML = "<strong>"+cveID+" copié. </strong>";
                printEphemeralMessage(event, messageHTML, 2000);

            }

            function add_functions_onclick_arrayOfCVE(arrayOfCVE=[]){
                var id = "";
                for(var i=0;i<arrayOfCVE.length;i++){
                    id = arrayOfCVE[i];

                    // CVE updateClipboard
                    document.getElementById('tableCVEs_Td_'+id+'_CVE').addEventListener('click', copyCVEname,false);

                    //v2
                    document.getElementById('tableCVEs_Td_'+id+'_v2').addEventListener('click', function(){
                        set_cartouche_FromCVEtd(this);
                    },false);

                    // v3
                    document.getElementById('tableCVEs_Td_'+id+'_v3').addEventListener('click', function(){
                        set_cartouche_FromCVEtd(this);
                    },false);

                    // v3 CNA
                    document.getElementById('tableCVEs_Td_'+id+'_v3_CNA').addEventListener('click', function(){
                        set_cartouche_FromCVEtd(this);
                    },false);
                }
            }

            function updateCartoucheFromStorageOrAutoNVD() {
                if(document.getElementById("v2boxFromGMstorage").checked){
                    document.getElementById('lock_v2').checked = false;
                    document.getElementById('lock_v3').checked = true;
                    stockageToFieldsV2();
                }else{
                    // set_v2_cartouche(cveMaxScoreV2);
                    var elemID = "tableCVEs_Td_"+cveMaxScoreV2+"_v2";
                    var elem = document.getElementById(elemID);
                    // console.log("updateCartoucheFromStorageOrAutoNVD 2 : "+elemID);
                    // console.log(elem);
                    set_cartouche_FromCVEtd(elem);
                }

                if(document.getElementById("v3boxFromGMstorage").checked){
                    document.getElementById('lock_v2').checked = true;
                    document.getElementById('lock_v3').checked = false;
                    stockageToFieldsV3();
                }else{
                    // set_v3_cartouche(cveMaxScoreV3);
                    elemID = "tableCVEs_Td_"+cveMaxScoreV3+"_v3"+cveMaxScoreV3_type+"";
                    elem = document.getElementById(elemID);
                    // console.log("updateCartoucheFromStorageOrAutoNVD 3 : "+elemID);
                    // console.log(elem);
                    set_cartouche_FromCVEtd(elem);
                }
            }

            function make_display_cartouche_out(item_before){
                // variables graphiques
                var font_size = "20px" ;
                var border_size = "0" ;
                var indentation_td = '40px';
                var greyedStyle = "background-color: #f0f0f0;"

                document.querySelector("html").setAttribute("xmlns","http://www.w3.org/1999/xhtml");
                // autres variables
                var list_input = [
                    {object : "div", id :'div_output', style : "font-size :"+font_size},
                    {object: "input", parent_id : "div_output", type:"hidden", id: "scriptVersionUsed", value: scriptVersion, class: "only_for_html_export"},
                    {object : "table", parent_id : "div_output" , id :'cartouche_table', class: "only_for_html_export", style:"display:block;", border: border_size},

                    {object: "tr", parent_id : "cartouche_table", id: "cartouche_table_tr1"},
                    {object: "td", parent_id : "cartouche_table_tr1", width: indentation_td, style :"padding-right:10px"},
                    //{object: "td", parent_id : "cartouche_table_tr1", appendChildText: "Gravité svg", style :"padding-right:10px"},
                    //{object: "td", parent_id : "cartouche_table_tr1", appendChildText: "Gravité img svg", style :"padding-right:10px"},
                    {object: "td", parent_id : "cartouche_table_tr1", appendChildText: "Gravité", style :"padding-right:10px"}, //Gravité img png
                    {object: "td", parent_id : "cartouche_table_tr1", appendChildText: "Score de base", style :"padding-right:10px"},
                    {object: "td", parent_id : "cartouche_table_tr1", appendChildText: "Score temporel", style :"padding-right:10px"},
                    {object: "td", parent_id : "cartouche_table_tr1", appendChildText: "Vecteur", style :"padding-right:10px", id : "cartouche_table_tr1_td5"},
                    {object : "input", parent_id : "cartouche_table_tr1_td5", type : "button", id : 'btn_add_RLof_RCc', size : 3, value : 'Ajouter RL:OF + RC:C', style :"margin-left:10px", class: "avoid_from_selection",tabindex:3},

                    {object: "tr", parent_id : "cartouche_table", id: "cartouche_table_tr2"},
                    {object: "td", parent_id : "cartouche_table_tr2", appendChildText: "v2"},
                    //{object: "td", parent_id : "cartouche_table_tr2", id: "cartouche_table_v2_pictogramme"},
                    //{object: "svg", parent_id: "cartouche_table_v2_pictogramme", width: "40", height: "40", id: "sous-titre-chart_v2", viewBox: "0 0 80 80"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "0", y: "60", width: "20", height: "20", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "20", y: "40", width: "20", height: "40", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "40", y: "20", width: "20", height: "60", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "60", y: "0", width: "20", height: "80", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},

                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "1", y: "67", width: "6", height: "12", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "21", y: "47", width: "6", height: "32", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "41", y: "27", width: "6", height: "52", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "61", y: "7", width: "6", height: "72", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},

                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "7", y: "67", width: "6", height: "12", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "27", y: "47", width: "6", height: "32", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "47", y: "27", width: "6", height: "52", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "67", y: "7", width: "6", height: "72", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},

                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "13", y: "67", width: "6", height: "12", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "33", y: "47", width: "6", height: "32", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "53", y: "27", width: "6", height: "52", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},
                    //{object: "rect", parent_id: "sous-titre-chart_v2", x: "73", y: "7", width: "6", height: "72", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},

                    //{object: "td", parent_id : "cartouche_table_tr2", id: "cartouche_table_v2_pictogramme_image_svg"},
                    //{object: "img", parent_id : "cartouche_table_v2_pictogramme_image_svg", id: "img_svg_v2", width: "40", height: "40"},
                    {object: "td", parent_id : "cartouche_table_tr2", id: "cartouche_table_v2_pictogramme_image_png"},
                    {object: "img", parent_id : "cartouche_table_v2_pictogramme_image_png", id: "img_png_v2", width: "40", height: "40"},
                    {object: "td", parent_id : "cartouche_table_tr2", id: "display_score_base_v2", appendChildText: "undefined"},
                    {object: "td", parent_id : "cartouche_table_tr2", id: "display_score_temp_v2", appendChildText: "undefined"},
                    {object: "td", parent_id : "cartouche_table_tr2", id: "display_vecteur_v2"},
                    {object: "a", parent_id : "display_vecteur_v2", id: "link_vecteur_v2", target: '_blank', appendChildText: "undefined"},
                    {object: "input", parent_id : "display_vecteur_v2", type:"text", id: "refCVE_vecteur_v2", class: "only_for_html_export inputTypeTextToHidden", placeholder:"CVE de référence pour ce vecteur", STYLE:greyedStyle, readonly:"readonly", size:"13", title:"La CVE qui correspond au score/vecteur", onclick:"window.open('https://cyberwatch.internet.np/cve_announcements/'+this.value)",tabindex:-1},
                    {object: "input", parent_id : "display_vecteur_v2", type:"text", id: "refCVE_vecteur_v2_source", class: "only_for_html_export inputTypeTextToHidden", placeholder:"origine du vecteur", title:"La source qui a fournit le vecteur",tabindex:1},
                    {object: "input", parent_id : "display_vecteur_v2", type:"text", id: "refCVE_v2_NVDpublishDate", class: "only_for_html_export inputTypeTextToHidden", STYLE:greyedStyle, readonly:"readonly", size:"8", title:"Date de publication sur le site NVD",tabindex:-1},
                    {object: "input", parent_id : "display_vecteur_v2", type:"text", id: "refCVE_v2_NVDmodifiedDate", class: "only_for_html_export inputTypeTextToHidden", STYLE:greyedStyle, readonly:"readonly", size:"8", title:"Date de mise à jour sur le site NVD",tabindex:-1},
                    {object: "input", parent_id : "display_vecteur_v2", type:"text", id: "refCVE_v2_NVDsource", class: "only_for_html_export inputTypeTextToHidden", STYLE:greyedStyle, readonly:"readonly", title:"L'éditeur responsable de la publication sur le site NVD",tabindex:-1},

                    {object: "tr", parent_id : "cartouche_table", id: "cartouche_table_tr3"},
                    {object: "td", parent_id : "cartouche_table_tr3", appendChildText: "v3.1"},
                    //{object: "td", parent_id : "cartouche_table_tr3", id: "cartouche_table_v3_pictogramme"},
                    //{object: "svg", parent_id: "cartouche_table_v3_pictogramme", width: "40", height: "40", id: "sous-titre-chart_v3", viewBox: "0 0 80 80"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "0", y: "60", width: "20", height: "20", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "20", y: "40", width: "20", height: "40", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "40", y: "20", width: "20", height: "60", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "60", y: "0", width: "20", height: "80", stroke: "black", fill: "white", strokewidth: "2", class: "chart_contour"},

                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "1", y: "67", width: "6", height: "12", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "21", y: "47", width: "6", height: "32", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "41", y: "27", width: "6", height: "52", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "61", y: "7", width: "6", height: "72", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_left"},

                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "7", y: "67", width: "6", height: "12", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "27", y: "47", width: "6", height: "32", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "47", y: "27", width: "6", height: "52", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "67", y: "7", width: "6", height: "72", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_center"},

                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "13", y: "67", width: "6", height: "12", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "33", y: "47", width: "6", height: "32", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "53", y: "27", width: "6", height: "52", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},
                    //{object: "rect", parent_id: "sous-titre-chart_v3", x: "73", y: "7", width: "6", height: "72", stroke: "black", fill: "black", strokewidth: "0", class: "chart_inside_right"},

                    //{object: "td", parent_id : "cartouche_table_tr3", id: "cartouche_table_v3_pictogramme_image_svg"},
                    //{object: "img", parent_id : "cartouche_table_v3_pictogramme_image_svg", id: "img_svg_v3", width: "40", height: "40"},
                    {object: "td", parent_id : "cartouche_table_tr3", id: "cartouche_table_v3_pictogramme_image_png"},
                    {object: "img", parent_id : "cartouche_table_v3_pictogramme_image_png", id: "img_png_v3", width: "40", height: "40"},
                    {object: "td", parent_id : "cartouche_table_tr3", id: "display_score_base_v3", appendChildText: "undefined"},
                    {object: "td", parent_id : "cartouche_table_tr3", id: "display_score_temp_v3", appendChildText: "undefined"},
                    {object: "td", parent_id : "cartouche_table_tr3", id: "display_vecteur_v3"},
                    {object: "a", parent_id : "display_vecteur_v3", id: "link_vecteur_v3", target: '_blank', appendChildText: "undefined"},
                    {object: "input", parent_id : "display_vecteur_v3", type:"text", id: "refCVE_vecteur_v3", class: "only_for_html_export inputTypeTextToHidden", placeholder:"CVE de référence pour ce vecteur", style:greyedStyle, readonly:"readonly", size:"13", title:"La CVE qui correspond au score/vecteur", onclick:"window.open('https://cyberwatch.internet.np/cve_announcements/'+this.value)",tabindex:-1},
                    {object: "input", parent_id : "display_vecteur_v3", type:"text", id: "refCVE_vecteur_v3_source", class: "only_for_html_export inputTypeTextToHidden", placeholder:"origine du vecteur", title:"La source qui a fournit le vecteur",tabindex:2},
                    {object: "input", parent_id : "display_vecteur_v3", type:"text", id: "refCVE_v3_NVDpublishDate", class: "only_for_html_export inputTypeTextToHidden", STYLE:greyedStyle, readonly:"readonly", size:"8", title:"Date de publication sur le site NVD",tabindex:-1},
                    {object: "input", parent_id : "display_vecteur_v3", type:"text", id: "refCVE_v3_NVDmodifiedDate", class: "only_for_html_export inputTypeTextToHidden", STYLE:greyedStyle, readonly:"readonly", size:"8", title:"Date de mise à jour sur le site NVD",tabindex:-1},
                    {object: "input", parent_id : "display_vecteur_v3", type:"text", id: "refCVE_v3_NVDsource", class: "only_for_html_export inputTypeTextToHidden", STYLE:greyedStyle, readonly:"readonly", title:"L'éditeur responsable de la publication sur le site NVD",tabindex:-1}
                ];



                if (item_before) {
                    make_html_from_list_of_dict(item_before, list_input);
                }

            }

            function make_div_Left_content(item_before, arrayOfCVEs = []){
                // variables graphiques
                var font_size = "20px" ;
                var bordeStyle = defaultTableBorder ;
                var indentation_td = '40px';
                var id, mapGetNVD, v3_Vector, v2_Vector, v3_BaseScore, v2_BaseScore, CW_url, CW_id;
                var commonStyle = "float: left; overflow: auto ;height:85vh; width:auto;";

                // suppression du footer
                //document.querySelector("footer").remove();
                document.querySelector("footer").style.display="none";

                var articleContent = document.getElementsByClassName('content')[0];
                articleContent.style = articleContent.style+";"+commonStyle

                // autres variables
                var list_input = [
                    {object : "div", id :'div_CVEs', class: "content", style : "font-size :"+font_size+";"+commonStyle},
                    {object: "table", parent_id : "div_CVEs", appendChildText: "Cliquer sur le score pour l'utiliser"},
                    {object: "table", parent_id : "div_CVEs", appendChildText: "En-tete rouge = travail en cours, vert = fini", id : "pProcessMessage"},
                    {object: "table", parent_id : "div_CVEs", id: "tableCVEs", border : bordeStyle},
                    {object: "input", type:"hidden", parent_id : "tableCVEs", id : "tableCVEs_sorting", value : "original"},
                    /*{object: "tr", parent_id : "tableCVEs", id: "tableCVEs_Tr_test"},
							{object: "td", parent_id : "tableCVEs_Tr_test", id: "tableCVEs_Td_test_CVE"},
								{object: "input", type:"text", parent_id : "tableCVEs_Td_test_CVE", id : "tableCVEs_Td_test_CVE_ipt"},
							{object: "td", parent_id : "tableCVEs_Tr_test", id: "tableCVEs_Td_test_v2"},
								{object: "p", parent_id : ("tableCVEs_Td_test_v2"), id : ("tableCVEs_Td_test_v2_score")},
								{object: "input", type:"hidden", parent_id : ("tableCVEs_Td_test_v2"), id : ("tableCVEs_Td_test_v2_vector")},
							{object: "td", parent_id : "tableCVEs_Tr_test", id: "tableCVEs_Td_test_v3"},
								{object: "p", parent_id : ("tableCVEs_Td_test_v3"), id : ("tableCVEs_Td_test_v3_score")},
								{object: "input", type:"hidden", parent_id : ("tableCVEs_Td_test_v3"), id : ("tableCVEs_Td_test_v3_vector")},*/
                    {object: "tr", parent_id : "tableCVEs", id: "tableCVEs_THs", style : "color:red;"},
                    {object: "th", parent_id : "tableCVEs_THs", style :"padding-right:10px", appendChildText: "CVE", id : 'tableCVEs_THs_CVE'},
                    {object: "th", parent_id : "tableCVEs_THs", style :"padding-right:10px", appendChildText: "from", id : 'tableCVEs_THs_Editeur'},
                    {object: "th", parent_id : "tableCVEs_THs", style :"padding-right:10px", appendChildText: "v2", id : 'tableCVEs_THs_v2'},
                    {object: "th", parent_id : "tableCVEs_THs", style :"padding-right:10px", appendChildText: "v3", id : 'tableCVEs_THs_v3'},
                    {object: "th", parent_id : "tableCVEs_THs", style :"padding-right:10px", appendChildText: "v3 CNA", id : 'tableCVEs_THs_v3_CNA'},
                    {object: "th", parent_id : "tableCVEs_THs", style :"padding-right:10px", appendChildText: "Liens", colspan:3}
                    //{object: "th", parent_id : "tableCVEs_THs", style :"padding-right:10px", appendChildText: "vuldb"}
                ];

                for(var i=0;i<arrayOfCVEs.length;i++){
                    id = arrayOfCVEs[i];

                    // récupération des valeurs NVD
                    list_input.push({object: "tr", parent_id : "tableCVEs", id: "tableCVEs_Tr_"+id});
                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_CVE", appendChildText: id, style : "color:red;", originalOrder : i});
                    list_input.push({object: "input", type:"hidden", parent_id : ("tableCVEs_Td_"+id+"_CVE"), id : ("tableCVEs_Td_"+id+"_NVDpublishDate")});
                    list_input.push({object: "input", type:"hidden", parent_id : ("tableCVEs_Td_"+id+"_CVE"), id : ("tableCVEs_Td_"+id+"_NVDmodifiedDate")});
                    list_input.push({object: "input", type:"hidden", parent_id : ("tableCVEs_Td_"+id+"_CVE"), id : ("tableCVEs_Td_"+id+"_NVDsource")});

                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_Editeur", originalOrder : i});

                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_v2", originalOrder : i});
                    list_input.push({object: "p", parent_id : ("tableCVEs_Td_"+id+"_v2"), id : ("tableCVEs_Td_"+id+"_v2_score")});
                    list_input.push({object: "input", type:"hidden", parent_id : ("tableCVEs_Td_"+id+"_v2"), id : ("tableCVEs_Td_"+id+"_v2_vector")});

                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_v3", originalOrder : i});
                    list_input.push({object: "p", parent_id : ("tableCVEs_Td_"+id+"_v3"), id : ("tableCVEs_Td_"+id+"_v3_score")});
                    list_input.push({object: "input", type:"hidden", parent_id : ("tableCVEs_Td_"+id+"_v3"), id : ("tableCVEs_Td_"+id+"_v3_vector")});

                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_v3_CNA", originalOrder : i});
                    list_input.push({object: "p", parent_id : ("tableCVEs_Td_"+id+"_v3_CNA"), id : ("tableCVEs_Td_"+id+"_v3_CNA_score")});
                    list_input.push({object: "input", type:"hidden", parent_id : ("tableCVEs_Td_"+id+"_v3_CNA"), id : ("tableCVEs_Td_"+id+"_v3_CNA_vector")});

                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_nvd"});
                    list_input.push({object: "a", parent_id : ("tableCVEs_Td_"+id+"_nvd"), href : 'https://nvd.nist.gov/vuln/detail/'+id, target: "_blank", appendChildText : "NVD"});

                    //list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_vuldb"});
                    //list_input.push({object: "a", parent_id :  "tableCVEs_Td_"+id+"_vuldb", click : "$('#search_vuldb_'+id).submit();", appendChildText : "vuldb"});

                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_google"});
                    list_input.push({object: "a", parent_id :  "tableCVEs_Td_"+id+"_google", href : 'https://www.google.com/search?q=%22'+id+'%22+cvss', target: "_blank", appendChildText : "google"});

                    list_input.push({object: "td", parent_id : "tableCVEs_Tr_"+id, id: "tableCVEs_Td_"+id+"_CW"});
                    CW_url = 'https://cyberwatch.internet.np/cve_announcements/'+id;
                    CW_id = "tableCVEs_Td_"+id+"_CW";
                    list_input.push({object: "a", parent_id : CW_id , href : CW_url, target: "_blank", appendChildText : "CW"});
                    colorURLavailability(CW_url, CW_id);
                }

                if (item_before) {
                    make_html_from_list_of_dict(item_before, list_input);
                }

            }

            function set_visibility_class(classe,visibility){
                const items = document.querySelectorAll(classe);

                items.forEach(item => {
                    item.style.display = visibility;
                });
            }

            function get_article_to_string(){
                set_visibility_class(".avoid_from_selection","none");
                set_visibility_class(".only_for_html_export","none");
                set_visibility_class(".avoid_from_selection_but_for_html_export","none");
                $('.article').selectText();
                var text_article = "";
                if (window.getSelection) {
                    text_article = window.getSelection().toString();
                } else if (document.selection && document.selection.type != "Control") {
                    text_article = document.selection.createRange().text;
                }

                set_visibility_class(".avoid_from_selection_but_for_html_export","block");
                set_visibility_class(".avoid_from_selection","block");
                set_visibility_class(".only_for_html_export","block");
                document.getSelection().removeAllRanges(); // supprime les selections
                return text_article;
            }

            function get_text_and_child_from_innerHTML(innerHTML,children,indent=0) {
                // découpe du texte entre le texte dans des balise et le reste
                var regex_balise_html_complete = "<.*?>";
                var regex_balise_html_type="(?<=<)(.*?)(?=[ >])",balise_html_type="";
                var resultat_regex;
                var texte_avant_balise="";
                var output="";

                var balise_begin=0,balise_end=0;
                // parcours pour chaque enfant, rechercher le texte avant
                for(var i=0;i<children.length;i++){
                    balise_html_type = children[i].nodeName.toLowerCase();
                    regex_balise_html_complete="<"+balise_html_type+".*?>";

                    //console.log("découpe de "+innerHTML);
                    resultat_regex = innerHTML.match(regex_balise_html_complete);
                    //console.log(i+"/"+children.length+" : "+balise_html_type+"\n=>");
                    //console.log(children);
                    //console.log(resultat_regex);
                    // get position start de la balise
                    balise_begin=resultat_regex.index;
                    // texte avant balise = substrin(balise_end,balise_begin)
                    //texte_avant_balise = get_indent(indent)+innerHTML.substring(0,balise_begin);  // test de correction de placement de texte interne une balise
                    texte_avant_balise = innerHTML.substring(0,balise_begin);
                    // néttoyage si le avant balise est rempli de vide
                    if(/^\s*$/.test(texte_avant_balise)) { texte_avant_balise = "";}
                    // get position fin de la balise
                    balise_end= resultat_regex.index+resultat_regex[0].length+children[i].innerHTML.length; // avant une fermeture de balise eventuelle
                    //console.log("fin de balise avant fermante : "+balise_end);
                    innerHTML=innerHTML.substring(balise_end);// avant une fermeture de balise eventuelle
                    // rechercher une balise fermante
                    regex_balise_html_complete="</"+balise_html_type+">";
                    resultat_regex = innerHTML.match(regex_balise_html_complete);
                    //console.log("recherche "+regex_balise_html_complete+" dans \n"+innerHTML);
                    //console.log(resultat_regex);
                    if(resultat_regex){
                        balise_end=resultat_regex.index+resultat_regex[0].length;
                        //console.log("a trouver balise fermante: "+resultat_regex.index+" fin de la balise avant fermante : "+balise_end+" ; taille de la balise fermante : "+resultat_regex[0].length+" ==== > "+balise_end);
                        // reste du texte apres la balise fermante
                        innerHTML=innerHTML.substring(balise_end);
                    }
                    //console.log("texte avant balise : "+texte_avant_balise);
                    //console.log("texte apres balise fermante eventuelle : \n"+innerHTML);
                    output += texte_avant_balise;

                    output += recursive_get_html(children[i],(indent+1));
                }
                // pour le dernier enfant rechercher le texte après

                output+=innerHTML;
                return output;
            }

            function dom_object_attributes_to_string(obj,array_attrib_to_exclude=[]){
                var str_obj="", attribut,attribut_name;
                // Get object of all {name: value}
                const attrs_names = obj.getAttributeNames();
                for(var i=0;i<attrs_names.length;i++){
                    attribut_name = attrs_names[i];
                    //console.log("attrib: "+attribut_name+" ; val : "+obj.getAttribute(attribut_name));
                    if(array_attrib_to_exclude.includes(attribut_name)){
                        // attribut a eviter, on passe à la suite
                        continue ;
                    }
                    attribut = obj.getAttribute(attribut_name);
                    if(Array.isArray(attribut)) {
                        str_obj =str_obj + " "+attrs_names[i]+'="';
                        for(var j=0;j<attribut.length;j++){
                            str_obj =str_obj +attribut[j]+" "
                        }
                        str_obj +='" ';
                    }else{
                        str_obj =str_obj + " "+attribut_name+'="'+attribut+'" ';
                    }
                }

                // vérification des propriétés qui ne sont pas des attributs
                if(obj.value && !(str_obj.includes("value="))){
                    // remplace des guillements eventuels par des double single quote
                    obj.value = (obj.value).replaceAll('"',"''")
                    // remplace chevrons par "155	233	9B	10011011	›	&#155;	&rsaquo;	Single right-pointing angle quotation mark"
                    obj.value = (obj.value).replaceAll('>',"›")
                    // remplace chevrons par "139	213	8B	10001011	‹	&#139;	&lsaquo;	Single left-pointing angle quotation"
                    obj.value = (obj.value).replaceAll('<',"‹")
                    str_obj =str_obj + ' value="'+(obj.value)+'"';
                }

                return str_obj;
            }

            function recursive_get_html(dom_object,indent=0) {
                var a_exclure=false;
                var output="";
                var object_type = dom_object.nodeName.toLowerCase();
                var tag_begin="";
                var tag_end = "";
                var indent_text = get_indent(indent);

                var temp_value, has_changed;//utilisé pour une alteration le temps de la recuperation du code

                var array_attrib_to_exclude=['class','style', 'onclick', 'placeholder'];//
                switch(object_type) {
                    case 'area':
                    case 'base':
                    case 'br':
                    case 'col':
                    case 'command':
                    case 'embed':
                    case 'hr':
                    case 'keygen':
                    case 'link':
                    case 'meta':
                    case 'param':
                    case 'source':
                    case 'track':
                    case 'wbr':
                        tag_begin = indent_text+"<"+object_type+" />\n";
                        tag_end="";
                        break;
                    case "input":
                        if(dom_object.classList.contains("inputTypeTextToHidden")){
                            //console.log(dom_object.id+" : "+dom_object.value);
                            tag_begin = indent_text+"<"+object_type+" "+ ((dom_object_attributes_to_string(dom_object, array_attrib_to_exclude)).replace('type="text"','type="hidden"'))+" />\n";
                        }else{
                            tag_begin = indent_text+"<"+object_type+" "+ dom_object_attributes_to_string(dom_object, array_attrib_to_exclude)+" />\n";
                        }
                        tag_end="";
                        break;
                    case "img":
                        // modification des sources d'images
                        has_changed = false;
                        if(dom_object.getAttribute("id") === "img_png_v2" || dom_object.getAttribute("id") === "img_png_v3") {
                            temp_value = dom_object.getAttribute("src");
                            dom_object.setAttribute("src", dom_object.getAttribute("src_portail"));
                            has_changed = true;
                        }
                        tag_begin = indent_text+"<"+object_type+" "+dom_object_attributes_to_string(dom_object)+" />\n";
                        tag_end="";
                        if(has_changed) {dom_object.setAttribute("src" ,temp_value);}
                        break;
                    case "a" :
                        // modification des liens pour le portail
                        has_changed = false;
                        if(dom_object.getAttribute("id") === "link_vecteur_v2"){
                            temp_value = dom_object.getAttribute("href");
                            //dom_object.setAttribute("href", "https://portail-calid.intradef.gouv.fr/index.php/publications/vulnerabilites/calculateur_cvss_v2");
                            has_changed = true;

                        }else if(dom_object.getAttribute("id") === "link_vecteur_v3") {
                            temp_value = dom_object.getAttribute("href");
                            //dom_object.setAttribute("href", "https://portail-calid.intradef.gouv.fr/index.php/publications/vulnerabilites/calculateur_cvss_v3");
                            has_changed = true;
                        }
                        tag_begin = indent_text+"<"+object_type+" "+dom_object_attributes_to_string(dom_object,array_attrib_to_exclude)+">";
                        tag_end="</"+object_type+">\n";
                        //if(has_changed) {dom_object.setAttribute("href" ,temp_value);}
                        break;
                    case "svg":
                        tag_begin = indent_text+"<"+object_type+" "+dom_object_attributes_to_string(dom_object,array_attrib_to_exclude)+">";
                        tag_end="</"+object_type+">\n";
                        break;
                    case "rect":
                        tag_begin = indent_text+"<"+object_type+" "+dom_object_attributes_to_string(dom_object,array_attrib_to_exclude)+"/>\n";
                        break;
                    case 'h1' : //passer les h1 en h3
                    case 'h2' : //passer les h2 en h3
                        object_type = 'h3';
                        // repris de default
                        tag_begin = indent_text+"<"+object_type+">";
                        tag_end="</"+object_type+">\n";
                        // avertissement si pas de break avant default
                        break;
                    case "td": // ajouter un espace apres les td
                    case 'table' : // conserver le style
                        // repris de default
                        tag_begin = indent_text+"<"+object_type+" "+dom_object_attributes_to_string(dom_object,['class'])+">";
                        tag_end="</"+object_type+">\n";//<td>&nbsp;</td>
                        // avertissement si pas de break avant default
                        break;
                    case "p": // correction du P des références CVE en le remplacant par un BR a la fin
                        // par defaut si pas trouvé l'id des LI
                        tag_begin = indent_text+"<"+object_type+">";
                        tag_end="</"+object_type+">\n";
                        if(dom_object.id) {
                            if((dom_object.id).includes("li_CVE_")) {
                                // il s'agit de li modifiées pour ajouté un ID
                                tag_begin = indent_text;
                                tag_end = "<br>";
                            }
                        }
                        break;
                    default:
                        tag_begin = indent_text+"<"+object_type+">";
                        tag_end="</"+object_type+">\n";
                }
                if(object_type=="li") {
                    //console.log(dom_object);
                }
                // objet a exclure via leur classe
                for (var ne_doit_pas_avoir of ['avoid_from_selection','meta-logo','meta-certfr','meta-affaire','meta-pub-date', 'row-toolbox']){ // ,''
                    a_exclure = a_exclure || dom_object.classList.contains(ne_doit_pas_avoir)
                }
                if(a_exclure){
                    //console.log("on exclut",dom_object)
                    // on ne prend pas la section de bas de page
                    // output devrait etre à  "";
                } else if (dom_object.classList.contains("post-type-title")){ // remlplacer le titre H2 par un titre H1
                    output += indent_text+"<h1>"+dom_object.innerText+"</h1>";
                }else if(dom_object.children.length>0) {
                    // enfant, on descend dans l'arborescence html
                    output += tag_begin;
                    output += "\n";
                    //output +=
                    output += get_text_and_child_from_innerHTML(dom_object.innerHTML,dom_object.children,(indent+1));
                    output += indent_text;
                    output += tag_end;
                }else{
                    //console.log(dom_object);
                    //console.log(dom_object.innerText);
                    // pas d'enfants, on recupere le innerText
                    output += tag_begin;
                    output += dom_object.innerText;
                    output += tag_end;
                    //console.log(output);
                }
                return output;
            }

            function get_article_to_html(){
                try{ // paliatif exclure div_input
                    document.getElementById("div_input").classList.add("avoid_from_selection")
                }catch{
                } // # DEBUG
                try{ // paliatif exclure div_top_section
                    document.getElementById("div_top_section").classList.add("avoid_from_selection")
                }catch{
                } // # DEBUG
                set_visibility_class(".avoid_from_selection","none");
                set_visibility_class(".only_for_html_export","block");
                set_visibility_class(".avoid_from_selection_but_for_html_export","block");
                var regexCVE = /^CVE-[0-9]{4}-([0-9]*)$/gi ;

                document.getElementById("link_vecteur_v2").href = "/index.php/publications/vulnerabilites/14645-description-et-application-cvss";
                document.getElementById("link_vecteur_v3").href = "/index.php/publications/vulnerabilites/14645-description-et-application-cvss";

                if(regexCVE.test(document.getElementById("refCVE_vecteur_v2").value)){
                    document.getElementById("link_vecteur_v2").href = "https://calid-cyberwatch-veille.intradef.gouv.fr/cve_announcements/"+(document.getElementById("refCVE_vecteur_v2").value).toUpperCase();
                }
                regexCVE = /^CVE-[0-9]{4}-([0-9]*)$/gi ; // repete car bug sinon ????
                if(regexCVE.test(document.getElementById("refCVE_vecteur_v3").value)){
                    document.getElementById("link_vecteur_v3").href = "https://calid-cyberwatch-veille.intradef.gouv.fr/cve_announcements/"+(document.getElementById("refCVE_vecteur_v3").value).toUpperCase();
                }


                var article = document.querySelector('article');
                var code_article = "<!DOCTYPE html>\n" + recursive_get_html(article);

                //console.log(article);
                //console.log(code_article);


                //set_visibility_class(".only_for_html_export","block");
                set_visibility_class(".avoid_from_selection","block");
                return code_article;
            }

            function stockageToFieldsV2(referenceANSSI = document.title){
                if(GM_getValue(referenceANSSI+".vecteurV2")) {
                    document.getElementById('lock_v2').checked = false;
                    update_options(GM_getValue(referenceANSSI+".vecteurV2"));
                    // lock v2
                    document.getElementById('lock_v2').checked = true;
                }
                if(GM_getValue(referenceANSSI+".CVEv2")) {document.getElementById("refCVE_vecteur_v2").value = GM_getValue(referenceANSSI+".CVEv2");}
                if(GM_getValue(referenceANSSI+".vecteurV2Source")) {document.getElementById("refCVE_vecteur_v2_source").value = GM_getValue(referenceANSSI+".vecteurV2Source");}
                if(GM_getValue(referenceANSSI+".refCVE_v2_NVDpublishDate")) {document.getElementById("refCVE_v2_NVDpublishDate").value = GM_getValue(referenceANSSI+".refCVE_v2_NVDpublishDate");}
                if(GM_getValue(referenceANSSI+".refCVE_v2_NVDmodifiedDate")) {document.getElementById("refCVE_v2_NVDmodifiedDate").value = GM_getValue(referenceANSSI+".refCVE_v2_NVDmodifiedDate");}
                if(GM_getValue(referenceANSSI+".refCVE_v2_NVDsource")) {document.getElementById("refCVE_v2_NVDsource").value = GM_getValue(referenceANSSI+".refCVE_v2_NVDsource");}
            }
            function stockageToFieldsV3(referenceANSSI = document.title){
                if(GM_getValue(referenceANSSI+".vecteurV3")) {
                    document.getElementById('lock_v3').checked = false;
                    update_options(GM_getValue(referenceANSSI+".vecteurV3"));
                }
                if(GM_getValue(referenceANSSI+".vecteurV2") || GM_getValue(referenceANSSI+".vecteurV3")) {
                    // unlock
                    //document.getElementById('lock_v2').checked = false;
                    //document.getElementById('lock_v3').checked = false;

                }
                if(GM_getValue(referenceANSSI+".CVEv3")) {document.getElementById("refCVE_vecteur_v3").value = GM_getValue(referenceANSSI+".CVEv3");}
                if(GM_getValue(referenceANSSI+".vecteurV3Source")) {document.getElementById("refCVE_vecteur_v3_source").value = GM_getValue(referenceANSSI+".vecteurV3Source");}
                if(GM_getValue(referenceANSSI+".refCVE_v3_NVDpublishDate")) {document.getElementById("refCVE_v3_NVDpublishDate").value = GM_getValue(referenceANSSI+".refCVE_v3_NVDpublishDate");}
                if(GM_getValue(referenceANSSI+".refCVE_v3_NVDmodifiedDate")) {document.getElementById("refCVE_v3_NVDmodifiedDate").value = GM_getValue(referenceANSSI+".refCVE_v3_NVDmodifiedDate");}
                if(GM_getValue(referenceANSSI+".refCVE_v3_NVDsource")) {document.getElementById("refCVE_v3_NVDsource").value = GM_getValue(referenceANSSI+".refCVE_v3_NVDsource");}
            }

            function stockageToFields(){
                stockageToFieldsV2();
                stockageToFieldsV3();
            }

            function fieldsToStockage(){
                var referenceANSSI = document.title;
                var titreText = $("h1").text();
                titreText = titreText.substr(titreText.indexOf(" ") + 1);

                //console.log("fieldsToStockage : CVE "+document.getElementById("refCVE_vecteur_v2").value);
                //console.log(document.getElementById("refCVE_vecteur_v2"));

                GM_setValue(referenceANSSI+".titre",titreText);
                GM_setValue(referenceANSSI+".vecteurV2",document.getElementById("link_vecteur_v2").innerText);
                GM_setValue(referenceANSSI+".scoreV2",document.getElementById("display_score_base_v2").innerText);
                GM_setValue(referenceANSSI+".vecteurV3",document.getElementById("link_vecteur_v3").innerText);
                GM_setValue(referenceANSSI+".scoreV3",document.getElementById("display_score_base_v3").innerText);
                GM_setValue(referenceANSSI+".CVEv2",document.getElementById("refCVE_vecteur_v2").value);
                GM_setValue(referenceANSSI+".CVEv3",document.getElementById("refCVE_vecteur_v3").value);
                GM_setValue(referenceANSSI+".vecteurV2Source",document.getElementById("refCVE_vecteur_v2_source").value);
                GM_setValue(referenceANSSI+".vecteurV3Source",document.getElementById("refCVE_vecteur_v3_source").value);

                GM_setValue(referenceANSSI+".refCVE_v2_NVDpublishDate",document.getElementById("refCVE_v2_NVDpublishDate").value);
                GM_setValue(referenceANSSI+".refCVE_v2_NVDmodifiedDate",document.getElementById("refCVE_v2_NVDmodifiedDate").value);
                GM_setValue(referenceANSSI+".refCVE_v2_NVDsource",document.getElementById("refCVE_v2_NVDsource").value);
                GM_setValue(referenceANSSI+".refCVE_v3_NVDpublishDate",document.getElementById("refCVE_v3_NVDpublishDate").value);
                GM_setValue(referenceANSSI+".refCVE_v3_NVDmodifiedDate",document.getElementById("refCVE_v3_NVDmodifiedDate").value);
                GM_setValue(referenceANSSI+".refCVE_v3_NVDsource",document.getElementById("refCVE_v3_NVDsource").value);
            }

            // fonction qui anime un message, retourne un interval ID
            function waitingAnimation(messageId, message="Work in Progress ", debug=false){
                if(debug){
                    console.log("waitingAnimation : creation de l'interval");
                }
                add_log_to_CMB("[waitingAnimation] creation de l'interval");
                var compteur=0; // utilisation en dehors de la fonction ashynchrone
                var compteurMax=3;
                var returnedInterval = setInterval(function intervalAnimation(messageId, message, compteurMax){
                    for(var i=0;i<compteur;i++){
                        message = message + ". ";
                    }
                    document.getElementById(messageId).innerText = message;
                    //console.log("ecriture DAYDAY");
                    compteur+=1;
                    if(compteur>compteurMax){compteur=0;}
                }, 250, messageId, message, compteurMax);
                if(debug){
                    console.log("retour de l'interval "+returnedInterval);
                }
                add_log_to_CMB("[waitingAnimation] retour de l'interval "+returnedInterval);
                return returnedInterval

            }

            // FONCTIONS\graphique + alteration de page />
            // < FONCTIONS\Click
            // < FONCTION\Click\get TXT
            $(document).delegate("#txt_gen","click",function(){
                var text_article = get_article_to_string();
                // remplacer cyberwatch np par cyberwatch dr
                //text_article = text_article.replaceAll("cyberwatch.internet.np","calid-cyberwatch-veille.intradef.gouv.fr");
                text_article = text_article.replace(/cyberwatch\.internet\.np/g,"calid-cyberwatch-veille.intradef.gouv.fr");

                text_article=text_article+"\n\nversion du script d'export utilisée : "+scriptVersion;
                text_article=text_article+"\nVecteur cvss V2 ("+document.getElementById("display_score_base_v2").innerText+") CVE : "+document.getElementById("refCVE_vecteur_v2").value+" from source : "+document.getElementById("refCVE_vecteur_v2_source").value+"\n";
                text_article+=document.getElementById("link_vecteur_v2").innerText; // ajout du cartouche input

                var lines = text_article.split('\n');
                lines.splice(1,1);
                var output = lines.join('\n');

                var title = $("h1").text();
                title = title.substr(title.indexOf(" ") + 1);
                var filename = '['+document.title+'] - '+title+'.txt';

                var a = window.document.createElement('a');
                a.href = window.URL.createObjectURL(new Blob([output], {type: 'text/plain'}));
                a.download = filename;
                document.body.appendChild(a);
                a.click();
                document.body.removeChild(a);

                fieldsToStockage();
            });
            // FONCTION\Click\get TXT />


            // < FONCTION\Click\get HTML
            $(document).delegate("#html_gen","click",async function(){
                //location.href = "javascript:void(update_cartouche_input());";// mise à jour de l'input box (pour eviter le cas d'un collage de code v3 avec l'attribut authentification déja changé
                var text_article = get_article_to_html();
                // remplacer cyberwatch np par cyberwatch dr
                //text_article = text_article.replaceAll("cyberwatch.internet.np","calid-cyberwatch-veille.intradef.gouv.fr");
                text_article = text_article.replace(/cyberwatch\.internet\.np/g,"calid-cyberwatch-veille.intradef.gouv.fr");

                var lines = text_article.split('\n');
                lines.splice(1,1);
                var output = lines.join('\n');

                var title = $("h1").text();
                title = title.substr(title.indexOf(" ") + 1);
                var filename = '['+document.title+'] - '+title+'.html';

                var a = window.document.createElement('a');
                a.href = window.URL.createObjectURL(new Blob([output], {type: 'text/plain'}));
                a.download = filename;
                document.body.appendChild(a);
                // mise à jour du stockage
                if(allowToStore()){
                    // mise à jour du local
                        fieldsToStockage();

                    // récupération du titre du fichier distant (combinaison des AVI et ALE les plus elevés
                        var newTitle = makeNewTitle();//title pasteBin
                        if(!newTitle){alert("erreur au niveau de makeNewTitle(); téléchargement impossible");return false;}


                    // Mise à jour du distant
                        var content = "";
                        content = extractGMstorage();
                        var fonctionErrPost = function(err, post){
                            if(err) {
                                console.log("err:"+err);
                                add_log_to_CMB("[MAIN] download HTML : erreur d'ajout sur le Git")
                            } else {
                                console.log("post:"+post);
                                add_log_to_CMB("[MAIN] download HTML : Reussite d'ajout sur le Git")
                            }
                        }
                        try{
                            await setGitStorageContent(newTitle, content,fonctionErrPost);
                        }catch (error){
                            console.log("[download] ERROR update storage :"+error.message);
                            add_log_to_CMB("[download] ERROR update storage :"+error.message);
                        }
                }
                // téléchargement.
                a.click();
                document.body.removeChild(a);

            });
            // FONCTION\Click\get HTML />
            // < FONCTIONS\Selection de text
            jQuery.fn.selectText = function(){
                var doc = document
                , element = this[0]
                , range, selection
                ;
                if (doc.body.createTextRange) {
                    range = document.body.createTextRange();
                    range.moveToElementText(element);
                    range.select();
                } else if (window.getSelection) {
                    selection = window.getSelection();
                    range = document.createRange();
                    range.selectNodeContents(element);
                    selection.removeAllRanges();
                    selection.addRange(range);
                }
            };
            // FONCTIONS\Selection de text />
        //}
        //if("FONCTIONS TABLEAU CVE"){
            // FONCTION\Click\set Sorting table>
            ///// https://www.guru99.com/quicksort-in-javascript.html
            // first call to quick sort
            // var sortedArray = quickSort(items, 0, items.length - 1);
            function swap(items, leftIndex, rightIndex){
                var temp = items[leftIndex];
                items[leftIndex] = items[rightIndex];
                items[rightIndex] = temp;
            }
            function partition(items, left, right) {
                var pivot   = items[Math.floor((right + left) / 2)], //middle element
                    i       = left, //left pointer
                    j       = right; //right pointer
                while (i <= j) {
                    while (items[i] < pivot) {
                        i++;
                    }
                    while (items[j] > pivot) {
                        j--;
                    }
                    if (i <= j) {
                        swap(items, i, j); //sawpping two elements
                        i++;
                        j--;
                    }
                }
                return i;
            }

            function quickSort(items, left, right) {
                var index;
                if (items.length > 1) {
                    index = partition(items, left, right); //index returned from partition
                    if (left < index - 1) { //more elements on the left side of the pivot
                        quickSort(items, left, index - 1);
                    }
                    if (index < right) { //more elements on the right side of the pivot
                        quickSort(items, index, right);
                    }
                }
                return items;
            }
            /////// FIN  https://www.guru99.com/quicksort-in-javascript.html

            function sortTable(tableID, columnToSort=0 ,ascending=true, compareType="string", originalOrderAttributeName="",verbose=true) {// compareType="string" / "float" / "int"
                var table, rows, switching, i, x, y, shouldSwitch;
                var xToCompare, yToCompare;
                table = document.getElementById(tableID);
                switching = true;
                /*Make a loop that will continue until no switching has been done:*/
                while (switching) {
                    //start by saying: no switching is done:
                    switching = false;
                    rows = table.rows;
                    /*Loop through all table rows (except the first, which contains table headers):*/
                    for (i = 1; i < (rows.length - 1); i++) {
                        //start by saying there should be no switching:
                        shouldSwitch = false;
                        /*Get the two elements you want to compare, one from current row and one from the next:*/
                        x = rows[i].getElementsByTagName("TD")[columnToSort];
                        y = rows[i + 1].getElementsByTagName("TD")[columnToSort];
                        //check if the two rows should switch place:
                        if(originalOrderAttributeName == "") {
                            xToCompare = x.innerText.toLowerCase();
                            yToCompare = y.innerText.toLowerCase();
                        }else{
                            xToCompare = x.getAttribute(originalOrderAttributeName).toLowerCase();
                            yToCompare = y.getAttribute(originalOrderAttributeName).toLowerCase();
                        }
                        if(verbose){console.log("sortTable (str): "+xToCompare+(ascending ?">":"<")+" "+yToCompare);}

                        switch(compareType){
                            case "string":
                                break;
                            case "float":
                                xToCompare = parseFloat(xToCompare);
                                if(isNaN(xToCompare)){xToCompare=-1;}
                                yToCompare = parseFloat(yToCompare);
                                if(isNaN(yToCompare)){yToCompare=-1;}
                                break;
                            case "int":
                                xToCompare = parseInt(xToCompare);
                                if(isNaN(xToCompare)){xToCompare=-1;}
                                yToCompare = parseInt(yToCompare);
                                if(isNaN(yToCompare)){yToCompare=-1;}
                                break;
                        }
                        if(verbose){console.log("sortTable ("+compareType+"): "+xToCompare+(ascending ?">":"<")+" "+yToCompare);}

                        if ( ( ascending ? (xToCompare > yToCompare) : (xToCompare < yToCompare) ) ) {
                            if(verbose){console.log(true);}
                            //if so, mark as a switch and break the loop:
                            shouldSwitch = true;
                            break;
                        }
                    }
                    if (shouldSwitch) {
                        /*If a switch has been marked, make the switch and mark that a switch has been done:*/
                        rows[i].parentNode.insertBefore(rows[i + 1], rows[i]);
                        switching = true;
                    }
                }
                return true;
            }

            function getHeaderIDfromIndexTableCVE(index){ // lié a getCompareTypeFromIndexTableCVE
                var listOfSortableHeaderID = ['tableCVEs_THs_CVE', "tableCVEs_THs_Editeur", "tableCVEs_THs_v2", "tableCVEs_THs_v3", "tableCVEs_THs_v3_CNA"];
                if(index<0 || index >= listOfSortableHeaderID.length){index=0}
                return listOfSortableHeaderID[index];
            }
            function getCompareTypeFromIndexTableCVE(index){ // lié a getHeaderIDfromIndexTableCVE
                var listOfCompareType = ['string', 'string', "float", "float", "float"];
                if(index<0 || index >= listOfCompareType.length){index=0}
                return listOfCompareType[index];
            }
            function makeSortingClick(clickableID, sortingStateID, tableID, origianlValueAttributeName, otherSortableHeaderToClean, columnToSort){
                // Ajout de l'action de tri en cliquant sur l'entete CVE
                $(document).delegate("#"+clickableID,"click",function(){
                    var debug=false;
                    // animation d'attente pendant le tri
                    var oldStyle = document.getElementById("tableCVEs_THs").style.color;
                    var oldMessage = document.getElementById("pProcessMessage").innerText;
                    document.getElementById("tableCVEs_THs").style.color = "orange";
                    document.getElementById("pProcessMessage").innerText = "";
                    //console.log("l'ancien message etait "+oldMessage);
                    var intervalAnimation = waitingAnimation("pProcessMessage", "tri en cours ");
                    var lstSortingValue=["original","ascending","descending"];
                    var actualSorting = document.getElementById(sortingStateID).value;

                    var ascending = true;
                    var originalOrderAttributeName="";
                    var compareType=getCompareTypeFromIndexTableCVE(columnToSort)

                    var nextSortingIndex = (lstSortingValue.indexOf(actualSorting)+1)%lstSortingValue.length; // donne le reste de la division par 3 (la longueur de la liste) de 1 + l'actuelle tri
                    if(debug){console.log("trie du tableau de "+(document.getElementById(sortingStateID).value)+" a "+lstSortingValue[nextSortingIndex]);}
                    document.getElementById(sortingStateID).value = lstSortingValue[nextSortingIndex];


                    var reg = /\[.*\]/
                    var oldText = document.getElementById(clickableID).innerText;
                    if(debug){console.log("index de lstSortingValue="+nextSortingIndex);}
                    switch(lstSortingValue[nextSortingIndex]){
                        case "original":
                            originalOrderAttributeName = origianlValueAttributeName;
                            document.getElementById(clickableID).innerText = oldText.replace(reg, "") + "";
                            compareType = "int";
                            break;
                        case "ascending":
                            document.getElementById(clickableID).innerText = oldText.replace(reg, "") + "[\u25B2]";
                            break;
                        case "descending":
                            document.getElementById(clickableID).innerText = oldText.replace(reg, "") + "[\u25BC]";
                            ascending=false;
                            break;
                    }
                    for(var i=0;i<otherSortableHeaderToClean.length;i++){
                        if(otherSortableHeaderToClean[i] == clickableID){continue;}
                        oldText = document.getElementById(otherSortableHeaderToClean[i]).innerText;
                        document.getElementById(otherSortableHeaderToClean[i]).innerText = oldText.replace(reg, "");
                    }
                    if(debug){console.log("to sort : "+tableID+"["+columnToSort+"] asc?"+ascending+" type :"+compareType+", ori:"+originalOrderAttributeName);}
                    var hasSorted = sortTable(tableID, columnToSort,ascending,compareType, originalOrderAttributeName,debug);

                    // fin de l'animation
                    if(debug){console.log("fin animation");}
                    add_log_to_CMB("[makeSortingClick] fin animation", 0);
                    clearInterval(intervalAnimation);
                    document.getElementById("pProcessMessage").innerText = oldMessage;
                    document.getElementById("tableCVEs_THs").style.color = oldStyle;

                });
            }

            // FONCTION\Click\set Sorting table />
        //}

        // FONCTIONS\Divers
        // FONCTIONS\Divers />
        // FONCTIONS />
    //}
    // < MAIN
    async function main(){
        // Initialisation
        //make_download_txt(); // crée le bouton de téléchargement au format txt
        var btn_id = make_download_html(); // crée le bouton de téléchargement au format html
        let className_itemBefore_top_section = "meta-post-type"
        let itemBefore_top_section = getLastItemOfClass(className_itemBefore_top_section, debug);
        if(itemBefore_top_section==null){
            itemBefore_top_section=document.body.firstChild; // au lieu de document.getElementById(btn_id)
        }
        make_top_section(itemBefore_top_section);


        // Mise à jour du stockage
        try{
            var voidvarStorage = await updateStorage();
        }catch (error){
            console.log("[MAIN] ERROR update storage :"+error.message);
            add_log_to_CMB("[MAIN] ERROR update storage :"+error.message);
        }

        let lastItemToolBox = getLastItemInTheToolBox();
        let DOM_toolbox=getParent(getParent(lastItemToolBox)); // <div toolbox><conteneur lastItemToolBox><lastItemToolBox>
        make_input_cartouche_out(DOM_toolbox);
        document.getElementById('lock_v2').addEventListener('change', computeCVSS,false);
        document.getElementById('lock_v3').addEventListener('change', computeCVSS,false);
        document.getElementById('v2boxFromGMstorage').addEventListener('change', updateCartoucheFromStorageOrAutoNVD,false);
        document.getElementById('v3boxFromGMstorage').addEventListener('change', updateCartoucheFromStorageOrAutoNVD,false);
        add_functions_onchange();
        make_display_cartouche_out(document.getElementsByClassName("meta-title")[0]);
        add_functions_onclick();

        stockageToFields();
        computeCVSS();


        // récupération des CVE
        var CWEs = [];
        var CVEs=[] ; //"CVE-2022-40684"];

        // Correction éventuelle des erreurs, très ponctuelle
        // var regex_CVE = /((CVE)\-([0-9]{4})\-[0-9]{1,5})/g;
        var regex_CVE = /((CVE)\-([0-9]{4})\-[0-9]*)/g;
        var regex_lienCVE = /(http[s]{0,1}:\/\/(.*))/g;

        if(true){
            CVEs=prepare_CVElistFromCurrentPage();
        }else{
            $('.article-content li').each(function(){
                var find_cve = $(this).text();
                if (find_cve.indexOf("Référence CVE") >= 0) {
                    // on est dans un li CVE
                    $(this).css("position","relative");
                    $(this).find('a:first').css({
                        'position' : 'absolute',
                        'right' : '66%'
                    });

                    // récupération de l'ID CVE
                    find_cve = find_cve.replace("http"," http") ; // le lien est colé à l'ID de la CVE sinon
                    var cve="", lien="";
                    // renvoie null dans certains cas alors que cela ne devrait pas
                    // cve = regex_CVE.exec(find_cve);
                    // cve =(Array.isArray(cve)?cve[0]:0);
                    // remplacement par :
                    var a1_txt_cve_space_splitted = find_cve.split(" ");
                    if (Array.isArray(a1_txt_cve_space_splitted)) {
                        for (var i = 0; i < a1_txt_cve_space_splitted.length; i++) { // recherche de l'id de la cve (du li)
                            //console.log("test : "+a1_txt_cve_space_splitted[i]);
                            cve = regex_CVE.exec(a1_txt_cve_space_splitted[i]);
                            cve =(Array.isArray(cve)?cve[0]:0);
                            //console.log("temp cve : "+cve);
                            if(cve) {break;}
                        }
                        for (i = 0; i < a1_txt_cve_space_splitted.length; i++) { // recherche de l'id de la cve (du li)
                            //console.log("test : "+a1_txt_cve_space_splitted[i]);
                            lien = regex_lienCVE.exec(a1_txt_cve_space_splitted[i]);
                            lien =(Array.isArray(lien)?lien[0]:"");
                            //console.log("temp cve : "+cve);
                            if(lien) {break;}
                        }
                    }
                    //console.log("lien et CVE trouvé ; CVE="+cve+", lien="+lien);

                    // comparaison du lien et de la CVE
                    var CVE_du_lien = regex_CVE.exec(lien);
                    CVE_du_lien =(Array.isArray(CVE_du_lien)?CVE_du_lien[0]:0);
                    if(CVE_du_lien && CVE_du_lien == cve){
                        //console.log("Lien de la CVE "+cve+" OK");
                    }else{
                        console.log("Lien de la CVE "+cve+" NOK");
                        add_log_to_CMB_PAGE("[MAIN] - foreach li : Lien de la CVE "+cve+" NOK", 0);
                    }


                    if(CVEs.includes(cve)){
                        // la CVE est déjà listée, on supprime la ligne
                        console.log("[MAIN] - foreach li : "+cve+" en doublon ; on supprime la ligne");
                        add_log_to_CMB_PAGE("[MAIN] - foreach li : "+cve+" en doublon ; on supprime la ligne", 0);
                        this.remove();
                        return 0;
                    }
                    var DomHTML = this.innerHTML
                    //console.log("avant : ");
                    //console.log(DomHTML);
                    var rx = new RegExp(/(.*?)<br(.*?)>/);
                    var lstMatch = rx.exec(DomHTML);
                    var replaceWithP = '<p id="li_CVE_'+cve+'">'+lstMatch[1]+"</p>";
                    //console.log("new txt : "+replaceWithP);
                    this.innerHTML = (DomHTML).replace(rx,replaceWithP);
                    //console.log("apres : ");
                    //console.log(this.innerHTML);
                    // vérification des liens
                    var regTestLiens = new RegExp(/<a(.*?)\/a>/);
                    lstMatch = regTestLiens.exec(DomHTML);
                    //console.log(lstMatch);
                    var resultatRegex = lstMatch[1];
                    if(resultatRegex.includes(cve)){
                        //console.log("Vérification des liens OK");
                    }else{
                        console.log("Vérification des liens NOK");
                        add_log_to_CMB_PAGE("[MAIN] - foreach li : Vérification des liens NOK", 0);
                        alert("Problème avec le lien de la cve "+cve);
                    }
                    // ajout du lien cyberwatch
                    //var url_cyberwatch = "https://calid-cyberwatch-veille.intradef.gouv.fr/cve_announcements/"+cve;
                    var url_cyberwatch = "https://cyberwatch.internet.np/cve_announcements/"+cve;
                    $(this).append("<br class = 'avoid_from_selection_but_for_html_export' /><a href='"+url_cyberwatch+"' target='_blank' style=' right: 66%;' class='only_for_html_export'>"+url_cyberwatch+"</a>");

                    // création de la div avec les différents ajouts de score
                    $(this).append('<div id="' + cve + '" class = "avoid_from_selection"></div>'); // avoid from selection est utilisé dans get_article_to_string pour cacher les div et ainsi pas les enregistrer dans le fichier texte généré
                    $('#'+cve).css({
                        'display' : 'inline-flex',
                        'position' : 'relative',
                        'left' : '36%'
                    });


                    var score_cveD = '<a href="https://www.google.com/search?q=%22'+cve+'%22" target="_blank">Google:'+cve+'</a>&nbsp;'
                    score_cveD = score_cveD +'<a style="background-color:black" href=https://www.cvedetails.com/cve-details.php?cve_id='+cve+' target="_blank" id="'+cve+'_cveDetails">??</a>';

                    $('#'+cve).append(score_cveD);
                    $('#'+cve).append('<span id="'+cve+'_NVD" style="margin-left:20px;"> </span>');
                    $('#'+cve).append('<span id="'+cve+'_vuldb"><img src="" width="22" height="22" style="margin: 0 0 3px 20px"><form id="search_vuldb_'+cve+'" action="https://vuldb.com/?search" method="post" target="_blank"><input name="search" type="hidden" value="'+cve+'"></form></span>');
                    //$('#'+cve).append('<span id="'+cve+'_sourceclear" style="margin-left:20px;"> <a href="https://www.sourceclear.com/vulnerability-database/search#query='+cve+' type:vulnerability" target="_blank"><img src="https://www.sourceclear.com/vulnerability-database/images/vcfavicon.ico" width="22" height="22"></a> </span>');
                    //$('#'+cve).append('<span id="'+cve+'_circl" style="margin-left:20px;"> <a href="https://cve.circl.lu/cve/'+cve+'" target="_blank"><img src="https://cve.circl.lu/static/img/favicon.ico" width="22" height="22"></a> </span>');
                    $('#'+cve).append('<span id="'+cve+'_feedly" style="margin-left:20px;"> <a href="https://feedly.com/cve/'+cve+'" target="_blank"><img src="https://feedly.com/favicon.ico" width="22" height="22"></a> </span>');
                    $("#"+cve+"_vuldb").mousedown(function(event) {
                        switch (event.which) {
                            case 1: // Left
                                $('#'+cve+'_vuldb').children(":first").css({
                                    'outline-style' : 'solid',
                                    'outline-width' : '3px',
                                    'outline-color' : '#2860ff'
                                });
                                $(document).delegate("#"+cve+"_vuldb","click",function(){
                                    updateClipboard(cve);
                                    $('#search_vuldb_'+cve).submit();
                                });
                                break;
                            case 2: // Middle
                                $('#'+cve+'_vuldb').children(":first").removeAttr('style');
                                $('#'+cve+'_vuldb').children(":first").css("margin","0 0 3px 20px");
                                break;
                            case 3: // Right
                                $('#'+cve+'_vuldb').children(":first").css("filter","invert(1)");
                                break;
                        }
                    });


                    //getCyberwatchScore(cve);
                    get_cveDetails(cve);
                    CVEs.push(cve);
                    //get_vuldb(cve[2]);
                }

            });
        }
 //debug(CVEs)

        if(CVEs.length == 0) {
            var tablesAct, tableAct_trs, k,l, cveTD;
            tablesAct = document.getElementsByClassName("specificStd");
            console.log(tablesAct);
            for(k=0;k<tablesAct.length;k++){
                cveTD = regex_CVE.exec(tablesAct[k].innerText);
                cveTD =(Array.isArray(cveTD)?cveTD[0]:0);
                if(cveTD) {CVEs.push(cveTD);}
            }
        }
        // ajout du tableau des CVE à gauche du content
        var unique = [...new Set(CVEs)];
        CVEs = unique;
        // CVEs = [ "CVE-2022-21476", "CVE-2022-21449"] // custom list if needed
        try{ // alimentation du tableau des CVEs
            make_div_Left_content(document.getElementById('header'),CVEs);
            compteurCVEtoLoad = CVEs.length;
            compteurCVEloaded = 0;
            var intervalAnimation = waitingAnimation("pProcessMessage","Chargement des Score NVD "); // pour exemple d'animation, utiliser l'avis https://www.cert.ssi.gouv.fr/avis/CERTFR-2022-AVI-615/ et commenter le test d'abscence de CVE
            for(var j=0;j<CVEs.length;j++){ loadScore(CVEs[j], intervalAnimation);}
            add_functions_onclick_arrayOfCVE(CVEs);
        } catch (error) {
            console.log("Erreur pendant la phase de création du tableau : "+error);
        }
        try{ // création du tri
            if(CVEs.length<=0) {
                clearInterval(intervalAnimation);
                document.getElementById("pProcessMessage").innerText = "Pas de CVE"
                document.getElementById("tableCVEs_THs_CVE").innerText = "Pas de CVE"
                document.getElementById("tableCVEs_THs").style.color = "green";
                document.getElementById("refCVE_vecteur_v2_source").value = "Pas de CVE"
                document.getElementById("refCVE_vecteur_v3_source").value = "Pas de CVE"
            }else{
                var s = (CVEs.length == 1?"":"s");
                document.getElementById("tableCVEs_THs_CVE").innerText = "CVE"+s+" ("+CVEs.length+")";

                // Ajout de l'action de tri en cliquant sur l'entete CVE
                var listOfSortableHeaderID = ['tableCVEs_THs_CVE', "tableCVEs_THs_Editeur", "tableCVEs_THs_v2", "tableCVEs_THs_v3", "tableCVEs_THs_v3_CNA"]; // lié a getCompareTypeFromIndexTableCVE getHeaderIDfromIndexTableCVE
                for(var HeaderID=0;HeaderID<listOfSortableHeaderID.length;HeaderID++){
                    makeSortingClick(listOfSortableHeaderID[HeaderID], "tableCVEs_sorting", "tableCVEs", "originalorder", listOfSortableHeaderID, HeaderID);
                }
            }
        } catch (error) {
            console.log("Erreur pendant la phase de création du tri : "+error);
        }


        if(CVEs.length==1){
            document.getElementById('refCVE_vecteur_v2').value = CVEs[0];
            document.getElementById('refCVE_vecteur_v3').value = CVEs[0];
        }
        //getCVSS_fromStorage("CVE-2022-1012", "CVEv2");
    }
    window.addEventListener('load', function() {
        setTimeout(function() {
            //console.clear();
            main(); }, 100);
    }, false);


    // MAIN />
})();