The XQuery specification is a formal recommendation from the W3C XQuery Working Group. MarkLogic 10 implements the W3C XQuery 1.0 Recommendation (http://www.w3.org/TR/xquery/). To maximize compatibility with MarkLogic Server and to offer strict XQuery compliance to those who desire it, as well as to include extensions to the language to make it easier to build applications, MarkLogic Server supports two dialects of XQuery. This chapter describes these dialects, and includes the following sections:
MarkLogic Server supports the following dialects of XQuery:
You can use library modules from different dialects together, as described in Rules For Combining the Dialects. Each dialect has a different set of pre-defined namespaces, as described in Predefined Namespace Prefixes for Each Dialect.
For a module to use the MarkLogic Server enhanced dialect, use the following for the XQuery version declaration on the first line of the XQuery module:
xquery version "1.0-ml";
Note the semi-colon at the end of the declaration, which is required in 1.0-ml
. The enhanced dialect has the XQuery 1.0 syntax and also includes various extensions to the language such as try
/catch
. This dialect is the default for new App Servers, and is considered the preferred dialect for new applications. For more details on the enhanced 1.0-ml
dialect, see MarkLogic Server Enhanced XQuery Language.
For a module to use the MarkLogic Server strict dialect (1.0
), use the following for the XQuery version declaration on the first line of the XQuery module:
xquery version "1.0";
Note the semi-colon at the end of the declaration, which is required in 1.0
. The strict mode is for compatibility with other XQuery 1.0 processors; if you write a library in 1.0
, you can use it with MarkLogic Server and you can also use it with other conforming processors. Similarly, you can use modules that are written in standard XQuery with MarkLogic Server.
To use the MarkLogic Server built-in functions in 1.0
, you must bind a prefix (for example, xdmp
) to the namespace for the MarkLogic Server functions; there is no need to import a library for these built-in functions, but you do need to bind the namespace to a prefix. To use the xdmp
functions in 1.0
, add prolog entries for the namespace bindings you are using in your query, as in the following example:
xquery version "1.0"; declare namespace xdmp = "http://marklogic.com/xdmp"; xdmp:version()
MarkLogic Server has a very flexible way of combining the different XQuery dialects. You can import a library module written in any of the three dialects into any main or library module. For example, you might find an open source standards-compliant module that you found on the internet which is written in the strict XQuery 1.0 dialect. You can then import this module into any MarkLogic Server XQuery program, regardless of dialect, and then use those functions in your code.
When writing modules of different dialects, the best practice is to always use the XQuery version declaration as the first line, indicating which dialect the module is written in. That way, if the module is written in a different dialect than the default dialect for the App Server or the program, it will still work correctly (for details, see Inheriting the Default XQuery Version From the App Server).
You can use the xdmp:dialect
attribute to specify which dialect expressions are evaluated in an XSLT stylesheet. For details, see xdmp:dialect Attribute.
If you are writing new XQuery code, the best practice is to use the 1.0-ml
dialect. If you are updating code that was written in previous versions of MarkLogic Server, migrate that code to 1.0-ml
. This section describes things to think about when migrating your application code and includes the following parts:
XQuery 0.9-ml
has been deprecated and will be removed in a future release. If you have code written in XQuery dialect 0.9-ml
, you need to migrate your XQuery code to either enhanced XQuery 1.0-ml
or strict XQuery 1.0
. The differences between the dialects are mostly syntax changes in the prolog, but there are also some other differences that might cause subtle changes in behavior. For details on the differences between the XQuery dialects in 0.9-ml
and 1.0-ml
, see XQuery Changes from 0.9-ml to 1.0-ml. When you migrate XQuery code to 1.0-ml
(or to 1.0
), there are several ways you can go about it:
0.9-ml
declaration to the first line of each XQuery file. Then, as you migrate a module, you can change the declaration to 1.0-ml
and make any needed syntax changes to that module.MarkLogic has deprecated XQuery 0.9-ml
and will be removing it in a future release.
While MarkLogic Server currently allows 0.9-ml
code to run without changes, the dialect is deprecated and will be removed in a future release. The new XQuery dialect 1.0-ml
has enhancements. Because you can mix modules in the old dialect with modules in the new, you can perform your migration one module at a time. This section highlights the major syntax and semantic changes between the XQuery used in 0.9-ml
and enhanced XQuery dialect 1.0-ml
. The changes fall into the following categories:
The following syntax has changed from XQuery 0.9-ml to XQuery 1.0-ml:
define
now use declare
. :=
syntax (for details and an example, see Declaring Variables).The following semantic changes have occurred from XQuery 0.9-ml to XQuery 1.0-ml:
namespace
keyword and a prefix for the namespace, for example:module namespace my = "my-namespace";
Functions in a main module must be put into the local:
namespace.
empty-sequence()
The 0.9-ml
dialect had you specify empty()
for the empty sequence.
0.9-ml
and returns false in 1.0-ml
(and throws an exception in 1.0
):(: returns true in 0.9-ml, false in 1.0-ml, and throws XDMP-EFFBOOLVALUE in 1.0 :) fn:boolean((fn:false(), fn:false()))
This change might affect applications that have if/then/else
statements where the if
test returns a sequence of boolean values. In these cases, you might see the if
statement evaluating to false
in cases where it previously evaluated to true
, causing the else
statement to be evaluated instead of the then
statement.
substring((), 1); substring("",1)
In the first line, ()
is parsed identically to ""
in the second line.
The functions affected by this are:
min
and max
offsets are now enforced. [5 to 1]
is now evaluated as ()
.xs
namespace prefix; previously it used the xdt
prefix. Any code you have that uses the xdt
namespace prefix will require a change to the xs
prefix. For example, if you have code that uses xdt:dayTimeDuration
, change it to xs:dayTimeDuration
.element()
tests in 0.9-ml
are equivalent to schema-element()
test in 1.0
and 1.0-ml
. Any code you have with element()
tests might not match some elements that previously matched. For example, substitution elements previously would match the base element name, but will now only match with schema-element()
test in 1.0
and 1.0-ml
.1.0-ml
has has function mapping enabled by default.fn:node-kind
does not exist in 1.0
and 1.0-ml
(it is replaced by xdmp:node-kind).The following built-in functions have changed:
1.0-ml
has implemented this function differently. It is more likely to throw an error when it has trouble resolving a relative URI reference.fn:get-local-name-from-QName |
fn:local-name-from-QName |
fn:input |
fn:doc |
fn:expanded-QName |
fn:QName |
fn:node-kind | xdmp:node-kind |
fn:get-minutes-from-dayTimeDuration |
fn:minutes-from-duration |
fn:get-month-from-date |
fn:month-from-date |
fn:get-seconds-from-dateTime |
fn:seconds-from-dateTime |
fn:get-year-from-dateTime |
fn:year-from-dateTime |
fn:get-day-from-date |
fn:day-from-date |
fn:get-seconds-from-dayTimeDuration |
fn:seconds-from-dayTimeDuration |
fn:get-year-from-date |
fn:year-from-date |
fn:get-days-from-duration |
fn:days-from-duration |
fn:get-timezone-from-date |
fn:timezone-from-date |
fn:get-namespace-uri-from-QName |
fn:namespace-uri-from-QName |
fn:get-seconds-from-time |
fn:seconds-from-time |
fn:get-minutes-from-time |
fn:minutes-from-time |
fn:get-timezone-from-time |
fn:timezone-from-time |
fn:get-hours-from-dateTime |
fn:hours-from-dateTime |
fn:get-months-from-yearMonthDuration |
fn:months-from-duration |
fn:get-day-from-dateTime |
fn:day-from-dateTime |
fn:get-month-from-dateTime |
fn:month-from-dateTime |
fn:get-hours-from-dayTimeDuration |
fn:hours-from-duration |
fn:get-minutes-from-dateTime |
fn:minutes-from-dateTime |
fn:get-years-from-yearMonthDuration |
fn:years-from-duration |
fn:get-hours-from-time |
fn:hours-from-time |
fn:get-timezone-from-dateTime |
fn:timezone-from-dateTime |
fn:get-namespace-uri-for-prefix |
fn:namespace-uri-for-prefix |
fn:get-in-scope-namespaces |
fn:in-scope-prefixes |
fn:subtract-dateTimes-yielding-dayTimeDuration
Use the minus sign operator (-
).fn:distinct-nodes
You can write an XQuery loop to replace this function or use a similar function from the FunctX library functx:distinct-nodes
. Read more at http://www.xqueryfunctions.com/xq/functx_distinct-nodes.html. fn:subtract-dateTimes-yielding-yearMonthDuration
Use the minus sign (-) operator to get xs:dayTimeDuration
.fn:string-pad
You must now write an XQuery loop to replace this function.fn:context-item
Use . or fn:current()
as appropriate.Each App Server has a setting for the default XQuery version. Any requests against that App Server that do not have explicitly specify an XQuery version declaration are treated as the default XQuery version value. Because of the way a request inherits it default XQuery version from the App Server environment, requests without an explicit declaration can be treated differently by different App Servers (if the App Servers have different default XQuery values). Therefore, it is best practice to specify the XQuery version in each module.
The task server does not allow you to specify a default XQuery version, and if there is no explicit version declaration in the XQuery code evaluated on the task server, the default XQuery version is determined as follows:
1.0-ml
.This makes it especially important to use XQuery version declarations in modules used by CPF or modules called from triggers. For details on CPF, see the Content Processing Framework Guide.
To ensure your code is always evaluated in the dialect in which you have written it, regardless of the context in which it is run, the best practice is to begin each XQuery module with a XQuery version declaration. For the syntax of the version declaration, see XQuery Version Declaration.
You specify the dialect for an XQuery module with a version declaration. The version declaration is optional, and comes before the prolog in an XQuery module. It is best practice to put the XQuery version declaration in your code as the first line in the module, as having it there ensures it will work as expected in any environment. For example, to specify 1.0-ml
as the XQuery version, begin your XQuery module with the following:
xquery version "1.0-ml";
In most cases, porting any XQuery code used in 0.9-ml
to the 1.0-ml
dialect will be straightforward. The bulk of the differences are syntax changes in the prolog. As stated earlier, you do not need to port all of your code at one time. A sensible approach is to migrate your code one XQuery module at a time. This section outlines the basic steps to follow when migrating your XQuery code.
The following are some basic steps to take when migrating 0.9-ml
XQuery code to 1.0-ml
:
0.9-ml
, the declarations will be as follows:xquery version "0.9-ml"
If you use the xquery version "0.9-ml" declaration, you will get a warning message. MarkLogic recommends that you move to either XQuery dialect 1.0-ml
or 1.0
as soon as possible.
1.0-ml
and add a semi-colon to the line so it appears as follows:xquery version "1.0-ml";
1.0
syntax (change define
to declare
, add semi-colons, and so on, as described in XQuery Changes from 0.9-ml to 1.0-ml). For the prolog syntax, see XQuery Prolog, the W3C specification (http://www.w3.org/TR/xquery/#id-grammar), or a third-party XQuery book.local:
prefix, which is predefined.xdt
namespace prefix, change the prefix to xs
(for example, change xdt:dayTimeDuration
to xs:dayTimeDuration
).fn
namespace URI, you must change the namespace URI of that module; you cannot use the URI bound to the fn
namespace prefix as the URI for a library module in 1.0
or 1.0-ml
. If you do change the namespace URI of a library module, you must also change the URI in any import module
statements in other modules that call the library.0.9-ml
and 1.0-ml
(see XQuery Changes from 0.9-ml to 1.0-ml). Check for any changes due to function mapping, which is described in Function Mapping.