1 /* This file is part of the YAZ toolkit.
2 * Copyright (C) 1995-2011 Index Data
3 * See the file LICENSE for details.
7 * \brief Implements SRW/SRU package encoding and decoding
16 #include <yaz/wrbuf.h>
18 #include <libxml/parser.h>
19 #include <libxml/tree.h>
24 static void add_XML_n(xmlNodePtr ptr, const char *elem, char *val, int len,
29 xmlDocPtr doc = xmlParseMemory(val,len);
32 xmlNodePtr c = xmlNewChild(ptr, ns_ptr, BAD_CAST elem, 0);
33 xmlNodePtr t = xmlDocGetRootElement(doc);
34 xmlAddChild(c, xmlCopyNode(t,1));
40 xmlNodePtr add_xsd_string_n(xmlNodePtr ptr, const char *elem, const char *val,
45 xmlNodePtr c = xmlNewChild(ptr, 0, BAD_CAST elem, 0);
46 xmlNodePtr t = xmlNewTextLen(BAD_CAST val, len);
53 xmlNodePtr add_xsd_string_ns(xmlNodePtr ptr, const char *elem, const char *val,
58 xmlNodePtr c = xmlNewChild(ptr, ns_ptr, BAD_CAST elem, 0);
59 xmlNodePtr t = xmlNewText(BAD_CAST val);
66 xmlNodePtr add_xsd_string(xmlNodePtr ptr, const char *elem, const char *val)
68 return add_xsd_string_ns(ptr, elem, val, 0);
71 static void add_xsd_integer(xmlNodePtr ptr, const char *elem,
77 sprintf(str, ODR_INT_PRINTF, *val);
78 xmlNewTextChild(ptr, 0, BAD_CAST elem, BAD_CAST str);
82 static int match_element(xmlNodePtr ptr, const char *elem)
84 if (ptr->type == XML_ELEMENT_NODE && !xmlStrcmp(ptr->name, BAD_CAST elem))
93 static int match_xsd_string_n(xmlNodePtr ptr, const char *elem, ODR o,
97 struct _xmlAttr *attr;
99 if (!match_element(ptr, elem))
102 for (attr = ptr->properties; attr; attr = attr->next)
103 if (!strcmp(attr->name, "type") &&
104 attr->children && attr->children->type == XML_TEXT_NODE)
106 const char *t = strchr(attr->children->content, ':');
110 t = attr->children->content;
111 if (!strcmp(t, "string"))
118 if (!ptr || ptr->type != XML_TEXT_NODE)
123 *val = odr_strdup(o, (const char *) ptr->content);
125 *len = xmlStrlen(ptr->content);
130 static int match_xsd_string(xmlNodePtr ptr, const char *elem, ODR o,
133 return match_xsd_string_n(ptr, elem, o, val, 0);
136 static int match_xsd_XML_n2(xmlNodePtr ptr, const char *elem, ODR o,
137 char **val, int *len, int fixup_root)
140 int no_root_nodes = 0;
142 if (!match_element(ptr, elem))
145 buf = xmlBufferCreate();
147 /* Copy each element nodes at top.
148 In most cases there is only one root node.. At least one server
149 http://www.theeuropeanlibrary.org/sru/sru.pl
150 has multiple root nodes in recordData.
152 for (ptr = ptr->children; ptr; ptr = ptr->next)
154 if (ptr->type == XML_ELEMENT_NODE)
156 /* copy node to get NS right (bug #740). */
157 xmlNode *tmp = xmlCopyNode(ptr, 1);
159 xmlNodeDump(buf, tmp->doc, tmp, 0, 0);
165 if (no_root_nodes != 1 && fixup_root)
167 /* does not appear to be an XML document. Make it so */
168 xmlBufferAddHead(buf, (const xmlChar *) "<yaz_record>", -1);
169 xmlBufferAdd(buf, (const xmlChar *) "</yaz_record>", -1);
171 *val = (char *) odr_malloc(o, buf->use + 1);
172 memcpy(*val, buf->content, buf->use);
173 (*val)[buf->use] = '\0';
183 static int match_xsd_XML_n(xmlNodePtr ptr, const char *elem, ODR o,
184 char **val, int *len)
186 return match_xsd_XML_n2(ptr, elem, o, val, len, 0);
189 static int match_xsd_integer(xmlNodePtr ptr, const char *elem, ODR o,
193 struct _xmlAttr *attr;
195 if (!match_element(ptr, elem))
198 for (attr = ptr->properties; attr; attr = attr->next)
199 if (!strcmp(attr->name, "type") &&
200 attr->children && attr->children->type == XML_TEXT_NODE)
202 const char *t = strchr(attr->children->content, ':');
206 t = attr->children->content;
207 if (!strcmp(t, "integer"))
214 if (!ptr || ptr->type != XML_TEXT_NODE)
216 *val = odr_intdup(o, odr_atoi((const char *) ptr->content));
220 char *yaz_negotiate_sru_version(char *input_ver)
225 if (!strcmp(input_ver, "1.1"))
227 return "1.2"; /* our latest supported version */
230 static int yaz_srw_record(ODR o, xmlNodePtr pptr, Z_SRW_record *rec,
231 Z_SRW_extra_record **extra,
232 void *client_data, const char *ns)
234 if (o->direction == ODR_DECODE)
236 Z_SRW_extra_record ex;
239 int pack = Z_SRW_recordPacking_string;
241 xmlNodePtr data_ptr = 0;
242 rec->recordSchema = 0;
243 rec->recordData_buf = 0;
244 rec->recordData_len = 0;
245 rec->recordPosition = 0;
248 ex.extraRecordData_buf = 0;
249 ex.extraRecordData_len = 0;
250 ex.recordIdentifier = 0;
252 for (ptr = pptr->children; ptr; ptr = ptr->next)
255 if (match_xsd_string(ptr, "recordSchema", o,
258 else if (match_xsd_string(ptr, "recordPacking", o, &spack))
261 pack = yaz_srw_str_to_pack(spack);
263 else if (match_xsd_integer(ptr, "recordPosition", o,
264 &rec->recordPosition))
266 else if (match_element(ptr, "recordData"))
268 /* save position of Data until after the loop
269 then we will know the packing (hopefully), and
270 unpacking is done once
274 else if (match_xsd_XML_n(ptr, "extraRecordData", o,
275 &ex.extraRecordData_buf,
276 &ex.extraRecordData_len) )
279 match_xsd_string(ptr, "recordIdentifier", o,
280 &ex.recordIdentifier);
286 case Z_SRW_recordPacking_XML:
287 match_xsd_XML_n2(data_ptr, "recordData", o,
288 &rec->recordData_buf, &rec->recordData_len, 1);
290 case Z_SRW_recordPacking_URL:
291 /* just store it as a string.
292 leave it to the backend to collect the document */
293 match_xsd_string_n(data_ptr, "recordData", o,
294 &rec->recordData_buf, &rec->recordData_len);
296 case Z_SRW_recordPacking_string:
297 match_xsd_string_n(data_ptr, "recordData", o,
298 &rec->recordData_buf, &rec->recordData_len);
302 rec->recordPacking = pack;
303 if (ex.extraRecordData_buf || ex.recordIdentifier)
305 *extra = (Z_SRW_extra_record *)
306 odr_malloc(o, sizeof(Z_SRW_extra_record));
307 memcpy(*extra, &ex, sizeof(Z_SRW_extra_record));
310 else if (o->direction == ODR_ENCODE)
312 xmlNodePtr ptr = pptr;
313 int pack = rec->recordPacking;
314 const char *spack = yaz_srw_pack_to_str(pack);
316 add_xsd_string(ptr, "recordSchema", rec->recordSchema);
318 add_xsd_string(ptr, "recordPacking", spack);
321 case Z_SRW_recordPacking_string:
322 add_xsd_string_n(ptr, "recordData", rec->recordData_buf,
323 rec->recordData_len);
325 case Z_SRW_recordPacking_XML:
326 add_XML_n(ptr, "recordData", rec->recordData_buf,
327 rec->recordData_len, 0);
329 case Z_SRW_recordPacking_URL:
330 add_xsd_string_n(ptr, "recordData", rec->recordData_buf,
331 rec->recordData_len);
334 if (rec->recordPosition)
335 add_xsd_integer(ptr, "recordPosition", rec->recordPosition );
338 if ((*extra)->recordIdentifier)
339 add_xsd_string(ptr, "recordIdentifier",
340 (*extra)->recordIdentifier);
341 if ((*extra)->extraRecordData_buf)
342 add_XML_n(ptr, "extraRecordData",
343 (*extra)->extraRecordData_buf,
344 (*extra)->extraRecordData_len, 0);
350 static int yaz_srw_records(ODR o, xmlNodePtr pptr, Z_SRW_record **recs,
351 Z_SRW_extra_record ***extra,
352 int *num, void *client_data, const char *ns)
354 if (o->direction == ODR_DECODE)
359 for (ptr = pptr->children; ptr; ptr = ptr->next)
361 if (ptr->type == XML_ELEMENT_NODE &&
362 !xmlStrcmp(ptr->name, BAD_CAST "record"))
367 *recs = (Z_SRW_record *) odr_malloc(o, *num * sizeof(**recs));
368 *extra = (Z_SRW_extra_record **) odr_malloc(o, *num * sizeof(**extra));
369 for (i = 0, ptr = pptr->children; ptr; ptr = ptr->next)
371 if (ptr->type == XML_ELEMENT_NODE &&
372 !xmlStrcmp(ptr->name, BAD_CAST "record"))
374 yaz_srw_record(o, ptr, *recs + i, *extra + i, client_data, ns);
379 else if (o->direction == ODR_ENCODE)
382 for (i = 0; i < *num; i++)
384 xmlNodePtr rptr = xmlNewChild(pptr, 0, BAD_CAST "record",
386 yaz_srw_record(o, rptr, (*recs)+i, (*extra ? *extra + i : 0),
393 static int yaz_srw_version(ODR o, xmlNodePtr pptr, Z_SRW_recordVersion *rec,
394 void *client_data, const char *ns)
396 if (o->direction == ODR_DECODE)
399 rec->versionType = 0;
400 rec->versionValue = 0;
401 for (ptr = pptr->children; ptr; ptr = ptr->next)
404 if (match_xsd_string(ptr, "versionType", o,
408 match_xsd_string(ptr, "versionValue", o, &rec->versionValue);
411 else if (o->direction == ODR_ENCODE)
413 xmlNodePtr ptr = pptr;
414 add_xsd_string(ptr, "versionType", rec->versionType);
415 add_xsd_string(ptr, "versionValue", rec->versionValue);
420 static int yaz_srw_versions(ODR o, xmlNodePtr pptr,
421 Z_SRW_recordVersion **vers,
422 int *num, void *client_data, const char *ns)
424 if (o->direction == ODR_DECODE)
429 for (ptr = pptr->children; ptr; ptr = ptr->next)
431 if (ptr->type == XML_ELEMENT_NODE &&
432 !xmlStrcmp(ptr->name, BAD_CAST "recordVersion"))
437 *vers = (Z_SRW_recordVersion *) odr_malloc(o, *num * sizeof(**vers));
438 for (i = 0, ptr = pptr->children; ptr; ptr = ptr->next)
440 if (ptr->type == XML_ELEMENT_NODE &&
441 !xmlStrcmp(ptr->name, BAD_CAST "recordVersion"))
443 yaz_srw_version(o, ptr, *vers + i, client_data, ns);
448 else if (o->direction == ODR_ENCODE)
451 for (i = 0; i < *num; i++)
453 xmlNodePtr rptr = xmlNewChild(pptr, 0, BAD_CAST "version",
455 yaz_srw_version(o, rptr, (*vers)+i, client_data, ns);
461 Z_FacetTerm *yaz_sru_proxy_get_facet_term_count(ODR odr, xmlNodePtr node)
465 WRBUF wrbuf = wrbuf_alloc();
466 Z_FacetTerm *facet_term;
467 const char *freq_string = yaz_element_attribute_value_get(
468 node, "facetvalue", "est_representation");
470 freq = odr_atoi(freq_string);
474 for (child = node->children; child ; child = child->next)
476 if (child->type == XML_TEXT_NODE)
477 wrbuf_puts(wrbuf, (const char *) child->content);
479 facet_term = facet_term_create_cstr(odr, wrbuf_cstr(wrbuf), freq);
480 wrbuf_destroy(wrbuf);
484 static Z_FacetField *yaz_sru_proxy_decode_facet_field(ODR odr, xmlNodePtr ptr)
486 Z_AttributeList *list;
487 Z_FacetField *facet_field;
492 const char* name = yaz_element_attribute_value_get(ptr, "facet", "code");
493 yaz_log(YLOG_DEBUG, "sru-proxy facet type: %s", name);
495 list = yaz_use_attribute_create(odr, name);
496 for (node = ptr->children; node; node = node->next) {
497 if (match_element(node, "facetvalue"))
500 facet_field = facet_field_create(odr, list, num_terms);
502 for (node = ptr->children; node; node = node->next)
504 if (match_element(node, "facetvalue"))
506 facet_field_term_set(odr, facet_field,
507 yaz_sru_proxy_get_facet_term_count(odr, node),
515 static int yaz_sru_proxy_decode_facets(ODR o, xmlNodePtr root,
516 Z_FacetList **facetList)
520 for (ptr = root->children; ptr; ptr = ptr->next)
522 if (match_element(ptr, "facets"))
525 Z_FacetList *facet_list;
527 for (node = ptr->children; node; node= node->next)
529 if (node->type == XML_ELEMENT_NODE)
532 facet_list = facet_list_create(o, num_facets);
534 for (node = ptr->children; node; node= node->next)
536 if (match_element(node, "facet"))
538 facet_list_field_set(
540 yaz_sru_proxy_decode_facet_field(o, node), num_facets);
544 *facetList = facet_list;
553 static int yaz_srw_decode_diagnostics(ODR o, xmlNodePtr pptr,
554 Z_SRW_diagnostic **recs, int *num,
555 void *client_data, const char *ns)
560 for (ptr = pptr; ptr; ptr = ptr->next)
562 if (ptr->type == XML_ELEMENT_NODE &&
563 !xmlStrcmp(ptr->name, BAD_CAST "diagnostic"))
568 *recs = (Z_SRW_diagnostic *) odr_malloc(o, *num * sizeof(**recs));
569 for (i = 0; i < *num; i++)
572 (*recs)[i].details = 0;
573 (*recs)[i].message = 0;
575 for (i = 0, ptr = pptr; ptr; ptr = ptr->next)
577 if (ptr->type == XML_ELEMENT_NODE &&
578 !xmlStrcmp(ptr->name, BAD_CAST "diagnostic"))
582 (*recs)[i].details = 0;
583 (*recs)[i].message = 0;
584 for (rptr = ptr->children; rptr; rptr = rptr->next)
586 if (match_xsd_string(rptr, "uri", o,
589 else if (match_xsd_string(rptr, "details", o,
590 &(*recs)[i].details))
593 match_xsd_string(rptr, "message", o, &(*recs)[i].message);
601 int sru_decode_surrogate_diagnostics(const char *buf, size_t len,
602 Z_SRW_diagnostic **diag,
606 xmlDocPtr doc = xmlParseMemory(buf, len);
609 xmlNodePtr ptr = xmlDocGetRootElement(doc);
610 while (ptr && ptr->type != XML_ELEMENT_NODE)
613 && !xmlStrcmp(ptr->ns->href,
614 BAD_CAST "http://www.loc.gov/zing/srw/diagnostic/"))
616 ret = yaz_srw_decode_diagnostics(odr, ptr, diag, num, 0, 0);
623 static int yaz_srw_diagnostics(ODR o, xmlNodePtr pptr, Z_SRW_diagnostic **recs,
624 int *num, void *client_data, const char *ns)
626 if (o->direction == ODR_DECODE)
628 return yaz_srw_decode_diagnostics(o, pptr->children, recs, num, client_data, ns);
630 else if (o->direction == ODR_ENCODE)
634 xmlNewNs(pptr, BAD_CAST YAZ_XMLNS_DIAG_v1_1, BAD_CAST "diag" );
635 for (i = 0; i < *num; i++)
637 const char *std_diag = "info:srw/diagnostic/1/";
638 const char *ucp_diag = "info:srw/diagnostic/12/";
639 xmlNodePtr rptr = xmlNewChild(pptr, ns_diag,
640 BAD_CAST "diagnostic", 0);
641 add_xsd_string(rptr, "uri", (*recs)[i].uri);
642 if ((*recs)[i].message)
643 add_xsd_string(rptr, "message", (*recs)[i].message);
644 else if ((*recs)[i].uri )
646 if (!strncmp((*recs)[i].uri, std_diag, strlen(std_diag)))
648 int no = atoi((*recs)[i].uri + strlen(std_diag));
649 const char *message = yaz_diag_srw_str(no);
651 add_xsd_string(rptr, "message", message);
653 else if (!strncmp((*recs)[i].uri, ucp_diag, strlen(ucp_diag)))
655 int no = atoi((*recs)[i].uri + strlen(ucp_diag));
656 const char *message = yaz_diag_sru_update_str(no);
658 add_xsd_string(rptr, "message", message);
661 add_xsd_string(rptr, "details", (*recs)[i].details);
667 static int yaz_srw_term(ODR o, xmlNodePtr pptr, Z_SRW_scanTerm *term,
668 void *client_data, const char *ns)
670 if (o->direction == ODR_DECODE)
674 term->numberOfRecords = 0;
675 term->displayTerm = 0;
676 term->whereInList = 0;
677 for (ptr = pptr->children; ptr; ptr = ptr->next)
679 if (match_xsd_string(ptr, "value", o, &term->value))
681 else if (match_xsd_integer(ptr, "numberOfRecords", o,
682 &term->numberOfRecords))
684 else if (match_xsd_string(ptr, "displayTerm", o,
688 match_xsd_string(ptr, "whereInList", o, &term->whereInList);
691 else if (o->direction == ODR_ENCODE)
693 xmlNodePtr ptr = pptr;
694 add_xsd_string(ptr, "value", term->value);
695 add_xsd_integer(ptr, "numberOfRecords", term->numberOfRecords);
696 add_xsd_string(ptr, "displayTerm", term->displayTerm);
697 add_xsd_string(ptr, "whereInList", term->whereInList);
702 static int yaz_srw_terms(ODR o, xmlNodePtr pptr, Z_SRW_scanTerm **terms,
703 int *num, void *client_data, const char *ns)
705 if (o->direction == ODR_DECODE)
710 for (ptr = pptr->children; ptr; ptr = ptr->next)
712 if (ptr->type == XML_ELEMENT_NODE &&
713 !xmlStrcmp(ptr->name, BAD_CAST "term"))
718 *terms = (Z_SRW_scanTerm *) odr_malloc(o, *num * sizeof(**terms));
719 for (i = 0, ptr = pptr->children; ptr; ptr = ptr->next, i++)
721 if (ptr->type == XML_ELEMENT_NODE &&
722 !xmlStrcmp(ptr->name, BAD_CAST "term"))
723 yaz_srw_term(o, ptr, (*terms)+i, client_data, ns);
726 else if (o->direction == ODR_ENCODE)
729 for (i = 0; i < *num; i++)
731 xmlNodePtr rptr = xmlNewChild(pptr, 0, BAD_CAST "term", 0);
732 yaz_srw_term(o, rptr, (*terms)+i, client_data, ns);
738 int yaz_srw_codec(ODR o, void * vptr, Z_SRW_PDU **handler_data,
739 void *client_data, const char *ns)
741 xmlNodePtr pptr = (xmlNodePtr) vptr;
742 if (o->direction == ODR_DECODE)
744 Z_SRW_PDU **p = handler_data;
745 xmlNodePtr method = pptr->children;
748 while (method && method->type == XML_TEXT_NODE)
749 method = method->next;
753 if (method->type != XML_ELEMENT_NODE)
756 *p = yaz_srw_get_core_v_1_1(o);
758 if (!xmlStrcmp(method->name, BAD_CAST "searchRetrieveRequest"))
760 xmlNodePtr ptr = method->children;
761 Z_SRW_searchRetrieveRequest *req;
763 (*p)->which = Z_SRW_searchRetrieve_request;
764 req = (*p)->u.request = (Z_SRW_searchRetrieveRequest *)
765 odr_malloc(o, sizeof(*req));
766 req->query_type = Z_SRW_query_type_cql;
768 req->sort_type = Z_SRW_sort_type_none;
770 req->startRecord = 0;
771 req->maximumRecords = 0;
772 req->recordSchema = 0;
773 req->recordPacking = 0;
774 req->recordXPath = 0;
775 req->resultSetTTL = 0;
779 for (; ptr; ptr = ptr->next)
781 if (match_xsd_string(ptr, "version", o,
784 else if (match_xsd_string(ptr, "query", o,
786 req->query_type = Z_SRW_query_type_cql;
787 else if (match_xsd_string(ptr, "pQuery", o,
789 req->query_type = Z_SRW_query_type_pqf;
790 else if (match_xsd_string(ptr, "xQuery", o,
792 req->query_type = Z_SRW_query_type_xcql;
793 else if (match_xsd_integer(ptr, "startRecord", o,
796 else if (match_xsd_integer(ptr, "maximumRecords", o,
797 &req->maximumRecords))
799 else if (match_xsd_string(ptr, "recordPacking", o,
800 &req->recordPacking))
802 else if (match_xsd_string(ptr, "recordSchema", o,
805 else if (match_xsd_string(ptr, "recordXPath", o,
808 else if (match_xsd_integer(ptr, "resultSetTTL", o,
811 else if (match_xsd_string(ptr, "sortKeys", o,
812 &req->sort.sortKeys))
813 req->sort_type = Z_SRW_sort_type_sort;
814 else if (match_xsd_string(ptr, "stylesheet", o,
818 match_xsd_string(ptr, "database", o, &req->database);
820 if (!req->query.cql && !req->query.pqf && !req->query.xcql)
822 /* should put proper diagnostic here */
826 else if (!xmlStrcmp(method->name, BAD_CAST "searchRetrieveResponse"))
828 xmlNodePtr ptr = method->children;
829 Z_SRW_searchRetrieveResponse *res;
831 (*p)->which = Z_SRW_searchRetrieve_response;
832 res = (*p)->u.response = (Z_SRW_searchRetrieveResponse *)
833 odr_malloc(o, sizeof(*res));
835 res->numberOfRecords = 0;
836 res->resultSetId = 0;
837 res->resultSetIdleTime = 0;
839 res->num_records = 0;
840 res->diagnostics = 0;
841 res->num_diagnostics = 0;
842 res->nextRecordPosition = 0;
844 res->suggestions = 0;
846 for (; ptr; ptr = ptr->next)
848 if (match_xsd_string(ptr, "version", o,
851 else if (match_xsd_XML_n(ptr, "extraResponseData", o,
852 &(*p)->extraResponseData_buf,
853 &(*p)->extraResponseData_len))
855 else if (match_xsd_integer(ptr, "numberOfRecords", o,
856 &res->numberOfRecords))
858 else if (match_xsd_string(ptr, "resultSetId", o,
861 else if (match_xsd_integer(ptr, "resultSetIdleTime", o,
862 &res->resultSetIdleTime))
864 else if (match_element(ptr, "records"))
865 yaz_srw_records(o, ptr, &res->records,
867 &res->num_records, client_data, ns);
868 else if (match_xsd_integer(ptr, "nextRecordPosition", o,
869 &res->nextRecordPosition))
871 else if (match_element(ptr, "diagnostics"))
872 yaz_srw_diagnostics(o, ptr, &res->diagnostics,
873 &res->num_diagnostics,
875 else if (match_element(ptr, "facet_analysis"))
876 yaz_sru_proxy_decode_facets(o, ptr, &res->facetList);
879 else if (!xmlStrcmp(method->name, BAD_CAST "explainRequest"))
881 Z_SRW_explainRequest *req;
882 xmlNodePtr ptr = method->children;
884 (*p)->which = Z_SRW_explain_request;
885 req = (*p)->u.explain_request = (Z_SRW_explainRequest *)
886 odr_malloc(o, sizeof(*req));
887 req->recordPacking = 0;
890 for (; ptr; ptr = ptr->next)
892 if (match_xsd_string(ptr, "version", o,
895 else if (match_xsd_XML_n(ptr, "extraResponseData", o,
896 &(*p)->extraResponseData_buf,
897 &(*p)->extraResponseData_len))
899 else if (match_xsd_string(ptr, "stylesheet", o,
902 else if (match_xsd_string(ptr, "recordPacking", o,
903 &req->recordPacking))
906 match_xsd_string(ptr, "database", o, &req->database);
909 else if (!xmlStrcmp(method->name, BAD_CAST "explainResponse"))
911 Z_SRW_explainResponse *res;
912 xmlNodePtr ptr = method->children;
914 (*p)->which = Z_SRW_explain_response;
915 res = (*p)->u.explain_response = (Z_SRW_explainResponse*)
916 odr_malloc(o, sizeof(*res));
917 res->diagnostics = 0;
918 res->num_diagnostics = 0;
919 res->record.recordSchema = 0;
920 res->record.recordData_buf = 0;
921 res->record.recordData_len = 0;
922 res->record.recordPosition = 0;
924 for (; ptr; ptr = ptr->next)
926 if (match_xsd_string(ptr, "version", o,
929 else if (match_xsd_XML_n(ptr, "extraResponseData", o,
930 &(*p)->extraResponseData_buf,
931 &(*p)->extraResponseData_len))
933 else if (match_element(ptr, "record"))
934 yaz_srw_record(o, ptr, &res->record, &res->extra_record,
936 else if (match_element(ptr, "diagnostics"))
937 yaz_srw_diagnostics(o, ptr, &res->diagnostics,
938 &res->num_diagnostics,
943 else if (!xmlStrcmp(method->name, BAD_CAST "scanRequest"))
945 Z_SRW_scanRequest *req;
946 xmlNodePtr ptr = method->children;
948 (*p)->which = Z_SRW_scan_request;
949 req = (*p)->u.scan_request = (Z_SRW_scanRequest *)
950 odr_malloc(o, sizeof(*req));
951 req->query_type = Z_SRW_query_type_cql;
952 req->scanClause.cql = 0;
953 req->responsePosition = 0;
954 req->maximumTerms = 0;
958 for (; ptr; ptr = ptr->next)
960 if (match_xsd_string(ptr, "version", o,
963 else if (match_xsd_XML_n(ptr, "extraResponseData", o,
964 &(*p)->extraResponseData_buf,
965 &(*p)->extraResponseData_len))
967 else if (match_xsd_string(ptr, "scanClause", o,
968 &req->scanClause.cql))
970 else if (match_xsd_string(ptr, "pScanClause", o,
971 &req->scanClause.pqf))
973 req->query_type = Z_SRW_query_type_pqf;
975 else if (match_xsd_integer(ptr, "responsePosition", o,
976 &req->responsePosition))
978 else if (match_xsd_integer(ptr, "maximumTerms", o,
981 else if (match_xsd_string(ptr, "stylesheet", o,
985 match_xsd_string(ptr, "database", o, &req->database);
988 else if (!xmlStrcmp(method->name, BAD_CAST "scanResponse"))
990 Z_SRW_scanResponse *res;
991 xmlNodePtr ptr = method->children;
993 (*p)->which = Z_SRW_scan_response;
994 res = (*p)->u.scan_response = (Z_SRW_scanResponse *)
995 odr_malloc(o, sizeof(*res));
998 res->diagnostics = 0;
999 res->num_diagnostics = 0;
1001 for (; ptr; ptr = ptr->next)
1003 if (match_xsd_string(ptr, "version", o,
1004 &(*p)->srw_version))
1006 else if (match_xsd_XML_n(ptr, "extraResponseData", o,
1007 &(*p)->extraResponseData_buf,
1008 &(*p)->extraResponseData_len))
1010 else if (match_element(ptr, "terms"))
1011 yaz_srw_terms(o, ptr, &res->terms,
1012 &res->num_terms, client_data,
1014 else if (match_element(ptr, "diagnostics"))
1015 yaz_srw_diagnostics(o, ptr, &res->diagnostics,
1016 &res->num_diagnostics,
1025 neg_version = yaz_negotiate_sru_version((*p)->srw_version);
1027 (*p)->srw_version = neg_version;
1029 else if (o->direction == ODR_ENCODE)
1031 Z_SRW_PDU **p = handler_data;
1035 if ((*p)->which == Z_SRW_searchRetrieve_request)
1037 Z_SRW_searchRetrieveRequest *req = (*p)->u.request;
1038 ptr = xmlNewChild(pptr, 0, BAD_CAST "searchRetrieveRequest", 0);
1039 ns_srw = xmlNewNs(ptr, BAD_CAST ns, BAD_CAST "zs");
1040 xmlSetNs(ptr, ns_srw);
1042 if ((*p)->srw_version)
1043 add_xsd_string(ptr, "version", (*p)->srw_version);
1044 switch (req->query_type)
1046 case Z_SRW_query_type_cql:
1047 add_xsd_string(ptr, "query", req->query.cql);
1049 case Z_SRW_query_type_xcql:
1050 add_xsd_string(ptr, "xQuery", req->query.xcql);
1052 case Z_SRW_query_type_pqf:
1053 add_xsd_string(ptr, "pQuery", req->query.pqf);
1056 add_xsd_integer(ptr, "startRecord", req->startRecord);
1057 add_xsd_integer(ptr, "maximumRecords", req->maximumRecords);
1058 add_xsd_string(ptr, "recordPacking", req->recordPacking);
1059 add_xsd_string(ptr, "recordSchema", req->recordSchema);
1060 add_xsd_string(ptr, "recordXPath", req->recordXPath);
1061 add_xsd_integer(ptr, "resultSetTTL", req->resultSetTTL);
1062 switch (req->sort_type)
1064 case Z_SRW_sort_type_none:
1066 case Z_SRW_sort_type_sort:
1067 add_xsd_string(ptr, "sortKeys", req->sort.sortKeys);
1069 case Z_SRW_sort_type_xSort:
1070 add_xsd_string(ptr, "xSortKeys", req->sort.xSortKeys);
1073 add_xsd_string(ptr, "stylesheet", req->stylesheet);
1074 add_xsd_string(ptr, "database", req->database);
1076 else if ((*p)->which == Z_SRW_searchRetrieve_response)
1078 Z_SRW_searchRetrieveResponse *res = (*p)->u.response;
1079 ptr = xmlNewChild(pptr, 0, BAD_CAST "searchRetrieveResponse", 0);
1080 ns_srw = xmlNewNs(ptr, BAD_CAST ns, BAD_CAST "zs");
1081 xmlSetNs(ptr, ns_srw);
1083 if ((*p)->srw_version)
1084 add_xsd_string(ptr, "version", (*p)->srw_version);
1085 add_xsd_integer(ptr, "numberOfRecords", res->numberOfRecords);
1086 add_xsd_string(ptr, "resultSetId", res->resultSetId);
1087 add_xsd_integer(ptr, "resultSetIdleTime", res->resultSetIdleTime);
1088 if (res->num_records)
1090 xmlNodePtr rptr = xmlNewChild(ptr, 0, BAD_CAST "records", 0);
1091 yaz_srw_records(o, rptr, &res->records, &res->extra_records,
1095 add_xsd_integer(ptr, "nextRecordPosition",
1096 res->nextRecordPosition);
1097 if (res->num_diagnostics)
1099 xmlNodePtr rptr = xmlNewChild(ptr, 0, BAD_CAST "diagnostics",
1101 yaz_srw_diagnostics(o, rptr, &res->diagnostics,
1102 &res->num_diagnostics, client_data, ns);
1105 else if ((*p)->which == Z_SRW_explain_request)
1107 Z_SRW_explainRequest *req = (*p)->u.explain_request;
1108 ptr = xmlNewChild(pptr, 0, BAD_CAST "explainRequest", 0);
1109 ns_srw = xmlNewNs(ptr, BAD_CAST ns, BAD_CAST "zs");
1110 xmlSetNs(ptr, ns_srw);
1112 add_xsd_string(ptr, "version", (*p)->srw_version);
1113 add_xsd_string(ptr, "recordPacking", req->recordPacking);
1114 add_xsd_string(ptr, "stylesheet", req->stylesheet);
1115 add_xsd_string(ptr, "database", req->database);
1117 else if ((*p)->which == Z_SRW_explain_response)
1119 Z_SRW_explainResponse *res = (*p)->u.explain_response;
1120 ptr = xmlNewChild(pptr, 0, BAD_CAST "explainResponse", 0);
1121 ns_srw = xmlNewNs(ptr, BAD_CAST ns, BAD_CAST "zs");
1122 xmlSetNs(ptr, ns_srw);
1124 add_xsd_string(ptr, "version", (*p)->srw_version);
1127 xmlNodePtr ptr1 = xmlNewChild(ptr, 0, BAD_CAST "record", 0);
1128 yaz_srw_record(o, ptr1, &res->record, &res->extra_record,
1131 if (res->num_diagnostics)
1133 xmlNodePtr rptr = xmlNewChild(ptr, 0, BAD_CAST "diagnostics",
1135 yaz_srw_diagnostics(o, rptr, &res->diagnostics,
1136 &res->num_diagnostics, client_data, ns);
1139 else if ((*p)->which == Z_SRW_scan_request)
1141 Z_SRW_scanRequest *req = (*p)->u.scan_request;
1142 ptr = xmlNewChild(pptr, 0, BAD_CAST "scanRequest", 0);
1143 ns_srw = xmlNewNs(ptr, BAD_CAST ns, BAD_CAST "zs");
1144 xmlSetNs(ptr, ns_srw);
1146 add_xsd_string(ptr, "version", (*p)->srw_version);
1147 switch (req->query_type)
1149 case Z_SRW_query_type_cql:
1150 add_xsd_string(ptr, "scanClause", req->scanClause.cql);
1152 case Z_SRW_query_type_pqf:
1153 add_xsd_string(ptr, "pScanClause", req->scanClause.pqf);
1156 add_xsd_integer(ptr, "responsePosition", req->responsePosition);
1157 add_xsd_integer(ptr, "maximumTerms", req->maximumTerms);
1158 add_xsd_string(ptr, "stylesheet", req->stylesheet);
1159 add_xsd_string(ptr, "database", req->database);
1161 else if ((*p)->which == Z_SRW_scan_response)
1163 Z_SRW_scanResponse *res = (*p)->u.scan_response;
1164 ptr = xmlNewChild(pptr, 0, BAD_CAST "scanResponse", 0);
1165 ns_srw = xmlNewNs(ptr, BAD_CAST ns, BAD_CAST "zs");
1166 xmlSetNs(ptr, ns_srw);
1168 add_xsd_string(ptr, "version", (*p)->srw_version);
1172 xmlNodePtr rptr = xmlNewChild(ptr, 0, BAD_CAST "terms", 0);
1173 yaz_srw_terms(o, rptr, &res->terms, &res->num_terms,
1176 if (res->num_diagnostics)
1178 xmlNodePtr rptr = xmlNewChild(ptr, 0, BAD_CAST "diagnostics",
1180 yaz_srw_diagnostics(o, rptr, &res->diagnostics,
1181 &res->num_diagnostics, client_data, ns);
1186 if (ptr && (*p)->extraResponseData_len)
1187 add_XML_n(ptr, "extraResponseData",
1188 (*p)->extraResponseData_buf,
1189 (*p)->extraResponseData_len, ns_srw);
1196 int yaz_ucp_codec(ODR o, void * vptr, Z_SRW_PDU **handler_data,
1197 void *client_data, const char *ns_ucp_str)
1199 xmlNodePtr pptr = (xmlNodePtr) vptr;
1200 const char *ns_srw_str = YAZ_XMLNS_SRU_v1_1;
1201 if (o->direction == ODR_DECODE)
1203 Z_SRW_PDU **p = handler_data;
1204 xmlNodePtr method = pptr->children;
1206 while (method && method->type == XML_TEXT_NODE)
1207 method = method->next;
1211 if (method->type != XML_ELEMENT_NODE)
1214 *p = yaz_srw_get_core_v_1_1(o);
1216 if (!xmlStrcmp(method->name, BAD_CAST "updateRequest"))
1218 xmlNodePtr ptr = method->children;
1219 Z_SRW_updateRequest *req;
1222 (*p)->which = Z_SRW_update_request;
1223 req = (*p)->u.update_request = (Z_SRW_updateRequest *)
1224 odr_malloc(o, sizeof(*req));
1228 req->recordVersions = 0;
1229 req->num_recordVersions = 0;
1231 req->extra_record = 0;
1232 req->extraRequestData_buf = 0;
1233 req->extraRequestData_len = 0;
1234 req->stylesheet = 0;
1236 for (; ptr; ptr = ptr->next)
1238 if (match_xsd_string(ptr, "version", o,
1239 &(*p)->srw_version))
1241 else if (match_xsd_string(ptr, "action", o,
1245 if (!strcmp(oper, "info:srw/action/1/delete"))
1246 req->operation = "delete";
1247 else if (!strcmp(oper,"info:srw/action/1/replace" ))
1248 req->operation = "replace";
1249 else if (!strcmp(oper, "info:srw/action/1/create"))
1250 req->operation = "insert";
1253 else if (match_xsd_string(ptr, "recordIdentifier", o,
1256 else if (match_element(ptr, "recordVersions" ) )
1257 yaz_srw_versions( o, ptr, &req->recordVersions,
1258 &req->num_recordVersions, client_data,
1260 else if (match_element(ptr, "record"))
1262 req->record = yaz_srw_get_record(o);
1263 yaz_srw_record(o, ptr, req->record, &req->extra_record,
1264 client_data, ns_ucp_str);
1266 else if (match_xsd_string(ptr, "stylesheet", o,
1270 match_xsd_string(ptr, "database", o, &req->database);
1273 else if (!xmlStrcmp(method->name, BAD_CAST "updateResponse"))
1275 xmlNodePtr ptr = method->children;
1276 Z_SRW_updateResponse *res;
1278 (*p)->which = Z_SRW_update_response;
1279 res = (*p)->u.update_response = (Z_SRW_updateResponse *)
1280 odr_malloc(o, sizeof(*res));
1282 res->operationStatus = 0;
1284 res->recordVersions = 0;
1285 res->num_recordVersions = 0;
1286 res->diagnostics = 0;
1287 res->num_diagnostics = 0;
1289 res->extra_record = 0;
1290 res->extraResponseData_buf = 0;
1291 res->extraResponseData_len = 0;
1293 for (; ptr; ptr = ptr->next)
1295 if (match_xsd_string(ptr, "version", o,
1296 &(*p)->srw_version))
1298 else if (match_xsd_string(ptr, "operationStatus", o,
1299 &res->operationStatus ))
1301 else if (match_xsd_string(ptr, "recordIdentifier", o,
1304 else if (match_element(ptr, "recordVersions" ))
1305 yaz_srw_versions(o, ptr, &res->recordVersions,
1306 &res->num_recordVersions,
1307 client_data, ns_ucp_str);
1308 else if (match_element(ptr, "record"))
1310 res->record = yaz_srw_get_record(o);
1311 yaz_srw_record(o, ptr, res->record, &res->extra_record,
1312 client_data, ns_ucp_str);
1314 else if (match_element(ptr, "diagnostics"))
1315 yaz_srw_diagnostics(o, ptr, &res->diagnostics,
1316 &res->num_diagnostics,
1317 client_data, ns_ucp_str);
1320 else if (!xmlStrcmp(method->name, BAD_CAST "explainUpdateRequest"))
1323 else if (!xmlStrcmp(method->name, BAD_CAST "explainUpdateResponse"))
1332 else if (o->direction == ODR_ENCODE)
1334 Z_SRW_PDU **p = handler_data;
1335 xmlNsPtr ns_ucp, ns_srw;
1337 if ((*p)->which == Z_SRW_update_request)
1339 Z_SRW_updateRequest *req = (*p)->u.update_request;
1340 xmlNodePtr ptr = xmlNewChild(pptr, 0, BAD_CAST "updateRequest", 0);
1341 ns_ucp = xmlNewNs(ptr, BAD_CAST ns_ucp_str, BAD_CAST "zu");
1342 xmlSetNs(ptr, ns_ucp);
1343 ns_srw = xmlNewNs(ptr, BAD_CAST ns_srw_str, BAD_CAST "zs");
1345 add_xsd_string_ns(ptr, "version", (*p)->srw_version, ns_srw);
1346 add_xsd_string(ptr, "action", req->operation);
1347 add_xsd_string(ptr, "recordIdentifier", req->recordId );
1348 if (req->recordVersions)
1349 yaz_srw_versions( o, ptr, &req->recordVersions,
1350 &req->num_recordVersions,
1351 client_data, ns_ucp_str);
1352 if (req->record && req->record->recordData_len)
1354 xmlNodePtr rptr = xmlNewChild(ptr, 0, BAD_CAST "record", 0);
1355 xmlSetNs(rptr, ns_srw);
1356 yaz_srw_record(o, rptr, req->record, &req->extra_record,
1357 client_data, ns_ucp_str);
1359 if (req->extraRequestData_len)
1361 add_XML_n(ptr, "extraRequestData",
1362 req->extraRequestData_buf,
1363 req->extraRequestData_len, ns_srw);
1365 add_xsd_string(ptr, "stylesheet", req->stylesheet);
1366 add_xsd_string(ptr, "database", req->database);
1368 else if ((*p)->which == Z_SRW_update_response)
1370 Z_SRW_updateResponse *res = (*p)->u.update_response;
1371 xmlNodePtr ptr = xmlNewChild(pptr, 0, (xmlChar *)
1372 "updateResponse", 0);
1373 ns_ucp = xmlNewNs(ptr, BAD_CAST ns_ucp_str, BAD_CAST "zu");
1374 xmlSetNs(ptr, ns_ucp);
1375 ns_srw = xmlNewNs(ptr, BAD_CAST ns_srw_str, BAD_CAST "zs");
1377 add_xsd_string_ns(ptr, "version", (*p)->srw_version, ns_srw);
1378 add_xsd_string(ptr, "operationStatus", res->operationStatus );
1379 add_xsd_string(ptr, "recordIdentifier", res->recordId );
1380 if (res->recordVersions)
1381 yaz_srw_versions(o, ptr, &res->recordVersions,
1382 &res->num_recordVersions,
1383 client_data, ns_ucp_str);
1384 if (res->record && res->record->recordData_len)
1386 xmlNodePtr rptr = xmlNewChild(ptr, 0, BAD_CAST "record", 0);
1387 xmlSetNs(rptr, ns_srw);
1388 yaz_srw_record(o, rptr, res->record, &res->extra_record,
1389 client_data, ns_ucp_str);
1391 if (res->num_diagnostics)
1394 xmlNewNs(pptr, BAD_CAST YAZ_XMLNS_DIAG_v1_1,
1397 xmlNodePtr rptr = xmlNewChild(ptr, ns_diag, BAD_CAST "diagnostics", 0);
1398 yaz_srw_diagnostics(o, rptr, &res->diagnostics,
1399 &res->num_diagnostics, client_data,
1402 if (res->extraResponseData_len)
1403 add_XML_n(ptr, "extraResponseData",
1404 res->extraResponseData_buf,
1405 res->extraResponseData_len, ns_srw);
1419 * c-file-style: "Stroustrup"
1420 * indent-tabs-mode: nil
1422 * vim: shiftwidth=4 tabstop=8 expandtab