3

我正在使用xmlstarlet从以“v”结尾的liquibase XML 更改日志中提取changeSet节点。viewName

但是,xmlstarlet 抱怨ends-withXPATH 函数不存在:

$ xmlstarlet sel -N x="http://www.liquibase.org/xml/ns/dbchangelog" -t -m \
"/x:databaseChangeLog/x:changeSet[x:createView[ends-with(@viewName, 'v')]]" \
-c . public.db.changelog.xml

xmlXPathCompOpEval: function ends-with not found
Unregistered function
Stack usage errror
xmlXPathCompiledEval: 3 objects left on the stack.
runtime error: element for-each
Failed to evaluate the 'select' expression.
None of the XPaths matched; to match a node in the default namespace
use '_' as the prefix (see section 5.1 in the manual).
For instance, use /_:node instead of /node

XML 看起来有点像这样:

<databaseChangeLog xmlns="http://www.liquibase.org/xml/ns/dbchangelog" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.1.xsd">
  <changeSet id="1391529990457-3">
    <createView viewName="myviewnamev"><!-- view definition here --></createView>
  </changeSet>
  <changeSet id="1391529990457-4">
    <createView viewName="anotherviewname"><!-- view definition here --></createView>
  </changeSet>
</databaseChangeLog>

我确实知道 XPATH 表达式在其他方面是正确的,因为如果我将选择标准更改为,x:createView[@viewName="myviewnamev"]那么它只会正确选择该changeLog条目。

如何让 xmlstarlet 正确使用ends-with?或者,是否有另一种方法来完成我想做的事情?

4

2 回答 2

5

xmlstarlet仅支持 XPath 1.0,不提供ends-with($string, $token)功能。您需要使用substring,string-length和 字符串比较来构建您自己的模式:

substring($string, string-length($string) - string-length($token) + 1) = $token]

应用于您的查询,它应该如下所示(我“预先计算”了字符串长度):

/x:databaseChangeLog/x:changeSet[x:createView[
  substring(@viewName, string-length(@viewName)) = 'v']
]

或者,您可能想要寻找更强大的 XPath 2.0/XQuery 引擎。

于 2014-02-06T11:52:43.803 回答
0

以@jens-erat 的回答为基础...

[substring(@viewName, string-length(@viewName)) = 'v']

可以通过将谓词放在@viewName 属性上来使更干燥:

[@viewName[substring(., string-length(.)) = 'v']]

导致:

$ xmlstarlet sel -N x="http://www.liquibase.org/xml/ns/dbchangelog" -t -m \
"/x:databaseChangeLog/x:changeSet[x:createView
   [@viewName[substring(., string-length(.)) = 'v']]
]" -c . public.db.changelog.xml
于 2021-10-29T06:08:59.167 回答