bytic /
utility
| 1 | <?php |
||
| 2 | |||
| 3 | /** @noinspection PhpComposerExtensionStubsInspection */ |
||
| 4 | |||
| 5 | namespace Nip\Utility\Xml; |
||
| 6 | |||
| 7 | use DOMDocument; |
||
| 8 | use DOMText; |
||
| 9 | use SebastianBergmann\CodeCoverage\XmlException; |
||
|
0 ignored issues
–
show
|
|||
| 10 | use SimpleXMLElement; |
||
| 11 | |||
| 12 | /** |
||
| 13 | * Class FromArrayBuilder |
||
| 14 | * @package Nip\Utility\Xml |
||
| 15 | */ |
||
| 16 | class FromArrayBuilder |
||
| 17 | { |
||
| 18 | protected $input; |
||
| 19 | |||
| 20 | /** |
||
| 21 | * @var DOMDocument |
||
| 22 | */ |
||
| 23 | protected $dom; |
||
| 24 | protected $options = []; |
||
| 25 | |||
| 26 | |||
| 27 | /** |
||
| 28 | * @param $input |
||
| 29 | * @param array $options |
||
| 30 | * |
||
| 31 | * @return DOMDocument|SimpleXMLElement |
||
| 32 | */ |
||
| 33 | 1 | public static function build($input, array $options = []) |
|
| 34 | { |
||
| 35 | 1 | if (is_object($input) && method_exists($input, 'toArray') && is_callable([$input, 'toArray'])) { |
|
| 36 | $input = $input->toArray(); |
||
| 37 | } |
||
| 38 | 1 | if (!is_array($input) || count($input) !== 1) { |
|
| 39 | throw new XmlException('Invalid input.'); |
||
| 40 | } |
||
| 41 | 1 | $key = key($input); |
|
| 42 | 1 | if (is_int($key)) { |
|
| 43 | throw new XmlException('The key of input must be alphanumeric'); |
||
| 44 | } |
||
| 45 | |||
| 46 | 1 | return (new static($input, $options))->buildXml(); |
|
| 47 | } |
||
| 48 | |||
| 49 | /** |
||
| 50 | * FromArrayBuilder constructor. |
||
| 51 | * |
||
| 52 | * @param $input |
||
| 53 | * @param array $options |
||
| 54 | */ |
||
| 55 | 1 | protected function __construct($input, array $options = []) |
|
| 56 | { |
||
| 57 | 1 | $this->input = $input; |
|
| 58 | |||
| 59 | $defaults = [ |
||
| 60 | 1 | 'format' => 'tags', |
|
| 61 | 1 | 'version' => '1.0', |
|
| 62 | 1 | 'encoding' => mb_internal_encoding(), |
|
| 63 | 1 | 'return' => 'simplexml', |
|
| 64 | 'pretty' => false, |
||
| 65 | ]; |
||
| 66 | 1 | $options += $defaults; |
|
| 67 | |||
| 68 | 1 | $this->options = $options; |
|
| 69 | |||
| 70 | 1 | $this->dom = new DOMDocument($options['version'], $options['encoding']); |
|
| 71 | 1 | if ($options['pretty']) { |
|
| 72 | $this->dom->formatOutput = true; |
||
| 73 | } |
||
| 74 | 1 | } |
|
| 75 | |||
| 76 | /** |
||
| 77 | * @return SimpleXMLElement|DOMDocument |
||
| 78 | */ |
||
| 79 | 1 | protected function buildXml() |
|
| 80 | { |
||
| 81 | 1 | $this->addData($this->dom, $this->dom, $this->input, $this->options['format']); |
|
| 82 | |||
| 83 | 1 | $return = strtolower($this->options['return']); |
|
| 84 | 1 | if ($return === 'simplexml' || $return === 'simplexmlelement') { |
|
| 85 | 1 | return new SimpleXMLElement($this->dom->saveXML()); |
|
| 86 | } |
||
| 87 | |||
| 88 | return $this->dom; |
||
| 89 | } |
||
| 90 | |||
| 91 | /** |
||
| 92 | * Recursive method to create childs from array |
||
| 93 | * |
||
| 94 | * @param \DOMDocument $dom Handler to DOMDocument |
||
| 95 | * @param \DOMDocument|\DOMElement $node Handler to DOMElement (child) |
||
| 96 | * @param array $data Array of data to append to the $node. |
||
| 97 | * @param string $format Either 'attributes' or 'tags'. This determines where nested keys go. |
||
| 98 | * |
||
| 99 | * @return void |
||
| 100 | * @throws XmlException |
||
| 101 | */ |
||
| 102 | 1 | protected function addData(DOMDocument $dom, $node, &$data, $format): void |
|
| 103 | { |
||
| 104 | 1 | if (empty($data) || !is_array($data)) { |
|
| 105 | return; |
||
| 106 | } |
||
| 107 | 1 | foreach ($data as $key => $value) { |
|
| 108 | 1 | if (!is_string($key)) { |
|
| 109 | throw new XmlException('Invalid array'); |
||
| 110 | } |
||
| 111 | 1 | $this->addDataItem($dom, $node, $key, $value, $format); |
|
| 112 | } |
||
| 113 | 1 | } |
|
| 114 | |||
| 115 | /** |
||
| 116 | * @param DOMDocument $dom |
||
| 117 | * @param $node |
||
| 118 | * @param $key |
||
| 119 | * @param $value |
||
| 120 | * @param $format |
||
| 121 | */ |
||
| 122 | 1 | protected function addDataItem(DOMDocument $dom, $node, $key, $value, $format): void |
|
| 123 | { |
||
| 124 | 1 | if (is_object($value) && method_exists($value, 'toArray') && is_callable([$value, 'toArray'])) { |
|
| 125 | $value = $value->toArray(); |
||
| 126 | } |
||
| 127 | |||
| 128 | 1 | if (!is_array($value)) { |
|
| 129 | 1 | if (is_bool($value)) { |
|
| 130 | $value = (int)$value; |
||
| 131 | 1 | } elseif ($value === null) { |
|
| 132 | $value = ''; |
||
| 133 | } |
||
| 134 | 1 | $isNamespace = strpos($key, 'xmlns:'); |
|
| 135 | 1 | if ($isNamespace !== false) { |
|
| 136 | /** @psalm-suppress PossiblyUndefinedMethod */ |
||
| 137 | $node->setAttributeNS('http://www.w3.org/2000/xmlns/', $key, (string)$value); |
||
| 138 | return; |
||
| 139 | } |
||
| 140 | 1 | if ($key[0] !== '@' && $format === 'tags') { |
|
| 141 | 1 | if (!is_numeric($value)) { |
|
| 142 | // Escape special characters |
||
| 143 | // https://www.w3.org/TR/REC-xml/#syntax |
||
| 144 | // https://bugs.php.net/bug.php?id=36795 |
||
| 145 | 1 | $child = $dom->createElement($key, ''); |
|
| 146 | 1 | $child->appendChild(new DOMText((string)$value)); |
|
| 147 | } else { |
||
| 148 | 1 | $child = $dom->createElement($key, (string)$value); |
|
| 149 | } |
||
| 150 | 1 | $node->appendChild($child); |
|
| 151 | } else { |
||
| 152 | if ($key[0] === '@') { |
||
| 153 | $key = substr($key, 1); |
||
| 154 | } |
||
| 155 | $attribute = $dom->createAttribute($key); |
||
| 156 | $attribute->appendChild($dom->createTextNode((string)$value)); |
||
| 157 | 1 | $node->appendChild($attribute); |
|
| 158 | } |
||
| 159 | } else { |
||
| 160 | 1 | if ($key[0] === '@') { |
|
| 161 | throw new XmlException('Invalid array'); |
||
| 162 | } |
||
| 163 | 1 | if (is_numeric(implode('', array_keys($value)))) { |
|
| 164 | // List |
||
| 165 | foreach ($value as $item) { |
||
| 166 | $itemData = compact('dom', 'node', 'key', 'format'); |
||
| 167 | $itemData['value'] = $item; |
||
| 168 | $this->createChild($itemData); |
||
| 169 | } |
||
| 170 | } else { |
||
| 171 | // Struct |
||
| 172 | 1 | $this->createChild(compact('dom', 'node', 'key', 'value', 'format')); |
|
| 173 | } |
||
| 174 | } |
||
| 175 | 1 | } |
|
| 176 | |||
| 177 | /** |
||
| 178 | * Helper to _fromArray(). It will create childs of arrays |
||
| 179 | * |
||
| 180 | * @param array $data Array with information to create childs |
||
| 181 | * |
||
| 182 | * @return void |
||
| 183 | */ |
||
| 184 | 1 | protected function createChild(array $data): void |
|
| 185 | { |
||
| 186 | $data += [ |
||
| 187 | 1 | 'dom' => null, |
|
| 188 | 'node' => null, |
||
| 189 | 'key' => null, |
||
| 190 | 'value' => null, |
||
| 191 | 'format' => null, |
||
| 192 | ]; |
||
| 193 | |||
| 194 | 1 | $value = $data['value']; |
|
| 195 | 1 | $dom = $data['dom']; |
|
| 196 | 1 | $key = $data['key']; |
|
| 197 | 1 | $format = $data['format']; |
|
| 198 | 1 | $node = $data['node']; |
|
| 199 | |||
| 200 | 1 | $childNS = $childValue = null; |
|
| 201 | 1 | if (is_object($value) && method_exists($value, 'toArray') && is_callable([$value, 'toArray'])) { |
|
| 202 | $value = $value->toArray(); |
||
| 203 | } |
||
| 204 | 1 | if (is_array($value)) { |
|
| 205 | 1 | if (isset($value['@'])) { |
|
| 206 | $childValue = (string)$value['@']; |
||
| 207 | unset($value['@']); |
||
| 208 | } |
||
| 209 | 1 | if (isset($value['xmlns:'])) { |
|
| 210 | $childNS = $value['xmlns:']; |
||
| 211 | 1 | unset($value['xmlns:']); |
|
| 212 | } |
||
| 213 | } elseif (!empty($value) || $value === 0 || $value === '0') { |
||
| 214 | $childValue = (string)$value; |
||
| 215 | } |
||
| 216 | |||
| 217 | 1 | $child = $dom->createElement($key); |
|
| 218 | 1 | if ($childValue !== null) { |
|
| 219 | $child->appendChild($dom->createTextNode($childValue)); |
||
| 220 | } |
||
| 221 | 1 | if ($childNS) { |
|
| 222 | $child->setAttribute('xmlns', $childNS); |
||
| 223 | } |
||
| 224 | |||
| 225 | 1 | $this->addData($dom, $child, $value, $format); |
|
| 226 | 1 | $node->appendChild($child); |
|
| 227 | 1 | } |
|
| 228 | } |
||
| 229 |
The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g.
excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths