1
|
|
|
<?php |
2
|
|
|
/** |
3
|
|
|
* Created by PhpStorm. |
4
|
|
|
* User: twhiston |
5
|
|
|
* Date: 27/11/16 |
6
|
|
|
* Time: 13:57 |
7
|
|
|
*/ |
8
|
|
|
|
9
|
|
|
namespace twhiston\simplexml_debug; |
10
|
|
|
|
11
|
|
|
|
12
|
|
|
/** |
13
|
|
|
* Class SxmlDebug |
14
|
|
|
* |
15
|
|
|
* @package twhiston\simplexml_debug |
16
|
|
|
*/ |
17
|
|
|
class SxmlDebug { |
18
|
|
|
|
19
|
|
|
|
20
|
|
|
/** |
21
|
|
|
* Character to use for indenting strings |
22
|
|
|
*/ |
23
|
|
|
const INDENT = "\t"; |
24
|
|
|
/** |
25
|
|
|
* How much of a string to extract |
26
|
|
|
*/ |
27
|
|
|
const EXTRACT_SIZE = 15; |
28
|
|
|
|
29
|
|
|
/** |
30
|
|
|
* Output a summary of the node or list of nodes referenced by a particular |
31
|
|
|
* SimpleXML object Rather than attempting a recursive inspection, presents |
32
|
|
|
* statistics aimed at understanding what your SimpleXML code is doing. |
33
|
|
|
* |
34
|
|
|
* @param \SimpleXMLElement $sxml The object to inspect |
35
|
|
|
* @return string output string |
36
|
|
|
* |
37
|
|
|
*/ |
38
|
5 |
|
public static function dump(\SimpleXMLElement $sxml) { |
39
|
|
|
|
40
|
5 |
|
$dump = ''; |
41
|
|
|
// Note that the header is added at the end, so we can add stats |
42
|
5 |
|
$dump .= '[' . PHP_EOL; |
43
|
|
|
|
44
|
|
|
// SimpleXML objects can be either a single node, or (more commonly) a list of 0 or more nodes |
45
|
|
|
// I haven't found a reliable way of distinguishing between the two cases |
46
|
|
|
// Note that for a single node, foreach($node) acts like foreach($node->children()) |
47
|
|
|
// Numeric array indexes, however, operate consistently: $node[0] just returns the node |
48
|
5 |
|
$item_index = 0; |
49
|
5 |
|
while (isset($sxml[$item_index])) { |
50
|
|
|
|
51
|
|
|
/** @var \SimpleXMLElement $item */ |
52
|
5 |
|
$item = $sxml[$item_index]; |
53
|
5 |
|
$item_index++; |
54
|
|
|
|
55
|
|
|
// It's surprisingly hard to find something which behaves consistently differently for an attribute and an element within SimpleXML |
56
|
|
|
// The below relies on the fact that the DOM makes a much clearer distinction |
57
|
|
|
// Note that this is not an expensive conversion, as we are only swapping PHP wrappers around an existing LibXML resource |
58
|
5 |
|
if (dom_import_simplexml($item) instanceOf \DOMAttr) { |
59
|
1 |
|
$dump .= self::dumpAddAttribute($item); |
60
|
|
|
} else { |
61
|
4 |
|
$dump .= self::dumpAddElement($item); |
62
|
|
|
} |
63
|
|
|
} |
64
|
5 |
|
$dump .= ']' . PHP_EOL; |
65
|
|
|
|
66
|
|
|
// Add on the header line, with the total number of items output |
67
|
5 |
|
return self::getHeaderLine($item_index) . $dump; |
68
|
|
|
} |
69
|
|
|
|
70
|
|
|
/** |
71
|
|
|
* @param \SimpleXMLElement $item |
72
|
|
|
* @return string |
73
|
|
|
*/ |
74
|
5 |
|
private static function dumpAddNamespace(\SimpleXMLElement $item): string { |
75
|
|
|
|
76
|
5 |
|
$dump = ''; |
77
|
|
|
// To what namespace does this attribute belong? Returns array( alias => URI ) |
78
|
5 |
|
$ns = $item->getNamespaces(FALSE); |
79
|
5 |
|
if (!empty($ns)) { |
80
|
3 |
|
$dump .= self::INDENT . self::INDENT . 'Namespace: \'' . reset($ns) . |
81
|
3 |
|
'\'' . |
82
|
3 |
|
PHP_EOL; |
83
|
3 |
|
if (key($ns) == '') { |
84
|
1 |
|
$dump .= self::INDENT . self::INDENT . '(Default Namespace)' . PHP_EOL; |
85
|
|
|
} else { |
86
|
2 |
|
$dump .= self::INDENT . self::INDENT . 'Namespace Alias: \'' . |
87
|
2 |
|
key($ns) . |
88
|
2 |
|
'\'' . |
89
|
2 |
|
PHP_EOL; |
90
|
|
|
} |
91
|
|
|
} |
92
|
|
|
|
93
|
5 |
|
return $dump; |
94
|
|
|
} |
95
|
|
|
|
96
|
|
|
/** |
97
|
|
|
* @param $title |
98
|
|
|
* @param $data |
99
|
|
|
* @param int $indent |
100
|
|
|
* @param bool $backtick |
101
|
|
|
* @return string |
102
|
|
|
*/ |
103
|
5 |
|
private static function dumpGetLine($title, |
104
|
|
|
$data, |
105
|
|
|
$indent = 1, |
106
|
|
|
$backtick = TRUE): string { |
107
|
5 |
|
return str_repeat(self::INDENT, $indent) . $title . ': ' . |
108
|
5 |
|
($backtick ? '\'' : '') . $data . |
109
|
5 |
|
($backtick ? '\'' : '') . PHP_EOL; |
110
|
|
|
} |
111
|
|
|
|
112
|
|
|
/** |
113
|
|
|
* @param \SimpleXMLElement $item |
114
|
|
|
* @return string |
115
|
|
|
*/ |
116
|
1 |
|
private static function dumpAddAttribute(\SimpleXMLElement $item): string { |
117
|
|
|
|
118
|
1 |
|
$dump = self::INDENT . 'Attribute {' . PHP_EOL; |
119
|
|
|
|
120
|
1 |
|
$dump .= self::dumpAddNamespace($item); |
121
|
|
|
|
122
|
1 |
|
$dump .= self::dumpGetLine('Name', $item->getName(), 2); |
123
|
1 |
|
$dump .= self::dumpGetLine('Value', (string) $item, 2); |
124
|
|
|
|
125
|
1 |
|
$dump .= self::INDENT . '}' . PHP_EOL; |
126
|
1 |
|
return $dump; |
127
|
|
|
|
128
|
|
|
} |
129
|
|
|
|
130
|
|
|
/** |
131
|
|
|
* @param \SimpleXMLElement $item |
132
|
|
|
* @return string |
133
|
|
|
*/ |
134
|
4 |
|
private static function dumpAddElement(\SimpleXMLElement $item): string { |
135
|
|
|
|
136
|
4 |
|
$dump = self::INDENT . 'Element {' . PHP_EOL; |
137
|
|
|
|
138
|
4 |
|
$dump .= self::dumpAddNamespace($item); |
139
|
|
|
|
140
|
4 |
|
$dump .= self::dumpGetLine('Name', $item->getName(), 2); |
141
|
4 |
|
$dump .= self::dumpGetLine('String Content', (string) $item, 2); |
142
|
|
|
|
143
|
|
|
// Now some statistics about attributes and children, by namespace |
144
|
|
|
|
145
|
|
|
// This returns all namespaces used by this node and all its descendants, |
146
|
|
|
// whether declared in this node, in its ancestors, or in its descendants |
147
|
4 |
|
$all_ns = $item->getNamespaces(TRUE); |
148
|
|
|
// If the default namespace is never declared, it will never show up using the below code |
149
|
4 |
|
if (!array_key_exists('', $all_ns)) { |
150
|
3 |
|
$all_ns[''] = NULL; |
151
|
|
|
} |
152
|
|
|
|
153
|
4 |
|
foreach ($all_ns as $ns_alias => $ns_uri) { |
154
|
4 |
|
$children = $item->children($ns_uri); |
155
|
4 |
|
$attributes = $item->attributes($ns_uri); |
156
|
|
|
|
157
|
|
|
// Somewhat confusingly, in the case where a parent element is missing the xmlns declaration, |
158
|
|
|
// but a descendant adds it, SimpleXML will look ahead and fill $all_ns[''] incorrectly |
159
|
|
|
if ( |
160
|
4 |
|
empty($ns_alias) |
161
|
|
|
&& |
162
|
4 |
|
NULL !== $ns_uri |
163
|
|
|
&& |
164
|
4 |
|
count($children) === 0 |
165
|
|
|
&& |
166
|
4 |
|
count($attributes) === 0 |
167
|
|
|
) { |
168
|
|
|
// Try looking for a default namespace without a known URI |
169
|
|
|
$ns_uri = NULL; |
170
|
|
|
$children = $item->children($ns_uri); |
171
|
|
|
$attributes = $item->attributes($ns_uri); |
172
|
|
|
} |
173
|
|
|
|
174
|
|
|
// Don't show zero-counts, as they're not that useful |
175
|
4 |
|
if (count($children) === 0 && count($attributes) === 0) { |
176
|
2 |
|
continue; |
177
|
|
|
} |
178
|
|
|
|
179
|
4 |
|
$ns_label = (($ns_alias === '') ? 'Default Namespace' : |
180
|
4 |
|
"Namespace $ns_alias"); |
181
|
|
|
|
182
|
4 |
|
$dump .= self::INDENT . self::INDENT . 'Content in ' . $ns_label . |
183
|
4 |
|
PHP_EOL; |
184
|
|
|
|
185
|
4 |
|
if (NULL !== $ns_uri) { |
186
|
3 |
|
$dump .= self::dumpGetLine('Namespace URI', $ns_uri, 3); |
187
|
|
|
} |
188
|
|
|
|
189
|
|
|
|
190
|
4 |
|
$dump .= self::dumpGetLine('Children', |
191
|
4 |
|
self::dumpGetChildDetails($children), |
192
|
4 |
|
3, |
193
|
4 |
|
FALSE); |
194
|
|
|
|
195
|
|
|
|
196
|
4 |
|
$dump .= self::dumpGetLine('Attributes', |
197
|
4 |
|
self::dumpGetAttributeDetails($attributes), |
198
|
4 |
|
3, |
199
|
4 |
|
FALSE); |
200
|
|
|
} |
201
|
|
|
|
202
|
4 |
|
return $dump . self::INDENT . '}' . PHP_EOL; |
203
|
|
|
} |
204
|
|
|
|
205
|
|
|
/** |
206
|
|
|
* @param \SimpleXMLElement $children |
207
|
|
|
* @return string |
208
|
|
|
*/ |
209
|
4 |
|
private static function dumpGetChildDetails(\SimpleXMLElement $children): string { |
210
|
|
|
// Count occurrence of child element names, rather than listing them all out |
211
|
4 |
|
$child_names = []; |
212
|
4 |
|
foreach ($children as $sx_child) { |
213
|
|
|
// Below is a rather clunky way of saying $child_names[ $sx_child->getName() ]++; |
214
|
|
|
// which avoids Notices about unset array keys |
215
|
3 |
|
$child_node_name = $sx_child->getName(); |
216
|
3 |
|
if (array_key_exists($child_node_name, $child_names)) { |
217
|
|
|
$child_names[$child_node_name]++; |
218
|
|
|
} else { |
219
|
3 |
|
$child_names[$child_node_name] = 1; |
220
|
|
|
} |
221
|
|
|
} |
222
|
4 |
|
ksort($child_names); |
223
|
4 |
|
$child_name_output = []; |
224
|
4 |
|
foreach ($child_names as $name => $count) { |
225
|
3 |
|
$child_name_output[] = "$count '$name'"; |
226
|
|
|
} |
227
|
|
|
|
228
|
4 |
|
$childrenString = count($children); |
229
|
|
|
// Don't output a trailing " - " if there are no children |
230
|
4 |
|
if (count($children) > 0) { |
231
|
3 |
|
$childrenString .= ' - ' . implode(', ', $child_name_output); |
232
|
|
|
} |
233
|
4 |
|
return $childrenString; |
234
|
|
|
} |
235
|
|
|
|
236
|
|
|
/** |
237
|
|
|
* @param \SimpleXMLElement $attributes |
238
|
|
|
* @return string |
239
|
|
|
*/ |
240
|
4 |
|
private static function dumpGetAttributeDetails(\SimpleXMLElement $attributes): string { |
241
|
|
|
// Attributes can't be duplicated, but I'm going to put them in alphabetical order |
242
|
4 |
|
$attribute_names = []; |
243
|
4 |
|
foreach ($attributes as $sx_attribute) { |
244
|
1 |
|
$attribute_names[] = "'" . $sx_attribute->getName() . "'"; |
245
|
|
|
} |
246
|
4 |
|
ksort($attribute_names); |
247
|
|
|
|
248
|
4 |
|
$attString = count($attributes); |
249
|
|
|
// Don't output a trailing " - " if there are no attributes |
250
|
4 |
|
if (count($attributes) > 0) { |
251
|
1 |
|
$attString .= ' - ' . implode(', ', $attribute_names); |
252
|
|
|
} |
253
|
4 |
|
return $attString; |
254
|
|
|
} |
255
|
|
|
|
256
|
|
|
/** |
257
|
|
|
* @param $index |
258
|
|
|
* @return string |
259
|
|
|
*/ |
260
|
9 |
|
private static function getHeaderLine($index): string { |
261
|
|
|
|
262
|
9 |
|
return 'SimpleXML object (' . $index . ' item' . |
263
|
9 |
|
($index > 1 ? 's' : '') . ')' . PHP_EOL; |
264
|
|
|
} |
265
|
|
|
|
266
|
|
|
/** |
267
|
|
|
* Output a tree-view of the node or list of nodes referenced by a particular |
268
|
|
|
* SimpleXML object Unlike simplexml_dump(), this processes the entire XML |
269
|
|
|
* tree recursively, while attempting to be more concise and readable than |
270
|
|
|
* the XML itself. Additionally, the output format is designed as a hint of |
271
|
|
|
* the syntax needed to traverse the object. |
272
|
|
|
* |
273
|
|
|
* @param \SimpleXMLElement $sxml The object to inspect |
274
|
|
|
* @param boolean $include_string_content Default false. If true, |
275
|
|
|
* will summarise textual |
276
|
|
|
* content, as well as child |
277
|
|
|
* elements and attribute |
278
|
|
|
* names |
279
|
|
|
* @return null|string Nothing, or output, depending on $return param |
280
|
|
|
* |
281
|
|
|
*/ |
282
|
4 |
|
public static function tree(\SimpleXMLElement $sxml, |
283
|
|
|
$include_string_content = FALSE): string { |
284
|
|
|
|
285
|
4 |
|
$dump = ''; |
286
|
|
|
// Note that the header is added at the end, so we can add stats |
287
|
|
|
|
288
|
|
|
// The initial object passed in may be a single node or a list of nodes, so we need an outer loop first |
289
|
|
|
// Note that for a single node, foreach($node) acts like foreach($node->children()) |
290
|
|
|
// Numeric array indexes, however, operate consistently: $node[0] just returns the node |
291
|
4 |
|
$root_item_index = 0; |
292
|
4 |
|
while (isset($sxml[$root_item_index])) { |
293
|
4 |
|
$root_item = $sxml[$root_item_index]; |
294
|
|
|
|
295
|
|
|
// Special case if the root is actually an attribute |
296
|
|
|
// It's surprisingly hard to find something which behaves consistently differently for an attribute and an element within SimpleXML |
297
|
|
|
// The below relies on the fact that the DOM makes a much clearer distinction |
298
|
|
|
// Note that this is not an expensive conversion, as we are only swapping PHP wrappers around an existing LibXML resource |
299
|
4 |
|
if (dom_import_simplexml($root_item) instanceOf \DOMAttr) { |
300
|
|
|
// To what namespace does this attribute belong? Returns array( alias => URI ) |
301
|
|
|
$ns = $root_item->getNamespaces(FALSE); |
302
|
|
|
if (key($ns)) { |
303
|
|
|
$dump .= key($ns) . ':'; |
304
|
|
|
} |
305
|
|
|
$dump .= $root_item->getName() . '="' . (string) $root_item . '"' . |
306
|
|
|
PHP_EOL; |
307
|
|
|
} else { |
308
|
|
|
// Display the root node as a numeric key reference, plus a hint as to its tag name |
309
|
|
|
// e.g. '[42] // <Answer>' |
310
|
|
|
|
311
|
|
|
// To what namespace does this attribute belong? Returns array( alias => URI ) |
312
|
4 |
|
$ns = $root_item->getNamespaces(FALSE); |
313
|
4 |
|
if (key($ns)) { |
314
|
|
|
$root_node_name = key($ns) . ':' . $root_item->getName(); |
315
|
|
|
} else { |
316
|
4 |
|
$root_node_name = $root_item->getName(); |
317
|
|
|
} |
318
|
4 |
|
$dump .= "[$root_item_index] // <$root_node_name>" . PHP_EOL; |
319
|
|
|
|
320
|
|
|
// This function is effectively recursing depth-first through the tree, |
321
|
|
|
// but this is managed manually using a stack rather than actual recursion |
322
|
|
|
// Each item on the stack is of the form array(int $depth, SimpleXMLElement $element, string $header_row) |
323
|
4 |
|
$dump .= SxmlDebug::recursivelyProcessNode( |
324
|
|
|
$root_item, |
325
|
4 |
|
1, |
326
|
|
|
$include_string_content |
327
|
|
|
); |
328
|
|
|
} |
329
|
|
|
|
330
|
4 |
|
$root_item_index++; |
331
|
|
|
} |
332
|
|
|
|
333
|
|
|
// Add on the header line, with the total number of items output |
334
|
4 |
|
$dump = self::getHeaderLine($root_item_index) . $dump; |
335
|
|
|
|
336
|
4 |
|
return $dump; |
337
|
|
|
|
338
|
|
|
} |
339
|
|
|
|
340
|
|
|
|
341
|
|
|
/** |
342
|
|
|
* @param string $stringContent |
343
|
|
|
* @param $depth |
344
|
|
|
* @return string |
345
|
|
|
*/ |
346
|
1 |
|
private static function treeGetStringExtract(string $stringContent, |
347
|
|
|
$depth): string { |
348
|
1 |
|
$string_extract = preg_replace('/\s+/', ' ', trim($stringContent)); |
349
|
1 |
|
if (strlen($string_extract) > SxmlDebug::EXTRACT_SIZE) { |
350
|
1 |
|
$string_extract = substr($string_extract, 0, SxmlDebug::EXTRACT_SIZE) |
351
|
1 |
|
. '...'; |
352
|
|
|
} |
353
|
1 |
|
return (strlen($stringContent) > 0) ? |
354
|
1 |
|
str_repeat(SxmlDebug::INDENT, $depth) |
355
|
1 |
|
. '(string) ' |
356
|
1 |
|
. "'$string_extract'" |
357
|
1 |
|
. ' (' . strlen($stringContent) . ' chars)' |
358
|
1 |
|
. PHP_EOL : ''; |
359
|
|
|
|
360
|
|
|
} |
361
|
|
|
|
362
|
|
|
/** |
363
|
|
|
* @param \SimpleXMLElement $item |
364
|
|
|
* @return array |
365
|
|
|
*/ |
366
|
4 |
|
private static function treeGetNamespaces(\SimpleXMLElement $item): array { |
367
|
|
|
// To what namespace does this element belong? Returns array( alias => URI ) |
368
|
4 |
|
$item_ns = $item->getNamespaces(FALSE); |
369
|
4 |
|
if (empty($item_ns)) { |
370
|
3 |
|
$item_ns = ['' => NULL]; |
371
|
|
|
} |
372
|
|
|
|
373
|
|
|
// This returns all namespaces used by this node and all its descendants, |
374
|
|
|
// whether declared in this node, in its ancestors, or in its descendants |
375
|
4 |
|
$all_ns = $item->getNamespaces(TRUE); |
376
|
|
|
// If the default namespace is never declared, it will never show up using the below code |
377
|
4 |
|
if (!array_key_exists('', $all_ns)) { |
378
|
3 |
|
$all_ns[''] = NULL; |
379
|
|
|
} |
380
|
|
|
|
381
|
|
|
// Prioritise "current" namespace by merging into onto the beginning of the list |
382
|
|
|
// (it will be added to the beginning and the duplicate entry dropped) |
383
|
4 |
|
return array_merge($item_ns, $all_ns); |
384
|
|
|
} |
385
|
|
|
|
386
|
|
|
/** |
387
|
|
|
* @param \SimpleXMLElement $attributes |
388
|
|
|
* @param string $nsAlias |
389
|
|
|
* @param int $depth |
390
|
|
|
* @param bool $isCurrentNamespace |
391
|
|
|
* @param bool $includeStringContent |
392
|
|
|
* @return string |
393
|
|
|
*/ |
394
|
4 |
|
private static function treeProcessAttributes(\SimpleXMLElement $attributes, |
395
|
|
|
string $nsAlias, |
396
|
|
|
int $depth, |
397
|
|
|
bool $isCurrentNamespace, |
398
|
|
|
bool $includeStringContent): string { |
399
|
|
|
|
400
|
4 |
|
$dump = ''; |
401
|
4 |
|
if (count($attributes) > 0) { |
402
|
4 |
|
if (!$isCurrentNamespace) { |
403
|
1 |
|
$dump .= str_repeat(self::INDENT, $depth) |
404
|
1 |
|
. "->attributes('$nsAlias', true)" . PHP_EOL; |
405
|
|
|
} |
406
|
|
|
|
407
|
4 |
|
foreach ($attributes as $sx_attribute) { |
408
|
|
|
// Output the attribute |
409
|
4 |
|
if ($isCurrentNamespace) { |
410
|
|
|
// In current namespace |
411
|
|
|
// e.g. ['attribName'] |
412
|
3 |
|
$dump .= str_repeat(self::INDENT, $depth) |
413
|
3 |
|
. "['" . $sx_attribute->getName() . "']" |
414
|
3 |
|
. PHP_EOL; |
415
|
3 |
|
$string_display_depth = $depth + 1; |
416
|
|
|
} else { |
417
|
|
|
// After a call to ->attributes() |
418
|
|
|
// e.g. ->attribName |
419
|
1 |
|
$dump .= str_repeat(self::INDENT, $depth + 1) |
420
|
1 |
|
. '->' . $sx_attribute->getName() |
421
|
1 |
|
. PHP_EOL; |
422
|
1 |
|
$string_display_depth = $depth + 2; |
423
|
|
|
} |
424
|
|
|
|
425
|
4 |
|
if ($includeStringContent) { |
426
|
|
|
// Show a chunk of the beginning of the content string, collapsing whitespace HTML-style |
427
|
4 |
|
$dump .= self::treeGetStringExtract((string) $sx_attribute, |
428
|
|
|
$string_display_depth); |
429
|
|
|
} |
430
|
|
|
} |
431
|
|
|
} |
432
|
4 |
|
return $dump; |
433
|
|
|
} |
434
|
|
|
|
435
|
|
|
/** |
436
|
|
|
* @param \SimpleXMLElement $children |
437
|
|
|
* @param string $nsAlias |
438
|
|
|
* @param int $depth |
439
|
|
|
* @param bool $isCurrentNamespace |
440
|
|
|
* @param bool $includeStringContent |
441
|
|
|
* @return string |
442
|
|
|
*/ |
443
|
4 |
|
private static function treeProcessChildren(\SimpleXMLElement $children, |
444
|
|
|
string $nsAlias, |
445
|
|
|
int $depth, |
446
|
|
|
bool $isCurrentNamespace, |
447
|
|
|
bool $includeStringContent): string { |
448
|
|
|
|
449
|
4 |
|
$dump = ''; |
450
|
4 |
|
if (count($children) > 0) { |
451
|
4 |
|
if ($isCurrentNamespace) { |
452
|
4 |
|
$display_depth = $depth; |
453
|
|
|
} else { |
454
|
1 |
|
$dump .= str_repeat(self::INDENT, $depth) |
455
|
1 |
|
. "->children('$nsAlias', true)" . PHP_EOL; |
456
|
1 |
|
$display_depth = $depth + 1; |
457
|
|
|
} |
458
|
|
|
|
459
|
|
|
// Recurse through the children with headers showing how to access them |
460
|
4 |
|
$child_names = []; |
461
|
4 |
|
foreach ($children as $sx_child) { |
462
|
|
|
// Below is a rather clunky way of saying $child_names[ $sx_child->getName() ]++; |
463
|
|
|
// which avoids Notices about unset array keys |
464
|
4 |
|
$child_node_name = $sx_child->getName(); |
465
|
4 |
|
if (array_key_exists($child_node_name, $child_names)) { |
466
|
4 |
|
$child_names[$child_node_name]++; |
467
|
|
|
} else { |
468
|
4 |
|
$child_names[$child_node_name] = 1; |
469
|
|
|
} |
470
|
|
|
|
471
|
|
|
// e.g. ->Foo[0] |
472
|
4 |
|
$dump .= str_repeat(self::INDENT, $display_depth) |
473
|
4 |
|
. '->' . $sx_child->getName() |
474
|
4 |
|
. '[' . ($child_names[$child_node_name] - 1) . ']' |
475
|
4 |
|
. PHP_EOL; |
476
|
|
|
|
477
|
4 |
|
$dump .= self::recursivelyProcessNode( |
478
|
|
|
$sx_child, |
479
|
4 |
|
$display_depth + 1, |
480
|
|
|
$includeStringContent |
481
|
|
|
); |
482
|
|
|
} |
483
|
|
|
} |
484
|
4 |
|
return $dump; |
485
|
|
|
} |
486
|
|
|
|
487
|
|
|
/** |
488
|
|
|
* @param $item |
489
|
|
|
* @param $depth |
490
|
|
|
* @param $include_string_content |
491
|
|
|
* @return string |
492
|
|
|
*/ |
493
|
4 |
|
private static function recursivelyProcessNode(\SimpleXMLElement $item, |
494
|
|
|
$depth, |
495
|
|
|
$include_string_content): string { |
496
|
|
|
|
497
|
4 |
|
$dump = ''; |
498
|
|
|
|
499
|
4 |
|
if ($include_string_content) { |
500
|
|
|
// Show a chunk of the beginning of the content string, collapsing whitespace HTML-style |
501
|
1 |
|
$dump = self::treeGetStringExtract((string) $item, $depth); |
502
|
|
|
} |
503
|
|
|
|
504
|
4 |
|
$itemNs = self::treeGetNamespaces($item); |
505
|
4 |
|
foreach ($itemNs as $ns_alias => $ns_uri) { |
506
|
|
|
|
507
|
|
|
|
508
|
|
|
// If things are in the current namespace, display them a bit differently |
509
|
4 |
|
$is_current_namespace = ($ns_uri === reset($itemNs)); |
510
|
|
|
|
511
|
4 |
|
$dump .= self::treeProcessAttributes($item->attributes($ns_alias, TRUE), |
512
|
|
|
$ns_alias, |
513
|
|
|
$depth, |
514
|
|
|
$is_current_namespace, |
515
|
|
|
$include_string_content); |
516
|
4 |
|
$dump .= self::treeProcessChildren($item->children($ns_alias, TRUE), |
517
|
|
|
$ns_alias, |
518
|
|
|
$depth, |
519
|
|
|
$is_current_namespace, |
520
|
|
|
$include_string_content); |
521
|
|
|
} |
522
|
|
|
|
523
|
4 |
|
return $dump; |
524
|
|
|
} |
525
|
|
|
|
526
|
|
|
} |