Indices are set for a container using the container's index specification. You can specify an index either against a specific node and namespace, or you can define default indices that are applied to every node in the container.
You add, delete, and replace indices using the container's index specification. You can also iterate through the specification, so as to examine each of the indices declared for the container. Finally, if you want to retrieve all the indices maintained for a named node, you can use the index specification to find and retrieve them.
An API exists that allows you to retrieve all of the documents or nodes referenced by a given index.
For simple programs, managing the index specification and then setting it to the container (as is illustrated in the following examples) can be tedious. For this reason, BDB XML also provides index management functions directly on the container. Which set of functions your application uses is entirely up to your requirements and personal tastes.
Performing index modifications (for example, adding and replacing indices) on a container that already contains documents can be a very expensive operation — especially if the container holds a large number of documents, or very large documents, or both. This is because indexing a container requires BDB XML to traverse every document in the container.
If you are considering re-indexing a large container, be aware that the operation can take a long time to complete.
To add an index to a container:
Retrieve the index specification from the container.
Use XmlIndexSpecification::addIndex() to add the index to the container. You must provide to this method the namespace and node name to which the index is applied. You must also identify the indexing strategy.
If the index already exists for the specified node, then the method silently does nothing.
Set the updated index specification back to the container.
For example:
#include "DbXml.hpp"
...
using namespace DbXml;
...
// Get a manager object.
XmlManager myManager;
// Open a container
XmlContainer myContainer = 
    myManager.openContainer("exampleData.dbxml");
// Get the index specification
XmlIndexSpecification is = myContainer.getIndexSpecification();
// Get the index type and syntax. This is the equivalent of
//    "node-element-presence-none"
// index strategy
XmlIndexSpecification::Type idxType = 
    (XmlIndexSpecification::Type)
    (XmlIndexSpecification::PATH_NODE | 
    XmlIndexSpecification::NODE_ELEMENT | 
    XmlIndexSpecification::KEY_PRESENCE);
// The syntax must be NONE because this is a presence
// index.
XmlValue::Type syntaxType = XmlValue::NONE;
// Add the index. We're indexing "node1" using the default
// namespace. Note that we could also do this using:
// is.addIndex("", "node1", "node-element-presence-none");
is.addIndex("", "node1", idxType, syntaxType);
// Save the index specification back to the container.
XmlUpdateContext uc = myManager.createUpdateContext();
myContainer.setIndexSpecification(is, uc); 
      To delete an index from a container:
Retrieve the index specification from the container.
Use XmlIndexSpecification::deleteIndex() to delete the index from the index specification.
Set the updated index specification back to the container.
For example:
#include "DbXml.hpp"
...
using namespace DbXml;
...
// Get a manager object.
XmlManager myManager;
// Open a container
XmlContainer myContainer = 
    myManager.openContainer("exampleData.dbxml");
// Get the index specification
XmlIndexSpecification is = myContainer.getIndexSpecification();
// We will delete the equivalent of:
// XmlIndexSpecification::Type idxType = 
//     (XmlIndexSpecification::Type)
//     (XmlIndexSpecification::PATH_NODE | 
//     XmlIndexSpecification::NODE_ELEMENT | 
//     XmlIndexSpecification::KEY_PRESENCE);
// XmlValue::Type syntaxType = XmlValue::NONE;
// Delete the index. We're deleting the index from "node1" in
// the default namespace that has the syntax strategy identified
// above. Note that we could also do this using:
// is.deleteIndex("", "node1", idxType, syntaxType);
is.deleteIndex("", "node1", "node-element-presence-none");
// Save the index specification back to the container.
XmlUpdateContext uc = myManager.createUpdateContext();
myContainer.setIndexSpecification(is, uc); 
      You can replace the indices maintained for a specific node by using XmlIndexSpecification::replaceIndex(). When you replace the index for a specified node, all of the current indices for that node are deleted and the replacement index strategies that you provide are used in their place.
Note that all the indices for a specific node can be retrieved and specified as a space separated list in a single string. So if you set a node-element-equality-string and a node-element-presence index for a given node, then it's indices are identified as:
"node-element-equality-string node-element-presence"
For example:
#include "DbXml.hpp"
...
using namespace DbXml;
...
// Get a manager object.
XmlManager myManager;
// Open a container
XmlContainer myContainer = 
    myManager.openContainer("exampleData.dbxml");
// Get the index specification
XmlIndexSpecification is = myContainer.getIndexSpecification();
// Replace the index.
std::string idxString = 
    "node-element-equality-string node-element-presence";
is.replaceIndex("", "node1", idxString);
// Save the index specification back to the container.
XmlUpdateContext uc = myManager.createUpdateContext();
myContainer.setIndexSpecification(is, uc); 
      You can iterate over all the indices in a container using XmlIndexSpecification::next(). You can retrieve indices using either the string or enumerated format.
For example:
#include "DbXml.hpp"
...
using namespace DbXml;
...
// Get a manager object.
XmlManager myManager;
// Open a container
XmlContainer myContainer = 
    myManager.openContainer("exampleData.dbxml");
// Get the index specification
XmlIndexSpecification is = myContainer.getIndexSpecification();
// Iterate over all of the indices in the container. Note
// that we could use the enumerated types to retrieve
// the indices as well.
std::string uri, name, index;
int count = 0;
while(is.next(uri,name,index)) {
    // Print the index strategy to the console:
    std::cout << "For node: '" << name << "' found:\n"
              << "\tURI: " << uri
              << "\tIndex: " << index << std::endl;
    count ++;
}
std::cout << count << " indices found." << std::endl; 
      Default indices are indices that are applied to all applicable nodes in the container that are not otherwise indexed. For example, if you declare a default index for a metadata node, then all metadata nodes will be indexed according to that indexing strategy, unless some other indexing strategy is explicitly set for them. In this way, you can avoid the labor of specifying a given indexing strategy for all occurrences of a specific kind of a node.
You add, delete, and replace default indices using:
XmlIndexSpecification::addDefaultIndex()
XmlIndexSpecification::deleteDefaultIndex()
XmlIndexSpecification::replaceDefaultIndex()
When you work with a default index, you identify only the indexing strategy; you do not identify a URI or node name to which the strategy is to be applied.
Note that just as is the case with other indexing methods, you can use either strings or enumerated types to identify the index strategy.
For example, to add a default index to a container:
#include "DbXml.hpp"
...
using namespace DbXml;
...
// Get a manager object.
XmlManager myManager;
// Open a container
XmlContainer myContainer = 
    myManager.openContainer("exampleData.dbxml");
// Get the index specification
XmlIndexSpecification is = myContainer.getIndexSpecification();
// Get the index type and syntax. This is the equivalent of
//    "node-metadata-equality-string"
// index strategy
XmlIndexSpecification::Type idxType = 
    (XmlIndexSpecification::Type)
    (XmlIndexSpecification::PATH_NODE | 
    XmlIndexSpecification::NODE_METADATA | 
    XmlIndexSpecification::KEY_EQUALITY);
// Declare the syntax type:
XmlValue::Type syntaxType = XmlValue::STRING;
// Add the default index. Note that we could also do this using:
// is.addDefaultIndex("node-metadata-equality-string");
is.addDefaultIndex(idxType, syntaxType);
// Save the index specification back to the container.
XmlUpdateContext uc = myManager.createUpdateContext();
myContainer.setIndexSpecification(is, uc); 
      You can retrieve all of the values referenced by an index using XmlContainer::lookupIndex(). This method returns an XmlResults object that contains all of the nodes or documents for which the identified index has keys. Whether nodes or documents is return depends on several factors:
If your container is of type WholedocContainer, then entire documents are always returned in this method's results set.
If your container is of type NodeContainer, and if you specified DBXML_INDEX_NODES when you created your container, then this method returns the nodes to which the index's keys refer.
For example, every container is created with a default index that ensures the uniqueness of the document names in your container. The:
URI is http://www.sleepycat.com/2002/dbxml.
Node name is name.
Indexing strategy is unique-node-metadata-equality-string.
Given this, you can efficiently retrieve every document in the container using XmlContainer::lookupIndex() as follows:
#include "DbXml.hpp"
...
using namespace DbXml;
...
// Get a manager object.
XmlManager myManager;
// Open a container
XmlContainer myContainer = 
    myManager.openContainer("exampleData.dbxml");
XmlQueryContext qc = myManager.createQueryContext();
// Lookup the index
std::string uri = "http://www.sleepycat.com/2002/dbxml";
std::string name = "name";
std::string idxStrategy = "unique-node-metadata-equality-string";
// Now look it up. This returns every document in the container.
XmlResults res = myContainer.lookupIndex(qc, uri, name, idxStrategy);
// Iterate over the results set, printing each document in it
XmlDocument thedoc = myManager.createDocument();
while (res.next(thedoc)) {
    std::string dummyString;
    std::cout << thedoc.getName() << ": " 
              << thedoc.getContent(dummyString) << std::endl;
} 
        In the event that you want to lookup an edge index, you must provide the lookup method with both the node and the parent node that together comprise the XML edge.
For example, suppose you have the following document in your container:
<mydoc>
   <node1>
        <node2>
            node2 1 
        </node2>
        <node2>
            node2 2
        </node2>
   </node1>
</mydoc> 
        Further suppose you indexed the presence of the node1/node2 edges. In this case, you can lookup the values referred to by this index by doing the following:
#include "DbXml.hpp"
...
using namespace DbXml;
...
// Get a manager object.
XmlManager myManager;
// Open a container
XmlContainer myContainer = 
    myManager.openContainer("exampleData.dbxml");
XmlQueryContext qc = myManager.createQueryContext();
// Node to lookup
std::string uri = "";
std::string name = "node2";
// Parent node to lookup
std::string parentURI = "";
std::string parentName = "node1";
std::string idxStrategy = "edge-element-presence";
// Now look it up.
XmlResults res = myContainer.lookupIndex(qc, uri, name, 
    parentURI, parentName, idxStrategy);
// Iterate over the results set, printing each value in it
XmlValue retValue;
while (res.next(retValue)) {
    std::cout << "Found: " << retValue.asString() << std::endl;
}