-
Notifications
You must be signed in to change notification settings - Fork 1.9k
/
XMLFormatter.php
104 lines (89 loc) · 3.01 KB
/
XMLFormatter.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
<?php
declare(strict_types=1);
/**
* This file is part of CodeIgniter 4 framework.
*
* (c) CodeIgniter Foundation <admin@codeigniter.com>
*
* For the full copyright and license information, please view
* the LICENSE file that was distributed with this source code.
*/
namespace CodeIgniter\Format;
use CodeIgniter\Format\Exceptions\FormatException;
use Config\Format;
use SimpleXMLElement;
/**
* XML data formatter
*
* @see \CodeIgniter\Format\XMLFormatterTest
*/
class XMLFormatter implements FormatterInterface
{
/**
* Takes the given data and formats it.
*
* @param array|bool|float|int|object|string|null $data
*
* @return false|string (XML string | false)
*/
public function format($data)
{
$config = new Format();
// SimpleXML is installed but default
// but best to check, and then provide a fallback.
if (! extension_loaded('simplexml')) {
throw FormatException::forMissingExtension(); // @codeCoverageIgnore
}
$options = $config->formatterOptions['application/xml'] ?? 0;
$output = new SimpleXMLElement('<?xml version="1.0"?><response></response>', $options);
$this->arrayToXML((array) $data, $output);
return $output->asXML();
}
/**
* A recursive method to convert an array into a valid XML string.
*
* Written by CodexWorld. Received permission by email on Nov 24, 2016 to use this code.
*
* @see http://www.codexworld.com/convert-array-to-xml-in-php/
*
* @param SimpleXMLElement $output
*
* @return void
*/
protected function arrayToXML(array $data, &$output)
{
foreach ($data as $key => $value) {
$key = $this->normalizeXMLTag($key);
if (is_array($value)) {
$subnode = $output->addChild("{$key}");
$this->arrayToXML($value, $subnode);
} else {
$output->addChild("{$key}", htmlspecialchars("{$value}"));
}
}
}
/**
* Normalizes tags into the allowed by W3C.
* Regex adopted from this StackOverflow answer.
*
* @param int|string $key
*
* @return string
*
* @see https://stackoverflow.com/questions/60001029/invalid-characters-in-xml-tag-name
*/
protected function normalizeXMLTag($key)
{
$startChar = 'A-Z_a-z' .
'\\x{C0}-\\x{D6}\\x{D8}-\\x{F6}\\x{F8}-\\x{2FF}\\x{370}-\\x{37D}' .
'\\x{37F}-\\x{1FFF}\\x{200C}-\\x{200D}\\x{2070}-\\x{218F}' .
'\\x{2C00}-\\x{2FEF}\\x{3001}-\\x{D7FF}\\x{F900}-\\x{FDCF}' .
'\\x{FDF0}-\\x{FFFD}\\x{10000}-\\x{EFFFF}';
$validName = $startChar . '\\.\\d\\x{B7}\\x{300}-\\x{36F}\\x{203F}-\\x{2040}';
$key = (string) $key;
$key = trim($key);
$key = preg_replace("/[^{$validName}-]+/u", '', $key);
$key = preg_replace("/^[^{$startChar}]+/u", 'item$0', $key);
return preg_replace('/^(xml).*/iu', 'item$0', $key); // XML is a reserved starting word
}
}