This document is also available in these non-normative formats: XML.
Copyright © 2015 Christian Grün and Dannes Wessels, published by the EXPath Community Group under the W3C Community Contributor License Agreement (CLA). A human-readable summary is available.
This specification was published by the EXPath Community Group. It is not a W3C Standard nor is it on the W3C Standards Track. Please note that under the W3C Community Contributor License Agreement (CLA) there is a limited opt-out and other conditions apply.s Learn more about W3C Community and Business Groups.
This module provides an API for accessing the document database MongoDB. It defines functions to connect to the DBMS, retrieve documents from databases and collections, update resources, perform map-reduce queries, and execute server-side JavaScript functions and database commsands.
The module has been designed to be compatible with XQuery 3.1 and XPath 3.1, and later versions. It has been inspired by existing [MongoDB Drivers] and the [MongoDB Meta Driver] recommendations. Its initial version was based on [Mongrel].
1 Status of this document
2 Introduction
2.1 Namespace conventions
2.2 Error management
2.3 JSON Data
2.4 Query Execution
2.5 Test suite
3 Client Operations
3.1 mongo:connect
3.2 mongo:list-databases
3.3 mongo:close
4 Database Operations
4.1 mongo:list-collections
4.2 mongo:command
4.3 mongo:eval
4.4 mongo:drop-database
5 Collections: Read Operations
5.1 mongo:find
5.2 mongo:find-one
5.3 mongo:count
5.4 mongo:aggregate
5.5 mongo:group
5.6 mongo:map-reduce
6 Collections: Update Operations
6.1 mongo:find-and-modify
6.2 mongo:find-and-remove
6.3 mongo:insert
6.4 mongo:save
6.5 mongo:update
6.6 mongo:remove
6.7 mongo:drop-collection
This document is in an interim draft stage. Comments are welcomed at public-expath@w3.org mailing list (archive).
The module defined by this document defines functions and errors in the
namespace http://expath.org/ns/mongo
. In this document, the
mongo
prefix is bound to this namespace URI. Error codes are
defined in the same namespace and are displayed with the same prefix.
The err
prefix denotes the namespace for XPath and XQuery errors,
http://www.w3.org/2005/xqt-errors
, as defined in the
[XQuery 3.1] specification.
Error conditions are identified by a code (a QName
). When such
an error condition is reached in the evaluation of an expression, a dynamic
error is thrown, with the corresponding error code (as if the standard XPath
function error()
had been called).
The following errors apply to most functions of this specification:
[mongo:id] is raised if no database connection exists for a given client id.
[mongo:json] is raised if a supplied XQuery map cannot be converted to a valid JSON object.
[mongo:io] is raised if an unexpected error (connection failure, timeout) occurs while interacting with the database.
[mongo:name] is raised if an invalid database name is
specified. Database names must have 1-63 characters, and they must not
contain any of the following twelve characters:
/\. "$*<>:|?
[mongo:name] is also raised if an invalid collection name is specified. Collection names must have 1-117 characters, and they must not contain the dollar sign ($).
Since Version 3.1, maps and arrays are available in XQuery and XPath. This specification makes heavy use of the feature: all JSON objects and arrays are represented in the equivalent XQuery data types.
If JSON strings are preferred as input and output, the functions
fn:parse-json
and fn:serialize
of the
[XPath and XQuery Functions and Operators 3.1] specification can be used for conversion.
The following query parses a JSON string and returns an XQuery map. The "liberal"
option accepts deviation from [RFC 7159], such as the omission of
quotes on keys:
fn:parse-json('{ info: "Hello Universe" }', map { "liberal": true() })
The next example shows how XQuery maps can be serialized as JSON:
fn:serialize(map { 'info': 'Hello Universe' }, <output:serialization-parameters xmlns:output='http://www.w3.org/2010/xslt-xquery-serialization'> <output:method value='json'/> </output:serialization-parameters> )
All functions in this module are ·nondeterministic·. Non-deterministic functions may return different results when executed more than once. This is illustrated by two examples:
The mongo:find may return different results when called more than once, as the contents of the MongoDB database instance may have changed between the first and second call.
Calling mongo:insert may be successful for the first time, but it may fail when called twice, because the document to be added will already exist.
A query processor must ensure that non-deterministic functions are not relocated or rewritten in the query, and that its results are not cached at runtime.
A [MongoDB Test Suite] is provided to ensure compatibility across different implementations of the specification. It is based on the QT3 format; see [XML Query Test Suite] for more details.
mongo:connect
mongo:connect
($uri asxs:string
) asxs:string
Establishes a connection to MongoDB and returns a client id as string that identifies the opened connection.
The $uri
string follows the MongoDB URI format. It must
at least contain one host name, and it may be prefixed with the
mongo
scheme and suffixed with a port number.
Multiple hosts, e.g. for a replica set, are separated with commas.
The format of the returned client id string is implementation-defined, but all returned ids must be unique during the evaluation of a query.
[mongo:connect] is raised if the connection could not be established, possibly due to a wrong URL or a connection failure.
The following expression creates three connections to local MongoDB instances, using the default port, and returns the client ids as result:
mongo:connect("localhost"), mongo:connect("localhost:27017"), mongo:connect("mongo://localhost:27017")
The following function call connects to a replica set with three members, and distributes reads to the secondary:
mongo:connect("localhost,localhost:27018/?readPreference=secondary")
mongo:list-databases
mongo:list-databases
($client-id asxs:string
) asxs:string*
Returns the names of all databases on the connected server.
The connection is identified by the supplied $client-id
.
The following query lists all databases on localhost:
mongo:list-databases(mongo:connect("localhost"))
mongo:close
mongo:close
($client-id asxs:string
) asempty-sequence()
Closes an open database connection. The connection to be closed is
identified by the supplied $client-id
. When a database connection
is closed, the associated id is discarded and invalidated. As a consequence,
each database can be closed once.
A connection must be kept open as long as it has not explicitly been closed by the user, and as long as the query has not been fully evaluated. After query evaluation, an implementation must ensure that all remaining connections are automatically closed.
The following expression closes a connection that has just been opened:
mongo:close(mongo:connect("localhost"))
The module provides no function for creating new databases, because non-existing databases will automatically be created by MongoDB with the first write operation.
mongo:list-collections
mongo:list-collections
($client-id asxs:string
, $database asxs:string
) asxs:string*
Returns the names of all collections contained in a databases.
The connection is identified by the supplied $client-id
,
and the name of the database is supplied via $database
.
mongo:command
mongo:command
($client-id asxs:string
, $database asxs:string
, $command asmap(*)
) asmap(*)
Executes a [MongoDB Command], supplied via
$command
, and returns the result as a map.
The connection is identified by the supplied $client-id
,
and the name of the database is supplied via $database
.
The object returned by MongoDB contains the field ok
, which
must be parsed by the implementation to decide if command execution was
successful (indicated by the integer value 1
) or not
(0
). The field must be removed from the object before the
result is returned as map. If execution failed, the field
errmsg
can be parsed to return a proper error message.
[mongo:exec] is raised if command execution failed.
The following query clones a database from a remote MongoDB instance to the current host. The result will either be a map, which contains the result of the command execution, or an error:
let $id := mongo:connect("localhost") return try { mongo:command($client-id, map { "clone", 1 }) } catch mongo:exec { "Command execution failed: " || $err:description, }
mongo:eval
mongo:eval
($client-id asxs:string
, $database asxs:string
, $code asxs:string
) asitem()*
mongo:eval
($client-id asxs:string
, $database asxs:string
, $code asxs:string
, $args asitem()*
) asitem()*
Runs server-server JavaScript script code, supplied via $code
.
Function arguments can be supplied via $args
. Arguments can
be booleans, strings, numbers, arrays or maps. An error will be raised if
any other type is supplied.
Items of type xs:untypedAtomic are converted to strings.
The connection is identified by the supplied $client-id
,
and the name of the database is supplied via $database
.
Due to the different type systems of XQuery and JavaScript, it is not possible to losslessly convert all values to one language and back. To ensure compatibility, an implementation must obey the following conversion rules:
Conversion of XQuery arguments to JavaScript:
A value of type xs:boolean is converted to a boolean.
A value of type xs:string and xs:untypedAtomic are converted to a string.
A value of type xs:numeric is converted to a number (i.e. a double-precision floating-point format value).
A value of type map(*) is converted to an object. Its entries must be converted recursively according to the given rules.
A value of type array(*) is converted to an array. Its members must be converted recursively according to the given rules.
The error [mongo:type] is raised for any other type.
Conversion of JavaScript results to XQuery:
A boolean is converted to xs:boolean.
A string is converted to xs:string.
A number is converted to xs:double.
An object is converted to map(*). Its entries must be converted recursively according to the given rules.
An array is converted to array(*). Its members must be converted recursively according to the given rules.
The error [mongo:type] is raised for any other type.
[mongo:type] is raised if an XQuery argument cannot be converted to Javascript, or if a Javascript result cannot be converted to XQuery.
[mongo:exec] is raised if JavaScript execution failed.
The following query returns the result of an arithmetic expression:
let $id := mongo:connect("localhost") return mongo:eval($client-id, "db", 'function ( x, y ) { return x + y; }', (2, 5) )
mongo:drop-database
mongo:drop-database
($client-id asxs:string
, $database asxs:string
) asempty-sequence()
Drops a database. No operation will be performed if the database does not exist.
The connection is identified by the supplied $client-id
,
and the name of the database is supplied via $database
.
The following query drops five databases (provided they exist):
let $id := mongo:connect("localhost") for $no in 1 to 5 return mongo:drop-database($client-id, "database-" || $no)
mongo:find
mongo:find
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
) asmap(*)*
mongo:find
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
) asmap(*)*
mongo:find
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
, $options asmap(*)
) asmap(*)*
Returns documents of a collection. If a query is supplied via the
$query
argument, the documents are filtered by that query.
The $options
argument can have the following entries:
"fields": map(*)
: Restricts the returned fields.
The field _id
will always be returned.
"sort": map(*)
: Sorts the returned documents.
"limit": xs:integer
: Limits the number of returned
documents by the specified integer.
"skip": xs:integer
: Skips the number of specified
documents.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following expression queries an addressbook and selects all persons living in Tokyo. It sorts results by the names and returns the first 50 documents:
let $id := mongo:connect("localhost") return mongo:find($client-id, 'db', 'addressbook', map { "city": "Tokyo" }, map { "sort": map { "name": 1 }, "limit": 50 } )
mongo:find-one
mongo:find-one
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
) asmap(*)?
mongo:find-one
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
) asmap(*)?
mongo:find-one
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
, $options asmap(*)
) asmap(*)?
Returns the first document of a database that optionally matches a
supplied $query
.
The $options
argument can have have the following entries:
"fields": map(*)
: Restricts the returned fields.
The field _id
will always be returned.
"sort": map(*)
: Sorts the documents before returning
the first result.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following query returns the first document that matches the specified query:
let $id := mongo:connect("localhost") return mongo:find-one($client-id, 'db', 'addressbook', map { "name": "John Taylor" })
mongo:count
mongo:count
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
) asxs:integer
mongo:count
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
) asxs:integer
Counts documents in a collection. If a query is supplied via the
$query
argument, the documents are filtered by that query.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following expression counts the number of documents in the "addressbook" collection:
mongo:count(mongo:connect("localhost"), 'db', 'addressbook')
mongo:aggregate
mongo:aggregate
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $pipeline asmap(*)*
) asmap(*)*
Calculates aggregate values for the documents in a collection and returns
the results. The operations to be performed are supplied via
the $pipeline
argument.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following query selects all documents with "Tokyo" as city and returns their names:
let $id := mongo:connect("localhost") return mongo:aggregate($client-id, "db","addressbook", (map { "$match" : map { "city": "Tokyo" } }, map { "$project": map { "name": 1 } }) )
mongo:group
mongo:group
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $fields asmap(*)
, $reduce asxs:string
, $initial asmap(*)
) asmap(*)*
mongo:group
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $fields asmap(*)
, $reduce asxs:string
, $initial asmap(*)
, $options asmap(*)
) asmap(*)*
Groups documents in a collection by the supplied $fields
,
aggregates the documents via the $reduce
function, and
returns the results. $initial
provides an initial result
document, which will be modified by the reduce function.
The $options
argument can have the following entries:
"cond": map(*)
: Filters the documents before being
processed.
"finalize": xs:string
: Follows the reduce function
and modifies the output.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following groups documents with age > 60 by the city field and returns the summed up orders field:
let $id := mongo:connect("localhost") return mongo:group($client-id, "db", "addressbook", map { "city": 1 }, "function (curr, result) { result.orders += curr.orders; }", map { "orders": 0 }, map { "cond": map { "age": map { "$gt": 60 } } } )
mongo:map-reduce
mongo:map-reduce
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $map asxs:string
, $reduce asxs:string
) asmap(*)*
mongo:map-reduce
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $map asxs:string
, $reduce asxs:string
, $options asmap(*)
) asmap(*)*
Runs a map-reduce aggregation operation over the documents of a
collection. The map and reduce functions are supplied via the
$map
and $reduce
arguments.
The $options
argument can have the following entries:
"query": map(*)
: Filters the documents before being
processed.
"output": xs:string
: Specifies the output target
of the result.
"type": xs:string
: Specifies the output type. Allowed
values are INLINE
(default), REPLACE
,
MERGE
and REDUCE
.
"finalize": xs:string
: Follows the reduce function
and modifies the output.
"sort": map(*)
: Sorts the returned documents.
"limit": xs:integer
: Limits the number of returned
documents by the specified integer.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following query sums up the numbber of orders from all documents of the "addressbook" collection:
let $id := mongo:connect("localhost") return mongo:map-reduce($client-id, "db", "addressbook", 'function () { emit(this._id, this.orders) };', 'function (id, ordersArray) { return Array.sum(ordersArray); };' )
mongo:find-and-modify
mongo:find-and-modify
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
, $update asmap(*)
) asmap(*)?
mongo:find-and-modify
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
, $update asmap(*)
, $options asmap(*)
) asmap(*)?
Finds a document that has been selected via the $query
argument, modifies it according to the $update
argument, and
returns the document as it was before the modifications. An empty sequence
is returned if no document was found.
The $options
argument can have the following entries:
"fields": map(*)
: Restricts the returned fields.
The field _id
will always be returned.
"sort": map(*)
: Sorts the documents before choosing the
first one as candidate for modification.
"new": xs:boolean
: Returns the modified document
rather than the original.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following query modifies a document with the specified id:
let $id := mongo:connect("localhost") return mongo:find-and-modify($client-id, 'db', 'addressbook', map { "_id": 123 }, map { "name": "Naomi Matsuo", "city": "Tokyo", "country": "Japan" } )
mongo:find-and-remove
mongo:find-and-remove
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
) asmap(*)?
Finds a document that has been selected via the $query
argument and returns it after removing it from the database. An empty
sequence is returned if no document was found.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following query removes a document with the specified id:
let $id := mongo:connect("localhost") return mongo:find-and-remove($client-id, 'db', 'addressbook', map { "_id": 123 })
mongo:insert
mongo:insert
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $documents asmap(*)*
) asempty-sequence()
Inserts documents into a collection. If the collection does not exists on
the server, it will be created. If the new document does not contain an
_id
field, it will be added.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
[mongo:write] is raised if a document cannot be inserted. This can e.g. happen if document with an identical id already exists.
The following expression inserts two new documents into a collection:
let $id := mongo:connect("localhost") let $docs := (map { 'name': 'John Daniels' }, map { 'name': 'Jack Walker' }) return mongo:insert($client-id, 'db', 'addressbook', $docs)
mongo:save
mongo:save
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $document asmap(*)
) asempty-sequence()
Updates an existing document in a collection or inserts a new document.
If the supplied document has no _id
field, or if the id does
not exist in the collection, the document will be inserted. Otherwise,
the existing document will be replaced.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following expression updates or inserts a single document:
let $id := mongo:connect("localhost") let $doc := map { '_id': 123, name': 'Hans Schmid' } return mongo:save($client-id, 'db', 'addressbook', $doc)
mongo:update
mongo:update
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
, $update asmap(*)
) asempty-sequence()
mongo:update
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
, $update asmap(*)
, $options asmap(*)
) asempty-sequence()
Finds one or more document that have been selected via the
$query
argument and modifies them according to the
$update
argument.
The $options
argument can have have the following entries:
"upsert": xs:boolean
: Inserts a new document if no
document matches the query criteria.
"multi": xs:boolean
: Updates all documents in the
collection that match the update query. Otherwise, updates only
one document.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
[mongo:write] is raised if a document cannot be inserted. This can e.g. happen if a user tries to change the id of the document.
The following update applies to all documents in the addressed collection.
It replaces the value of the info
field with null
,
or adds a new name/value pair if the field does not exist:
let $id := mongo:connect("localhost") return mongo:update($client-id, 'db', 'addressbook', map { }, map { '$set': map { 'info': () } }, map { 'upsert': true(), 'multi': true() } )
mongo:remove
mongo:remove
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
, $query asmap(*)
) asempty-sequence()
Removes documents from a collection that are selected by the
$query
argument.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following expression removes all documents from a collection:
mongo:remove(mongo:connect("localhost"), 'db', 'addressbook', map { })
mongo:drop-collection
mongo:drop-collection
($client-id asxs:string
, $database asxs:string
, $collection asxs:string
) asempty-sequence()
Drops a collection. No operation will be performed if the collection does not exist.
The connection is identified by the supplied $client-id
,
and the name of the database and collection are supplied via
$database
and $collection
.
The following query drops five collections in a database:
let $id := mongo:connect("localhost") for $no in 1 to 5 return mongo:drop-collection($client-id, "database", "collection-" || $no)