# XPath Transformation Service Transforms an [XML](https://www.w3.org/XML/) input using an [XPath](https://www.w3.org/TR/xpath/#section-Expressions) expression. ## Examples ### Basic Example Given a retrieved XML **Input XML** ```xml 0 450 10 ``` The XPath `/PTZStatus/AbsoluteHigh/azimuth/text()` returns the document ``` 450 ``` ## Advanced Example Given a retrieved XML (e.g. from an HIK Vision device with the namespace `xmlns="http://www.hikvision.com/ver20/XMLSchema"`): **Input XML** ```xml 0 450 10 ``` A simple xpath query to fetch the Azimut value does not work as it does not address the namespace. There are two ways to address the namespace. - Simple path which may not work in complex XML. - With full qualified path. The XPath - `[name()='PTZStatus']/*[name()='AbsoluteHigh']/*[name()='azimuth']/` - `/*[local-name()='PTZStatus' and namespace-uri()='http://www.hikvision.com/ver20/XMLSchema']/*[local-name()='AbsoluteHigh' and namespace-uri()='http://www.hikvision.com/ver20/XMLSchema']/*[local-name()='azimuth' and namespace-uri()='http://www.hikvision.com/ver20/XMLSchema']` returns ``` 450 ``` ### In Setup #### .items ```java String Temperature_xml "Temperature [XPATH(/*[name()='PTZStatus']/*[name()='AbsoluteHigh']/*[name()='azimuth']/):%s °C]" {...} Number Temperature "Temperature [%.1f °C]" ``` ### .rules ```java rule "Convert XML to Item Type Number" when Item Temperature_xml changed then // use the transformation service to retrieve the value // Simple val mytest = transform("XPATH", "/*[name()='PTZStatus'] /*[name()='AbsoluteHigh'] /*[name()='azimuth'] /text()", Temperature_xml.state.toString ) // Fully qualified val mytest = transform("XPATH", "/*[local-name()='PTZStatus' and namespace-uri()='http://www.hikvision.com/ver20/XMLSchema'] /*[local-name()='AbsoluteHigh' and namespace-uri()='http://www.hikvision.com/ver20/XMLSchema'] /*[local-name()='azimuth' and namespace-uri()='http://www.hikvision.com/ver20/XMLSchema'] /text()", Temperature_xml.state.toString ) // post the new value to the Number Item Temperature.postUpdate( newValue ) end ``` Now the resulting Number can also be used in the label to [change the color](https://docs.openhab.org/configuration/sitemaps.html#label-and-value-colors) or in a rule as value for comparison. ## Usage as a Profile The functionality of this `TransformationService` can be used in a `Profile` on an `ItemChannelLink` too. To do so, it can be configured in the `.items` file as follows: ```java String { channel=""[profile="transform:XPATH", function="", sourceFormat=""]} ``` The XPath expression to be executed has to be set in the `function` parameter. The parameter `sourceFormat` is optional and can be used to format the input value **before** the transformation, i.e. `%.3f`. If omitted the default is `%s`, so the input value will be put into the transformation without any format changes. Please note: This profile is a one-way transformation, i.e. only values from a device towards the item are changed, the other direction is left untouched. ## Further Reading - An [introduction](https://www.w3schools.com/xml/xpath_intro.asp) to XPath at W3School - An informative explanation of [common mistakes](https://qxf2.com/blog/common-xpath-mistakes/). - Online validation tools like [this](https://www.freeformatter.com/xpath-tester.html) to check the syntax.