Consistently use "superuser" instead of "super user"
[pgsql.git] / src / test / regress / expected / xml_2.out
blob048426028175d2ed467d260f3f4370c713d5c742
1 CREATE TABLE xmltest (
2     id int,
3     data xml
4 );
5 INSERT INTO xmltest VALUES (1, '<value>one</value>');
6 INSERT INTO xmltest VALUES (2, '<value>two</value>');
7 INSERT INTO xmltest VALUES (3, '<wrong');
8 ERROR:  invalid XML content
9 LINE 1: INSERT INTO xmltest VALUES (3, '<wrong');
10                                        ^
11 DETAIL:  line 1: Couldn't find end of Start Tag wrong line 1
12 SELECT * FROM xmltest;
13  id |        data        
14 ----+--------------------
15   1 | <value>one</value>
16   2 | <value>two</value>
17 (2 rows)
19 SELECT xmlcomment('test');
20  xmlcomment  
21 -------------
22  <!--test-->
23 (1 row)
25 SELECT xmlcomment('-test');
26   xmlcomment  
27 --------------
28  <!---test-->
29 (1 row)
31 SELECT xmlcomment('test-');
32 ERROR:  invalid XML comment
33 SELECT xmlcomment('--test');
34 ERROR:  invalid XML comment
35 SELECT xmlcomment('te st');
36   xmlcomment  
37 --------------
38  <!--te st-->
39 (1 row)
41 SELECT xmlconcat(xmlcomment('hello'),
42                  xmlelement(NAME qux, 'foo'),
43                  xmlcomment('world'));
44                xmlconcat                
45 ----------------------------------------
46  <!--hello--><qux>foo</qux><!--world-->
47 (1 row)
49 SELECT xmlconcat('hello', 'you');
50  xmlconcat 
51 -----------
52  helloyou
53 (1 row)
55 SELECT xmlconcat(1, 2);
56 ERROR:  argument of XMLCONCAT must be type xml, not type integer
57 LINE 1: SELECT xmlconcat(1, 2);
58                          ^
59 SELECT xmlconcat('bad', '<syntax');
60 ERROR:  invalid XML content
61 LINE 1: SELECT xmlconcat('bad', '<syntax');
62                                 ^
63 DETAIL:  line 1: Couldn't find end of Start Tag syntax line 1
64 SELECT xmlconcat('<foo/>', NULL, '<?xml version="1.1" standalone="no"?><bar/>');
65   xmlconcat   
66 --------------
67  <foo/><bar/>
68 (1 row)
70 SELECT xmlconcat('<?xml version="1.1"?><foo/>', NULL, '<?xml version="1.1" standalone="no"?><bar/>');
71              xmlconcat             
72 -----------------------------------
73  <?xml version="1.1"?><foo/><bar/>
74 (1 row)
76 SELECT xmlconcat(NULL);
77  xmlconcat 
78 -----------
80 (1 row)
82 SELECT xmlconcat(NULL, NULL);
83  xmlconcat 
84 -----------
86 (1 row)
88 SELECT xmlelement(name element,
89                   xmlattributes (1 as one, 'deuce' as two),
90                   'content');
91                    xmlelement                   
92 ------------------------------------------------
93  <element one="1" two="deuce">content</element>
94 (1 row)
96 SELECT xmlelement(name element,
97                   xmlattributes ('unnamed and wrong'));
98 ERROR:  unnamed XML attribute value must be a column reference
99 LINE 2:                   xmlattributes ('unnamed and wrong'));
100                                          ^
101 SELECT xmlelement(name element, xmlelement(name nested, 'stuff'));
102                 xmlelement                 
103 -------------------------------------------
104  <element><nested>stuff</nested></element>
105 (1 row)
107 SELECT xmlelement(name employee, xmlforest(name, age, salary as pay)) FROM emp;
108                               xmlelement                              
109 ----------------------------------------------------------------------
110  <employee><name>sharon</name><age>25</age><pay>1000</pay></employee>
111  <employee><name>sam</name><age>30</age><pay>2000</pay></employee>
112  <employee><name>bill</name><age>20</age><pay>1000</pay></employee>
113  <employee><name>jeff</name><age>23</age><pay>600</pay></employee>
114  <employee><name>cim</name><age>30</age><pay>400</pay></employee>
115  <employee><name>linda</name><age>19</age><pay>100</pay></employee>
116 (6 rows)
118 SELECT xmlelement(name duplicate, xmlattributes(1 as a, 2 as b, 3 as a));
119 ERROR:  XML attribute name "a" appears more than once
120 LINE 1: ...ment(name duplicate, xmlattributes(1 as a, 2 as b, 3 as a));
121                                                               ^
122 SELECT xmlelement(name num, 37);
123   xmlelement   
124 ---------------
125  <num>37</num>
126 (1 row)
128 SELECT xmlelement(name foo, text 'bar');
129    xmlelement   
130 ----------------
131  <foo>bar</foo>
132 (1 row)
134 SELECT xmlelement(name foo, xml 'bar');
135    xmlelement   
136 ----------------
137  <foo>bar</foo>
138 (1 row)
140 SELECT xmlelement(name foo, text 'b<a/>r');
141        xmlelement        
142 -------------------------
143  <foo>b&lt;a/&gt;r</foo>
144 (1 row)
146 SELECT xmlelement(name foo, xml 'b<a/>r');
147     xmlelement     
148 -------------------
149  <foo>b<a/>r</foo>
150 (1 row)
152 SELECT xmlelement(name foo, array[1, 2, 3]);
153                                xmlelement                                
154 -------------------------------------------------------------------------
155  <foo><element>1</element><element>2</element><element>3</element></foo>
156 (1 row)
158 SET xmlbinary TO base64;
159 SELECT xmlelement(name foo, bytea 'bar');
160    xmlelement    
161 -----------------
162  <foo>YmFy</foo>
163 (1 row)
165 SET xmlbinary TO hex;
166 SELECT xmlelement(name foo, bytea 'bar');
167     xmlelement     
168 -------------------
169  <foo>626172</foo>
170 (1 row)
172 SELECT xmlelement(name foo, xmlattributes(true as bar));
173     xmlelement     
174 -------------------
175  <foo bar="true"/>
176 (1 row)
178 SELECT xmlelement(name foo, xmlattributes('2009-04-09 00:24:37'::timestamp as bar));
179             xmlelement            
180 ----------------------------------
181  <foo bar="2009-04-09T00:24:37"/>
182 (1 row)
184 SELECT xmlelement(name foo, xmlattributes('infinity'::timestamp as bar));
185 ERROR:  timestamp out of range
186 DETAIL:  XML does not support infinite timestamp values.
187 SELECT xmlelement(name foo, xmlattributes('<>&"''' as funny, xml 'b<a/>r' as funnier));
188                          xmlelement                         
189 ------------------------------------------------------------
190  <foo funny="&lt;&gt;&amp;&quot;'" funnier="b&lt;a/&gt;r"/>
191 (1 row)
193 SELECT xmlparse(content '');
194  xmlparse 
195 ----------
197 (1 row)
199 SELECT xmlparse(content '  ');
200  xmlparse 
201 ----------
202    
203 (1 row)
205 SELECT xmlparse(content 'abc');
206  xmlparse 
207 ----------
208  abc
209 (1 row)
211 SELECT xmlparse(content '<abc>x</abc>');
212    xmlparse   
213 --------------
214  <abc>x</abc>
215 (1 row)
217 SELECT xmlparse(content '<invalidentity>&</invalidentity>');
218 ERROR:  invalid XML content
219 DETAIL:  line 1: xmlParseEntityRef: no name
220 <invalidentity>&</invalidentity>
221                 ^
222 line 1: chunk is not well balanced
223 SELECT xmlparse(content '<undefinedentity>&idontexist;</undefinedentity>');
224 ERROR:  invalid XML content
225 DETAIL:  line 1: Entity 'idontexist' not defined
226 <undefinedentity>&idontexist;</undefinedentity>
227                              ^
228 line 1: chunk is not well balanced
229 SELECT xmlparse(content '<invalidns xmlns=''&lt;''/>');
230          xmlparse          
231 ---------------------------
232  <invalidns xmlns='&lt;'/>
233 (1 row)
235 SELECT xmlparse(content '<relativens xmlns=''relative''/>');
236             xmlparse            
237 --------------------------------
238  <relativens xmlns='relative'/>
239 (1 row)
241 SELECT xmlparse(content '<twoerrors>&idontexist;</unbalanced>');
242 ERROR:  invalid XML content
243 DETAIL:  line 1: Entity 'idontexist' not defined
244 <twoerrors>&idontexist;</unbalanced>
245                        ^
246 line 1: Opening and ending tag mismatch: twoerrors line 1 and unbalanced
247 line 1: chunk is not well balanced
248 SELECT xmlparse(content '<nosuchprefix:tag/>');
249       xmlparse       
250 ---------------------
251  <nosuchprefix:tag/>
252 (1 row)
254 SELECT xmlparse(document '   ');
255 ERROR:  invalid XML document
256 DETAIL:  line 1: Start tag expected, '<' not found
257 SELECT xmlparse(document 'abc');
258 ERROR:  invalid XML document
259 DETAIL:  line 1: Start tag expected, '<' not found
262 SELECT xmlparse(document '<abc>x</abc>');
263    xmlparse   
264 --------------
265  <abc>x</abc>
266 (1 row)
268 SELECT xmlparse(document '<invalidentity>&</abc>');
269 ERROR:  invalid XML document
270 DETAIL:  line 1: xmlParseEntityRef: no name
271 <invalidentity>&</abc>
272                 ^
273 line 1: Opening and ending tag mismatch: invalidentity line 1 and abc
274 SELECT xmlparse(document '<undefinedentity>&idontexist;</abc>');
275 ERROR:  invalid XML document
276 DETAIL:  line 1: Entity 'idontexist' not defined
277 <undefinedentity>&idontexist;</abc>
278                              ^
279 line 1: Opening and ending tag mismatch: undefinedentity line 1 and abc
280 SELECT xmlparse(document '<invalidns xmlns=''&lt;''/>');
281          xmlparse          
282 ---------------------------
283  <invalidns xmlns='&lt;'/>
284 (1 row)
286 SELECT xmlparse(document '<relativens xmlns=''relative''/>');
287             xmlparse            
288 --------------------------------
289  <relativens xmlns='relative'/>
290 (1 row)
292 SELECT xmlparse(document '<twoerrors>&idontexist;</unbalanced>');
293 ERROR:  invalid XML document
294 DETAIL:  line 1: Entity 'idontexist' not defined
295 <twoerrors>&idontexist;</unbalanced>
296                        ^
297 line 1: Opening and ending tag mismatch: twoerrors line 1 and unbalanced
298 SELECT xmlparse(document '<nosuchprefix:tag/>');
299       xmlparse       
300 ---------------------
301  <nosuchprefix:tag/>
302 (1 row)
304 SELECT xmlpi(name foo);
305   xmlpi  
306 ---------
307  <?foo?>
308 (1 row)
310 SELECT xmlpi(name xml);
311 ERROR:  invalid XML processing instruction
312 DETAIL:  XML processing instruction target name cannot be "xml".
313 SELECT xmlpi(name xmlstuff);
314     xmlpi     
315 --------------
316  <?xmlstuff?>
317 (1 row)
319 SELECT xmlpi(name foo, 'bar');
320     xmlpi    
321 -------------
322  <?foo bar?>
323 (1 row)
325 SELECT xmlpi(name foo, 'in?>valid');
326 ERROR:  invalid XML processing instruction
327 DETAIL:  XML processing instruction cannot contain "?>".
328 SELECT xmlpi(name foo, null);
329  xmlpi 
330 -------
332 (1 row)
334 SELECT xmlpi(name xml, null);
335 ERROR:  invalid XML processing instruction
336 DETAIL:  XML processing instruction target name cannot be "xml".
337 SELECT xmlpi(name xmlstuff, null);
338  xmlpi 
339 -------
341 (1 row)
343 SELECT xmlpi(name "xml-stylesheet", 'href="mystyle.css" type="text/css"');
344                          xmlpi                         
345 -------------------------------------------------------
346  <?xml-stylesheet href="mystyle.css" type="text/css"?>
347 (1 row)
349 SELECT xmlpi(name foo, '   bar');
350     xmlpi    
351 -------------
352  <?foo bar?>
353 (1 row)
355 SELECT xmlroot(xml '<foo/>', version no value, standalone no value);
356  xmlroot 
357 ---------
358  <foo/>
359 (1 row)
361 SELECT xmlroot(xml '<foo/>', version '2.0');
362            xmlroot           
363 -----------------------------
364  <?xml version="2.0"?><foo/>
365 (1 row)
367 SELECT xmlroot(xml '<foo/>', version no value, standalone yes);
368                    xmlroot                    
369 ----------------------------------------------
370  <?xml version="1.0" standalone="yes"?><foo/>
371 (1 row)
373 SELECT xmlroot(xml '<?xml version="1.1"?><foo/>', version no value, standalone yes);
374                    xmlroot                    
375 ----------------------------------------------
376  <?xml version="1.0" standalone="yes"?><foo/>
377 (1 row)
379 SELECT xmlroot(xmlroot(xml '<foo/>', version '1.0'), version '1.1', standalone no);
380                    xmlroot                   
381 ---------------------------------------------
382  <?xml version="1.1" standalone="no"?><foo/>
383 (1 row)
385 SELECT xmlroot('<?xml version="1.1" standalone="yes"?><foo/>', version no value, standalone no);
386                    xmlroot                   
387 ---------------------------------------------
388  <?xml version="1.0" standalone="no"?><foo/>
389 (1 row)
391 SELECT xmlroot('<?xml version="1.1" standalone="yes"?><foo/>', version no value, standalone no value);
392  xmlroot 
393 ---------
394  <foo/>
395 (1 row)
397 SELECT xmlroot('<?xml version="1.1" standalone="yes"?><foo/>', version no value);
398                    xmlroot                    
399 ----------------------------------------------
400  <?xml version="1.0" standalone="yes"?><foo/>
401 (1 row)
403 SELECT xmlroot (
404   xmlelement (
405     name gazonk,
406     xmlattributes (
407       'val' AS name,
408       1 + 1 AS num
409     ),
410     xmlelement (
411       NAME qux,
412       'foo'
413     )
414   ),
415   version '1.0',
416   standalone yes
418                                          xmlroot                                          
419 ------------------------------------------------------------------------------------------
420  <?xml version="1.0" standalone="yes"?><gazonk name="val" num="2"><qux>foo</qux></gazonk>
421 (1 row)
423 SELECT xmlserialize(content data as character varying(20)) FROM xmltest;
424     xmlserialize    
425 --------------------
426  <value>one</value>
427  <value>two</value>
428 (2 rows)
430 SELECT xmlserialize(content 'good' as char(10));
431  xmlserialize 
432 --------------
433  good      
434 (1 row)
436 SELECT xmlserialize(document 'bad' as text);
437 ERROR:  not an XML document
438 SELECT xml '<foo>bar</foo>' IS DOCUMENT;
439  ?column? 
440 ----------
442 (1 row)
444 SELECT xml '<foo>bar</foo><bar>foo</bar>' IS DOCUMENT;
445  ?column? 
446 ----------
448 (1 row)
450 SELECT xml '<abc/>' IS NOT DOCUMENT;
451  ?column? 
452 ----------
454 (1 row)
456 SELECT xml 'abc' IS NOT DOCUMENT;
457  ?column? 
458 ----------
460 (1 row)
462 SELECT '<>' IS NOT DOCUMENT;
463 ERROR:  invalid XML content
464 LINE 1: SELECT '<>' IS NOT DOCUMENT;
465                ^
466 DETAIL:  line 1: StartTag: invalid element name
469 SELECT xmlagg(data) FROM xmltest;
470                 xmlagg                
471 --------------------------------------
472  <value>one</value><value>two</value>
473 (1 row)
475 SELECT xmlagg(data) FROM xmltest WHERE id > 10;
476  xmlagg 
477 --------
479 (1 row)
481 SELECT xmlelement(name employees, xmlagg(xmlelement(name name, name))) FROM emp;
482                                                            xmlelement                                                           
483 --------------------------------------------------------------------------------------------------------------------------------
484  <employees><name>sharon</name><name>sam</name><name>bill</name><name>jeff</name><name>cim</name><name>linda</name></employees>
485 (1 row)
487 -- Check mapping SQL identifier to XML name
488 SELECT xmlpi(name ":::_xml_abc135.%-&_");
489                       xmlpi                      
490 -------------------------------------------------
491  <?_x003A_::_x005F_xml_abc135._x0025_-_x0026__?>
492 (1 row)
494 SELECT xmlpi(name "123");
495      xmlpi     
496 ---------------
497  <?_x0031_23?>
498 (1 row)
500 PREPARE foo (xml) AS SELECT xmlconcat('<foo/>', $1);
501 SET XML OPTION DOCUMENT;
502 EXECUTE foo ('<bar/>');
503   xmlconcat   
504 --------------
505  <foo/><bar/>
506 (1 row)
508 EXECUTE foo ('bad');
509 ERROR:  invalid XML document
510 LINE 1: EXECUTE foo ('bad');
511                      ^
512 DETAIL:  line 1: Start tag expected, '<' not found
515 SELECT xml '<!DOCTYPE a><a/><b/>';
516 ERROR:  invalid XML document
517 LINE 1: SELECT xml '<!DOCTYPE a><a/><b/>';
518                    ^
519 DETAIL:  line 1: Extra content at the end of the document
520 <!DOCTYPE a><a/><b/>
521                 ^
522 SET XML OPTION CONTENT;
523 EXECUTE foo ('<bar/>');
524   xmlconcat   
525 --------------
526  <foo/><bar/>
527 (1 row)
529 EXECUTE foo ('good');
530  xmlconcat  
531 ------------
532  <foo/>good
533 (1 row)
535 SELECT xml '<!-- in SQL:2006+ a doc is content too--> <?y z?> <!DOCTYPE a><a/>';
536                                 xml                                 
537 --------------------------------------------------------------------
538  <!-- in SQL:2006+ a doc is content too--> <?y z?> <!DOCTYPE a><a/>
539 (1 row)
541 SELECT xml '<?xml version="1.0"?> <!-- hi--> <!DOCTYPE a><a/>';
542              xml              
543 ------------------------------
544   <!-- hi--> <!DOCTYPE a><a/>
545 (1 row)
547 SELECT xml '<!DOCTYPE a><a/>';
548        xml        
549 ------------------
550  <!DOCTYPE a><a/>
551 (1 row)
553 SELECT xml '<!-- hi--> oops <!DOCTYPE a><a/>';
554 ERROR:  invalid XML content
555 LINE 1: SELECT xml '<!-- hi--> oops <!DOCTYPE a><a/>';
556                    ^
557 DETAIL:  line 1: StartTag: invalid element name
558 <!-- hi--> oops <!DOCTYPE a><a/>
559                  ^
560 SELECT xml '<!-- hi--> <oops/> <!DOCTYPE a><a/>';
561 ERROR:  invalid XML content
562 LINE 1: SELECT xml '<!-- hi--> <oops/> <!DOCTYPE a><a/>';
563                    ^
564 DETAIL:  line 1: StartTag: invalid element name
565 <!-- hi--> <oops/> <!DOCTYPE a><a/>
566                     ^
567 SELECT xml '<!DOCTYPE a><a/><b/>';
568 ERROR:  invalid XML content
569 LINE 1: SELECT xml '<!DOCTYPE a><a/><b/>';
570                    ^
571 DETAIL:  line 1: Extra content at the end of the document
572 <!DOCTYPE a><a/><b/>
573                 ^
574 -- Test backwards parsing
575 CREATE VIEW xmlview1 AS SELECT xmlcomment('test');
576 CREATE VIEW xmlview2 AS SELECT xmlconcat('hello', 'you');
577 CREATE VIEW xmlview3 AS SELECT xmlelement(name element, xmlattributes (1 as ":one:", 'deuce' as two), 'content&');
578 CREATE VIEW xmlview4 AS SELECT xmlelement(name employee, xmlforest(name, age, salary as pay)) FROM emp;
579 CREATE VIEW xmlview5 AS SELECT xmlparse(content '<abc>x</abc>');
580 CREATE VIEW xmlview6 AS SELECT xmlpi(name foo, 'bar');
581 CREATE VIEW xmlview7 AS SELECT xmlroot(xml '<foo/>', version no value, standalone yes);
582 CREATE VIEW xmlview8 AS SELECT xmlserialize(content 'good' as char(10));
583 CREATE VIEW xmlview9 AS SELECT xmlserialize(content 'good' as text);
584 SELECT table_name, view_definition FROM information_schema.views
585   WHERE table_name LIKE 'xmlview%' ORDER BY 1;
586  table_name |                                                  view_definition                                                  
587 ------------+-------------------------------------------------------------------------------------------------------------------
588  xmlview1   |  SELECT xmlcomment('test'::text) AS xmlcomment;
589  xmlview2   |  SELECT XMLCONCAT('hello'::xml, 'you'::xml) AS "xmlconcat";
590  xmlview3   |  SELECT XMLELEMENT(NAME element, XMLATTRIBUTES(1 AS ":one:", 'deuce' AS two), 'content&') AS "xmlelement";
591  xmlview4   |  SELECT XMLELEMENT(NAME employee, XMLFOREST(emp.name AS name, emp.age AS age, emp.salary AS pay)) AS "xmlelement"+
592             |    FROM emp;
593  xmlview5   |  SELECT XMLPARSE(CONTENT '<abc>x</abc>'::text STRIP WHITESPACE) AS "xmlparse";
594  xmlview6   |  SELECT XMLPI(NAME foo, 'bar'::text) AS "xmlpi";
595  xmlview7   |  SELECT XMLROOT('<foo/>'::xml, VERSION NO VALUE, STANDALONE YES) AS "xmlroot";
596  xmlview8   |  SELECT (XMLSERIALIZE(CONTENT 'good'::xml AS character(10)))::character(10) AS "xmlserialize";
597  xmlview9   |  SELECT XMLSERIALIZE(CONTENT 'good'::xml AS text) AS "xmlserialize";
598 (9 rows)
600 -- Text XPath expressions evaluation
601 SELECT xpath('/value', data) FROM xmltest;
602         xpath         
603 ----------------------
604  {<value>one</value>}
605  {<value>two</value>}
606 (2 rows)
608 SELECT xpath(NULL, NULL) IS NULL FROM xmltest;
609  ?column? 
610 ----------
613 (2 rows)
615 SELECT xpath('', '<!-- error -->');
616 ERROR:  empty XPath expression
617 CONTEXT:  SQL function "xpath" statement 1
618 SELECT xpath('//text()', '<local:data xmlns:local="http://127.0.0.1"><local:piece id="1">number one</local:piece><local:piece id="2" /></local:data>');
619      xpath      
620 ----------------
621  {"number one"}
622 (1 row)
624 SELECT xpath('//loc:piece/@id', '<local:data xmlns:local="http://127.0.0.1"><local:piece id="1">number one</local:piece><local:piece id="2" /></local:data>', ARRAY[ARRAY['loc', 'http://127.0.0.1']]);
625  xpath 
626 -------
627  {1,2}
628 (1 row)
630 SELECT xpath('//loc:piece', '<local:data xmlns:local="http://127.0.0.1"><local:piece id="1">number one</local:piece><local:piece id="2" /></local:data>', ARRAY[ARRAY['loc', 'http://127.0.0.1']]);
631                                                                      xpath                                                                      
632 ------------------------------------------------------------------------------------------------------------------------------------------------
633  {"<local:piece xmlns:local=\"http://127.0.0.1\" id=\"1\">number one</local:piece>","<local:piece xmlns:local=\"http://127.0.0.1\" id=\"2\"/>"}
634 (1 row)
636 SELECT xpath('//loc:piece', '<local:data xmlns:local="http://127.0.0.1" xmlns="http://127.0.0.2"><local:piece id="1"><internal>number one</internal><internal2/></local:piece><local:piece id="2" /></local:data>', ARRAY[ARRAY['loc', 'http://127.0.0.1']]);
637                                                                                                    xpath                                                                                                    
638 ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
639  {"<local:piece xmlns:local=\"http://127.0.0.1\" xmlns=\"http://127.0.0.2\" id=\"1\"><internal>number one</internal><internal2/></local:piece>","<local:piece xmlns:local=\"http://127.0.0.1\" id=\"2\"/>"}
640 (1 row)
642 SELECT xpath('//b', '<a>one <b>two</b> three <b>etc</b></a>');
643           xpath          
644 -------------------------
645  {<b>two</b>,<b>etc</b>}
646 (1 row)
648 SELECT xpath('//text()', '<root>&lt;</root>');
649  xpath  
650 --------
651  {&lt;}
652 (1 row)
654 SELECT xpath('//@value', '<root value="&lt;"/>');
655  xpath  
656 --------
657  {&lt;}
658 (1 row)
660 SELECT xpath('''<<invalid>>''', '<root/>');
661            xpath           
662 ---------------------------
663  {&lt;&lt;invalid&gt;&gt;}
664 (1 row)
666 SELECT xpath('count(//*)', '<root><sub/><sub/></root>');
667  xpath 
668 -------
669  {3}
670 (1 row)
672 SELECT xpath('count(//*)=0', '<root><sub/><sub/></root>');
673   xpath  
674 ---------
675  {false}
676 (1 row)
678 SELECT xpath('count(//*)=3', '<root><sub/><sub/></root>');
679  xpath  
680 --------
681  {true}
682 (1 row)
684 SELECT xpath('name(/*)', '<root><sub/><sub/></root>');
685  xpath  
686 --------
687  {root}
688 (1 row)
690 SELECT xpath('/nosuchtag', '<root/>');
691  xpath 
692 -------
693  {}
694 (1 row)
696 SELECT xpath('root', '<root/>');
697    xpath   
698 -----------
699  {<root/>}
700 (1 row)
702 -- Round-trip non-ASCII data through xpath().
703 DO $$
704 DECLARE
705   xml_declaration text := '<?xml version="1.0" encoding="ISO-8859-1"?>';
706   degree_symbol text;
707   res xml[];
708 BEGIN
709   -- Per the documentation, except when the server encoding is UTF8, xpath()
710   -- may not work on non-ASCII data.  The untranslatable_character and
711   -- undefined_function traps below, currently dead code, will become relevant
712   -- if we remove this limitation.
713   IF current_setting('server_encoding') <> 'UTF8' THEN
714     RAISE LOG 'skip: encoding % unsupported for xpath',
715       current_setting('server_encoding');
716     RETURN;
717   END IF;
719   degree_symbol := convert_from('\xc2b0', 'UTF8');
720   res := xpath('text()', (xml_declaration ||
721     '<x>' || degree_symbol || '</x>')::xml);
722   IF degree_symbol <> res[1]::text THEN
723     RAISE 'expected % (%), got % (%)',
724       degree_symbol, convert_to(degree_symbol, 'UTF8'),
725       res[1], convert_to(res[1]::text, 'UTF8');
726   END IF;
727 EXCEPTION
728   -- character with byte sequence 0xc2 0xb0 in encoding "UTF8" has no equivalent in encoding "LATIN8"
729   WHEN untranslatable_character
730   -- default conversion function for encoding "UTF8" to "MULE_INTERNAL" does not exist
731   OR undefined_function
732   -- unsupported XML feature
733   OR feature_not_supported THEN
734     RAISE LOG 'skip: %', SQLERRM;
737 -- Test xmlexists and xpath_exists
738 SELECT xmlexists('//town[text() = ''Toronto'']' PASSING BY REF '<towns><town>Bidford-on-Avon</town><town>Cwmbran</town><town>Bristol</town></towns>');
739  xmlexists 
740 -----------
742 (1 row)
744 SELECT xmlexists('//town[text() = ''Cwmbran'']' PASSING BY REF '<towns><town>Bidford-on-Avon</town><town>Cwmbran</town><town>Bristol</town></towns>');
745  xmlexists 
746 -----------
748 (1 row)
750 SELECT xmlexists('count(/nosuchtag)' PASSING BY REF '<root/>');
751  xmlexists 
752 -----------
754 (1 row)
756 SELECT xpath_exists('//town[text() = ''Toronto'']','<towns><town>Bidford-on-Avon</town><town>Cwmbran</town><town>Bristol</town></towns>'::xml);
757  xpath_exists 
758 --------------
760 (1 row)
762 SELECT xpath_exists('//town[text() = ''Cwmbran'']','<towns><town>Bidford-on-Avon</town><town>Cwmbran</town><town>Bristol</town></towns>'::xml);
763  xpath_exists 
764 --------------
766 (1 row)
768 SELECT xpath_exists('count(/nosuchtag)', '<root/>'::xml);
769  xpath_exists 
770 --------------
772 (1 row)
774 INSERT INTO xmltest VALUES (4, '<menu><beers><name>Budvar</name><cost>free</cost><name>Carling</name><cost>lots</cost></beers></menu>'::xml);
775 INSERT INTO xmltest VALUES (5, '<menu><beers><name>Molson</name><cost>free</cost><name>Carling</name><cost>lots</cost></beers></menu>'::xml);
776 INSERT INTO xmltest VALUES (6, '<myns:menu xmlns:myns="http://myns.com"><myns:beers><myns:name>Budvar</myns:name><myns:cost>free</myns:cost><myns:name>Carling</myns:name><myns:cost>lots</myns:cost></myns:beers></myns:menu>'::xml);
777 INSERT INTO xmltest VALUES (7, '<myns:menu xmlns:myns="http://myns.com"><myns:beers><myns:name>Molson</myns:name><myns:cost>free</myns:cost><myns:name>Carling</myns:name><myns:cost>lots</myns:cost></myns:beers></myns:menu>'::xml);
778 SELECT COUNT(id) FROM xmltest WHERE xmlexists('/menu/beer' PASSING data);
779  count 
780 -------
781      0
782 (1 row)
784 SELECT COUNT(id) FROM xmltest WHERE xmlexists('/menu/beer' PASSING BY REF data BY REF);
785  count 
786 -------
787      0
788 (1 row)
790 SELECT COUNT(id) FROM xmltest WHERE xmlexists('/menu/beers' PASSING BY REF data);
791  count 
792 -------
793      2
794 (1 row)
796 SELECT COUNT(id) FROM xmltest WHERE xmlexists('/menu/beers/name[text() = ''Molson'']' PASSING BY REF data);
797  count 
798 -------
799      1
800 (1 row)
802 SELECT COUNT(id) FROM xmltest WHERE xpath_exists('/menu/beer',data);
803  count 
804 -------
805      0
806 (1 row)
808 SELECT COUNT(id) FROM xmltest WHERE xpath_exists('/menu/beers',data);
809  count 
810 -------
811      2
812 (1 row)
814 SELECT COUNT(id) FROM xmltest WHERE xpath_exists('/menu/beers/name[text() = ''Molson'']',data);
815  count 
816 -------
817      1
818 (1 row)
820 SELECT COUNT(id) FROM xmltest WHERE xpath_exists('/myns:menu/myns:beer',data,ARRAY[ARRAY['myns','http://myns.com']]);
821  count 
822 -------
823      0
824 (1 row)
826 SELECT COUNT(id) FROM xmltest WHERE xpath_exists('/myns:menu/myns:beers',data,ARRAY[ARRAY['myns','http://myns.com']]);
827  count 
828 -------
829      2
830 (1 row)
832 SELECT COUNT(id) FROM xmltest WHERE xpath_exists('/myns:menu/myns:beers/myns:name[text() = ''Molson'']',data,ARRAY[ARRAY['myns','http://myns.com']]);
833  count 
834 -------
835      1
836 (1 row)
838 CREATE TABLE query ( expr TEXT );
839 INSERT INTO query VALUES ('/menu/beers/cost[text() = ''lots'']');
840 SELECT COUNT(id) FROM xmltest, query WHERE xmlexists(expr PASSING BY REF data);
841  count 
842 -------
843      2
844 (1 row)
846 -- Test xml_is_well_formed and variants
847 SELECT xml_is_well_formed_document('<foo>bar</foo>');
848  xml_is_well_formed_document 
849 -----------------------------
851 (1 row)
853 SELECT xml_is_well_formed_document('abc');
854  xml_is_well_formed_document 
855 -----------------------------
857 (1 row)
859 SELECT xml_is_well_formed_content('<foo>bar</foo>');
860  xml_is_well_formed_content 
861 ----------------------------
863 (1 row)
865 SELECT xml_is_well_formed_content('abc');
866  xml_is_well_formed_content 
867 ----------------------------
869 (1 row)
871 SET xmloption TO DOCUMENT;
872 SELECT xml_is_well_formed('abc');
873  xml_is_well_formed 
874 --------------------
876 (1 row)
878 SELECT xml_is_well_formed('<>');
879  xml_is_well_formed 
880 --------------------
882 (1 row)
884 SELECT xml_is_well_formed('<abc/>');
885  xml_is_well_formed 
886 --------------------
888 (1 row)
890 SELECT xml_is_well_formed('<foo>bar</foo>');
891  xml_is_well_formed 
892 --------------------
894 (1 row)
896 SELECT xml_is_well_formed('<foo>bar</foo');
897  xml_is_well_formed 
898 --------------------
900 (1 row)
902 SELECT xml_is_well_formed('<foo><bar>baz</foo>');
903  xml_is_well_formed 
904 --------------------
906 (1 row)
908 SELECT xml_is_well_formed('<local:data xmlns:local="http://127.0.0.1"><local:piece id="1">number one</local:piece><local:piece id="2" /></local:data>');
909  xml_is_well_formed 
910 --------------------
912 (1 row)
914 SELECT xml_is_well_formed('<pg:foo xmlns:pg="http://postgresql.org/stuff">bar</my:foo>');
915  xml_is_well_formed 
916 --------------------
918 (1 row)
920 SELECT xml_is_well_formed('<pg:foo xmlns:pg="http://postgresql.org/stuff">bar</pg:foo>');
921  xml_is_well_formed 
922 --------------------
924 (1 row)
926 SELECT xml_is_well_formed('<invalidentity>&</abc>');
927  xml_is_well_formed 
928 --------------------
930 (1 row)
932 SELECT xml_is_well_formed('<undefinedentity>&idontexist;</abc>');
933  xml_is_well_formed 
934 --------------------
936 (1 row)
938 SELECT xml_is_well_formed('<invalidns xmlns=''&lt;''/>');
939  xml_is_well_formed 
940 --------------------
942 (1 row)
944 SELECT xml_is_well_formed('<relativens xmlns=''relative''/>');
945  xml_is_well_formed 
946 --------------------
948 (1 row)
950 SELECT xml_is_well_formed('<twoerrors>&idontexist;</unbalanced>');
951  xml_is_well_formed 
952 --------------------
954 (1 row)
956 SET xmloption TO CONTENT;
957 SELECT xml_is_well_formed('abc');
958  xml_is_well_formed 
959 --------------------
961 (1 row)
963 -- Since xpath() deals with namespaces, it's a bit stricter about
964 -- what's well-formed and what's not. If we don't obey these rules
965 -- (i.e. ignore namespace-related errors from libxml), xpath()
966 -- fails in subtle ways. The following would for example produce
967 -- the xml value
968 --   <invalidns xmlns='<'/>
969 -- which is invalid because '<' may not appear un-escaped in
970 -- attribute values.
971 -- Since different libxml versions emit slightly different
972 -- error messages, we suppress the DETAIL in this test.
973 \set VERBOSITY terse
974 SELECT xpath('/*', '<invalidns xmlns=''&lt;''/>');
975 ERROR:  could not parse XML document
976 \set VERBOSITY default
977 -- Again, the XML isn't well-formed for namespace purposes
978 SELECT xpath('/*', '<nosuchprefix:tag/>');
979 ERROR:  could not parse XML document
980 DETAIL:  line 1: Namespace prefix nosuchprefix on tag is not defined
981 <nosuchprefix:tag/>
982                  ^
983 CONTEXT:  SQL function "xpath" statement 1
984 -- XPath deprecates relative namespaces, but they're not supposed to
985 -- throw an error, only a warning.
986 SELECT xpath('/*', '<relativens xmlns=''relative''/>');
987 WARNING:  line 1: xmlns: URI relative is not absolute
988 <relativens xmlns='relative'/>
989                             ^
990                 xpath                 
991 --------------------------------------
992  {"<relativens xmlns=\"relative\"/>"}
993 (1 row)
995 -- External entity references should not leak filesystem information.
996 SELECT XMLPARSE(DOCUMENT '<!DOCTYPE foo [<!ENTITY c SYSTEM "/etc/passwd">]><foo>&c;</foo>');
997                             xmlparse                             
998 -----------------------------------------------------------------
999  <!DOCTYPE foo [<!ENTITY c SYSTEM "/etc/passwd">]><foo>&c;</foo>
1000 (1 row)
1002 SELECT XMLPARSE(DOCUMENT '<!DOCTYPE foo [<!ENTITY c SYSTEM "/etc/no.such.file">]><foo>&c;</foo>');
1003                                xmlparse                                
1004 -----------------------------------------------------------------------
1005  <!DOCTYPE foo [<!ENTITY c SYSTEM "/etc/no.such.file">]><foo>&c;</foo>
1006 (1 row)
1008 -- This might or might not load the requested DTD, but it mustn't throw error.
1009 SELECT XMLPARSE(DOCUMENT '<!DOCTYPE chapter PUBLIC "-//OASIS//DTD DocBook XML V4.1.2//EN" "http://www.oasis-open.org/docbook/xml/4.1.2/docbookx.dtd"><chapter>&nbsp;</chapter>');
1010                                                                        xmlparse                                                                       
1011 ------------------------------------------------------------------------------------------------------------------------------------------------------
1012  <!DOCTYPE chapter PUBLIC "-//OASIS//DTD DocBook XML V4.1.2//EN" "http://www.oasis-open.org/docbook/xml/4.1.2/docbookx.dtd"><chapter>&nbsp;</chapter>
1013 (1 row)
1015 -- XMLPATH tests
1016 CREATE TABLE xmldata(data xml);
1017 INSERT INTO xmldata VALUES('<ROWS>
1018 <ROW id="1">
1019   <COUNTRY_ID>AU</COUNTRY_ID>
1020   <COUNTRY_NAME>Australia</COUNTRY_NAME>
1021   <REGION_ID>3</REGION_ID>
1022 </ROW>
1023 <ROW id="2">
1024   <COUNTRY_ID>CN</COUNTRY_ID>
1025   <COUNTRY_NAME>China</COUNTRY_NAME>
1026   <REGION_ID>3</REGION_ID>
1027 </ROW>
1028 <ROW id="3">
1029   <COUNTRY_ID>HK</COUNTRY_ID>
1030   <COUNTRY_NAME>HongKong</COUNTRY_NAME>
1031   <REGION_ID>3</REGION_ID>
1032 </ROW>
1033 <ROW id="4">
1034   <COUNTRY_ID>IN</COUNTRY_ID>
1035   <COUNTRY_NAME>India</COUNTRY_NAME>
1036   <REGION_ID>3</REGION_ID>
1037 </ROW>
1038 <ROW id="5">
1039   <COUNTRY_ID>JP</COUNTRY_ID>
1040   <COUNTRY_NAME>Japan</COUNTRY_NAME>
1041   <REGION_ID>3</REGION_ID><PREMIER_NAME>Sinzo Abe</PREMIER_NAME>
1042 </ROW>
1043 <ROW id="6">
1044   <COUNTRY_ID>SG</COUNTRY_ID>
1045   <COUNTRY_NAME>Singapore</COUNTRY_NAME>
1046   <REGION_ID>3</REGION_ID><SIZE unit="km">791</SIZE>
1047 </ROW>
1048 </ROWS>');
1049 -- XMLTABLE with columns
1050 SELECT  xmltable.*
1051    FROM (SELECT data FROM xmldata) x,
1052         LATERAL XMLTABLE('/ROWS/ROW'
1053                          PASSING data
1054                          COLUMNS id int PATH '@id',
1055                                   _id FOR ORDINALITY,
1056                                   country_name text PATH 'COUNTRY_NAME/text()' NOT NULL,
1057                                   country_id text PATH 'COUNTRY_ID',
1058                                   region_id int PATH 'REGION_ID',
1059                                   size float PATH 'SIZE',
1060                                   unit text PATH 'SIZE/@unit',
1061                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified');
1062  id | _id | country_name | country_id | region_id | size | unit | premier_name  
1063 ----+-----+--------------+------------+-----------+------+------+---------------
1064   1 |   1 | Australia    | AU         |         3 |      |      | not specified
1065   2 |   2 | China        | CN         |         3 |      |      | not specified
1066   3 |   3 | HongKong     | HK         |         3 |      |      | not specified
1067   4 |   4 | India        | IN         |         3 |      |      | not specified
1068   5 |   5 | Japan        | JP         |         3 |      |      | Sinzo Abe
1069   6 |   6 | Singapore    | SG         |         3 |  791 | km   | not specified
1070 (6 rows)
1072 CREATE VIEW xmltableview1 AS SELECT  xmltable.*
1073    FROM (SELECT data FROM xmldata) x,
1074         LATERAL XMLTABLE('/ROWS/ROW'
1075                          PASSING data
1076                          COLUMNS id int PATH '@id',
1077                                   _id FOR ORDINALITY,
1078                                   country_name text PATH 'COUNTRY_NAME/text()' NOT NULL,
1079                                   country_id text PATH 'COUNTRY_ID',
1080                                   region_id int PATH 'REGION_ID',
1081                                   size float PATH 'SIZE',
1082                                   unit text PATH 'SIZE/@unit',
1083                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified');
1084 SELECT * FROM xmltableview1;
1085  id | _id | country_name | country_id | region_id | size | unit | premier_name  
1086 ----+-----+--------------+------------+-----------+------+------+---------------
1087   1 |   1 | Australia    | AU         |         3 |      |      | not specified
1088   2 |   2 | China        | CN         |         3 |      |      | not specified
1089   3 |   3 | HongKong     | HK         |         3 |      |      | not specified
1090   4 |   4 | India        | IN         |         3 |      |      | not specified
1091   5 |   5 | Japan        | JP         |         3 |      |      | Sinzo Abe
1092   6 |   6 | Singapore    | SG         |         3 |  791 | km   | not specified
1093 (6 rows)
1095 \sv xmltableview1
1096 CREATE OR REPLACE VIEW public.xmltableview1 AS
1097  SELECT "xmltable".id,
1098     "xmltable"._id,
1099     "xmltable".country_name,
1100     "xmltable".country_id,
1101     "xmltable".region_id,
1102     "xmltable".size,
1103     "xmltable".unit,
1104     "xmltable".premier_name
1105    FROM ( SELECT xmldata.data
1106            FROM xmldata) x,
1107     LATERAL XMLTABLE(('/ROWS/ROW'::text) PASSING (x.data) COLUMNS id integer PATH ('@id'::text), _id FOR ORDINALITY, country_name text PATH ('COUNTRY_NAME/text()'::text) NOT NULL, country_id text PATH ('COUNTRY_ID'::text), region_id integer PATH ('REGION_ID'::text), size double precision PATH ('SIZE'::text), unit text PATH ('SIZE/@unit'::text), premier_name text DEFAULT ('not specified'::text) PATH ('PREMIER_NAME'::text))
1108 EXPLAIN (COSTS OFF) SELECT * FROM xmltableview1;
1109                QUERY PLAN                
1110 -----------------------------------------
1111  Nested Loop
1112    ->  Seq Scan on xmldata
1113    ->  Table Function Scan on "xmltable"
1114 (3 rows)
1116 EXPLAIN (COSTS OFF, VERBOSE) SELECT * FROM xmltableview1;
1117                                                                                                                                                                                                                             QUERY PLAN                                                                                                                                                                                                                            
1118 ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
1119  Nested Loop
1120    Output: "xmltable".id, "xmltable"._id, "xmltable".country_name, "xmltable".country_id, "xmltable".region_id, "xmltable".size, "xmltable".unit, "xmltable".premier_name
1121    ->  Seq Scan on public.xmldata
1122          Output: xmldata.data
1123    ->  Table Function Scan on "xmltable"
1124          Output: "xmltable".id, "xmltable"._id, "xmltable".country_name, "xmltable".country_id, "xmltable".region_id, "xmltable".size, "xmltable".unit, "xmltable".premier_name
1125          Table Function Call: XMLTABLE(('/ROWS/ROW'::text) PASSING (xmldata.data) COLUMNS id integer PATH ('@id'::text), _id FOR ORDINALITY, country_name text PATH ('COUNTRY_NAME/text()'::text) NOT NULL, country_id text PATH ('COUNTRY_ID'::text), region_id integer PATH ('REGION_ID'::text), size double precision PATH ('SIZE'::text), unit text PATH ('SIZE/@unit'::text), premier_name text DEFAULT ('not specified'::text) PATH ('PREMIER_NAME'::text))
1126 (7 rows)
1128 -- XMLNAMESPACES tests
1129 SELECT * FROM XMLTABLE(XMLNAMESPACES('http://x.y' AS zz),
1130                       '/zz:rows/zz:row'
1131                       PASSING '<rows xmlns="http://x.y"><row><a>10</a></row></rows>'
1132                       COLUMNS a int PATH 'zz:a');
1133  a  
1134 ----
1135  10
1136 (1 row)
1138 CREATE VIEW xmltableview2 AS SELECT * FROM XMLTABLE(XMLNAMESPACES('http://x.y' AS zz),
1139                       '/zz:rows/zz:row'
1140                       PASSING '<rows xmlns="http://x.y"><row><a>10</a></row></rows>'
1141                       COLUMNS a int PATH 'zz:a');
1142 SELECT * FROM xmltableview2;
1143  a  
1144 ----
1145  10
1146 (1 row)
1148 SELECT * FROM XMLTABLE(XMLNAMESPACES(DEFAULT 'http://x.y'),
1149                       '/rows/row'
1150                       PASSING '<rows xmlns="http://x.y"><row><a>10</a></row></rows>'
1151                       COLUMNS a int PATH 'a');
1152 ERROR:  DEFAULT namespace is not supported
1153 SELECT * FROM XMLTABLE('.'
1154                        PASSING '<foo/>'
1155                        COLUMNS a text PATH 'foo/namespace::node()');
1156                   a                   
1157 --------------------------------------
1158  http://www.w3.org/XML/1998/namespace
1159 (1 row)
1161 -- used in prepare statements
1162 PREPARE pp AS
1163 SELECT  xmltable.*
1164    FROM (SELECT data FROM xmldata) x,
1165         LATERAL XMLTABLE('/ROWS/ROW'
1166                          PASSING data
1167                          COLUMNS id int PATH '@id',
1168                                   _id FOR ORDINALITY,
1169                                   country_name text PATH 'COUNTRY_NAME' NOT NULL,
1170                                   country_id text PATH 'COUNTRY_ID',
1171                                   region_id int PATH 'REGION_ID',
1172                                   size float PATH 'SIZE',
1173                                   unit text PATH 'SIZE/@unit',
1174                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified');
1175 EXECUTE pp;
1176  id | _id | country_name | country_id | region_id | size | unit | premier_name  
1177 ----+-----+--------------+------------+-----------+------+------+---------------
1178   1 |   1 | Australia    | AU         |         3 |      |      | not specified
1179   2 |   2 | China        | CN         |         3 |      |      | not specified
1180   3 |   3 | HongKong     | HK         |         3 |      |      | not specified
1181   4 |   4 | India        | IN         |         3 |      |      | not specified
1182   5 |   5 | Japan        | JP         |         3 |      |      | Sinzo Abe
1183   6 |   6 | Singapore    | SG         |         3 |  791 | km   | not specified
1184 (6 rows)
1186 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS "COUNTRY_NAME" text, "REGION_ID" int);
1187  COUNTRY_NAME | REGION_ID 
1188 --------------+-----------
1189  India        |         3
1190  Japan        |         3
1191 (2 rows)
1193 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS id FOR ORDINALITY, "COUNTRY_NAME" text, "REGION_ID" int);
1194  id | COUNTRY_NAME | REGION_ID 
1195 ----+--------------+-----------
1196   1 | India        |         3
1197   2 | Japan        |         3
1198 (2 rows)
1200 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS id int PATH '@id', "COUNTRY_NAME" text, "REGION_ID" int);
1201  id | COUNTRY_NAME | REGION_ID 
1202 ----+--------------+-----------
1203   4 | India        |         3
1204   5 | Japan        |         3
1205 (2 rows)
1207 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS id int PATH '@id');
1208  id 
1209 ----
1210   4
1211   5
1212 (2 rows)
1214 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS id FOR ORDINALITY);
1215  id 
1216 ----
1217   1
1218   2
1219 (2 rows)
1221 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS id int PATH '@id', "COUNTRY_NAME" text, "REGION_ID" int, rawdata xml PATH '.');
1222  id | COUNTRY_NAME | REGION_ID |                             rawdata                              
1223 ----+--------------+-----------+------------------------------------------------------------------
1224   4 | India        |         3 | <ROW id="4">                                                    +
1225     |              |           |   <COUNTRY_ID>IN</COUNTRY_ID>                                   +
1226     |              |           |   <COUNTRY_NAME>India</COUNTRY_NAME>                            +
1227     |              |           |   <REGION_ID>3</REGION_ID>                                      +
1228     |              |           | </ROW>
1229   5 | Japan        |         3 | <ROW id="5">                                                    +
1230     |              |           |   <COUNTRY_ID>JP</COUNTRY_ID>                                   +
1231     |              |           |   <COUNTRY_NAME>Japan</COUNTRY_NAME>                            +
1232     |              |           |   <REGION_ID>3</REGION_ID><PREMIER_NAME>Sinzo Abe</PREMIER_NAME>+
1233     |              |           | </ROW>
1234 (2 rows)
1236 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS id int PATH '@id', "COUNTRY_NAME" text, "REGION_ID" int, rawdata xml PATH './*');
1237  id | COUNTRY_NAME | REGION_ID |                                                           rawdata                                                           
1238 ----+--------------+-----------+-----------------------------------------------------------------------------------------------------------------------------
1239   4 | India        |         3 | <COUNTRY_ID>IN</COUNTRY_ID><COUNTRY_NAME>India</COUNTRY_NAME><REGION_ID>3</REGION_ID>
1240   5 | Japan        |         3 | <COUNTRY_ID>JP</COUNTRY_ID><COUNTRY_NAME>Japan</COUNTRY_NAME><REGION_ID>3</REGION_ID><PREMIER_NAME>Sinzo Abe</PREMIER_NAME>
1241 (2 rows)
1243 SELECT * FROM xmltable('/root' passing '<root><element>a1a<!-- aaaa -->a2a<?aaaaa?> <!--z-->  bbbb<x>xxx</x>cccc</element></root>' COLUMNS element text);
1244        element        
1245 ----------------------
1246  a1aa2a   bbbbxxxcccc
1247 (1 row)
1249 SELECT * FROM xmltable('/root' passing '<root><element>a1a<!-- aaaa -->a2a<?aaaaa?> <!--z-->  bbbb<x>xxx</x>cccc</element></root>' COLUMNS element text PATH 'element/text()'); -- should fail
1250 ERROR:  more than one value returned by column XPath expression
1251 -- CDATA test
1252 select * from xmltable('d/r' passing '<d><r><c><![CDATA[<hello> &"<>!<a>foo</a>]]></c></r><r><c>2</c></r></d>' columns c text);
1253             c            
1254 -------------------------
1255  <hello> &"<>!<a>foo</a>
1257 (2 rows)
1259 -- XML builtin entities
1260 SELECT * FROM xmltable('/x/a' PASSING '<x><a><ent>&apos;</ent></a><a><ent>&quot;</ent></a><a><ent>&amp;</ent></a><a><ent>&lt;</ent></a><a><ent>&gt;</ent></a></x>' COLUMNS ent text);
1261  ent 
1262 -----
1268 (5 rows)
1270 SELECT * FROM xmltable('/x/a' PASSING '<x><a><ent>&apos;</ent></a><a><ent>&quot;</ent></a><a><ent>&amp;</ent></a><a><ent>&lt;</ent></a><a><ent>&gt;</ent></a></x>' COLUMNS ent xml);
1271        ent        
1272 ------------------
1273  <ent>'</ent>
1274  <ent>"</ent>
1275  <ent>&amp;</ent>
1276  <ent>&lt;</ent>
1277  <ent>&gt;</ent>
1278 (5 rows)
1280 EXPLAIN (VERBOSE, COSTS OFF)
1281 SELECT  xmltable.*
1282    FROM (SELECT data FROM xmldata) x,
1283         LATERAL XMLTABLE('/ROWS/ROW'
1284                          PASSING data
1285                          COLUMNS id int PATH '@id',
1286                                   _id FOR ORDINALITY,
1287                                   country_name text PATH 'COUNTRY_NAME' NOT NULL,
1288                                   country_id text PATH 'COUNTRY_ID',
1289                                   region_id int PATH 'REGION_ID',
1290                                   size float PATH 'SIZE',
1291                                   unit text PATH 'SIZE/@unit',
1292                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified');
1293                                                                                                                                                                                                                         QUERY PLAN                                                                                                                                                                                                                         
1294 -----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
1295  Nested Loop
1296    Output: "xmltable".id, "xmltable"._id, "xmltable".country_name, "xmltable".country_id, "xmltable".region_id, "xmltable".size, "xmltable".unit, "xmltable".premier_name
1297    ->  Seq Scan on public.xmldata
1298          Output: xmldata.data
1299    ->  Table Function Scan on "xmltable"
1300          Output: "xmltable".id, "xmltable"._id, "xmltable".country_name, "xmltable".country_id, "xmltable".region_id, "xmltable".size, "xmltable".unit, "xmltable".premier_name
1301          Table Function Call: XMLTABLE(('/ROWS/ROW'::text) PASSING (xmldata.data) COLUMNS id integer PATH ('@id'::text), _id FOR ORDINALITY, country_name text PATH ('COUNTRY_NAME'::text) NOT NULL, country_id text PATH ('COUNTRY_ID'::text), region_id integer PATH ('REGION_ID'::text), size double precision PATH ('SIZE'::text), unit text PATH ('SIZE/@unit'::text), premier_name text DEFAULT ('not specified'::text) PATH ('PREMIER_NAME'::text))
1302 (7 rows)
1304 -- test qual
1305 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS "COUNTRY_NAME" text, "REGION_ID" int) WHERE "COUNTRY_NAME" = 'Japan';
1306  COUNTRY_NAME | REGION_ID 
1307 --------------+-----------
1308  Japan        |         3
1309 (1 row)
1311 EXPLAIN (VERBOSE, COSTS OFF)
1312 SELECT xmltable.* FROM xmldata, LATERAL xmltable('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]' PASSING data COLUMNS "COUNTRY_NAME" text, "REGION_ID" int) WHERE "COUNTRY_NAME" = 'Japan';
1313                                                                                     QUERY PLAN                                                                                    
1314 ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
1315  Nested Loop
1316    Output: "xmltable"."COUNTRY_NAME", "xmltable"."REGION_ID"
1317    ->  Seq Scan on public.xmldata
1318          Output: xmldata.data
1319    ->  Table Function Scan on "xmltable"
1320          Output: "xmltable"."COUNTRY_NAME", "xmltable"."REGION_ID"
1321          Table Function Call: XMLTABLE(('/ROWS/ROW[COUNTRY_NAME="Japan" or COUNTRY_NAME="India"]'::text) PASSING (xmldata.data) COLUMNS "COUNTRY_NAME" text, "REGION_ID" integer)
1322          Filter: ("xmltable"."COUNTRY_NAME" = 'Japan'::text)
1323 (8 rows)
1325 -- should to work with more data
1326 INSERT INTO xmldata VALUES('<ROWS>
1327 <ROW id="10">
1328   <COUNTRY_ID>CZ</COUNTRY_ID>
1329   <COUNTRY_NAME>Czech Republic</COUNTRY_NAME>
1330   <REGION_ID>2</REGION_ID><PREMIER_NAME>Milos Zeman</PREMIER_NAME>
1331 </ROW>
1332 <ROW id="11">
1333   <COUNTRY_ID>DE</COUNTRY_ID>
1334   <COUNTRY_NAME>Germany</COUNTRY_NAME>
1335   <REGION_ID>2</REGION_ID>
1336 </ROW>
1337 <ROW id="12">
1338   <COUNTRY_ID>FR</COUNTRY_ID>
1339   <COUNTRY_NAME>France</COUNTRY_NAME>
1340   <REGION_ID>2</REGION_ID>
1341 </ROW>
1342 </ROWS>');
1343 INSERT INTO xmldata VALUES('<ROWS>
1344 <ROW id="20">
1345   <COUNTRY_ID>EG</COUNTRY_ID>
1346   <COUNTRY_NAME>Egypt</COUNTRY_NAME>
1347   <REGION_ID>1</REGION_ID>
1348 </ROW>
1349 <ROW id="21">
1350   <COUNTRY_ID>SD</COUNTRY_ID>
1351   <COUNTRY_NAME>Sudan</COUNTRY_NAME>
1352   <REGION_ID>1</REGION_ID>
1353 </ROW>
1354 </ROWS>');
1355 SELECT  xmltable.*
1356    FROM (SELECT data FROM xmldata) x,
1357         LATERAL XMLTABLE('/ROWS/ROW'
1358                          PASSING data
1359                          COLUMNS id int PATH '@id',
1360                                   _id FOR ORDINALITY,
1361                                   country_name text PATH 'COUNTRY_NAME' NOT NULL,
1362                                   country_id text PATH 'COUNTRY_ID',
1363                                   region_id int PATH 'REGION_ID',
1364                                   size float PATH 'SIZE',
1365                                   unit text PATH 'SIZE/@unit',
1366                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified');
1367  id | _id |  country_name  | country_id | region_id | size | unit | premier_name  
1368 ----+-----+----------------+------------+-----------+------+------+---------------
1369   1 |   1 | Australia      | AU         |         3 |      |      | not specified
1370   2 |   2 | China          | CN         |         3 |      |      | not specified
1371   3 |   3 | HongKong       | HK         |         3 |      |      | not specified
1372   4 |   4 | India          | IN         |         3 |      |      | not specified
1373   5 |   5 | Japan          | JP         |         3 |      |      | Sinzo Abe
1374   6 |   6 | Singapore      | SG         |         3 |  791 | km   | not specified
1375  10 |   1 | Czech Republic | CZ         |         2 |      |      | Milos Zeman
1376  11 |   2 | Germany        | DE         |         2 |      |      | not specified
1377  12 |   3 | France         | FR         |         2 |      |      | not specified
1378  20 |   1 | Egypt          | EG         |         1 |      |      | not specified
1379  21 |   2 | Sudan          | SD         |         1 |      |      | not specified
1380 (11 rows)
1382 SELECT  xmltable.*
1383    FROM (SELECT data FROM xmldata) x,
1384         LATERAL XMLTABLE('/ROWS/ROW'
1385                          PASSING data
1386                          COLUMNS id int PATH '@id',
1387                                   _id FOR ORDINALITY,
1388                                   country_name text PATH 'COUNTRY_NAME' NOT NULL,
1389                                   country_id text PATH 'COUNTRY_ID',
1390                                   region_id int PATH 'REGION_ID',
1391                                   size float PATH 'SIZE',
1392                                   unit text PATH 'SIZE/@unit',
1393                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified')
1394   WHERE region_id = 2;
1395  id | _id |  country_name  | country_id | region_id | size | unit | premier_name  
1396 ----+-----+----------------+------------+-----------+------+------+---------------
1397  10 |   1 | Czech Republic | CZ         |         2 |      |      | Milos Zeman
1398  11 |   2 | Germany        | DE         |         2 |      |      | not specified
1399  12 |   3 | France         | FR         |         2 |      |      | not specified
1400 (3 rows)
1402 EXPLAIN (VERBOSE, COSTS OFF)
1403 SELECT  xmltable.*
1404    FROM (SELECT data FROM xmldata) x,
1405         LATERAL XMLTABLE('/ROWS/ROW'
1406                          PASSING data
1407                          COLUMNS id int PATH '@id',
1408                                   _id FOR ORDINALITY,
1409                                   country_name text PATH 'COUNTRY_NAME' NOT NULL,
1410                                   country_id text PATH 'COUNTRY_ID',
1411                                   region_id int PATH 'REGION_ID',
1412                                   size float PATH 'SIZE',
1413                                   unit text PATH 'SIZE/@unit',
1414                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified')
1415   WHERE region_id = 2;
1416                                                                                                                                                                                                                         QUERY PLAN                                                                                                                                                                                                                         
1417 -----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
1418  Nested Loop
1419    Output: "xmltable".id, "xmltable"._id, "xmltable".country_name, "xmltable".country_id, "xmltable".region_id, "xmltable".size, "xmltable".unit, "xmltable".premier_name
1420    ->  Seq Scan on public.xmldata
1421          Output: xmldata.data
1422    ->  Table Function Scan on "xmltable"
1423          Output: "xmltable".id, "xmltable"._id, "xmltable".country_name, "xmltable".country_id, "xmltable".region_id, "xmltable".size, "xmltable".unit, "xmltable".premier_name
1424          Table Function Call: XMLTABLE(('/ROWS/ROW'::text) PASSING (xmldata.data) COLUMNS id integer PATH ('@id'::text), _id FOR ORDINALITY, country_name text PATH ('COUNTRY_NAME'::text) NOT NULL, country_id text PATH ('COUNTRY_ID'::text), region_id integer PATH ('REGION_ID'::text), size double precision PATH ('SIZE'::text), unit text PATH ('SIZE/@unit'::text), premier_name text DEFAULT ('not specified'::text) PATH ('PREMIER_NAME'::text))
1425          Filter: ("xmltable".region_id = 2)
1426 (8 rows)
1428 -- should fail, NULL value
1429 SELECT  xmltable.*
1430    FROM (SELECT data FROM xmldata) x,
1431         LATERAL XMLTABLE('/ROWS/ROW'
1432                          PASSING data
1433                          COLUMNS id int PATH '@id',
1434                                   _id FOR ORDINALITY,
1435                                   country_name text PATH 'COUNTRY_NAME' NOT NULL,
1436                                   country_id text PATH 'COUNTRY_ID',
1437                                   region_id int PATH 'REGION_ID',
1438                                   size float PATH 'SIZE' NOT NULL,
1439                                   unit text PATH 'SIZE/@unit',
1440                                   premier_name text PATH 'PREMIER_NAME' DEFAULT 'not specified');
1441 ERROR:  null is not allowed in column "size"
1442 -- if all is ok, then result is empty
1443 -- one line xml test
1444 WITH
1445    x AS (SELECT proname, proowner, procost::numeric, pronargs,
1446                 array_to_string(proargnames,',') as proargnames,
1447                 case when proargtypes <> '' then array_to_string(proargtypes::oid[],',') end as proargtypes
1448            FROM pg_proc WHERE proname = 'f_leak'),
1449    y AS (SELECT xmlelement(name proc,
1450                            xmlforest(proname, proowner,
1451                                      procost, pronargs,
1452                                      proargnames, proargtypes)) as proc
1453            FROM x),
1454    z AS (SELECT xmltable.*
1455            FROM y,
1456                 LATERAL xmltable('/proc' PASSING proc
1457                                  COLUMNS proname name,
1458                                          proowner oid,
1459                                          procost float,
1460                                          pronargs int,
1461                                          proargnames text,
1462                                          proargtypes text))
1463    SELECT * FROM z
1464    EXCEPT SELECT * FROM x;
1465  proname | proowner | procost | pronargs | proargnames | proargtypes 
1466 ---------+----------+---------+----------+-------------+-------------
1467 (0 rows)
1469 -- multi line xml test, result should be empty too
1470 WITH
1471    x AS (SELECT proname, proowner, procost::numeric, pronargs,
1472                 array_to_string(proargnames,',') as proargnames,
1473                 case when proargtypes <> '' then array_to_string(proargtypes::oid[],',') end as proargtypes
1474            FROM pg_proc),
1475    y AS (SELECT xmlelement(name data,
1476                            xmlagg(xmlelement(name proc,
1477                                              xmlforest(proname, proowner, procost,
1478                                                        pronargs, proargnames, proargtypes)))) as doc
1479            FROM x),
1480    z AS (SELECT xmltable.*
1481            FROM y,
1482                 LATERAL xmltable('/data/proc' PASSING doc
1483                                  COLUMNS proname name,
1484                                          proowner oid,
1485                                          procost float,
1486                                          pronargs int,
1487                                          proargnames text,
1488                                          proargtypes text))
1489    SELECT * FROM z
1490    EXCEPT SELECT * FROM x;
1491  proname | proowner | procost | pronargs | proargnames | proargtypes 
1492 ---------+----------+---------+----------+-------------+-------------
1493 (0 rows)
1495 CREATE TABLE xmltest2(x xml, _path text);
1496 INSERT INTO xmltest2 VALUES('<d><r><ac>1</ac></r></d>', 'A');
1497 INSERT INTO xmltest2 VALUES('<d><r><bc>2</bc></r></d>', 'B');
1498 INSERT INTO xmltest2 VALUES('<d><r><cc>3</cc></r></d>', 'C');
1499 INSERT INTO xmltest2 VALUES('<d><r><dc>2</dc></r></d>', 'D');
1500 SELECT xmltable.* FROM xmltest2, LATERAL xmltable('/d/r' PASSING x COLUMNS a int PATH '' || lower(_path) || 'c');
1501  a 
1507 (4 rows)
1509 SELECT xmltable.* FROM xmltest2, LATERAL xmltable(('/d/r/' || lower(_path) || 'c') PASSING x COLUMNS a int PATH '.');
1510  a 
1516 (4 rows)
1518 SELECT xmltable.* FROM xmltest2, LATERAL xmltable(('/d/r/' || lower(_path) || 'c') PASSING x COLUMNS a int PATH 'x' DEFAULT ascii(_path) - 54);
1519  a  
1520 ----
1521  11
1522  12
1523  13
1524  14
1525 (4 rows)
1527 -- XPath result can be boolean or number too
1528 SELECT * FROM XMLTABLE('*' PASSING '<a>a</a>' COLUMNS a xml PATH '.', b text PATH '.', c text PATH '"hi"', d boolean PATH '. = "a"', e integer PATH 'string-length(.)');
1529     a     | b | c  | d | e 
1530 ----------+---+----+---+---
1531  <a>a</a> | a | hi | t | 1
1532 (1 row)
1535 SELECT * FROM XMLTABLE('*' PASSING '<e>pre<!--c1--><?pi arg?><![CDATA[&ent1]]><n2>&amp;deep</n2>post</e>' COLUMNS x xml PATH 'node()', y xml PATH '/');
1536 -[ RECORD 1 ]-----------------------------------------------------------
1537 x | pre<!--c1--><?pi arg?><![CDATA[&ent1]]><n2>&amp;deep</n2>post
1538 y | <e>pre<!--c1--><?pi arg?><![CDATA[&ent1]]><n2>&amp;deep</n2>post</e>+
1539   | 
1542 SELECT * FROM XMLTABLE('.' PASSING XMLELEMENT(NAME a) columns a varchar(20) PATH '"<foo/>"', b xml PATH '"<foo/>"');
1543    a    |      b       
1544 --------+--------------
1545  <foo/> | &lt;foo/&gt;
1546 (1 row)