IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Contribuez Discussion :

Système de requêtes sur des données JSON [Sources]


Sujet :

Contribuez

  1. #1
    Membre confirmé
    Profil pro
    Inscrit en
    Février 2009
    Messages
    354
    Détails du profil
    Informations personnelles :
    Localisation : France, Sarthe (Pays de la Loire)

    Informations forums :
    Inscription : Février 2009
    Messages : 354
    Points : 491
    Points
    491
    Par défaut Système de requêtes sur des données JSON
    Ce script permet de faire des sélections via une syntaxe ce rapprochant des sélecteurs css. Mais aussi de tester si un objet valide un sélecteur.
    Il n'y a que 9 sélecteurs définit pour le moment, mais permettent déjà de faire pas mal de truc intéressant :

    * : sélectionne tout les éléments définissant l'objet récursivement (ses descendants)
    > : sélectionne tout les éléments définissant l'objet (ses enfants)
    :where : qui est une condition à passer pour valider ou pas une sélection
    :whereMatch : qui est une condition définit par une règle à passer pour valider ou pas une sélection
    :whereUnique : qui est une condition à passer pour valider ou pas une sélection, et dont la validation interrompra la recherche d'autre occurrence (un peux comme une clé primaire)
    :whereMin : qui définit un minimum
    :whereMax : qui définit un maximum
    dutexte : une propriété spécifique, peux être un chiffre pour un tableaux
    [+du texte+] : une propriété spécifique,s'accordant pas à la syntaxe js

    la clause :where et :whereMatch, passe par une référence à l'objet en court "object" via le nom de variable "object"

    :whereMin et :whereMax accèpte deux arguments.
    le 1er argument de définit la proprité devant satisfaire la condition sous forme d'une règle
    le 2er argument est optionnelle , il définit qu'elle fonction , ou classe permet de comparer la valeur à une autre.

    par exemple : JSON.select(Articles, "> :whereMax(> date, new Date) > titre"));


    exemple d'une requête complexe : Séléction des titres des commentaire de fred des articles de Marion délisse et dont au moins un des commentaires est édité par Joakim

    > :where(object.auteur == 'Marion délisse'):whereMatch(object.commentaire, :where(object.auteur == 'Joakim')) commentaire > :where(object.auteur == 'fred') > titre

    voir ci-dessous
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138
    139
    140
    141
    142
    143
    144
    145
    146
    147
    148
    149
    150
    151
    152
    153
    154
    155
    156
    157
    158
    159
    160
    161
    162
    163
    164
    165
    166
    167
    168
    169
    170
    171
    172
    173
    174
    175
    176
    177
    178
    179
    180
    181
    182
    183
    184
    185
    186
    187
    188
    189
    190
    191
    192
    193
    194
    195
    196
    197
    198
    199
    200
    201
    202
    203
    204
    205
    206
    207
    208
    209
    210
    211
    212
    213
    214
    215
    216
    217
    218
    219
    220
    221
    222
    223
    224
    225
    226
    227
    228
    229
    230
    231
    232
    233
    234
    235
    236
    237
    238
    239
    240
    241
    242
    243
    244
    245
    246
    247
    248
    249
    250
    251
    252
    253
    254
    255
    256
    257
    258
    259
    260
    261
    262
    263
    264
    265
    266
    267
    268
    269
    270
    271
    272
    273
    274
    275
    276
    277
    278
    279
    280
    281
    282
    283
    284
    285
    286
    287
    288
    289
    290
    291
    292
    293
    294
    295
    296
    297
    298
    299
    300
     
    <!DOCTYPE HTML>
    <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="fr" lang="fr">
    <head>
        <title>JSONSelector</title>
    	<script type="text/javascript">
     
     
    var JSON = window.JSON || {};
     
    JSON.each = function(object, fn){
    	if(object.each){
    		object.each(fn);
    	}else if(object.constructor == Object){
    		for(var i in object)
    			fn(object[i], i);
    	}else if(object.constructor == Array){
    		for(var i = 0, l = object.length; i < l; i++)
    			fn(object[i], i);
    	}
    };
     
    (function(){
     
    	var cache = {select : {}, match : {}};
     
    	function getLastProperty(resProperty, lastProperty){
    	}
     
    	function getCondition(rule, si){
    		for(var i = si, l = rule.length, j = 1; i < l; i++){
    			if(rule[i] == '(' && rule[i-1] != '\\'){
    				j++;
    			}else if(rule[i] == ')' && rule[i-1] != '\\')
    				j--;
    			if(!j)
    				return [rule.slice(si, i), rule.slice(i + 1)];
    		}
    	}
     
    	function addslashes(str) {
    		str = str.replace(/\\/g,'\\\\');
    		str = str.replace(/\'/g,'\\\'');
    		str = str.replace(/\"/g,'\\"');
    		str = str.replace(/\0/g,'\\0');
    		return str;
    	}
     
    	/*
    	 $1 = une descendance direct ou indirect
    	 $2 = une descendance direct >	
    	 $3 = une propriété	
    	 $4 = une propriété ne s'accordant pas à la syntaxe js [+..+]
    	 $5 = la valeur de la proprité $4
    	 $6 = clause where ou whereMatch + tout le reste de la règle (parssé récurssivement)
    	 $7 = clause whereMatch
    	 $8 = les clauses commencant par :
    	*/
     
    	var GROUPE_REG = /((\s*>\s*)|\s*\*\s*)((\[\+([^\+\]]+)\+\])|[\w\d$_]*)|(:where((Match)*(Unique)*(Min)*(Max)*)\(.*)|(:\w+)/g;
     
     
    	function query(selector, match, min, max, rest){
    		var closes = [],//objets des fermetures des fonctions 
    		lastProperty = 'object',
    		combiProperty = 'object',
    		mm = min || max,
    		minRule, maxRule, currentProperty;
     
    		var start = (match ? "var valid = {};try{" : mm || rest  ? "" : "var res = [];" + (/whereUnique/.test(selector) ? "var unique = {};" : ""));
     
    		var fn = start +selector.replace(GROUPE_REG, function($0, $1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11){
    			if($1){
    				combiProperty = lastProperty;
    				currentProperty = $3 || ($5 ? addslashes($5) : null);
     
    				if(currentProperty){
    					if($2) lastProperty += "['"+currentProperty+"']";
    					else lastProperty = "object['"+currentProperty+"']";
    				}else lastProperty = 'object';
     
    				if($2){
    					closes.push(currentProperty  ? "}" : "});");
    					return currentProperty ? "if("+combiProperty+".hasOwnProperty('"+currentProperty+"')){" : "JSON.each("+combiProperty+",function(object){";
    				}else{
    					closes.push("JSON.each(object, arguments.callee);})("+combiProperty+");");
    					if(currentProperty)closes.push("}");
    					return "(function(object){"+(currentProperty? "if(object.hasOwnProperty('"+currentProperty+"')){" : "");
    				}
    			}
    			if($6){
    				if($10 || $11){
    					var c = getCondition($6, $7 ? 7 + $7.length : 7);
    					c[0] = c[0].split(/,((\s|\w|\d|_)*)$/gm);
    					if($10) minRule= c; else maxRule = c;
    					return '';
    				}
    				if($9)
    					closes.push("throw unique;}catch(e){}");
    				closes.push('}');
    				var conds = getCondition($6, $7 ? 7 + $7.length : 7);
    				if($8)
    					return "if(JSON.match("+lastProperty+",\""+conds[0]+"\")){" + conds[1].replace(GROUPE_REG, arguments.callee);
    				return ($9 ? "try{" : "") +"if("+(conds[0]).replace("object", lastProperty)+"){" + conds[1].replace(GROUPE_REG, arguments.callee);
    			}
    		})+ (match ? "throw valid;" : mm ? 
    			mm[1] ? "var mm = "+mm[1]+"("+lastProperty+");if(!minMax || mm"+(min ? "<" : ">")+"minMax){minMax = mm;_object = __object;}" 
    			: "if(!minMax || "+lastProperty+(min ? "<" : ">")+"minMax){minMax = "+lastProperty+";_object = __object;}"
    			: "res.push("+lastProperty+");")+ closes.reverse().join('');
     
    		if(minRule || maxRule){
    			var m = minRule || maxRule;
    			fn += "var minMax, _object, __object; JSON.each(res, function(object){__object = object;"+query(m[0][0], false, minRule ? minRule[0] : false, maxRule ? maxRule[0]: false)+"});" 
    			+ (!m[1] ? "res = [_object];" : "object = _object;res = [];if(object){" + query(m[1], false, false, false, true))+"}";
    		}
     
    		var end = (match ? "}catch(e){if(e == valid)return true;}return false;" : mm || rest? "" : "return res;");
     
    		if(mm || rest) return fn + end;
     
    		fn = new Function("object", fn + end); 
    		//met en cache
    		cache[match ? "match" : "select"][selector] = fn;
    		//pour voire comment les fonctions sont créer
    		document.getElementById("de").innerHTML += "function du sélécteur : " + selector + "\n" + fn + "\n\n";
    		return fn;
    	}
     
     
     
    	JSON.select = function(object, selector){
    		return  (cache.select[selector] || query(selector))(object);
    	}
     
    	JSON.match = function(object, selector){
    		return (cache.match[selector] || query(selector, true))(object);
    	}
     
    })();
     
     
    	</script>
    </head>
    <body>
    <!-- pour voire ce que donnes les fonctions;-->
    <pre id = 'de'></pre>
    <div id = 'test'></div>
    </body> 
     
    <script type="text/javascript">
     
    	//une JSON simple pour l'exemple. 
    	//Un ensemble d'article , ccontenant des commentaire, eux même pouvant contenir des commentaire
    	var Articles = [{
    		titre : 'les abeilles chantent',
    		auteur : 'Marion délisse',
    		date : "10 06 2007",
    		commentaires : [{
    			titre : 'Pas mal!',
    			auteur : 'fred',
    			date : "10 07 2007",
    			contenue : 'du text....',
    			commentaires : [{
    				titre : 'Merci ;)',
    				date : "10 12 2007",
    				auteur : 'Marion délisse',
    				contenue : 'du text....'
    			}]
    		},{
    			titre : 'Bof',
    			auteur : 'jean-jean le blérot',
    			date : "10 08 2007",
    			contenue : 'du text....',
    			commentaires : [{
    				titre : 'Hmm',
    				auteur : 'Marion délisse',
    				date : "10 09 2007",
    				contenue : 'du text....'
    			},{
    				titre : 'N\'importe quoi',
    				date : "10 10 2007",
    				auteur : 'fred',
    				contenue : 'du text....'
    			},{
    				titre : 'L\'écoute pas mon bébé',
    				date : "10 11 2007",
    				auteur : 'Joakim',
    				contenue : 'du text....'
    			}]
    		}]
    	},{
    		titre : 'Jojo le boss !!!!',
    		auteur : 'Joakim',
    		date : "10 06 2008",
    		commentaires : [{
    			titre : 'Comment t\es tros fort',
    			auteur : 'fred',
    			date : "10 07 2008",
    			contenue : 'du text....',
    			commentaires : [{
    				titre : 'Je sais !! ',
    				auteur : 'Joakim',
    				date : "10 08 2008",
    				contenue : 'du text....',
    				commentaires : [{
    					titre : 'Ca va les chevilles ?',
    					date : "10 09 2008",
    					auteur : 'jean-jean le blérot',
    					contenue : 'du text....',
    				}]
    			}, {
    				titre : 'Bisous mon héro',
    				date : "10 10 2008",
    				auteur : 'Marion délisse',
    				contenue : 'du text....'
    			}]
    		}]
    	},{
    		titre : 'les abeilles chantent plus',
    		auteur : 'Marion délisse',
    		date : "10 06 2009",
    		commentaires : [{
    			titre : 'C\'est triste',
    			date : "10 07 2009",
    			auteur : 'fred',
    			contenue : 'du text....',
    			commentaires : [{
    				titre : 'C\'est vrai',
    				date : "10 08 2009",
    				auteur : 'Marion délisse',
    				contenue : 'du text....'
    			}]
    		}]
    	}];
     
     
    	//séléctionne les objets Articles.
    	alert("Séléction des objets Articles :" 
    	+JSON.select(Articles, ">").length);
    	//explication : on séléctionne l'ensemble des enfants de Articles par le sélécteur de descendance directe >. 
    	//exemple pas du tout utile vu qu'il renvoie juste une copie de Articles
     
    	//Séléction des titre des Articles.
    	alert("Séléction des titre des Articles :" 
    	+JSON.select(Articles, "> > titre"));
    	//explication : on recherche pour chaque article de Articles  > une propriété titre descandant directement de l'article (deuxième > plus la la prorpiété titre) 
    	//les espace entre les > ne sont pas obligatoire
     
    	//Séléction de l'ensemble des objets possèdant un titres (commentaire compris).
    	alert("Séléction de l'ensemble des objets possèdant un titres (commentaire compris) :" 
    	+JSON.select(Articles, "*:where(object.titre)").length);
    	//explication : on ne séléctionne  tout les objets * 
    	//qui définissent un titre , via la clause where et la référence à l'objet en court "object"
     
    	//Séléction de l'auteur de l'article les abeilles chantent
    	alert("Séléction de l'auteur de l'article les abeilles chantent :" 
    	+JSON.select(Articles, "> :whereUnique(object.titre == 'les abeilles chantent') > auteur"));
    	//explication : on séléctionne l'ensemble des articles de Articles  (>) 
    	//puis pour chaque article on ne séléctionne que celui  qui à comme titre les abeilles chantent via la clause whereUnique et la référence à l'objet en court "object"
    	//whereUnique stop la recherche au premier test positif, il peux servir pour définir des clé primaire
    	//puis on séléctionne la propriété auteur directement descendante (fille) par > auteur
     
    	//Séléction du titre de le l'article le plus récent
    	alert("Séléction du titre le l'article le plus récent :" +
    	JSON.select(Articles, "> :whereMax(> date, new Date) > titre"));
    	//explication : on séléctionne l'ensemble des articles de Articles  (>) 
    	//puis on ne séléctionne  celui  qui la date la plus récente :whereMax(> date, new Date)
    	//le 1er argument de whereMax définit la proprité devant satisfaire la condition sous forme d'une règle
    	//le 2er argument est optionnelle , il définit qu'elle fonction , ou classe permet de comparer la valeur à une autre
    	// ici on transforme une string , en date , mais vous pouvait passer tout sorte d'objet
     
     
    	//Séléction du titre le l'article qui  comprends le commentaire le plus vieux
    	alert("Séléction du titre le l'article qui  comprends le commentaire le plus vieux :" +
    	JSON.select(Articles, "> :whereMin(* commentaires > > date, new Date) > titre"));
    	//explication : on séléctionne l'ensemble des articles de Articles  (>) 
    	//puis on ne séléctionne  celui  qui à le commentaire le plus vieux :whereMin(* commentaire > date, new Date)
    		//détail de ->  * commentaires > date , on se trouve dans un article, donc on séléctionne tout les tableaux de commentaires possible commentaires (* commentaires)
    		//puis pour chaque commentaire (>) on test  la date ( >date)
     
    	//Séléction des titres des commentaires des articles de Marion délisse et dont au moin un des commentaires est édité par Joakim
    	alert("Séléction des titres des commentaire des articles de Marion délisse et dont au moin un des commentaires est édité par Joakim : " 
    	+JSON.select(Articles, "> :where(object.auteur == 'Marion délisse'):whereMatch(* commentaires > :where(object.auteur == 'Joakim')) > titre"));
    	//explication : on séléctionne l'ensemble des articles de Articles  (>) 
    	//puis pour chaque article on ne séléctionne que celui  qui à comme auteur Marion délisse via la clause where et la référence à l'objet en court "object"
    	//de ses commentaire, on test si au moin un commentaire est éditer par joakim 
    			//détail de -> whereMatch(* commentaires > :where(object.auteur == 'Joakim'))
    			// * commentaires -> on séléctionne tout les object commentaire descendant 
    			// > -> puis pour chaque commentaire
    			// :where(object.auteur == 'Joakim') ->  si l'auteur == joakim
    			//le teste passe ....
    	//enfin on ne séléctionn que les titre  (> titre)
     
    	//Séléction des titres des commentaires de fred des articles de Marion délisse et dont au moin un des commentaires est édité par Joakim
    	alert("Séléction des titres des commentaires de fred des articles de Marion délisse et dont au moin un des commentaires est édité par Joakim : " +
    	JSON.select(Articles, "> :where(object.auteur == 'Marion délisse'):whereMatch(* commentaires > :where(object.auteur == 'Joakim')) * commentaires > :where(object.auteur == 'fred') > titre"));
    	//explication : je vous laisse devinner ;)
     
    	//A vous de bidouiller maintenant :) 
    </script>
    Utile pour des services web, renvoyant du json....

  2. #2
    Rédacteur

    Homme Profil pro
    Responsable de projet
    Inscrit en
    Mai 2009
    Messages
    634
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : Responsable de projet
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Mai 2009
    Messages : 634
    Points : 3 511
    Points
    3 511
    Par défaut
    Salut,

    le concept est vraiment pas mal.

    Bonne continuation!

    ++

  3. #3
    Membre confirmé
    Profil pro
    Inscrit en
    Février 2009
    Messages
    354
    Détails du profil
    Informations personnelles :
    Localisation : France, Sarthe (Pays de la Loire)

    Informations forums :
    Inscription : Février 2009
    Messages : 354
    Points : 491
    Points
    491
    Par défaut
    merci ornitho13

Discussions similaires

  1. Requête sur des données en ligne
    Par Oncle_Pete dans le forum SQL
    Réponses: 6
    Dernier message: 27/03/2012, 16h50
  2. Réponses: 1
    Dernier message: 18/03/2012, 09h59
  3. Requête sur des données "chainées"
    Par jbrasselet dans le forum SQL
    Réponses: 4
    Dernier message: 17/03/2011, 08h45
  4. Réponses: 5
    Dernier message: 25/09/2009, 18h44
  5. Travailler sur des données qui doivent être triées
    Par haypo dans le forum XML/XSL et SOAP
    Réponses: 2
    Dernier message: 19/07/2003, 17h13

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo