From fe7ec61e4cfe3f8112da2ef24a7ba25e218146c7 Mon Sep 17 00:00:00 2001 From: Tim van Dijen Date: Thu, 12 Sep 2024 16:52:34 +0200 Subject: [PATCH] Add exclusion-list for ExtendableAttributesTrait --- src/ExtendableAttributesTrait.php | 66 +++++++++++++-------- tests/Utils/ExtendableAttributesElement.php | 5 ++ tests/XML/ExtendableAttributesTest.php | 21 +++++++ 3 files changed, 67 insertions(+), 25 deletions(-) diff --git a/src/ExtendableAttributesTrait.php b/src/ExtendableAttributesTrait.php index 644cc6e7..61918222 100644 --- a/src/ExtendableAttributesTrait.php +++ b/src/ExtendableAttributesTrait.php @@ -93,9 +93,9 @@ public function getAttributesNS(): array * * @return array $attributes */ - protected static function getAttributesNSFromXML(DOMElement $xml, NS|array $namespace = null): array - { + protected static function getAttributesNSFromXML(DOMElement $xml, NS|array $namespace = null): array { $namespace = $namespace ?? static::XS_ANY_ATTR_NAMESPACE; + $exclusionList = static::getAttributeExclusions(); $attributes = []; // Validate namespace value @@ -103,28 +103,18 @@ protected static function getAttributesNSFromXML(DOMElement $xml, NS|array $name // Must be one of the predefined values Assert::oneOf($namespace, NS::cases()); - if ($namespace === NS::ANY) { - foreach ($xml->attributes as $a) { - $attributes[] = new Attribute($a->namespaceURI, $a->prefix, $a->localName, $a->nodeValue); - } - } elseif ($namespace === NS::LOCAL) { - foreach ($xml->attributes as $a) { - if ($a->namespaceURI === null) { - $attributes[] = new Attribute($a->namespaceURI, $a->prefix, $a->localName, $a->nodeValue); - } - } - } elseif ($namespace === NS::OTHER) { - foreach ($xml->attributes as $a) { - if (!in_array($a->namespaceURI, [static::NS, null], true)) { - $attributes[] = new Attribute($a->namespaceURI, $a->prefix, $a->localName, $a->nodeValue); - } - } - } elseif ($namespace === NS::TARGET) { - foreach ($xml->attributes as $a) { - if ($a->namespaceURI === static::NS) { - $attributes[] = new Attribute($a->namespaceURI, $a->prefix, $a->localName, $a->nodeValue); - } + foreach ($xml->attributes as $a) { + if (in_array([$a->namespaceURI, $a->localName], $exclusionList, true)) { + continue; + } elseif ($namespace === NS::OTHER && in_array($a->namespaceURI, [static::NS, null], true)) { + continue; + } elseif ($namespace === NS::TARGET && $a->namespaceURI !== static::NS) { + continue; + } elseif ($namespace === NS::LOCAL && $a->namespaceURI !== null) { + continue; } + + $attributes[] = new Attribute($a->namespaceURI, $a->prefix, $a->localName, $a->nodeValue); } } else { // Array must be non-empty and cannot contain ##any or ##other @@ -144,9 +134,13 @@ protected static function getAttributesNSFromXML(DOMElement $xml, NS|array $name } foreach ($xml->attributes as $a) { - if (in_array($a->namespaceURI, $namespace, true)) { - $attributes[] = new Attribute($a->namespaceURI, $a->prefix, $a->localName, $a->nodeValue); + if (in_array([$a->namespaceURI, $a->localName], $exclusionList, true)) { + continue; + } elseif (!in_array($a->namespaceURI, $namespace, true)) { + continue; } + + $attributes[] = new Attribute($a->namespaceURI, $a->prefix, $a->localName, $a->nodeValue); } } @@ -230,6 +224,13 @@ function (Attribute $attr) { } } + $exclusionList = static::getAttributeExclusions(); + foreach ($attributes as $i => $attr) { + if (in_array([$attr->getNamespaceURI(), $attr->getAttrName()], $exclusionList, true)) { + unset($attributes[$i]); + } + } + $this->namespacedAttributes = $attributes; } @@ -249,4 +250,19 @@ public function getAttributeNamespace(): array|NS return static::XS_ANY_ATTR_NAMESPACE; } + + + /** + * Get the exclusions list for getAttributeNSFromXML. + * + * @return array + */ + public static function getAttributeExclusions(): array + { + if (defined('static::XS_ANY_ATTR_EXCLUSIONS')) { + return static::XS_ANY_ATTR_EXCLUSIONS; + } + + return []; + } } diff --git a/tests/Utils/ExtendableAttributesElement.php b/tests/Utils/ExtendableAttributesElement.php index ff022e8a..05d36c28 100644 --- a/tests/Utils/ExtendableAttributesElement.php +++ b/tests/Utils/ExtendableAttributesElement.php @@ -33,6 +33,11 @@ class ExtendableAttributesElement extends AbstractElement /** @var string|\SimpleSAML\XML\XsNamespace */ public const XS_ANY_ATTR_NAMESPACE = NS::ANY; + /** @var array */ + public const XS_ANY_ATTR_EXCLUSIONS = [ + ['urn:x-simplesamlphp:namespace', 'attr3'], + ]; + /** * Get the namespace for the element. diff --git a/tests/XML/ExtendableAttributesTest.php b/tests/XML/ExtendableAttributesTest.php index 17603f47..66574f5e 100644 --- a/tests/XML/ExtendableAttributesTest.php +++ b/tests/XML/ExtendableAttributesTest.php @@ -49,6 +49,7 @@ public function testMarshalling(): void [ new Attribute('urn:x-simplesamlphp:namespace', 'ssp', 'attr1', 'testval1'), new Attribute('urn:x-simplesamlphp:namespace', 'ssp', 'attr2', 'testval2'), + new Attribute('urn:x-simplesamlphp:namespace', 'ssp', 'attr3', 'testval3'), ], ); @@ -57,4 +58,24 @@ public function testMarshalling(): void strval($extendableElement), ); } + + + /** + */ + public function testGetAttributesNSFromXML(): void + { + $elt = ExtendableAttributesElement::fromXML(self::$xmlRepresentation->documentElement); + $attributes = $elt->getAttributesNS(); + + $this->assertCount(2, $attributes); + $this->assertEquals($attributes[0]->getNamespaceURI(), 'urn:x-simplesamlphp:namespace'); + $this->assertEquals($attributes[0]->getNamespacePrefix(), 'ssp'); + $this->assertEquals($attributes[0]->getAttrName(), 'attr1'); + $this->assertEquals($attributes[0]->getAttrValue(), 'testval1'); + + $this->assertEquals($attributes[1]->getNamespaceURI(), 'urn:x-simplesamlphp:namespace'); + $this->assertEquals($attributes[1]->getNamespacePrefix(), 'ssp'); + $this->assertEquals($attributes[1]->getAttrName(), 'attr2'); + $this->assertEquals($attributes[1]->getAttrValue(), 'testval2'); + } }