xslt transformation for given xml - html

This is my xml file for which I am to write an xsl file. Purpose for this xsl is to use for xml to html translation using java xml transform.
I am trying to implement this logic in my xsl file, but unable to get so.
for each loop Id = PO1
if segment Id = PO1
if element Id = PO103
print its value (i.e. CS)
if loop Id = PID
if segment Id = PID
if element Id = PID05
print its value (i.e. BANANAS, ...)
Any help would be appreciated.

Try it this way:
<xsl:for-each select="//loop[#Id='PO1']/segment[#Id='PO1']/element[#Id='PO103']
|
//loop[#Id='PID']/segment[#Id='PID']/element[#Id='PID05'] ">
<!-- whatever "print" means goes here -->
</xsl:for-each>
P.S. Please post your XML as code, not as picture.

Thank you user3016153, with your help I could solve my problem like this :
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/">
<html>
<body>
<h2>HTML generated by XSLT stylesheet from EDI input</h2>
<table border="1">
<tr bgcolor="#9acd32">
<th>Title</th>
<th>Value</th>
</tr>
<xsl:for-each select="//loop[#Id='PO1']/segment[#Id='PO1']/element[#Id='PO103']
|
//loop[#Id='PID']/segment[#Id='PID']/element[#Id='PID05'] ">
<tr>
<td><xsl:value-of select="#Id"/></td>
<td><xsl:value-of select="."/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>

Related

How do I fix this xml and xsl code because it doesn't work

First attempt to code xml and use xsl for html output.
I took the W3 examples and modified them for my needs but I've missed something as looking at the xml in a browser does work correctly. None of the html code shows in the browser but the items info from the xml shows as long list.
What shows in browser:
SCALARADD Image1inSCALARADD Add one value to an image. A B A same as B allowed redLevel, greenLevel, blueLevel TRIMFILL Image1inTRIMFILL Trim an image and fill the trim area. A B A same as B allowed trimTopIndex, trimBottomIndex, trimLeftIndex, trimRightIndex fillRedLevel, fillGreenLevel, fillBlueLevel
Here is the xml file:
<?xml version="1.0" encoding="UTF-8"?>
<?xml-stylesheet type="text/xsl" href="routines.xsl"?>
<family>
<Image1in>
<routine>SCALARADD</routine>
<symbolic> Image1inSCALARADD </symbolic>
<operation> Add one value to an image. </operation>
<inputLayer1> A </inputLayer1>
<inputLayer2></inputLayer2>
<outputLayer1> B </outputLayer1>
<commentLayer> A same as B allowed </commentLayer>
<inputLong></inputLong>
<inputDouble> redLevel, greenLevel, blueLevel </inputDouble>
<outputLong></outputLong>
<outputDouble></outputDouble>
<comment> </comment>
</Image1in>
<Image1in>
<routine>TRIMFILL</routine>
<symbolic> Image1inTRIMFILL </symbolic>
<operation> Trim an image and fill the trim area. </operation>
<inputLayer1> A </inputLayer1>
<inputLayer2></inputLayer2>
<outputLayer1> B </outputLayer1>
<commentLayer> A same as B allowed </commentLayer>
<inputLong> trimTopIndex, trimBottomIndex, trimLeftIndex, trimRightIndex </inputLong>
<inputDouble> fillRedLevel, fillGreenLevel, fillBlueLevel </inputDouble>
<outputLong></outputLong>
<outputDouble></outputDouble>
<comment> </comment>
</Image1in>
</family>
Here is the xsl file:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/">
<html>
<body>
<h2>Routines</h2>
<table border="2">
<tr bgcolor="#7f7f7f">
<td style="text-align:left">Title</td>
<td style="text-align:left">Symbol</td>
<td style="text-align:left">Operation</td>
<td style="text-align:left">Input Layer 1</td>
</tr>
<xsl:for-each select="family/Image1In">
<xsl:sort select="routine"/>
<tr>
<td><xsl:value-of select="routine"/></td>
<td><xsl:value-of select="symbolic"/></td>
<td><xsl:value-of select="operation"/></td>
<td><xsl:value-of select="inputLayer1"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
I appreciate any help someone can share. Spent whole weekend try to fix.
Regards,
RONC
A browser is not a good environment for testing XSLT. As it happens, your XSLT has a mistake of using:
<xsl:for-each select="family/Image1In">
instead of:
<xsl:for-each select="family/Image1in">
XML is case-sensitive, and your expression selects nothing.
However, you should still be seeing the header row of the table. What you see instead - the raw text of of the XML - suggests that your browser is applying the "same origin" security rule and blocking the request for the stylesheet.
You can try this code <xsl:for-each select="family/Image1in"> insted of <xsl:for-each select="family/Image1In"> Or if you want to remove both leading and trailing space remove use normalize-space().
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/">
<html>
<body>
<h2>Routines</h2>
<table border="2">
<tr bgcolor="#7f7f7f">
<td style="text-align:left">Title</td>
<td style="text-align:left">Symbol</td>
<td style="text-align:left">Operation</td>
<td style="text-align:left">Input Layer 1</td>
</tr>
<xsl:for-each select="family/Image1in">
<xsl:sort select="routine"/>
<tr>
<td><xsl:value-of select="normalize-space(routine)"/></td>
<td><xsl:value-of select="normalize-space(symbolic)"/></td>
<td><xsl:value-of select="normalize-space(operation)"/></td>
<td><xsl:value-of select="normalize-space(inputLayer1)"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
DEMO: https://xsltfiddle.liberty-development.net/3NSSEv5

