Completed
Push — master ( 5e529b...7a5a40 )
by Asmir
8s
created

SchemaReader::getDOM()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 11
Code Lines 7

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 8
CRAP Score 2

Importance

Changes 0
Metric Value
dl 0
loc 11
ccs 8
cts 8
cp 1
rs 9.4285
c 0
b 0
f 0
cc 2
eloc 7
nc 2
nop 1
crap 2
1
<?php
2
3
declare(strict_types=1);
4
5
namespace GoetasWebservices\XML\XSDReader;
6
7
use Closure;
8
use DOMDocument;
9
use DOMElement;
10
use DOMNode;
11
use GoetasWebservices\XML\XSDReader\Documentation\DocumentationReader;
12
use GoetasWebservices\XML\XSDReader\Documentation\StandardDocumentationReader;
13
use GoetasWebservices\XML\XSDReader\Exception\IOException;
14
use GoetasWebservices\XML\XSDReader\Exception\TypeException;
15
use GoetasWebservices\XML\XSDReader\Schema\Attribute\Attribute;
16
use GoetasWebservices\XML\XSDReader\Schema\Attribute\AttributeContainer;
17
use GoetasWebservices\XML\XSDReader\Schema\Attribute\AttributeDef;
18
use GoetasWebservices\XML\XSDReader\Schema\Attribute\AttributeItem;
19
use GoetasWebservices\XML\XSDReader\Schema\Attribute\Group as AttributeGroup;
20
use GoetasWebservices\XML\XSDReader\Schema\Element\Element;
21
use GoetasWebservices\XML\XSDReader\Schema\Element\ElementContainer;
22
use GoetasWebservices\XML\XSDReader\Schema\Element\ElementDef;
23
use GoetasWebservices\XML\XSDReader\Schema\Element\ElementRef;
24
use GoetasWebservices\XML\XSDReader\Schema\Element\Group;
25
use GoetasWebservices\XML\XSDReader\Schema\Element\GroupRef;
26
use GoetasWebservices\XML\XSDReader\Schema\Element\InterfaceSetMinMax;
27
use GoetasWebservices\XML\XSDReader\Schema\Exception\TypeNotFoundException;
28
use GoetasWebservices\XML\XSDReader\Schema\Inheritance\Base;
29
use GoetasWebservices\XML\XSDReader\Schema\Inheritance\Extension;
30
use GoetasWebservices\XML\XSDReader\Schema\Inheritance\Restriction;
31
use GoetasWebservices\XML\XSDReader\Schema\Item;
32
use GoetasWebservices\XML\XSDReader\Schema\Schema;
33
use GoetasWebservices\XML\XSDReader\Schema\SchemaItem;
34
use GoetasWebservices\XML\XSDReader\Schema\Type\BaseComplexType;
35
use GoetasWebservices\XML\XSDReader\Schema\Type\ComplexType;
36
use GoetasWebservices\XML\XSDReader\Schema\Type\ComplexTypeSimpleContent;
37
use GoetasWebservices\XML\XSDReader\Schema\Type\SimpleType;
38
use GoetasWebservices\XML\XSDReader\Schema\Type\Type;
39
use GoetasWebservices\XML\XSDReader\Utils\UrlUtils;
40
41
class SchemaReader
42
{
43
    public const XSD_NS = 'http://www.w3.org/2001/XMLSchema';
44
45
    public const XML_NS = 'http://www.w3.org/XML/1998/namespace';
46
47
    /**
48
     * @var DocumentationReader
49
     */
50
    private $documentationReader;
51
52
    /**
53
     * @var Schema[]
54
     */
55
    private $loadedFiles = array();
56
57
    /**
58
     * @var Schema[][]
59
     */
60
    private $loadedSchemas = array();
61
62
    /**
63
     * @var string[]
64
     */
65
    protected $knownLocationSchemas = [
66
        'http://www.w3.org/2001/xml.xsd' => (
67
            __DIR__.'/Resources/xml.xsd'
68
        ),
69
        'http://www.w3.org/2001/XMLSchema.xsd' => (
70
            __DIR__.'/Resources/XMLSchema.xsd'
71
        ),
72
        'http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd' => (
73
            __DIR__.'/Resources/oasis-200401-wss-wssecurity-secext-1.0.xsd'
74
        ),
75
        'http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd' => (
76
            __DIR__.'/Resources/oasis-200401-wss-wssecurity-utility-1.0.xsd'
77
        ),
78
        'https://www.w3.org/TR/xmldsig-core/xmldsig-core-schema.xsd' => (
79
            __DIR__.'/Resources/xmldsig-core-schema.xsd'
80
        ),
81
        'http://www.w3.org/TR/xmldsig-core/xmldsig-core-schema.xsd' => (
82
            __DIR__.'/Resources/xmldsig-core-schema.xsd'
83
        ),
84
    ];
85
86
    /**
87
     * @var string[]
88
     */
89
    protected static $globalSchemaInfo = array(
90
        self::XML_NS => 'http://www.w3.org/2001/xml.xsd',
91
        self::XSD_NS => 'http://www.w3.org/2001/XMLSchema.xsd',
92
    );
93
94 2
    private function extractErrorMessage(): \Exception
95
    {
96 2
        $errors = array();
97
98 2
        foreach (libxml_get_errors() as $error) {
99 1
            $errors[] = sprintf("Error[%s] code %s: %s in '%s' at position %s:%s", $error->level, $error->code, trim($error->message), $error->file, $error->line, $error->column);
100
        }
101 2
        $e = new \Exception(implode('; ', $errors));
102 2
        libxml_use_internal_errors(false);
103
104 2
        return $e;
105
    }
106
107 71
    public function __construct(DocumentationReader $documentationReader = null)
108
    {
109 71
        if (null === $documentationReader) {
110 71
            $documentationReader = new StandardDocumentationReader();
111
        }
112 71
        $this->documentationReader = $documentationReader;
113 71
    }
114
115 1
    public function addKnownSchemaLocation(string $remote, string $local): void
116
    {
117 1
        $this->knownLocationSchemas[$remote] = $local;
118 1
    }
119
120 60
    private function loadAttributeGroup(
121
        Schema $schema,
122
        DOMElement $node
123
    ): Closure {
124 60
        $attGroup = new AttributeGroup($schema, $node->getAttribute('name'));
125 60
        $attGroup->setDoc($this->getDocumentation($node));
126 60
        $schema->addAttributeGroup($attGroup);
127
128 60
        return function () use ($schema, $node, $attGroup): void {
129 60
            SchemaReader::againstDOMNodeList(
130 60
                $node,
131 60
                function (
132
                    DOMElement $node,
133
                    DOMElement $childNode
134
                ) use (
135 60
                    $schema,
136 60
                    $attGroup
137
                ): void {
138 60
                    switch ($childNode->localName) {
139 60
                        case 'attribute':
140 60
                            $attribute = $this->getAttributeFromAttributeOrRef(
141 60
                                $childNode,
142 60
                                $schema,
143 60
                                $node
144
                            );
145 60
                            $attGroup->addAttribute($attribute);
146 60
                            break;
147 60
                        case 'attributeGroup':
148 1
                            $this->findSomethingLikeAttributeGroup(
149 1
                                $schema,
150 1
                                $node,
151 1
                                $childNode,
152 1
                                $attGroup
153
                            );
154 1
                            break;
155
                    }
156 60
                }
157
            );
158 60
        };
159
    }
160
161 60
    private function getAttributeFromAttributeOrRef(
162
        DOMElement $childNode,
163
        Schema $schema,
164
        DOMElement $node
165
    ): AttributeItem {
166 60
        if ($childNode->hasAttribute('ref')) {
167 60
            $attribute = $this->findAttributeItem($schema, $node, $childNode->getAttribute('ref'));
168
        } else {
169
            /**
170
             * @var Attribute
171
             */
172 60
            $attribute = $this->loadAttribute($schema, $childNode);
173
        }
174
175 60
        return $attribute;
176
    }
177
178 60
    private function loadAttribute(
179
        Schema $schema,
180
        DOMElement $node
181
    ): Attribute {
182 60
        $attribute = new Attribute($schema, $node->getAttribute('name'));
183 60
        $attribute->setDoc($this->getDocumentation($node));
184 60
        $this->fillItem($attribute, $node);
185
186 60
        if ($node->hasAttribute('nillable')) {
187 1
            $attribute->setNil($node->getAttribute('nillable') == 'true');
188
        }
189 60
        if ($node->hasAttribute('form')) {
190 1
            $attribute->setQualified($node->getAttribute('form') == 'qualified');
191
        }
192 60
        if ($node->hasAttribute('use')) {
193 60
            $attribute->setUse($node->getAttribute('use'));
194
        }
195
196 60
        return $attribute;
197
    }
198
199 60
    private function loadAttributeOrElementDef(
200
        Schema $schema,
201
        DOMElement $node,
202
        bool $attributeDef
203
    ): Closure {
204 60
        $name = $node->getAttribute('name');
205 60
        if ($attributeDef) {
206 60
            $attribute = new AttributeDef($schema, $name);
207 60
            $schema->addAttribute($attribute);
208
        } else {
209 60
            $attribute = new ElementDef($schema, $name);
210 60
            $schema->addElement($attribute);
211
        }
212
213 60
        return function () use ($attribute, $node): void {
214 60
            $this->fillItem($attribute, $node);
215 60
        };
216
    }
217
218 60
    private function loadAttributeDef(Schema $schema, DOMElement $node): Closure
219
    {
220 60
        return $this->loadAttributeOrElementDef($schema, $node, true);
221
    }
222
223 60
    private function getDocumentation(DOMElement $node): string
224
    {
225 60
        return $this->documentationReader->get($node);
226
    }
227
228
    /**
229
     * @return Closure[]
230
     */
231 60
    private function schemaNode(Schema $schema, DOMElement $node, Schema $parent = null): array
232
    {
233 60
        $this->setSchemaThingsFromNode($schema, $node, $parent);
234 60
        $functions = array();
235
236 60
        self::againstDOMNodeList(
237 60
            $node,
238 60
            function (
239
                DOMElement $node,
240
                DOMElement $childNode
241
            ) use (
242 60
                $schema,
243 60
                &$functions
244
            ): void {
245 60
                $callback = null;
246
247 60
                switch ($childNode->localName) {
248 60
                    case 'attributeGroup':
249 60
                        $callback = $this->loadAttributeGroup($schema, $childNode);
250 60
                        break;
251 60
                    case 'include':
252 60
                    case 'import':
253 60
                        $callback = $this->loadImport($schema, $childNode);
254 60
                        break;
255 60
                    case 'element':
256 60
                        $callback = $this->loadElementDef($schema, $childNode);
257 60
                        break;
258 60
                    case 'attribute':
259 60
                        $callback = $this->loadAttributeDef($schema, $childNode);
260 60
                        break;
261 60
                    case 'group':
262 60
                        $callback = $this->loadGroup($schema, $childNode);
263 60
                        break;
264 60
                    case 'complexType':
265 60
                        $callback = $this->loadComplexType($schema, $childNode);
266 60
                        break;
267 60
                    case 'simpleType':
268 60
                        $callback = $this->loadSimpleType($schema, $childNode);
269 60
                        break;
270
                }
271
272 60
                if ($callback instanceof Closure) {
273 60
                    $functions[] = $callback;
274
                }
275 60
            }
276
        );
277
278 60
        return $functions;
279
    }
280
281 60
    private function loadGroupRef(Group $referenced, DOMElement $node): GroupRef
282
    {
283 60
        $ref = new GroupRef($referenced);
284 60
        $ref->setDoc($this->getDocumentation($node));
285
286 60
        self::maybeSetMax($ref, $node);
287 60
        self::maybeSetMin($ref, $node);
288
289 60
        return $ref;
290
    }
291
292 60
    private static function maybeSetMax(InterfaceSetMinMax $ref, DOMElement $node): void
293
    {
294 60
        if ($node->hasAttribute('maxOccurs')) {
295 60
            $ref->setMax($node->getAttribute('maxOccurs') == 'unbounded' ? -1 : (int) $node->getAttribute('maxOccurs'));
296
        }
297 60
    }
298
299 60
    private static function maybeSetMin(InterfaceSetMinMax $ref, DOMElement $node): void
300
    {
301 60
        if ($node->hasAttribute('minOccurs')) {
302 60
            $ref->setMin((int) $node->getAttribute('minOccurs'));
303 60
            if ($ref->getMin() > $ref->getMax() && $ref->getMax() !== -1) {
304 7
                $ref->setMax($ref->getMin());
305
            }
306
        }
307 60
    }
308
309 60
    private function loadSequence(ElementContainer $elementContainer, DOMElement $node, int $max = null): void
310
    {
311
        $max =
312
            (
313 60
                (is_int($max) && (bool) $max) ||
314 60
                $node->getAttribute('maxOccurs') == 'unbounded' ||
315 60
                $node->getAttribute('maxOccurs') > 1
316
            )
317 60
                ? 2
318 60
                : null;
319
320 60
        self::againstDOMNodeList(
321 60
            $node,
322 60
            function (
323
                DOMElement $node,
324
                DOMElement $childNode
325
            ) use (
326 60
                $elementContainer,
327 60
                $max
328
            ): void {
329 60
                $this->loadSequenceChildNode(
330 60
                    $elementContainer,
331 60
                    $node,
332 60
                    $childNode,
333 60
                    $max
334
                );
335 60
            }
336
        );
337 60
    }
338
339 60
    private function loadSequenceChildNode(
340
        ElementContainer $elementContainer,
341
        DOMElement $node,
342
        DOMElement $childNode,
343
        ? int $max
344
    ): void {
345 60
        switch ($childNode->localName) {
346 60
            case 'sequence':
347 60
            case 'choice':
348 60
            case 'all':
349 60
                $this->loadSequence(
350 60
                    $elementContainer,
351 60
                    $childNode,
352 60
                    $max
353
                );
354 60
                break;
355 60
            case 'element':
356 60
                $this->loadSequenceChildNodeLoadElement(
357 60
                    $elementContainer,
358 60
                    $node,
359 60
                    $childNode,
360 60
                    $max
361
                );
362 60
                break;
363 60
            case 'group':
364 60
                $this->addGroupAsElement(
365 60
                    $elementContainer->getSchema(),
366 60
                    $node,
367 60
                    $childNode,
368 60
                    $elementContainer
369
                );
370 60
                break;
371
        }
372 60
    }
373
374 60
    private function loadSequenceChildNodeLoadElement(
375
        ElementContainer $elementContainer,
376
        DOMElement $node,
377
        DOMElement $childNode,
378
        ? int $max
379
    ): void {
380 60
        if ($childNode->hasAttribute('ref')) {
381 60
            $element = new ElementRef(
382 60
                $this->findElement($elementContainer->getSchema(), $node, $childNode->getAttribute('ref'))
383
            );
384 60
            $element->setDoc($this->getDocumentation($childNode));
385
386 60
            self::maybeSetMax($element, $childNode);
387 60
            self::maybeSetMin($element, $childNode);
388 60
            if ($childNode->hasAttribute('nillable')) {
389
                $element->setNil($childNode->getAttribute('nillable') == 'true');
390
            }
391 60
            if ($childNode->hasAttribute('form')) {
392 60
                $element->setQualified($childNode->getAttribute('form') == 'qualified');
393
            }
394
        } else {
395 60
            $element = $this->loadElement(
396 60
                $elementContainer->getSchema(),
397 60
                $childNode
398
            );
399
        }
400 60
        if ($max > 1) {
401
            /*
402
            * although one might think the typecast is not needed with $max being `? int $max` after passing > 1,
403
            * phpstan@a4f89fa still thinks it's possibly null.
404
            * see https://github.com/phpstan/phpstan/issues/577 for related issue
405
            */
406 60
            $element->setMax((int) $max);
407
        }
408 60
        $elementContainer->addElement($element);
409 60
    }
410
411 60
    private function addGroupAsElement(
412
        Schema $schema,
413
        DOMElement $node,
414
        DOMElement $childNode,
415
        ElementContainer $elementContainer
416
    ): void {
417 60
        $referencedGroup = $this->findGroup(
418 60
            $schema,
419 60
            $node,
420 60
            $childNode->getAttribute('ref')
421
        );
422
423 60
        $group = $this->loadGroupRef($referencedGroup, $childNode);
424 60
        $elementContainer->addElement($group);
425 60
    }
426
427 60
    private function loadGroup(Schema $schema, DOMElement $node): Closure
428
    {
429 60
        $group = new Group($schema, $node->getAttribute('name'));
430 60
        $group->setDoc($this->getDocumentation($node));
431 60
        $groupOriginal = $group;
432
433 60
        if ($node->hasAttribute('maxOccurs') || $node->hasAttribute('maxOccurs')) {
434 1
            $group = new GroupRef($group);
435
436 1
            if ($node->hasAttribute('maxOccurs')) {
437 1
                self::maybeSetMax($group, $node);
438
            }
439 1
            if ($node->hasAttribute('minOccurs')) {
440 1
                self::maybeSetMin($group, $node);
441
            }
442
        }
443
444 60
        $schema->addGroup($group);
445
446 60
        return function () use ($groupOriginal, $node): void {
447 60
            static::againstDOMNodeList(
448 60
                $node,
449 60
                function (DOMelement $node, DOMElement $childNode) use ($groupOriginal): void {
450 60
                    switch ($childNode->localName) {
451 60
                        case 'sequence':
452 60
                        case 'choice':
453 60
                        case 'all':
454 60
                            $this->loadSequence($groupOriginal, $childNode);
455 60
                            break;
456
                    }
457 60
                }
458
            );
459 60
        };
460
    }
461
462 60
    private function loadComplexType(Schema $schema, DOMElement $node, Closure $callback = null): Closure
463
    {
464
        /**
465
         * @var bool
466
         */
467 60
        $isSimple = false;
468
469 60
        self::againstDOMNodeList(
470 60
            $node,
471 60
            function (
472
                DOMElement $node,
473
                DOMElement $childNode
474
            ) use (
475 60
                &$isSimple
476
            ): void {
477 60
                if ($isSimple) {
478 1
                    return;
479
                }
480 60
                if ($childNode->localName === 'simpleContent') {
481 2
                    $isSimple = true;
482
                }
483 60
            }
484
        );
485
486 60
        $type = $isSimple ? new ComplexTypeSimpleContent($schema, $node->getAttribute('name')) : new ComplexType($schema, $node->getAttribute('name'));
487
488 60
        $type->setDoc($this->getDocumentation($node));
489 60
        if ($node->getAttribute('name')) {
490 60
            $schema->addType($type);
491
        }
492
493 60
        return function () use ($type, $node, $schema, $callback): void {
494 60
            $this->fillTypeNode($type, $node, true);
495
496 60
            self::againstDOMNodeList(
497 60
                $node,
498 60
                function (
499
                    DOMElement $node,
500
                    DOMElement $childNode
501
                ) use (
502 60
                    $schema,
503 60
                    $type
504
                ): void {
505 60
                    $this->loadComplexTypeFromChildNode(
506 60
                        $type,
507 60
                        $node,
508 60
                        $childNode,
509 60
                        $schema
510
                    );
511 60
                }
512
            );
513
514 60
            if ($callback instanceof Closure) {
515 60
                call_user_func($callback, $type);
516
            }
517 60
        };
518
    }
519
520 60
    private function loadComplexTypeFromChildNode(
521
        BaseComplexType $type,
522
        DOMElement $node,
523
        DOMElement $childNode,
524
        Schema $schema
525
    ): void {
526 60
        switch ($childNode->localName) {
527 60
            case 'sequence':
528 60
            case 'choice':
529 60
            case 'all':
530 60
                if ($type instanceof ElementContainer) {
531 60
                    $this->loadSequence(
532 60
                        $type,
533 60
                        $childNode
534
                    );
535
                }
536 60
                break;
537 60
            case 'attribute':
538 60
                $this->addAttributeFromAttributeOrRef(
539 60
                    $type,
540 60
                    $childNode,
541 60
                    $schema,
542 60
                    $node
543
                );
544 60
                break;
545 60
            case 'attributeGroup':
546 2
                $this->findSomethingLikeAttributeGroup(
547 2
                    $schema,
548 2
                    $node,
549 2
                    $childNode,
550 2
                    $type
551
                );
552 2
                break;
553 60
            case 'group':
554
                if (
555 1
                    $type instanceof ComplexType
556
                ) {
557 1
                    $this->addGroupAsElement(
558 1
                        $schema,
559 1
                        $node,
560 1
                        $childNode,
561 1
                        $type
562
                    );
563
                }
564 1
                break;
565
        }
566 60
    }
567
568 60
    private function loadSimpleType(Schema $schema, DOMElement $node, Closure $callback = null): Closure
569
    {
570 60
        $type = new SimpleType($schema, $node->getAttribute('name'));
571 60
        $type->setDoc($this->getDocumentation($node));
572 60
        if ($node->getAttribute('name')) {
573 60
            $schema->addType($type);
574
        }
575
576 60
        return function () use ($type, $node, $callback): void {
577 60
            $this->fillTypeNode($type, $node, true);
578
579 60
            self::againstDOMNodeList(
580 60
                $node,
581 60
                function (DOMElement $node, DOMElement $childNode) use ($type): void {
582 60
                    switch ($childNode->localName) {
583 60
                        case 'union':
584 60
                            $this->loadUnion($type, $childNode);
585 60
                            break;
586 60
                        case 'list':
587 60
                            $this->loadList($type, $childNode);
588 60
                            break;
589
                    }
590 60
                }
591
            );
592
593 60
            if ($callback instanceof Closure) {
594 60
                call_user_func($callback, $type);
595
            }
596 60
        };
597
    }
598
599 60
    private function loadList(SimpleType $type, DOMElement $node): void
600
    {
601 60
        if ($node->hasAttribute('itemType')) {
602
            /**
603
             * @var SimpleType
604
             */
605 60
            $listType = $this->findSomeType($type, $node, 'itemType');
606 60
            $type->setList($listType);
607
        } else {
608 60
            self::againstDOMNodeList(
609 60
                $node,
610 60
                function (
611
                    DOMElement $node,
612
                    DOMElement $childNode
613
                ) use (
614 60
                    $type
615
                ): void {
616 60
                    $this->loadTypeWithCallback(
617 60
                        $type->getSchema(),
618 60
                        $childNode,
619 60
                        function (SimpleType $list) use ($type): void {
620 60
                            $type->setList($list);
621 60
                        }
622
                    );
623 60
                }
624
            );
625
        }
626 60
    }
627
628 60
    private function findSomeType(
629
        SchemaItem $fromThis,
630
        DOMElement $node,
631
        string $attributeName
632
    ): SchemaItem {
633 60
        return $this->findSomeTypeFromAttribute(
634 60
            $fromThis,
635 60
            $node,
636 60
            $node->getAttribute($attributeName)
637
        );
638
    }
639
640 60
    private function findSomeTypeFromAttribute(
641
        SchemaItem $fromThis,
642
        DOMElement $node,
643
        string $attributeName
644
    ): SchemaItem {
645 60
        $out = $this->findType(
646 60
            $fromThis->getSchema(),
647 60
            $node,
648 60
            $attributeName
649
        );
650
651 60
        return $out;
652
    }
653
654 60
    private function loadUnion(SimpleType $type, DOMElement $node): void
655
    {
656 60
        if ($node->hasAttribute('memberTypes')) {
657 60
            $types = preg_split('/\s+/', $node->getAttribute('memberTypes'));
658 60
            foreach ($types as $typeName) {
659
                /**
660
                 * @var SimpleType
661
                 */
662 60
                $unionType = $this->findSomeTypeFromAttribute(
663 60
                    $type,
664 60
                    $node,
665 60
                    $typeName
666
                );
667 60
                $type->addUnion($unionType);
668
            }
669
        }
670 60
        self::againstDOMNodeList(
671 60
            $node,
672 60
            function (
673
                DOMElement $node,
674
                DOMElement $childNode
675
            ) use (
676 60
                $type
677
            ): void {
678 60
                $this->loadTypeWithCallback(
679 60
                    $type->getSchema(),
680 60
                    $childNode,
681 60
                    function (SimpleType $unType) use ($type): void {
682 60
                        $type->addUnion($unType);
683 60
                    }
684
                );
685 60
            }
686
        );
687 60
    }
688
689 60
    private function fillTypeNode(Type $type, DOMElement $node, bool $checkAbstract = false): void
690
    {
691 60
        if ($checkAbstract) {
692 60
            $type->setAbstract($node->getAttribute('abstract') === 'true' || $node->getAttribute('abstract') === '1');
693
        }
694
695 60
        self::againstDOMNodeList(
696 60
            $node,
697 60
            function (DOMElement $node, DOMElement $childNode) use ($type): void {
698 60
                switch ($childNode->localName) {
699 60
                    case 'restriction':
700 60
                        $this->loadRestriction($type, $childNode);
701 60
                        break;
702 60
                    case 'extension':
703 60
                        if ($type instanceof BaseComplexType) {
704 60
                            $this->loadExtension($type, $childNode);
705
                        }
706 60
                        break;
707 60
                    case 'simpleContent':
708 60
                    case 'complexContent':
709 60
                        $this->fillTypeNode($type, $childNode);
710 60
                        break;
711
                }
712 60
            }
713
        );
714 60
    }
715
716 60
    private function loadExtension(BaseComplexType $type, DOMElement $node): void
717
    {
718 60
        $extension = new Extension();
719 60
        $type->setExtension($extension);
720
721 60
        if ($node->hasAttribute('base')) {
722 60
            $this->findAndSetSomeBase(
723 60
                $type,
724 60
                $extension,
725 60
                $node
726
            );
727
        }
728 60
        $this->loadExtensionChildNodes($type, $node);
729 60
    }
730
731 60
    private function findAndSetSomeBase(
732
        Type $type,
733
        Base $setBaseOnThis,
734
        DOMElement $node
735
    ): void {
736
        /**
737
         * @var Type
738
         */
739 60
        $parent = $this->findSomeType($type, $node, 'base');
740 60
        $setBaseOnThis->setBase($parent);
741 60
    }
742
743 60
    private function loadExtensionChildNodes(
744
        BaseComplexType $type,
745
        DOMElement $node
746
    ): void {
747 60
        self::againstDOMNodeList(
748 60
            $node,
749 60
            function (
750
                DOMElement $node,
751
                DOMElement $childNode
752
            ) use (
753 60
                $type
754
            ): void {
755 60
                switch ($childNode->localName) {
756 60
                    case 'sequence':
757 60
                    case 'choice':
758 60
                    case 'all':
759 60
                        if ($type instanceof ElementContainer) {
760 60
                            $this->loadSequence(
761 60
                                $type,
762 60
                                $childNode
763
                            );
764
                        }
765 60
                        break;
766 60
                    case 'attribute':
767 60
                        $this->addAttributeFromAttributeOrRef(
768 60
                            $type,
769 60
                            $childNode,
770 60
                            $type->getSchema(),
771 60
                            $node
772
                        );
773 60
                        break;
774 60
                    case 'attributeGroup':
775 60
                        $this->findSomethingLikeAttributeGroup(
776 60
                            $type->getSchema(),
777 60
                            $node,
778 60
                            $childNode,
779 60
                            $type
780
                        );
781 60
                        break;
782
                }
783 60
            }
784
        );
785 60
    }
786
787 60
    private function loadRestriction(Type $type, DOMElement $node): void
788
    {
789 60
        $restriction = new Restriction();
790 60
        $type->setRestriction($restriction);
791 60
        if ($node->hasAttribute('base')) {
792 60
            $this->findAndSetSomeBase($type, $restriction, $node);
793
        } else {
794 60
            self::againstDOMNodeList(
795 60
                $node,
796 60
                function (
797
                    DOMElement $node,
798
                    DOMElement $childNode
799
                ) use (
800 60
                    $type,
801 60
                    $restriction
802
                ): void {
803 60
                    $this->loadTypeWithCallback(
804 60
                        $type->getSchema(),
805 60
                        $childNode,
806 60
                        function (Type $restType) use ($restriction): void {
807 60
                            $restriction->setBase($restType);
808 60
                        }
809
                    );
810 60
                }
811
            );
812
        }
813 60
        self::againstDOMNodeList(
814 60
            $node,
815 60
            function (
816
                DOMElement $node,
817
                DOMElement $childNode
818
            ) use (
819 60
                $restriction
820
            ): void {
821
                if (
822 60
                in_array(
823 60
                    $childNode->localName,
824
                    [
825 60
                        'enumeration',
826
                        'pattern',
827
                        'length',
828
                        'minLength',
829
                        'maxLength',
830
                        'minInclusive',
831
                        'maxInclusive',
832
                        'minExclusive',
833
                        'maxExclusive',
834
                        'fractionDigits',
835
                        'totalDigits',
836
                        'whiteSpace',
837
                    ],
838 60
                    true
839
                )
840
                ) {
841 60
                    $restriction->addCheck(
842 60
                        $childNode->localName,
843
                        [
844 60
                            'value' => $childNode->getAttribute('value'),
845 60
                            'doc' => $this->getDocumentation($childNode),
846
                        ]
847
                    );
848
                }
849 60
            }
850
        );
851 60
    }
852
853
    /**
854
     * @return mixed[]
855
     */
856 60
    private static function splitParts(DOMElement $node, string $typeName): array
857
    {
858 60
        $prefix = null;
859 60
        $name = $typeName;
860 60
        if (strpos($typeName, ':') !== false) {
861 60
            list($prefix, $name) = explode(':', $typeName);
862
        }
863
864 60
        $namespace = $node->lookupNamespaceUri($prefix ?: '');
865
866
        return array(
867 60
            $name,
868 60
            $namespace,
869 60
            $prefix,
870
        );
871
    }
872
873 60
    private function findAttributeItem(Schema $schema, DOMElement $node, string $typeName): AttributeItem
874
    {
875 60
        list($name, $namespace) = static::splitParts($node, $typeName);
876
877
        /**
878
         * @var string|null $namespace
879
         */
880 60
        $namespace = $namespace ?: $schema->getTargetNamespace();
881
882
        try {
883
            /**
884
             * @var AttributeItem $out
885
             */
886 60
            $out = $schema->findAttribute((string) $name, $namespace);
887
888 60
            return $out;
889
        } catch (TypeNotFoundException $e) {
890
            throw new TypeException(
891
                sprintf(
892
                    "Can't find %s named {%s}#%s, at line %d in %s ",
893
                    'attribute',
894
                    $namespace,
895
                    $name,
896
                    $node->getLineNo(),
897
                    $node->ownerDocument->documentURI
898
                ),
899
                0,
900
                $e
901
            );
902
        }
903
    }
904
905 60
    private function findAttributeGroup(Schema $schema, DOMElement $node, string $typeName): AttributeGroup
906
    {
907 60
        list($name, $namespace) = static::splitParts($node, $typeName);
908
909
        /**
910
         * @var string|null $namespace
911
         */
912 60
        $namespace = $namespace ?: $schema->getTargetNamespace();
913
914
        try {
915
            /**
916
             * @var AttributeGroup $out
917
             */
918 60
            $out = $schema->findAttributeGroup((string) $name, $namespace);
919
920 60
            return $out;
921
        } catch (TypeNotFoundException $e) {
922
            throw new TypeException(
923
                sprintf(
924
                    "Can't find %s named {%s}#%s, at line %d in %s ",
925
                    'attributegroup',
926
                    $namespace,
927
                    $name,
928
                    $node->getLineNo(),
929
                    $node->ownerDocument->documentURI
930
                ),
931
                0,
932
                $e
933
            );
934
        }
935
    }
936
937 60
    private function findElement(Schema $schema, DOMElement $node, string $typeName): ElementDef
938
    {
939 60
        list($name, $namespace) = static::splitParts($node, $typeName);
940
941
        /**
942
         * @var string|null $namespace
943
         */
944 60
        $namespace = $namespace ?: $schema->getTargetNamespace();
945
946
        try {
947 60
            return $schema->findElement((string) $name, $namespace);
948
        } catch (TypeNotFoundException $e) {
949
            throw new TypeException(
950
                sprintf(
951
                    "Can't find %s named {%s}#%s, at line %d in %s ",
952
                    'element',
953
                    $namespace,
954
                    $name,
955
                    $node->getLineNo(),
956
                    $node->ownerDocument->documentURI
957
                ),
958
                0,
959
                $e
960
            );
961
        }
962
    }
963
964 60
    private function findGroup(Schema $schema, DOMElement $node, string $typeName): Group
965
    {
966 60
        list($name, $namespace) = static::splitParts($node, $typeName);
967
968
        /**
969
         * @var string|null $namespace
970
         */
971 60
        $namespace = $namespace ?: $schema->getTargetNamespace();
972
973
        try {
974
            /**
975
             * @var Group $out
976
             */
977 60
            $out = $schema->findGroup((string) $name, $namespace);
978
979 60
            return $out;
980
        } catch (TypeNotFoundException $e) {
981
            throw new TypeException(
982
                sprintf(
983
                    "Can't find %s named {%s}#%s, at line %d in %s ",
984
                    'group',
985
                    $namespace,
986
                    $name,
987
                    $node->getLineNo(),
988
                    $node->ownerDocument->documentURI
989
                ),
990
                0,
991
                $e
992
            );
993
        }
994
    }
995
996 60
    private function findType(Schema $schema, DOMElement $node, string $typeName): SchemaItem
997
    {
998 60
        list($name, $namespace) = static::splitParts($node, $typeName);
999
1000
        /**
1001
         * @var string|null $namespace
1002
         */
1003 60
        $namespace = $namespace ?: $schema->getTargetNamespace();
1004
1005
        try {
1006
            /**
1007
             * @var SchemaItem $out
1008
             */
1009 60
            $out = $schema->findType((string) $name, $namespace);
1010
1011 60
            return $out;
1012
        } catch (TypeNotFoundException $e) {
1013
            throw new TypeException(
1014
                sprintf(
1015
                    "Can't find %s named {%s}#%s, at line %d in %s ",
1016
                    'type',
1017
                    $namespace,
1018
                    $name,
1019
                    $node->getLineNo(),
1020
                    $node->ownerDocument->documentURI
1021
                ),
1022
                0,
1023
                $e
1024
            );
1025
        }
1026
    }
1027
1028
    /**
1029
     * @return Closure
1030
     */
1031 60
    private function loadElementDef(Schema $schema, DOMElement $node): Closure
1032
    {
1033 60
        return $this->loadAttributeOrElementDef($schema, $node, false);
1034
    }
1035
1036 60
    private function fillItem(Item $element, DOMElement $node): void
1037
    {
1038
        /**
1039
         * @var bool
1040
         */
1041 60
        $skip = false;
1042 60
        self::againstDOMNodeList(
1043 60
            $node,
1044 60
            function (
1045
                DOMElement $node,
1046
                DOMElement $childNode
1047
            ) use (
1048 60
                $element,
1049 60
                &$skip
1050
            ): void {
1051
                if (
1052 60
                    !$skip &&
1053 60
                    in_array(
1054 60
                        $childNode->localName,
1055
                        [
1056 60
                            'complexType',
1057
                            'simpleType',
1058
                        ],
1059 60
                        true
1060
                    )
1061
                ) {
1062 60
                    $this->loadTypeWithCallback(
1063 60
                        $element->getSchema(),
1064 60
                        $childNode,
1065 60
                        function (Type $type) use ($element): void {
1066 60
                            $element->setType($type);
1067 60
                        }
1068
                    );
1069 60
                    $skip = true;
1070
                }
1071 60
            }
1072
        );
1073 60
        if ($skip) {
1074 60
            return;
1075
        }
1076 60
        $this->fillItemNonLocalType($element, $node);
1077 60
    }
1078
1079 60
    private function fillItemNonLocalType(Item $element, DOMElement $node): void
1080
    {
1081 60
        if ($node->getAttribute('type')) {
1082
            /**
1083
             * @var Type
1084
             */
1085 60
            $type = $this->findSomeType($element, $node, 'type');
1086
        } else {
1087
            /**
1088
             * @var Type
1089
             */
1090 60
            $type = $this->findSomeTypeFromAttribute(
1091 60
                $element,
1092 60
                $node,
1093 60
                ($node->lookupPrefix(self::XSD_NS).':anyType')
1094
            );
1095
        }
1096
1097 60
        $element->setType($type);
1098 60
    }
1099
1100 60
    private function loadImport(
1101
        Schema $schema,
1102
        DOMElement $node
1103
    ): Closure {
1104 60
        $namespace = $node->getAttribute('namespace');
1105 60
        $schemaLocation = $node->getAttribute('schemaLocation');
1106
1107
        // postpone schema loading
1108 60
        if ($namespace && !$schemaLocation && !isset(self::$globalSchemaInfo[$namespace])) {
1109 2
            return function () use ($schema, $namespace): void {
1110 2
                if (!empty($this->loadedSchemas[$namespace])) {
1111 2
                    foreach ($this->loadedSchemas[$namespace] as $s) {
1112 2
                        $schema->addSchema($s, $namespace);
1113
                    }
1114
                }
1115 2
            };
1116 60
        } elseif ($namespace && !$schemaLocation && !empty($this->loadedSchemas[$namespace])) {
1117 1
            foreach ($this->loadedSchemas[$namespace] as $s) {
1118 1
                $schema->addSchema($s, $namespace);
1119
            }
1120
        }
1121
1122 60
        $base = urldecode($node->ownerDocument->documentURI);
1123 60
        $file = UrlUtils::resolveRelativeUrl($base, $node->getAttribute('schemaLocation'));
1124
1125 60
        if (isset($this->loadedFiles[$file])) {
1126 60
            $schema->addSchema($this->loadedFiles[$file]);
1127
1128 60
            return function (): void {
1129 60
            };
1130
        }
1131
1132 2
        return $this->loadImportFresh($namespace, $schema, $file);
1133
    }
1134
1135 2
    private function createOrUseSchemaForNs(
1136
        Schema $schema,
1137
        string $namespace
1138
    ): Schema {
1139 2
        if (('' !== trim($namespace))) {
1140 1
            $newSchema = new Schema();
1141 1
            $newSchema->addSchema($this->getGlobalSchema());
1142 1
            $schema->addSchema($newSchema);
1143
        } else {
1144 1
            $newSchema = $schema;
1145
        }
1146
1147 2
        return $newSchema;
1148
    }
1149
1150
    private function loadImportFresh(
1151
        string $namespace,
1152
        Schema $schema,
1153
        string $file
1154
    ): Closure {
1155 2
        return function () use ($namespace, $schema, $file): void {
1156 2
            $dom = $this->getDOM(
1157 2
                isset($this->knownLocationSchemas[$file])
1158 1
                    ? $this->knownLocationSchemas[$file]
1159 2
                    : $file
1160
            );
1161
1162 2
            $schemaNew = $this->createOrUseSchemaForNs($schema, $namespace);
1163
1164 2
            $this->setLoadedFile($file, $schemaNew);
1165
1166 2
            $callbacks = $this->schemaNode($schemaNew, $dom->documentElement, $schema);
1167
1168 2
            foreach ($callbacks as $callback) {
1169 2
                $callback();
1170
            }
1171 2
        };
1172
    }
1173
1174
    /**
1175
     * @var Schema|null
1176
     */
1177
    protected $globalSchema;
1178
1179 60
    public function getGlobalSchema(): Schema
1180
    {
1181 60
        if (!($this->globalSchema instanceof Schema)) {
1182 60
            $callbacks = array();
1183 60
            $globalSchemas = array();
1184
            /**
1185
             * @var string $namespace
1186
             */
1187 60
            foreach (self::$globalSchemaInfo as $namespace => $uri) {
1188 60
                $this->setLoadedFile(
1189 60
                    $uri,
1190 60
                    $globalSchemas[$namespace] = $schema = new Schema()
1191
                );
1192 60
                $this->setLoadedSchema($namespace, $schema);
1193 60
                if ($namespace === self::XSD_NS) {
1194 60
                    $this->globalSchema = $schema;
1195
                }
1196 60
                $xml = $this->getDOM($this->knownLocationSchemas[$uri]);
1197 60
                $callbacks = array_merge($callbacks, $this->schemaNode($schema, $xml->documentElement));
1198
            }
1199
1200 60
            $globalSchemas[(string) static::XSD_NS]->addType(new SimpleType($globalSchemas[(string) static::XSD_NS], 'anySimpleType'));
1201 60
            $globalSchemas[(string) static::XSD_NS]->addType(new SimpleType($globalSchemas[(string) static::XSD_NS], 'anyType'));
1202
1203 60
            $globalSchemas[(string) static::XML_NS]->addSchema(
1204 60
                $globalSchemas[(string) static::XSD_NS],
1205 60
                (string) static::XSD_NS
1206
            );
1207 60
            $globalSchemas[(string) static::XSD_NS]->addSchema(
1208 60
                $globalSchemas[(string) static::XML_NS],
1209 60
                (string) static::XML_NS
1210
            );
1211
1212
            /**
1213
             * @var Closure
1214
             */
1215 60
            foreach ($callbacks as $callback) {
1216 60
                $callback();
1217
            }
1218
        }
1219
1220 60
        if (!($this->globalSchema instanceof Schema)) {
1221
            throw new TypeException('Globa schema not discoverd');
1222
        }
1223
1224 60
        return $this->globalSchema;
1225
    }
1226
1227
    /**
1228
     * @param DOMNode[] $nodes
1229
     */
1230 1
    public function readNodes(array $nodes, string $file = null): Schema
1231
    {
1232 1
        $rootSchema = new Schema();
1233 1
        $rootSchema->addSchema($this->getGlobalSchema());
1234
1235 1
        if ($file !== null) {
1236 1
            $this->setLoadedFile($file, $rootSchema);
1237
        }
1238
1239 1
        $all = array();
1240 1
        foreach ($nodes as $k => $node) {
1241 1
            if (($node instanceof \DOMElement) && $node->namespaceURI === self::XSD_NS && $node->localName == 'schema') {
1242 1
                $holderSchema = new Schema();
1243 1
                $holderSchema->addSchema($this->getGlobalSchema());
1244
1245 1
                $this->setLoadedSchemaFromElement($node, $holderSchema);
1246
1247 1
                $rootSchema->addSchema($holderSchema);
1248
1249 1
                $callbacks = $this->schemaNode($holderSchema, $node);
1250 1
                $all = array_merge($callbacks, $all);
1251
            }
1252
        }
1253
1254 1
        foreach ($all as $callback) {
1255 1
            call_user_func($callback);
1256
        }
1257
1258 1
        return $rootSchema;
1259
    }
1260
1261 59
    public function readNode(DOMElement $node, string $file = null): Schema
1262
    {
1263 59
        $rootSchema = new Schema();
1264 59
        $rootSchema->addSchema($this->getGlobalSchema());
1265
1266 59
        if ($file !== null) {
1267 58
            $this->setLoadedFile($file, $rootSchema);
1268
        }
1269
1270 59
        $this->setLoadedSchemaFromElement($node, $rootSchema);
1271
1272 59
        $callbacks = $this->schemaNode($rootSchema, $node);
1273
1274 59
        foreach ($callbacks as $callback) {
1275 53
            call_user_func($callback);
1276
        }
1277
1278 59
        return $rootSchema;
1279
    }
1280
1281
    /**
1282
     * @throws IOException
1283
     */
1284 58
    public function readString(string $content, string $file = 'schema.xsd'): Schema
1285
    {
1286 58
        $xml = new DOMDocument('1.0', 'UTF-8');
1287 58
        libxml_use_internal_errors(true);
1288 58
        if (!@$xml->loadXML($content)) {
1289 1
            throw new IOException("Can't load the schema", 0, $this->extractErrorMessage());
1290
        }
1291 57
        libxml_use_internal_errors(false);
1292 57
        $xml->documentURI = $file;
1293
1294 57
        return $this->readNode($xml->documentElement, $file);
1295
    }
1296
1297
    /**
1298
     * @throws IOException
1299
     */
1300 2
    public function readFile(string $file): Schema
1301
    {
1302 2
        $xml = $this->getDOM($file);
1303
1304 1
        return $this->readNode($xml->documentElement, $file);
1305
    }
1306
1307
    /**
1308
     * @throws IOException
1309
     */
1310 61
    private function getDOM(string $file): DOMDocument
1311
    {
1312 61
        $xml = new DOMDocument('1.0', 'UTF-8');
1313 61
        libxml_use_internal_errors(true);
1314 61
        if (!@$xml->load($file)) {
1315 1
            libxml_use_internal_errors(false);
1316 1
            throw new IOException("Can't load the file '$file'", 0, $this->extractErrorMessage());
1317
        }
1318 60
        libxml_use_internal_errors(false);
1319
1320 60
        return $xml;
1321
    }
1322
1323 60
    private static function againstDOMNodeList(
1324
        DOMElement $node,
1325
        Closure $againstNodeList
1326
    ): void {
1327 60
        $limit = $node->childNodes->length;
1328 60
        for ($i = 0; $i < $limit; $i += 1) {
1329
            /**
1330
             * @var DOMNode
1331
             */
1332 60
            $childNode = $node->childNodes->item($i);
1333
1334 60
            if ($childNode instanceof DOMElement) {
1335 60
                $againstNodeList(
1336 60
                    $node,
1337 60
                    $childNode
1338
                );
1339
            }
1340
        }
1341 60
    }
1342
1343 60
    private function loadTypeWithCallback(
1344
        Schema $schema,
1345
        DOMElement $childNode,
1346
        Closure $callback
1347
    ): void {
1348
        /**
1349
         * @var Closure|null $func
1350
         */
1351 60
        $func = null;
1352
1353 60
        switch ($childNode->localName) {
1354 60
            case 'complexType':
1355 60
                $func = $this->loadComplexType($schema, $childNode, $callback);
1356 60
                break;
1357 60
            case 'simpleType':
1358 60
                $func = $this->loadSimpleType($schema, $childNode, $callback);
1359 60
                break;
1360
        }
1361
1362 60
        if ($func instanceof Closure) {
1363 60
            call_user_func($func);
1364
        }
1365 60
    }
1366
1367 60
    private function loadElement(
1368
        Schema $schema,
1369
        DOMElement $node
1370
    ): Element {
1371 60
        $element = new Element($schema, $node->getAttribute('name'));
1372 60
        $element->setDoc($this->getDocumentation($node));
1373
1374 60
        $this->fillItem($element, $node);
1375
1376 60
        self::maybeSetMax($element, $node);
1377 60
        self::maybeSetMin($element, $node);
1378
1379 60
        $xp = new \DOMXPath($node->ownerDocument);
1380 60
        $xp->registerNamespace('xs', 'http://www.w3.org/2001/XMLSchema');
1381
1382 60
        if ($xp->query('ancestor::xs:choice', $node)->length) {
1383 60
            $element->setMin(0);
1384
        }
1385
1386 60
        if ($node->hasAttribute('nillable')) {
1387 3
            $element->setNil($node->getAttribute('nillable') == 'true');
1388
        }
1389 60
        if ($node->hasAttribute('form')) {
1390 4
            $element->setQualified($node->getAttribute('form') == 'qualified');
1391
        }
1392
1393 60
        return $element;
1394
    }
1395
1396 60
    private function addAttributeFromAttributeOrRef(
1397
        BaseComplexType $type,
1398
        DOMElement $childNode,
1399
        Schema $schema,
1400
        DOMElement $node
1401
    ): void {
1402 60
        $attribute = $this->getAttributeFromAttributeOrRef(
1403 60
            $childNode,
1404 60
            $schema,
1405 60
            $node
1406
        );
1407
1408 60
        $type->addAttribute($attribute);
1409 60
    }
1410
1411 60
    private function findSomethingLikeAttributeGroup(
1412
        Schema $schema,
1413
        DOMElement $node,
1414
        DOMElement $childNode,
1415
        AttributeContainer $addToThis
1416
    ): void {
1417 60
        $attribute = $this->findAttributeGroup($schema, $node, $childNode->getAttribute('ref'));
1418 60
        $addToThis->addAttribute($attribute);
1419 60
    }
1420
1421 60
    private function setLoadedFile(string $key, Schema $schema): void
1422
    {
1423 60
        $this->loadedFiles[$key] = $schema;
1424 60
    }
1425
1426 60
    private function setLoadedSchemaFromElement(DOMElement $node, Schema $schema): void
1427
    {
1428 60
        if ($node->hasAttribute('targetNamespace')) {
1429 60
            $this->setLoadedSchema($node->getAttribute('targetNamespace'), $schema);
1430
        }
1431 60
    }
1432
1433 60
    private function setLoadedSchema(string $namespace, Schema $schema): void
1434
    {
1435 60
        if (!isset($this->loadedSchemas[$namespace])) {
1436 60
            $this->loadedSchemas[$namespace] = array();
1437
        }
1438 60
        if (!in_array($schema, $this->loadedSchemas[$namespace], true)) {
1439 60
            $this->loadedSchemas[$namespace][] = $schema;
1440
        }
1441 60
    }
1442
1443 60
    private function setSchemaThingsFromNode(
1444
        Schema $schema,
1445
        DOMElement $node,
1446
        Schema $parent = null
1447
    ): void {
1448 60
        $schema->setDoc($this->getDocumentation($node));
1449
1450 60
        if ($node->hasAttribute('targetNamespace')) {
1451 60
            $schema->setTargetNamespace($node->getAttribute('targetNamespace'));
1452
        } elseif ($parent instanceof Schema) {
1453
            $schema->setTargetNamespace($parent->getTargetNamespace());
1454
        }
1455 60
        $schema->setElementsQualification($node->getAttribute('elementFormDefault') == 'qualified');
1456 60
        $schema->setAttributesQualification($node->getAttribute('attributeFormDefault') == 'qualified');
1457 60
        $schema->setDoc($this->getDocumentation($node));
1458 60
    }
1459
}
1460