2 ** $Id: pz2.js,v 1.13 2007-05-14 12:57:43 jakub Exp $
3 ** pz2.js - pazpar2's javascript client library.
6 //since explorer is flawed
8 window.Node = new Object();
10 Node.ATTRIBUTE_NODE = 2;
12 Node.CDATA_SECTION_NODE = 4;
13 Node.ENTITY_REFERENCE_NODE = 5;
15 Node.PROCESSING_INSTRUCTION_NODE = 7;
16 Node.COMMENT_NODE = 8;
17 Node.DOCUMENT_NODE = 9;
18 Node.DOCUMENT_TYPE_NODE = 10;
19 Node.DOCUMENT_FRAGMENT_NODE = 11;
20 Node.NOTATION_NODE = 12;
23 if(typeof window.jQuery == "undefined"){
24 throw new Error("pz2.js requires jQuery library");
26 // prevent execution of more than once
27 if(typeof window.pz2 == "undefined") {
28 window.undefined = window.undefined;
30 var pz2 = function(paramArray) {
34 // at least one callback required
36 throw new Error("An array with parameters has to be suplied when instantiating a class");
38 //supported pazpar2's protocol version
39 __myself.suppProtoVer = '1';
40 __myself.errorHandler = paramArray.errorhandler || null;
43 __myself.statCallback = paramArray.onstat || null;
44 __myself.showCallback = paramArray.onshow || null;
45 __myself.termlistCallback = paramArray.onterm || null;
46 __myself.recordCallback = paramArray.onrecord || null;
47 __myself.bytargetCallback = paramArray.onbytarget || null;
48 __myself.resetCallback = paramArray.onreset || null;
51 __myself.termKeys = paramArray.termlist || "subject";
53 // some configurational stuff
54 __myself.pz2String = "search.pz2";
55 __myself.keepAlive = 50000;
57 __myself.sessionID = null;
58 __myself.initStatusOK = false;
59 __myself.pingStatusOK = false;
60 __myself.searchStatusOK = false;
62 if ( paramArray.keepAlive < __myself.keepAlive )
63 __myself.keepAlive = paramArray.keepAlive;
66 __myself.currentSort = "relevance";
68 __myself.currentStart = 0;
69 __myself.currentNum = 20;
71 // last full record retrieved
72 __myself.currRecID = null;
74 __myself.currQuery = null;
77 __myself.statTime = paramArray.stattime || 2000;
78 __myself.statTimer = null;
79 __myself.termTime = paramArray.termtime || 1000;
80 __myself.termTimer = null;
81 __myself.showTime = paramArray.showtime || 1000;
82 __myself.showTimer = null;
83 __myself.showFastCount = 4;
84 __myself.bytargetTime = paramArray.bytargettime || 1000;
85 __myself.bytargetTimer = null;
88 __myself.dumpFactor = 500;
89 __myself.showCounter = 0;
90 __myself.termCounter = 0;
92 // active clients, updated by stat and show
93 // might be an issue since bytarget will poll accordingly
94 __myself.activeClients = 1;
97 $(document).ajaxError(
98 function (ajaxError, xhr, reqSettings, prevException)
100 if ( xhr.responseXML && xhr.responseXML.getElementsByTagName("error").length )
102 var errMsg = xhr.responseXML.getElementsByTagName("error")[0].childNodes[0].nodeValue;
103 var errCode = xhr.responseXML.getElementsByTagName("error")[0].getAttribute("code");
105 var err = new Error(errMsg);
108 if (__myself.errorHandler) {
109 __myself.errorHandler(err);
115 // ensure the errors are propagated
116 else if (prevException != undefined ) {
120 throw new Error("XMLHttpRequest error. STATUS: " + xhr.status + " STATUS TEXT: " + xhr.statusText);
124 // auto init session?
125 if (paramArray.autoInit !== false)
131 __myself.sessionID = null;
132 __myself.initStatusOK = false;
133 __myself.pingStatusOK = false;
134 __myself.searchStatusOK = false;
136 clearTimeout(__myself.statTimer);
137 clearTimeout(__myself.showTimer);
138 clearTimeout(__myself.termTimer);
139 clearTimeout(__myself.bytargetTimer);
141 __myself.resetCallback();
143 init: function ( sessionId )
146 if ( sessionId != undefined ) {
147 __myself.initStatusOK = true;
148 __myself.sessionID = sessionId;
152 $.get( __myself.pz2String,
153 { "command": "init" },
155 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
156 if ( data.getElementsByTagName("protocol")[0].childNodes[0].nodeValue != __myself.suppProtoVer )
157 throw new Error("Server's protocol not supported by the client");
158 __myself.initStatusOK = true;
159 __myself.sessionID = data.getElementsByTagName("session")[0].childNodes[0].nodeValue;
160 setTimeout("__myself.ping()", __myself.keepAlive);
163 // if it gets here the http return code was 200 (pz2 errors are 417)
164 // but the response was invalid, it should never occur
165 setTimeout("__myself.init()", 1000);
170 // no need to ping explicitly
173 if( !__myself.initStatusOK )
175 // session is not initialized code here
177 $.get( __myself.pz2String,
178 { "command": "ping", "session": __myself.sessionID },
180 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
181 __myself.pingStatusOK = true;
182 setTimeout("__myself.ping()", __myself.keepAlive);
185 // if it gets here the http return code was 200 (pz2 errors are 417)
186 // but the response was invalid, it should never occur
187 setTimeout("__myself.ping()", 1000);
191 search: function (query, num, sort, filter)
193 clearTimeout(__myself.statTimer);
194 clearTimeout(__myself.showTimer);
195 clearTimeout(__myself.termTimer);
196 clearTimeout(__myself.bytargetTimer);
198 __myself.showCounter = 0;
199 __myself.termCounter = 0;
201 if( !__myself.initStatusOK )
204 if( query !== undefined )
205 __myself.currQuery = query;
207 throw new Error("You need to supply query to the search command");
209 if( filter !== undefined )
210 var searchParams = { "command": "search", "session": __myself.sessionID, "query": __myself.currQuery, "filter": filter };
212 var searchParams = { "command": "search", "session": __myself.sessionID, "query": __myself.currQuery };
214 $.get( __myself.pz2String,
217 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
218 __myself.searchStatusOK = true;
220 __myself.show(0, num, sort);
221 if ( __myself.statCallback )
222 __myself.statTimer = setTimeout("__myself.stat()", __myself.statTime / 2);
223 if ( __myself.termlistCallback )
224 //__myself.termlist();
225 __myself.termTimer = setTimeout("__myself.termlist()", __myself.termTime / 2);
226 if ( __myself.bytargetCallback )
227 __myself.bytargetTimer = setTimeout("__myself.bytarget()", __myself.bytargetTime / 2);
230 // if it gets here the http return code was 200 (pz2 errors are 417)
231 // but the response was invalid, it should never occur
232 setTimeout("__myself.search(__myself.currQuery)", 1000);
238 if( !__myself.searchStatusOK )
240 // if called explicitly takes precedence
241 clearTimeout(__myself.statTimer);
243 $.get( __myself.pz2String,
244 { "command": "stat", "session": __myself.sessionID },
246 if ( data.getElementsByTagName("stat") ) {
247 var activeClients = Number( data.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue );
248 __myself.activeClients = activeClients;
250 "activeclients": activeClients,
251 "hits": Number( data.getElementsByTagName("hits")[0].childNodes[0].nodeValue ),
252 "records": Number( data.getElementsByTagName("records")[0].childNodes[0].nodeValue ),
253 "clients": Number( data.getElementsByTagName("clients")[0].childNodes[0].nodeValue ),
254 "initializing": Number( data.getElementsByTagName("initializing")[0].childNodes[0].nodeValue ),
255 "searching": Number( data.getElementsByTagName("searching")[0].childNodes[0].nodeValue ),
256 "presenting": Number( data.getElementsByTagName("presenting")[0].childNodes[0].nodeValue ),
257 "idle": Number( data.getElementsByTagName("idle")[0].childNodes[0].nodeValue ),
258 "failed": Number( data.getElementsByTagName("failed")[0].childNodes[0].nodeValue ),
259 "error": Number( data.getElementsByTagName("error")[0].childNodes[0].nodeValue )
261 __myself.statCallback(stat);
262 if (activeClients > 0)
263 __myself.statTimer = setTimeout("__myself.stat()", __myself.statTime);
266 // if it gets here the http return code was 200 (pz2 errors are 417)
267 // but the response was invalid, it should never occur
268 __myself.statTimer = setTimeout("__myself.stat()", __myself.statTime / 4);
272 show: function(start, num, sort)
274 if( !__myself.searchStatusOK )
276 // if called explicitly takes precedence
277 clearTimeout(__myself.showTimer);
279 if( sort !== undefined )
280 __myself.currentSort = sort;
281 if( start !== undefined )
282 __myself.currentStart = Number( start );
283 if( num !== undefined )
284 __myself.currentNum = Number( num );
286 $.get( __myself.pz2String,
287 { "command": "show", "session": __myself.sessionID, "start": __myself.currentStart,
288 "num": __myself.currentNum, "sort": __myself.currentSort, "block": 1 },
290 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
291 // first parse the status data send along with records
292 // this is strictly bound to the format
293 var activeClients = Number( data.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue );
295 "activeclients": activeClients,
296 "merged": Number( data.getElementsByTagName("merged")[0].childNodes[0].nodeValue ),
297 "total": Number( data.getElementsByTagName("total")[0].childNodes[0].nodeValue ),
298 "start": Number( data.getElementsByTagName("start")[0].childNodes[0].nodeValue ),
299 "num": Number( data.getElementsByTagName("num")[0].childNodes[0].nodeValue ),
302 // parse all the first-level nodes for all <hit> tags
303 var hits = data.getElementsByTagName("hit");
304 var hit = new Array();
305 for (i = 0; i < hits.length; i++) {
306 show.hits[i] = new Array();
307 show.hits[i]['location'] = new Array();
308 for ( j = 0; j < hits[i].childNodes.length; j++) {
310 if ( hits[i].childNodes[j].nodeType == Node.ELEMENT_NODE ) {
311 if (hits[i].childNodes[j].nodeName == 'location') {
312 var locNode = hits[i].childNodes[j];
313 var id = locNode.getAttribute('id');
314 show.hits[i]['location'][id] = {
315 "id": locNode.getAttribute("id"),
316 "name": locNode.getAttribute("name")
320 var nodeName = hits[i].childNodes[j].nodeName;
321 var nodeText = hits[i].childNodes[j].firstChild.nodeValue;
322 show.hits[i][nodeName] = nodeText;
327 __myself.showCallback(show);
328 __myself.showCounter++;
329 var delay = __myself.showTime;
330 if (__myself.showCounter > __myself.showFastCount)
332 if (activeClients > 0)
333 __myself.showTimer = setTimeout("__myself.show()", delay);
336 // if it gets here the http return code was 200 (pz2 errors are 417)
337 // but the response was invalid, it should never occur
338 __myself.showTimer = setTimeout("__myself.show()", __myself.showTime / 4);
344 if( !__myself.searchStatusOK )
347 if( id !== undefined )
348 __myself.currRecID = id;
350 $.get( __myself.pz2String,
351 { "command": "record", "session": __myself.sessionID, "id": __myself.currRecID },
354 var record = new Array();
355 if ( recordNode = data.getElementsByTagName("record")[0] ) {
356 for ( i = 0; i < recordNode.childNodes.length; i++) {
357 if ( recordNode.childNodes[i].nodeType == Node.ELEMENT_NODE ) {
358 var nodeName = recordNode.childNodes[i].nodeName;
359 var nodeText = recordNode.childNodes[i].firstChild.nodeValue;
360 record[nodeName] = nodeText;
363 // the location is hard coded
364 var locationNodes = recordNode.getElementsByTagName("location");
365 record["location"] = new Array();
366 for ( i = 0; i < locationNodes.length; i++ ) {
367 record["location"][i] = {
368 "id": locationNodes[i].getAttribute("id"),
369 "name": locationNodes[i].getAttribute("name")
371 for ( j = 0; j < locationNodes[i].childNodes.length; j++) {
372 if ( locationNodes[i].childNodes[j].nodeType == Node.ELEMENT_NODE ) {
373 var nodeName = locationNodes[i].childNodes[j].nodeName;
375 if (locationNodes[i].childNodes[j].firstChild)
376 nodeText = locationNodes[i].childNodes[j].firstChild.nodeValue;
379 record["location"][i][nodeName] = nodeText;
383 __myself.recordCallback(record);
386 // if it gets here the http return code was 200 (pz2 errors are 417)
387 // but the response was invalid, it should never occur
388 setTimeout("__myself.record(__myself.currRecID)", 1000);
394 if( !__myself.searchStatusOK )
396 // if called explicitly takes precedence
397 clearTimeout(__myself.termTimer);
399 $.get( __myself.pz2String,
400 { "command": "termlist", "session": __myself.sessionID, "name": __myself.termKeys },
402 if ( data.getElementsByTagName("termlist") ) {
403 var termList = { "activeclients": Number( data.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue ) };
404 var termLists = data.getElementsByTagName("list");
406 for (i = 0; i < termLists.length; i++) {
407 var listName = termLists[i].getAttribute('name');
408 termList[listName] = new Array();
409 var terms = termLists[i].getElementsByTagName('term');
410 //for each term in the list
411 for (j = 0; j < terms.length; j++) {
413 "name": terms[j].getElementsByTagName("name")[0].childNodes[0].nodeValue,
414 "freq": terms[j].getElementsByTagName("frequency")[0].childNodes[0].nodeValue
417 var termIdNode = terms[j].getElementsByTagName("id");
418 if(terms[j].getElementsByTagName("id").length)
419 term["id"] = termIdNode[0].childNodes[0].nodeValue;
421 termList[listName][j] = term;
425 __myself.termlistCallback(termList);
426 __myself.termCounter++;
427 if (termList["activeclients"] > 0)
428 __myself.termTimer = setTimeout("__myself.termlist()", (__myself.termTime + __myself.termCounter*__myself.dumpFactor));
431 // if it gets here the http return code was 200 (pz2 errors are 417)
432 // but the response was invalid, it should never occur
433 __myself.termTimer = setTimeout("__myself.termlist()", __myself.termTime / 4);
440 if( !__myself.searchStatusOK )
442 // if called explicitly takes precedence
443 clearTimeout(__myself.bytargetTimer);
445 $.get( __myself.pz2String,
446 { "command": "bytarget", "session": __myself.sessionID },
448 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
449 var targetNodes = data.getElementsByTagName("target");
450 var bytarget = new Array();
451 for ( i = 0; i < targetNodes.length; i++) {
452 bytarget[i] = new Array();
453 for( j = 0; j < targetNodes[i].childNodes.length; j++ ) {
454 if ( targetNodes[i].childNodes[j].nodeType == Node.ELEMENT_NODE ) {
455 var nodeName = targetNodes[i].childNodes[j].nodeName;
456 var nodeText = targetNodes[i].childNodes[j].firstChild.nodeValue;
457 bytarget[i][nodeName] = nodeText;
461 __myself.bytargetCallback(bytarget);
462 if ( __myself.activeClients > 0 )
463 __myself.bytargetTimer = setTimeout("__myself.bytarget()", __myself.bytargetTime);
466 // if it gets here the http return code was 200 (pz2 errors are 417)
467 // but the response was invalid, it should never occur
468 __myself.bytargetTimer = setTimeout("__myself.bytarget()", __myself.bytargetTime / 4);
472 // just for testing, probably shouldn't be here
473 showNext: function(page)
475 var step = page || 1;
476 __myself.show( ( step * __myself.currentNum ) + __myself.currentStart );
478 showPrev: function(page)
480 if (__myself.currentStart == 0 )
482 var step = page || 1;
483 var newStart = __myself.currentStart - (step * __myself.currentNum );
484 __myself.show( newStart > 0 ? newStart : 0 );
486 showPage: function(pageNum)
488 //var page = pageNum || 1;
489 __myself.show(pageNum * __myself.currentNum);