XPath doesn't recognize path

I'm writing an XSL transformer to convert XML to HTML. Here's my gun.xml:
<?xml version="1.0" encoding="utf-8"?>
<?xml-stylesheet type="text/xsl" href="guns.xslt"?>
<guns xsi:noNamespaceSchemaLocation="guns2.xsd"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<gun>
<model>revolver</model>
<handy>1</handy>
<origin>Britain</origin>
<ttc>20mm</ttc>
</gun>
</guns>
And guns.xslt here:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match = "/">
<html>
<body>
<h2>Gun Collection</h2>
<table border = "1">
<tr bgcolor = "#9acd32">
<th>Model</th>
<th>Origin</th>
<th>TTC</th>
</tr>
<xsl:for-each select="guns/gun">
<tr>
<td><xsl:value-of select = "model"/></td>
<td><xsl:value-of select = "handy"/></td>
<td><xsl:value-of select = "origin"/></td>
<td><xsl:value-of select = "ttc"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
Both files are located in the same package, but XPAth doesn't recognize "guns/gun" in for-each block. What am I missing?
You have declared in xml-stylesheet - type as "text/xsl", please change file extension from "guns.xslt" to "guns.xsl" and also change in your XML as below:
<?xml-stylesheet type="text/xsl" href="guns.xsl"?>
Then your result will be as below:

XSLT code returns empty rows when converting XML to HTML table

I am trying to convert an XML document to HTML using XSLT 1.0. I aim to display the XML rows in an HTML table. But my XSLT code returns empty rows in the HTML table. Here's what my XML looks like:
<?xml version="1.0" encoding="UTF-8"?>
<people>
<person id="ABCDE1234" firstname="John" lastname="Lewis" sex="male"/>
<person id="XYZWU6789" firstname="Marie" lastname="Claire" sex="female"/>
</people>
Here's my XSLT code:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/">
<html>
<head>
<title>XML to HTML using XSLT</title>
</head>
<body>
<table border="1" >
<tr>
<th>id</th>
<th>firstname</th>
<th>lastname</th>
<th>sex</th>
</tr>
<xsl:for-each select="people/person">
<tr>
<td><xsl:value-of select="id"/></td>
<td><xsl:value-of select="firstname"/></td>
<td><xsl:value-of select="lastname"/></td>
<td><xsl:value-of select="sex"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
However, my XSLT code runs properly when I reformat the XML file as follows:
<?xml version="1.0" encoding="UTF-8"?>
<people>
<person>
<id>ABCDE1234</id>
<firstname>John</firstname>
<lastname>Lewis</lastname>
<sex>Male</sex>
</person>
<person>
<id>XYZWU6789</id>
<firstname>Mary</firstname>
<lastname>Claire</lastname>
<sex>Female</sex>
</person>
</people>
Why is this happening? Any ideas?
Just add the # sign so that the values will recognize as attributes.
<td><xsl:value-of select="#id"/></td>
<td><xsl:value-of select="#firstname"/></td>
<td><xsl:value-of select="#lastname"/></td>
<td><xsl:value-of select="#sex"/></td>

How to process a namespaced XML document using XSLT?

