ReactOS 0.4.16-dev-2216-ga08d639
xdr.c
Go to the documentation of this file.
1/*
2 * XDR (XML-Data Reduced) -> XSD (XML Schema Document) conversion
3 *
4 * Copyright 2010 Adam Martinson for CodeWeavers
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 */
20
21#include <assert.h>
22#include <libxml/tree.h>
23
24#include "wine/debug.h"
25
26/* Both XDR and XSD are valid XML
27 * We just convert the doc tree, no need for a parser.
28 */
29
31
32static const xmlChar DT_prefix[] = "dt";
33static const xmlChar DT_href[] = "urn:schemas-microsoft-com:datatypes";
34static const xmlChar XDR_href[] = "urn:schemas-microsoft-com:xml-data";
35static const xmlChar XSD_prefix[] = "xsd";
36static const xmlChar XSD_href[] = "http://www.w3.org/2001/XMLSchema";
37
38static const xmlChar xs_all[] = "all";
39static const xmlChar xs_annotation[] = "annotation";
40static const xmlChar xs_any[] = "any";
41static const xmlChar xs_anyAttribute[] = "anyAttribute";
42static const xmlChar xs_attribute[] = "attribute";
43static const xmlChar xs_AttributeType[] = "AttributeType";
44static const xmlChar xs_base[] = "base";
45static const xmlChar xs_choice[] = "choice";
46static const xmlChar xs_complexType[] = "complexType";
47static const xmlChar xs_content[] = "content";
48static const xmlChar xs_datatype[] = "datatype";
49static const xmlChar xs_default[] = "default";
50static const xmlChar xs_description[] = "description";
51static const xmlChar xs_documentation[] = "documentation";
52static const xmlChar xs_element[] = "element";
53static const xmlChar xs_ElementType[] = "ElementType";
54static const xmlChar xs_eltOnly[] = "eltOnly";
55static const xmlChar xs_enumeration[] = "enumeration";
56static const xmlChar xs_extension[] = "extension";
57static const xmlChar xs_group[] = "group";
58static const xmlChar xs_lax[] = "lax";
59static const xmlChar xs_length[] = "length";
60static const xmlChar xs_many[] = "many";
61static const xmlChar xs_maxOccurs[] = "maxOccurs";
62static const xmlChar xs_minOccurs[] = "minOccurs";
63static const xmlChar xs_mixed[] = "mixed";
64static const xmlChar xs_model[] = "model";
65static const xmlChar xs_name[] = "name";
66static const xmlChar xs_namespace[] = "namespace";
67static const xmlChar xs_no[] = "no";
68static const xmlChar xs_open[] = "open";
69static const xmlChar xs_optional[] = "optional";
70static const xmlChar xs_order[] = "order";
71static const xmlChar xs_processContents[] = "processContents";
72static const xmlChar xs_ref[] = "ref";
73static const xmlChar xs_required[] = "required";
74static const xmlChar xs_restriction[] = "restriction";
75static const xmlChar xs_schema[] = "schema";
76static const xmlChar xs_seq[] = "seq";
77static const xmlChar xs_sequence[] = "sequence";
78static const xmlChar xs_simpleContent[] = "simpleContent";
79static const xmlChar xs_simpleType[] = "simpleType";
80static const xmlChar xs_strict[] = "strict";
81static const xmlChar xs_targetNamespace[] = "targetNamespace";
82static const xmlChar xs_textOnly[] = "textOnly";
83static const xmlChar xs_true[] = "true";
84static const xmlChar xs_type[] = "type";
85static const xmlChar xs_unbounded[] = "unbounded";
86static const xmlChar xs_use[] = "use";
87static const xmlChar xs_value[] = "value";
88static const xmlChar xs_values[] = "values";
89static const xmlChar xs_xsd_string[] = "xsd:string";
90
91typedef enum _CONTENT_TYPE
92{
98
99typedef enum _ORDER_TYPE
100{
105
106#define FOREACH_CHILD(node, child) \
107 for (child = node->children; child != NULL; child = child->next) \
108 if (child->type == XML_ELEMENT_NODE)
109
110#define FOREACH_ATTR(node, attr) \
111 for (attr = node->properties; attr != NULL; attr = attr->next)
112
113#define FOREACH_NS(node, ns) \
114 for (ns = node->nsDef; ns != NULL; ns = ns->next)
115
116static inline xmlNodePtr get_schema(xmlNodePtr node)
117{
118 return xmlDocGetRootElement(node->doc);
119}
120
121static inline xmlNodePtr get_child(xmlNodePtr node, xmlChar const* name)
122{
123 xmlNodePtr child = NULL;
124 if (node)
125 {
127 {
128 if (xmlStrEqual(child->name, name))
129 break;
130 }
131 }
132
133 return child;
134}
135
136static inline xmlNodePtr get_child_with_attr(xmlNodePtr node, xmlChar const* name,
137 xmlChar const* attr_ns, xmlChar const* attr_name,
138 xmlChar const* attr_val)
139{
140 xmlChar* str;
141 if (node)
142 {
144 {
145 if (xmlStrEqual(node->name, name))
146 {
147 str = (attr_ns != NULL)? xmlGetNsProp(node, attr_name, attr_ns) :
148 xmlGetProp(node, attr_name);
149 if (str)
150 {
151 if (xmlStrEqual(str, attr_val))
152 {
153 xmlFree(str);
154 return node;
155 }
156 xmlFree(str);
157 }
158 }
159 }
160 }
161
162 return NULL;
163}
164
165static inline xmlNsPtr get_dt_ns(xmlNodePtr node)
166{
167 xmlNsPtr ns;
168
170 assert(node != NULL);
171
173 {
174 if (xmlStrEqual(ns->href, DT_href))
175 break;
176 }
177
178 return ns;
179}
180
181static inline xmlChar* get_dt_type(xmlNodePtr xdr)
182{
183 xmlChar* str = xmlGetNsProp(xdr, xs_type, DT_href);
184 if (!str)
185 {
186 xmlNodePtr datatype = get_child(xdr, xs_datatype);
187 if (datatype)
188 str = xmlGetNsProp(datatype, xs_type, DT_href);
189 }
190 return str;
191}
192
193static inline xmlChar* get_attr_val(xmlAttrPtr attr)
194{
195 return xmlNodeGetContent((xmlNodePtr)attr);
196}
197
198static inline xmlNodePtr add_any_child(xmlNodePtr parent, BOOL set_occurs)
199{
200 xmlNodePtr child = xmlNewChild(parent, NULL, xs_any, NULL);
201 if (set_occurs)
202 {
203 xmlSetProp(child, xs_minOccurs, BAD_CAST "0");
204 xmlSetProp(child, xs_maxOccurs, xs_unbounded);
205 }
206 xmlSetProp(child, xs_processContents, xs_strict);
207 return child;
208}
209
210static inline xmlNodePtr add_anyAttribute_child(xmlNodePtr parent)
211{
212 xmlNodePtr child = xmlNewChild(parent, NULL, xs_anyAttribute, NULL);
213 xmlSetProp(child, xs_processContents, xs_lax);
214 return child;
215}
216
217static inline xmlAttrPtr copy_prop_ignore_ns(xmlAttrPtr xdr_attr, xmlNodePtr node)
218{
219 xmlChar* str = get_attr_val(xdr_attr);
220 xmlAttrPtr attr = xmlSetProp(node, xdr_attr->name, str);
221 xmlFree(str);
222 return attr;
223}
224static inline xmlAttrPtr XDR_A_default(xmlAttrPtr xdr_attr, xmlNodePtr node)
225{
226 TRACE("(%p, %p)\n", xdr_attr, node);
227
228 return copy_prop_ignore_ns(xdr_attr, node);
229}
230
231static inline xmlAttrPtr XDR_A_dt_type(xmlAttrPtr xdr_attr, xmlNodePtr node)
232{
233 xmlChar* str = get_attr_val(xdr_attr);
234 xmlAttrPtr attr;
235
236 TRACE("(%p, %p)\n", xdr_attr, node);
237
239 attr = NULL;
240 else
241 attr = xmlSetNsProp(node, get_dt_ns(node), DT_prefix, str);
242 xmlFree(str);
243 return attr;
244}
245
246static xmlAttrPtr XDR_A_maxOccurs(xmlAttrPtr xdr_attr, xmlNodePtr node)
247{
248 xmlChar* str = get_attr_val(xdr_attr);
249 xmlAttrPtr attr;
250
251 TRACE("(%p, %p)\n", xdr_attr, node);
252
253 if (xmlStrEqual(str, BAD_CAST "*"))
254 attr = xmlSetProp(node, xs_maxOccurs, xs_unbounded);
255 else
256 attr = copy_prop_ignore_ns(xdr_attr, node);
257
258 xmlFree(str);
259 return attr;
260}
261
262static inline xmlAttrPtr XDR_A_minOccurs(xmlAttrPtr xdr_attr, xmlNodePtr node)
263{
264 TRACE("(%p, %p)\n", xdr_attr, node);
265
266 return copy_prop_ignore_ns(xdr_attr, node);
267}
268
269static inline xmlAttrPtr XDR_A_name(xmlAttrPtr xdr_attr, xmlNodePtr node)
270{
271 TRACE("(%p, %p)\n", xdr_attr, node);
272
273 return copy_prop_ignore_ns(xdr_attr, node);
274}
275
276static xmlAttrPtr XDR_A_type(xmlAttrPtr xdr_attr, xmlNodePtr node)
277{
278 xmlChar* str = get_attr_val(xdr_attr);
279 xmlAttrPtr attr = xmlSetProp(node, xs_ref, str);
280
281 TRACE("(%p, %p)\n", xdr_attr, node);
282
283 xmlFree(str);
284 return attr;
285}
286
287static xmlAttrPtr XDR_A_required(xmlAttrPtr xdr_attr, xmlNodePtr node)
288{
289 xmlChar* str = get_attr_val(xdr_attr);
290 xmlAttrPtr attr;
291
292 TRACE("(%p, %p)\n", xdr_attr, node);
293
294 if (xmlStrEqual(str, xs_no))
295 attr = xmlSetProp(node, xs_use, xs_optional);
296 else /* yes */
297 attr = xmlSetProp(node, xs_use, xs_required);
298 xmlFree(str);
299 return attr;
300}
301
302static xmlNodePtr XDR_E_description(xmlNodePtr xdr, xmlNodePtr parent)
303{
304 xmlNodePtr xsd_node = xmlNewChild(parent, NULL, xs_annotation, NULL);
305 xmlAttrPtr xdr_attr;
306
307 TRACE("(%p, %p)\n", xdr, parent);
308
309 xmlNewChild(xsd_node, NULL, xs_documentation, xdr->content);
310
311 FOREACH_ATTR(xdr, xdr_attr)
312 {
313 xmlCopyProp(xsd_node, xdr_attr);
314 }
315 return xsd_node;
316}
317
318static xmlNodePtr XDR_E_AttributeType(xmlNodePtr xdr, xmlNodePtr parent)
319{
320 xmlChar *str, *type = get_dt_type(xdr);
321 xmlNodePtr xsd_node, xsd_child, xdr_child;
322 xmlAttrPtr xdr_attr;
323
324 TRACE("(%p, %p)\n", xdr, parent);
325
326 xsd_node = xmlNewChild(parent, NULL, xs_attribute, NULL);
327
329 {
330 xmlChar *tmp, *tokBegin, *tokEnd = NULL;
331 xmlNodePtr xsd_enum;
332 xsd_child = xmlNewChild(xsd_node, NULL, xs_simpleType, NULL);
333 xsd_child = xmlNewChild(xsd_child, NULL, xs_restriction, NULL);
334 xmlSetProp(xsd_child, xs_base, xs_xsd_string);
335
336 tokBegin = str = xmlGetNsProp(xdr, xs_values, DT_href);
337 while (tokBegin && *tokBegin)
338 {
339 while (*tokBegin && isspace(*tokBegin))
340 ++tokBegin;
341 tokEnd = tokBegin;
342 while (*tokEnd && !isspace(*tokEnd))
343 ++tokEnd;
344 if (tokEnd == tokBegin)
345 break;
346 xsd_enum = xmlNewChild(xsd_child, NULL, xs_enumeration, NULL);
347 tmp = xmlStrndup(tokBegin, tokEnd-tokBegin);
348 xmlSetProp(xsd_enum, xs_value, tmp);
349 xmlFree(tmp);
350 tokBegin = tokEnd;
351 }
352 xmlFree(str);
353
354 }
355 else if (type)
356 {
358 str = xmlStrcat(str, BAD_CAST ":");
359 str = xmlStrcat(str, type);
360 xmlSetProp(xsd_node, xs_type, str);
361 xmlFree(str);
362 }
363 xmlFree(type);
364
365 FOREACH_ATTR(xdr, xdr_attr)
366 {
367 if (xmlStrEqual(xdr_attr->name, xs_default))
368 XDR_A_default(xdr_attr, xsd_node);
369 else if (xmlStrEqual(xdr_attr->name, xs_name))
370 XDR_A_name(xdr_attr, xsd_node);
371 else if (xmlStrEqual(xdr_attr->name, xs_type) && xdr_attr->ns == get_dt_ns(xdr))
372 XDR_A_dt_type(xdr_attr, xsd_node);
373 else if (xmlStrEqual(xdr_attr->name, xs_values) && xdr_attr->ns == get_dt_ns(xdr))
374 ; /* already handled */
375 else if (xmlStrEqual(xdr_attr->name, xs_required))
376 XDR_A_required(xdr_attr, xsd_node);
377 else
378 xmlCopyProp(xsd_node, xdr_attr);
379 }
380
381 FOREACH_CHILD(xdr, xdr_child)
382 {
383 if (xmlStrEqual(xdr_child->name, xs_datatype))
384 ; /* already handled */
385 else if (xmlStrEqual(xdr_child->name, xs_description))
386 XDR_E_description(xdr_child, xsd_node);
387 else
388 FIXME("unexpected child <%s>\n", xdr_child->name);
389 }
390
391 return xsd_node;
392}
393
394static xmlNodePtr XDR_E_attribute(xmlNodePtr xdr, xmlNodePtr parent)
395{
396 xmlChar* str = xmlGetProp(xdr, xs_type);
397 xmlNodePtr xsd_node, xdr_child, xdr_attrType;
398 xmlAttrPtr xdr_attr;
399
400 TRACE("(%p, %p)\n", xdr, parent);
401
402 xdr_attrType = get_child_with_attr(xdr->parent, xs_AttributeType, NULL, xs_name, str);
403 xmlFree(str);
404
405 if (xdr_attrType)
406 xsd_node = XDR_E_AttributeType(xdr_attrType, parent);
407 else
408 xsd_node = xmlNewChild(parent, NULL, xs_attribute, NULL);
409
410 FOREACH_ATTR(xdr, xdr_attr)
411 {
412 if (xmlStrEqual(xdr_attr->name, xs_default))
413 XDR_A_default(xdr_attr, xsd_node);
414 else if (xmlStrEqual(xdr_attr->name, xs_type) && !xdr_attrType)
415 XDR_A_type(xdr_attr, xsd_node);
416 else if (xmlStrEqual(xdr_attr->name, xs_required))
417 XDR_A_required(xdr_attr, xsd_node);
418 else
419 xmlCopyProp(xsd_node, xdr_attr);
420 }
421
422 FOREACH_CHILD(xdr, xdr_child)
423 {
424 FIXME("unexpected child <%s>\n", xdr_child->name);
425 }
426
427 return xsd_node;
428}
429
430static xmlNodePtr XDR_E_element(xmlNodePtr xdr, xmlNodePtr parent)
431{
432 xmlNodePtr xdr_child, xsd_node = xmlNewChild(parent, NULL, xs_element, NULL);
433 xmlAttrPtr xdr_attr;
434
435 FOREACH_ATTR(xdr, xdr_attr)
436 {
437 if (xmlStrEqual(xdr_attr->name, xs_type))
438 XDR_A_type(xdr_attr, xsd_node);
439 else if (xmlStrEqual(xdr_attr->name, xs_maxOccurs))
440 XDR_A_maxOccurs(xdr_attr, xsd_node);
441 else if (xmlStrEqual(xdr_attr->name, xs_minOccurs))
442 XDR_A_minOccurs(xdr_attr, xsd_node);
443 else
444 xmlCopyProp(xsd_node, xdr_attr);
445 }
446
447 FOREACH_CHILD(xdr, xdr_child)
448 {
449 FIXME("unexpected child <%s>\n", xdr_child->name);
450 }
451
452 return xsd_node;
453}
454
455static xmlNodePtr XDR_E_group(xmlNodePtr xdr, xmlNodePtr parent)
456{
457 xmlNodePtr xdr_child, xsd_node;
458 xmlChar* str = xmlGetProp(xdr, xs_order);
459 xmlAttrPtr xdr_attr;
460
461 TRACE("(%p, %p)\n", xdr, parent);
462
463 if (!str || xmlStrEqual(str, xs_seq))
464 xsd_node = xmlNewChild(parent, NULL, xs_sequence, NULL);
465 else if (xmlStrEqual(str, xs_many))
466 xsd_node = xmlNewChild(parent, NULL, xs_choice, NULL);
467 else /* one */
468 xsd_node = xmlNewChild(parent, NULL, xs_all, NULL);
469 xmlFree(str);
470
471 FOREACH_ATTR(xdr, xdr_attr)
472 {
473 if (xmlStrEqual(xdr_attr->name, xs_order))
474 ; /* already handled */
475 else if (xmlStrEqual(xdr_attr->name, xs_model))
476 ; /* ignored */
477 else if (xmlStrEqual(xdr_attr->name, xs_maxOccurs))
478 XDR_A_maxOccurs(xdr_attr, xsd_node);
479 else if (xmlStrEqual(xdr_attr->name, xs_minOccurs))
480 XDR_A_minOccurs(xdr_attr, xsd_node);
481 else
482 xmlCopyProp(xsd_node, xdr_attr);
483 }
484
485 FOREACH_CHILD(xdr, xdr_child)
486 {
487 if (xmlStrEqual(xdr_child->name, xs_description))
488 XDR_E_description(xdr_child, xsd_node);
489 else if (xmlStrEqual(xdr_child->name, xs_element))
490 XDR_E_element(xdr_child, xsd_node);
491 }
492
493 return xsd_node;
494}
495
496static xmlNodePtr XDR_E_ElementType(xmlNodePtr xdr, xmlNodePtr parent)
497{
498 xmlChar *str, *type = get_dt_type(xdr);
499 BOOL is_open = TRUE;
500 int n_attributes = 0, n_elements = 0, n_groups = 0;
503 xmlNodePtr xsd_node, xsd_type, xsd_child, xdr_child;
504 xmlAttrPtr xdr_attr;
505 xmlNsPtr dt_ns = get_dt_ns(parent);
506
507 TRACE("(%p, %p)\n", xdr, parent);
508
509 str = xmlGetProp(xdr, xs_model);
510 if (str && !xmlStrEqual(str, xs_open))
511 is_open = FALSE;
512 xmlFree(str);
513
514 if (type)
515 {
517 }
518 else
519 {
520 str = xmlGetProp(xdr, xs_content);
521 if (!str || xmlStrEqual(str, xs_mixed))
523 else if (xmlStrEqual(str, xs_eltOnly))
525 else if (xmlStrEqual(str, xs_textOnly))
527 else /* empty */
529 xmlFree(str);
530 }
531
532 str = xmlGetProp(xdr, xs_order);
533 if (!str || xmlStrEqual(str, xs_seq))
534 {
536 }
537 else if (xmlStrEqual(str, xs_many))
538 {
540 }
541 else /* one */
542 {
544 is_open = FALSE;
545 }
546 xmlFree(str);
547
548 FOREACH_CHILD(xdr, xdr_child)
549 {
550 if (xmlStrEqual(xdr_child->name, xs_element))
551 ++n_elements;
552 else if (xmlStrEqual(xdr_child->name, xs_group))
553 ++n_groups;
554 else if (xmlStrEqual(xdr_child->name, xs_attribute))
555 ++n_attributes;
556 }
557
558 xsd_node = xmlNewChild(parent, NULL, xs_element, NULL);
559 assert(xsd_node != NULL);
560 switch (content)
561 {
562 case CONTENT_MIXED:
563 case CONTENT_ELTONLY:
564 {
565 xmlNodePtr xsd_base;
566 xsd_type = xmlNewChild(xsd_node, NULL, xs_complexType, NULL);
567
568 if (content == CONTENT_MIXED)
569 xmlSetProp(xsd_type, xs_mixed, xs_true);
570
571 if (is_open)
572 xsd_base = xmlNewChild(xsd_type, NULL, xs_sequence, NULL);
573 else
574 xsd_base = xsd_type;
575
576 if (is_open && n_elements < 2 && !n_groups)
577 {/* no specific sequence of elements we need,
578 just has to start with the right one, if any */
579 if ((xdr_child = get_child(xdr, xs_element)))
580 {
581 xsd_child = XDR_E_element(xdr_child, xsd_base);
582 xmlUnsetProp(xsd_child, xs_maxOccurs);
583 }
584 }
585 else
586 {
587 switch (order)
588 {
589 case ORDER_SEQ:
590 xsd_child = xmlNewChild(xsd_base, NULL, xs_sequence, NULL);
591 break;
592 case ORDER_MANY:
593 xsd_child = xmlNewChild(xsd_base, NULL, xs_choice, NULL);
594 xmlSetProp(xsd_child, xs_maxOccurs, xs_unbounded);
595 break;
596 case ORDER_ONE:
597 xsd_child = xmlNewChild(xsd_base, NULL, xs_all, NULL);
598 break;
599 }
600
601 FOREACH_CHILD(xdr, xdr_child)
602 {
603 if (xmlStrEqual(xdr_child->name, xs_element))
604 XDR_E_element(xdr_child, xsd_child);
605 else if (xmlStrEqual(xdr_child->name, xs_group))
606 XDR_E_group(xdr_child, xsd_child);
607 }
608 }
609
610 if (n_attributes)
611 {
612 FOREACH_CHILD(xdr, xdr_child)
613 {
614 if (xmlStrEqual(xdr_child->name, xs_attribute))
615 XDR_E_attribute(xdr_child, xsd_type);
616 }
617 }
618
619 if (is_open)
620 {
621 add_any_child(xsd_base, TRUE);
622 add_anyAttribute_child(xsd_type);
623 }
624 }
625 break;
626 case CONTENT_TEXTONLY:
627 {
628 if (is_open)
629 {
630 xsd_type = xmlNewChild(xsd_node, NULL, xs_complexType, NULL);
631 if (type)
632 {
633 xsd_child = xmlNewChild(xsd_type, NULL, xs_simpleContent, NULL);
634 xsd_child = xmlNewChild(xsd_child, NULL, xs_extension, NULL);
636 str = xmlStrcat(str, BAD_CAST ":");
637 str = xmlStrcat(str, type);
638 xmlSetProp(xsd_child, xs_base, str);
639 xmlFree(str);
640 assert(dt_ns != NULL);
641 xmlSetNsProp(xsd_node, dt_ns, DT_prefix, type);
642 }
643 else
644 {
645 xmlSetProp(xsd_type, xs_mixed, xs_true);
646 xsd_child = xmlNewChild(xsd_type, NULL, xs_choice, NULL);
647 xmlSetProp(xsd_child, xs_minOccurs, BAD_CAST "0");
648 xmlSetProp(xsd_child, xs_maxOccurs, xs_unbounded);
649 xsd_child = add_any_child(xsd_child, FALSE);
650 xmlSetProp(xsd_child, xs_namespace, BAD_CAST "##other");
651 xsd_child = xsd_type;
652 }
653
654 if (n_attributes)
655 FOREACH_CHILD(xdr, xdr_child)
656 {
657 if (xmlStrEqual(xdr_child->name, xs_attribute))
658 XDR_E_attribute(xdr_child, xsd_child);
659 }
660
661 xmlNewChild(xsd_child, NULL, xs_anyAttribute, NULL);
662 }
663 else if (!n_attributes)
664 {
665 if (type)
666 {
668 str = xmlStrcat(str, BAD_CAST ":");
669 str = xmlStrcat(str, type);
670 xmlSetProp(xsd_node, xs_type, str);
671 xmlFree(str);
672 str = NULL;
673 xmlSetNsProp(xsd_node, dt_ns, DT_prefix, type);
674 }
675 else
676 {
677 xmlSetProp(xsd_node, xs_type, xs_xsd_string);
678 }
679 }
680 else
681 {
682 xsd_type = xmlNewChild(xsd_node, NULL, xs_complexType, NULL);
683 xsd_child = xmlNewChild(xsd_type, NULL, xs_simpleContent, NULL);
684 xsd_child = xmlNewChild(xsd_child, NULL, xs_extension, NULL);
685 xmlSetProp(xsd_child, xs_base, xs_xsd_string);
686
687 FOREACH_CHILD(xdr, xdr_child)
688 {
689 if (xmlStrEqual(xdr_child->name, xs_attribute))
690 XDR_E_attribute(xdr_child, xsd_child);
691 }
692 }
693 }
694 break;
695 case CONTENT_EMPTY: /* not allowed with model="open" */
696 {
697 if (n_attributes)
698 {
699 xsd_type = xmlNewChild(xsd_node, NULL, xs_complexType, NULL);
700
701 FOREACH_CHILD(xdr, xdr_child)
702 {
703 if (xmlStrEqual(xdr_child->name, xs_attribute))
704 XDR_E_attribute(xdr_child, xsd_type);
705 }
706 }
707 else
708 {
709 xsd_type = xmlNewChild(xsd_node, NULL, xs_simpleType, NULL);
710 xsd_child = xmlNewChild(xsd_type, NULL, xs_restriction, NULL);
711 xmlSetProp(xsd_child, xs_base, xs_xsd_string);
712 xsd_child = xmlNewChild(xsd_child, NULL, xs_length, NULL);
713 xmlSetProp(xsd_child, xs_value, BAD_CAST "0");
714 }
715 }
716 break;
717 }
718 xmlFree(type);
719
720 FOREACH_ATTR(xdr, xdr_attr)
721 {
722 if (xmlStrEqual(xdr_attr->name, xs_content))
723 ; /* already handled */
724 else if (xmlStrEqual(xdr_attr->name, xs_name))
725 XDR_A_name(xdr_attr, xsd_node);
726 else if (xmlStrEqual(xdr_attr->name, xs_type) && xdr_attr->ns == get_dt_ns(xdr))
727 XDR_A_dt_type(xdr_attr, xsd_node);
728 else if (xmlStrEqual(xdr_attr->name, xs_model))
729 ; /* already handled */
730 else if (xmlStrEqual(xdr_attr->name, xs_order))
731 ; /* already handled */
732 else
733 xmlCopyProp(xsd_node, xdr_attr);
734
735 }
736
737 FOREACH_CHILD(xdr, xdr_child)
738 {
739 if (xmlStrEqual(xdr_child->name, xs_attribute))
740 ; /* already handled */
741 else if (xmlStrEqual(xdr_child->name, xs_AttributeType))
742 ; /* handled through XDR_E_attribute when parent is not <Schema> */
743 else if (xmlStrEqual(xdr_child->name, xs_datatype))
744 ; /* already handled */
745 else if (xmlStrEqual(xdr_child->name, xs_description))
746 XDR_E_description(xdr_child, xsd_node);
747 else if (xmlStrEqual(xdr_child->name, xs_element))
748 ; /* already handled */
749 else if (xmlStrEqual(xdr_child->name, xs_group))
750 ; /* already handled */
751 else
752 FIXME("unexpected child <%s>\n", xdr_child->name);
753 }
754
755 return xsd_node;
756}
757
758static xmlNodePtr XDR_E_Schema(xmlNodePtr xdr, xmlNodePtr parent, xmlChar const* nsURI)
759{
760 xmlNodePtr xsd_node, xdr_child;
761 xmlNsPtr ns, xdr_ns;
762 xmlAttrPtr xdr_attr;
763
764 TRACE("(%p, %p)\n", xdr, parent);
765
766 xsd_node = xmlNewDocNode((xmlDocPtr)parent, NULL, xs_schema, NULL);
767 xmlDocSetRootElement((xmlDocPtr)parent, xsd_node);
768 assert(xsd_node != NULL);
769
770 if (nsURI && *nsURI) xmlNewNs(xsd_node, nsURI, NULL);
771 ns = xmlNewNs(xsd_node, XSD_href, XSD_prefix);
772 assert(ns != NULL);
773
774 xmlSetNs(xsd_node, ns);
775
776 if (nsURI && *nsURI) xmlSetProp(xsd_node, xs_targetNamespace, nsURI);
777
778 FOREACH_NS(xdr, xdr_ns)
779 {
780 /* TODO: special handling for dt namespace? */
781 assert(xdr_ns->href != NULL);
782 if (xmlStrEqual(xdr_ns->href, XDR_href))
783 ; /* ignored */
784 else if (xdr_ns->prefix != NULL)
785 xmlNewNs(xsd_node, xdr_ns->href, xdr_ns->prefix);
786 else
787 FIXME("unexpected default xmlns: %s\n", xdr_ns->href);
788 }
789
790 FOREACH_ATTR(xdr, xdr_attr)
791 {
792 xmlCopyProp(xsd_node, xdr_attr);
793 }
794
795 FOREACH_CHILD(xdr, xdr_child)
796 {
797 if (xmlStrEqual(xdr_child->name, xs_AttributeType))
798 XDR_E_AttributeType(xdr_child, xsd_node);
799 else if (xmlStrEqual(xdr_child->name, xs_description))
800 XDR_E_description(xdr_child, xsd_node);
801 else if (xmlStrEqual(xdr_child->name, xs_ElementType))
802 XDR_E_ElementType(xdr_child, xsd_node);
803 else
804 FIXME("unexpected child <%s>\n", xdr_child->name);
805 }
806
807 return xsd_node;
808}
809
810xmlDocPtr XDR_to_XSD_doc(xmlDocPtr xdr_doc, xmlChar const* nsURI)
811{
812 xmlDocPtr xsd_doc = xmlNewDoc(NULL);
813
814 TRACE("(%p)\n", xdr_doc);
815
816 XDR_E_Schema(get_schema((xmlNodePtr)xdr_doc), (xmlNodePtr)xsd_doc, nsURI);
817
818 return xsd_doc;
819}
#define isspace(c)
Definition: acclib.h:69
#define WINE_DEFAULT_DEBUG_CHANNEL(t)
Definition: precomp.h:23
#define FIXME(fmt,...)
Definition: precomp.h:53
#define NULL
Definition: types.h:112
#define TRUE
Definition: types.h:120
#define FALSE
Definition: types.h:117
content
Definition: atl_ax.c:994
#define assert(_expr)
Definition: assert.h:32
r parent
Definition: btrfs.c:3010
unsigned int BOOL
Definition: ntddk_ex.h:94
GLuint GLuint GLsizei GLenum type
Definition: gl.h:1545
GLuint GLdouble GLdouble GLint GLint order
Definition: glext.h:11194
static HWND child
Definition: cursoricon.c:298
const WCHAR * str
xmlFreeFunc xmlFree
Definition: globals.c:184
#define TRACE(s)
Definition: solgame.cpp:4
Definition: cookie.c:202
Definition: name.c:39
Definition: mxnamespace.c:38
Definition: dlist.c:348
static xmlNodePtr get_schema(xmlNodePtr node)
Definition: xdr.c:116
static const xmlChar xs_open[]
Definition: xdr.c:68
static const xmlChar xs_base[]
Definition: xdr.c:44
static const xmlChar xs_true[]
Definition: xdr.c:83
static xmlAttrPtr XDR_A_required(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:287
static xmlNodePtr get_child_with_attr(xmlNodePtr node, xmlChar const *name, xmlChar const *attr_ns, xmlChar const *attr_name, xmlChar const *attr_val)
Definition: xdr.c:136
static const xmlChar xs_annotation[]
Definition: xdr.c:39
static const xmlChar xs_optional[]
Definition: xdr.c:69
static xmlNodePtr add_anyAttribute_child(xmlNodePtr parent)
Definition: xdr.c:210
enum _CONTENT_TYPE CONTENT_TYPE
static const xmlChar xs_enumeration[]
Definition: xdr.c:55
static const xmlChar xs_processContents[]
Definition: xdr.c:71
static xmlNsPtr get_dt_ns(xmlNodePtr node)
Definition: xdr.c:165
static const xmlChar XSD_href[]
Definition: xdr.c:36
static const xmlChar xs_values[]
Definition: xdr.c:88
static const xmlChar xs_extension[]
Definition: xdr.c:56
static const xmlChar DT_prefix[]
Definition: xdr.c:32
static const xmlChar xs_xsd_string[]
Definition: xdr.c:89
#define FOREACH_ATTR(node, attr)
Definition: xdr.c:110
static const xmlChar xs_complexType[]
Definition: xdr.c:46
static xmlAttrPtr XDR_A_dt_type(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:231
static const xmlChar xs_choice[]
Definition: xdr.c:45
static xmlNodePtr XDR_E_group(xmlNodePtr xdr, xmlNodePtr parent)
Definition: xdr.c:455
static const xmlChar xs_strict[]
Definition: xdr.c:80
static const xmlChar XSD_prefix[]
Definition: xdr.c:35
static const xmlChar xs_seq[]
Definition: xdr.c:76
static const xmlChar xs_schema[]
Definition: xdr.c:75
static const xmlChar xs_name[]
Definition: xdr.c:65
static const xmlChar xs_maxOccurs[]
Definition: xdr.c:61
static xmlAttrPtr XDR_A_type(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:276
static xmlChar * get_attr_val(xmlAttrPtr attr)
Definition: xdr.c:193
static const xmlChar xs_description[]
Definition: xdr.c:50
static const xmlChar xs_restriction[]
Definition: xdr.c:74
static xmlAttrPtr XDR_A_name(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:269
static const xmlChar xs_type[]
Definition: xdr.c:84
static const xmlChar xs_simpleContent[]
Definition: xdr.c:78
static const xmlChar xs_mixed[]
Definition: xdr.c:63
static const xmlChar xs_no[]
Definition: xdr.c:67
static xmlAttrPtr XDR_A_minOccurs(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:262
static const xmlChar xs_documentation[]
Definition: xdr.c:51
static xmlNodePtr XDR_E_AttributeType(xmlNodePtr xdr, xmlNodePtr parent)
Definition: xdr.c:318
static const xmlChar xs_order[]
Definition: xdr.c:70
static xmlNodePtr add_any_child(xmlNodePtr parent, BOOL set_occurs)
Definition: xdr.c:198
static const xmlChar xs_many[]
Definition: xdr.c:60
static const xmlChar xs_namespace[]
Definition: xdr.c:66
static const xmlChar xs_all[]
Definition: xdr.c:38
static const xmlChar xs_ElementType[]
Definition: xdr.c:53
static const xmlChar xs_simpleType[]
Definition: xdr.c:79
static const xmlChar xs_minOccurs[]
Definition: xdr.c:62
static const xmlChar xs_length[]
Definition: xdr.c:59
static const xmlChar xs_attribute[]
Definition: xdr.c:42
static const xmlChar xs_model[]
Definition: xdr.c:64
xmlDocPtr XDR_to_XSD_doc(xmlDocPtr xdr_doc, xmlChar const *nsURI)
Definition: xdr.c:810
static const xmlChar xs_any[]
Definition: xdr.c:40
static xmlNodePtr XDR_E_Schema(xmlNodePtr xdr, xmlNodePtr parent, xmlChar const *nsURI)
Definition: xdr.c:758
static xmlNodePtr XDR_E_element(xmlNodePtr xdr, xmlNodePtr parent)
Definition: xdr.c:430
static const xmlChar xs_content[]
Definition: xdr.c:47
static const xmlChar xs_default[]
Definition: xdr.c:49
static const xmlChar xs_group[]
Definition: xdr.c:57
static const xmlChar xs_textOnly[]
Definition: xdr.c:82
static xmlNodePtr XDR_E_ElementType(xmlNodePtr xdr, xmlNodePtr parent)
Definition: xdr.c:496
static const xmlChar xs_use[]
Definition: xdr.c:86
static const xmlChar xs_ref[]
Definition: xdr.c:72
static const xmlChar xs_value[]
Definition: xdr.c:87
static xmlChar * get_dt_type(xmlNodePtr xdr)
Definition: xdr.c:181
static xmlAttrPtr XDR_A_default(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:224
static const xmlChar xs_unbounded[]
Definition: xdr.c:85
#define FOREACH_NS(node, ns)
Definition: xdr.c:113
static xmlNodePtr get_child(xmlNodePtr node, xmlChar const *name)
Definition: xdr.c:121
_ORDER_TYPE
Definition: xdr.c:100
@ ORDER_ONE
Definition: xdr.c:103
@ ORDER_MANY
Definition: xdr.c:102
@ ORDER_SEQ
Definition: xdr.c:101
static const xmlChar xs_AttributeType[]
Definition: xdr.c:43
static const xmlChar xs_sequence[]
Definition: xdr.c:77
enum _ORDER_TYPE ORDER_TYPE
static const xmlChar xs_required[]
Definition: xdr.c:73
static const xmlChar xs_lax[]
Definition: xdr.c:58
static xmlAttrPtr XDR_A_maxOccurs(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:246
static const xmlChar xs_datatype[]
Definition: xdr.c:48
static xmlNodePtr XDR_E_description(xmlNodePtr xdr, xmlNodePtr parent)
Definition: xdr.c:302
_CONTENT_TYPE
Definition: xdr.c:92
@ CONTENT_TEXTONLY
Definition: xdr.c:94
@ CONTENT_EMPTY
Definition: xdr.c:93
@ CONTENT_ELTONLY
Definition: xdr.c:95
@ CONTENT_MIXED
Definition: xdr.c:96
static const xmlChar xs_targetNamespace[]
Definition: xdr.c:81
static const xmlChar XDR_href[]
Definition: xdr.c:34
static const xmlChar xs_anyAttribute[]
Definition: xdr.c:41
#define FOREACH_CHILD(node, child)
Definition: xdr.c:106
static xmlAttrPtr copy_prop_ignore_ns(xmlAttrPtr xdr_attr, xmlNodePtr node)
Definition: xdr.c:217
static const xmlChar xs_eltOnly[]
Definition: xdr.c:54
static const xmlChar DT_href[]
Definition: xdr.c:33
static xmlNodePtr XDR_E_attribute(xmlNodePtr xdr, xmlNodePtr parent)
Definition: xdr.c:394
static const xmlChar xs_element[]
Definition: xdr.c:52
XMLPUBFUN xmlChar * xmlStrndup(const xmlChar *cur, int len)
Definition: xmlstring.c:45
XMLPUBFUN xmlChar * xmlStrcat(xmlChar *cur, const xmlChar *add)
Definition: xmlstring.c:524
#define BAD_CAST
Definition: xmlstring.h:35
XMLPUBFUN int xmlStrEqual(const xmlChar *str1, const xmlChar *str2)
Definition: xmlstring.c:162
unsigned char xmlChar
Definition: xmlstring.h:28
XMLPUBFUN xmlChar * xmlStrdup(const xmlChar *cur)
Definition: xmlstring.c:69