I am trying to process an XML file with XSLT to produce a list or table or (eventually) an SQL INSERT command with some of the values. I have been using the example from w3schools http://www.w3schools.com/xsl/tryxslt.asp?xmlfile=cdcatalog&xsltfile=cdcatalog.
My XML is very simple and I need to extract just the name and the rate of the hotel:
<?xml version="1.0" encoding="utf-8"?>
<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<soapenv:Body>
<getHotelsResponse xmlns="http://hotel.booking.vbooking.com">
<getHotelsReturn>
<address>
<number>589-591</number>
<postcode>08014</postcode>
<region>Catalonia</region>
<street>Carrer Vermell</street>
<town>Barcelona</town>
</address>
<name>Downtown Hotel</name>
<rate>235.0</rate>
</getHotelsReturn>
</getHotelsResponse>
</soapenv:Body>
</soapenv:Envelope>
The best XSLT I could build up from the w3schools was this one:
<?xml version="1.0" encoding="utf-8"?>
<!-- Edited by XMLSpyΠ-->
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/soapenv:Envelope/soapenv:Body/getHotelsResponse">
<html>
<body>
<h2>Hotels in Barcelona</h2>
<table border="1">
<tr bgcolor="#9acd32">
<th>Hotel name</th>
<th>Rate ($/night)</th>
</tr>
<xsl:for-each select="getHotelsReturn">
<tr>
<td><xsl:value-of select="name"/></td>
<td><xsl:value-of select="rate"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
Which should produce a similar result to what I had in the w3schools example, but it comes back as blank.
Can anyone please explain this? Thanks.
As pointed out by #keshlam, if elements in your input XML have namespaces, you must declare them in your XSLT stylesheet as well.
Find more detail on namespaces in a previous answer of mine.
The gist of the information you find there is: as far as the XSLT processor is concerned, a getHotelsReturn element is entirely different from a vb:getHotelsReturn element.
Stylesheet
<?xml version="1.0" encoding="utf-8"?>
<!-- Edited by XMLSpy?-->
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:vb="http://hotel.booking.vbooking.com"
xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/"
exclude-result-prefixes="soapenv vb">
<xsl:template match="/soapenv:Envelope/soapenv:Body/vb:getHotelsResponse">
<html>
<body>
<h2>Hotels in Barcelona</h2>
<table border="1">
<tr bgcolor="#9acd32">
<th>Hotel name</th>
<th>Rate ($/night)</th>
</tr>
<xsl:for-each select="vb:getHotelsReturn">
<tr>
<td><xsl:value-of select="vb:name"/></td>
<td><xsl:value-of select="vb:rate"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
Output
<html>
<body>
<h2>Hotels in Barcelona</h2>
<table border="1">
<tr bgcolor="#9acd32">
<th>Hotel name</th>
<th>Rate ($/night)</th>
</tr>
<tr>
<td>Downtown Hotel</td>
<td>235.0</td>
</tr>
</table>
</body>
</html>
Rendered HTML
You can use XPaths like this:
/Product1/Product2/Product3[#ValidYN = 'Y' and #ProductType = 'ABC']
/Product1/Product2/Product3[#ValidYN = 'Y' and #ProductType = 'DEF']
/Product1/Product2/Product3[#ValidYN = 'Y' and #ProductType = 'GHI']
Check this other question How to parse XML using XSLT?

XML to HTML using XSL.Unable to read data from node

I have a xml file with one of the nodes as:
<test-case time="0.077" name="TestingNunit.NUnitTestClass.NunitTestMethod1" asserts="1" success="False" result="Failure" executed="True">
<failure>
<message>
<![CDATA[ Expected: not 3.0d But was: 3.0d ]]>
</message>
<stack-trace>
<![CDATA[at TestingNunit.NUnitTestClass.NunitTestMethod1() in z:\UnitTestingSample\UIAutomation\TestingNunit\UnitTest1.cs:line 12 ]]>
</stack-trace>
</failure>
</test-case>
<test-case time="0.003" name="TestingNunit.NUnitTestClass.NunitTestMethod2" asserts="2" success="False" result="Failure" executed="True">
<failure>
<message>
<![CDATA[ Expected: 2.0d But was: 3.0d ]]>
</message>
<stack-trace>
<![CDATA[at TestingNunit.NUnitTestClass.NunitTestMethod2() in z:\UnitTestingSample\UIAutomation\TestingNunit\UnitTest1.cs:line 21 ]]>
</stack-trace>
</failure>
Now I want to read the data of all CDATA's". I am trying to generate a html file using XSL for which I need this data? I am able to generate data for normal nodes and their attributes but for a node written in this way I am unable to figure out a way to read it. This is my xsl file foreach loop
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl">
<xsl:output method="xml" indent="yes"/>
<xsl:template match="/">
<html>
<body>
<table border="1">
<tr bgcolor="#9acd32">
<th>Test Method</th>
<th>Execution Time</th>
<th>Asserts Performed</th>
<th>Success</th>
<th>Result</th>
<th>Executed</th>
<th>Failure Message</th>
<th>Stack Trace</th>
</tr>
<xsl:for-each select="test-results/test-suite/results/test-suite/results/test-suite/results/test-case">
<tr>
<td><xsl:value-of select="#name"/></td>
<td><xsl:value-of select="#time"/></td>
<td><xsl:value-of select="#asserts"/></td>
<td><xsl:value-of select="#success"/></td>
<td><xsl:value-of select="#result"/></td>
<td><xsl:value-of select="#executed"/></td>
<td><xsl:value-of select="//failure/message/text()" disable-output-escaping="no"/></td>
<td><xsl:value-of select="//failure/stack-trace/text()" disable-output-escaping="no"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
To read CDATAs (and then transform them) use the text() function in your XPath.
//test-case/failure/message/text()
//test-case/failure/stack-trace/text()
Try select="./failure/message/text()" instead of select="//failure/message/text()" (mind the ./ at the beginning).
Doing this the XPath relates to the node that is currently selected by the for-each loop. When using // it will select all nodes from the document ignoring the currently selected node. Imagine this like a lscommand on Linux: ls /* will select the top level elements regardless of your current work directory, while ls ./* will select the child elements of your current work directory.