Source: helma/Search.js

//
// A wrapper for Apache Lucene for use with Helma Object Publisher
// Copyright (c) 2005-2006 Robert Gaggl
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2.1 of the License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
//
//
// $Revision$
// $Author$
// $Date$
//


/**
 * @fileoverview Fields and methods of the helma.Search class
 * <br /><br />
 * To use this optional module, its repository needs to be added to the 
 * application, for example by calling app.addRepository('modules/helma/Search.js')
 */

// take care of any dependencies
app.addRepository('modules/helma/lucene-core.jar');
app.addRepository('modules/helma/lucene-analyzers.jar');


if (!global.helma) {
    global.helma = {};
}

/**
 * Constructs a new instance of helma.Search. This merely
 * checks if the Apache Lucene library is in the application
 * classpath.
 * @class This class provides functionality for
 * creating a fulltext search index based on Apache Lucene.
 * @returns A newly created instance of this prototype.
 * @constructor
 * @author Robert Gaggl <robert@nomatic.org> 
 */
helma.Search = function() {
    try {
        var c = java.lang.Class.forName("org.apache.lucene.analysis.Analyzer",
                                        true, app.getClassLoader());
        var pkg = Packages.org.apache.lucene.LucenePackage.get();
        var version = parseFloat(pkg.getSpecificationVersion());
        if (version < 2.2) {
            throw "Incompatible version";
        }
    } catch (e) {
        throw "helma.Search needs lucene.jar in version >= 2.2 \
               in lib/ext or application directory \
               [http://lucene.apache.org/]";
    }
    return this;
};


/***************************************
 ***** S T A T I C   M E T H O D S *****
 ***************************************/

/** @ignore */
helma.Search.toString = function() {
    return "[helma.Search]";
};

/**
 * Returns a new Analyzer instance depending on the key
 * passed as argument. Currently supported arguments are
 * "br" (BrazilianAnalyzer), "cn" (ChineseAnalyzer), "cz" (CzechAnalyzer),
 * "nl" (DutchAnalyzer), "fr" (FrenchAnalyzer), "de" (GermanAnalyzer),
 * "el" (GreekAnalyzer), "keyword" (KeywordAnalyzer), "ru" (RussianAnalyzer),
 * "simple" (SimpleAnalyzer), "snowball" (SnowballAnalyzer), "stop" (StopAnalyzer)
 * "whitespace" (WhitespaceAnalyzer). If no argument is given, a StandardAnalyzer
 * is returned.
 * @param {String} key The key identifying the analyzer
 * @returns A newly created Analyzer instance
 * @type {org.apache.lucene.analysis.Analyzer}
 */
helma.Search.getAnalyzer = function(key) {
    var pkg = Packages.org.apache.lucene;
    switch (key) {
        case "br":
            return new pkg.analysis.br.BrazilianAnalyzer();
        case "cn":
            return new pkg.analysis.cn.ChineseAnalyzer();
        case "cz":
            return new pkg.analysis.cz.CzechAnalyzer();
        case "nl":
            return new pkg.analysis.nl.DutchAnalyzer();
        case "fr":
            return new pkg.analysis.fr.FrenchAnalyzer();
        case "de":
            return new pkg.analysis.de.GermanAnalyzer();
        case "el":
            return new pkg.analysis.el.GreekAnalyzer();
        case "keyword":
            return new pkg.analysis.KeywordAnalyzer();
        case "ru":
            return new pkg.analysis.ru.RussianAnalyzer();
        case "simple":
            return new pkg.analysis.SimpleAnalyzer();
        case "snowball":
            return new pkg.analysis.snowball.SnowballAnalyzer();
        case "stop":
            return new pkg.analysis.StopAnalyzer();
        case "whitespace":
            return new pkg.analysis.WhitespaceAnalyzer();
        default:
            return new pkg.analysis.standard.StandardAnalyzer();
    }
};

/**
 * Constructs a new QueryFilter instance. This class
 * wraps a lucene QueryFilter.
 * @param {helma.Search.Query} q The query object to use as
 * the basis for the QueryFilter instance.
 * @returns A newly created QueryFilter instance.
 * @constructor
 */
helma.Search.QueryFilter = function(q) {
    var filter = new Packages.org.apache.lucene.search.QueryFilter(q.getQuery());

    /**
     * Returns the wrapped filter instance
     * @type org.apache.lucene.search.QueryFilter
     */
    this.getFilter = function() {
        return filter;
    };

    return this;
};

/** @ignore */
helma.Search.QueryFilter.prototype.toString = function() {
    return this.getFilter().toString();
};



/*********************************************
 ***** P R O T O T Y P E   M E T H O D S *****
 *********************************************/


/** @ignore */
helma.Search.prototype.toString = function() {
    return helma.Search.toString();
};

/**
 * Returns an instance of org.apache.lucene.store.FSDirectory. If
 * no index is present in the given directory, it is created on the fly.
 * @param {File|helma.File|java.io.File|String} dir The directory
 * where the index is located or should be created at.
 * @param {Boolean} create If true the index will be created, removing
 * any existing index in the same directory
 * @returns The index directory.
 * @type org.apache.lucene.store.FSDirectory
 */
helma.Search.prototype.getDirectory = function(dir, create) {
    if (!dir) {
        throw "helma.Search.getDirectory(): insufficient arguments.";
    }
    var d;
    if (dir.constructor == String) {
        d = new java.io.File(dir);
    } else if (dir.constructor == File || dir.constructor == helma.File) {
        d = new java.io.File(dir.getAbsolutePath());
    } else if (!((d = dir) instanceof java.io.File)) {
        throw "helma.Search.getDirectory(): " + dir + " is not a valid argument.";
    }
    return Packages.org.apache.lucene.store.FSDirectory.getDirectory(d,
               create === true || !d.exists());
};

/**
 * Returns a RAM directory object.
 * @param {File|helma.File|java.io.File|String} dir Optional directory
 * containing a Lucene index from which this RAM directory should be created.
 * @returns A RAM directory instance.
 * @type org.apache.lucene.store.RAMDirectory
 */
helma.Search.prototype.getRAMDirectory = function(dir) {
    if (dir != null) {
        var d;
        if (dir.constructor == String) {
            d = new java.io.File(dir);
        } else if (dir.constructor == File || dir.constructor == helma.File) {
            d = new java.io.File(dir.getAbsolutePath());
        } else if (!((d = dir) instanceof java.io.File)) {
            throw "helma.Search.getRAMDirectory(): " + dir + " is not a valid argument.";
        }
        if (!d.exists()) {
            throw "helma.Search.getRAMDirectory(): " + dir + " does not exist.";
        }
        return Packages.org.apache.lucene.store.RAMDirectory(d);
    }
    return Packages.org.apache.lucene.store.RAMDirectory();
};

/**
 * Creates a new Lucene index in the directory passed as
 * argument, using an optional analyzer, and returns an instance
 * of helma.Search.Index. Any already existing index in this
 * directory will be preserved.
 * @param {org.apache.lucene.store.Directory} dir The directory
 * where the index should be stored. This can be either a
 * FSDirectory or a RAMDirectory instance.
 * @param {org.apache.lucene.analysis.Analyzer} analyzer The analyzer to
 * use for the index. If not specified a StandardAnalyzer will be used.
 * @returns The index instance.
 * @type helma.Search.Index
 */
helma.Search.prototype.createIndex = function(dir, analyzer) {
    if (!dir || !(dir instanceof Packages.org.apache.lucene.store.Directory)) {
        throw "Index directory missing or invalid.";
    } else if (!analyzer) {
        // no analyzer given, use a StandardAnalyzer
        analyzer = helma.Search.getAnalyzer();
    }
    var index = new helma.Search.Index(dir, analyzer);
    if (!Packages.org.apache.lucene.index.IndexReader.indexExists(dir)) {
        index.create();
    }
    return index;
};


/*********************
 ***** I N D E X *****
 *********************/


/**
 * Creates a new instance of helma.Search.Index
 * @class Instances of this class represent a Lucene search index
 * located in either a directory on disk or in RAM. This class
 * provides various methods for modifying the underlying Lucene index.
 * @param {org.apache.lucene.store.Directory} directory The directory
 * where the Lucene index is located at.
 * @param {org.apache.lucene.analysis.Analyzer} analyzer The analyzer
 * to use when modifying the index.
 * @constructor
 */
helma.Search.Index = function(directory, analyzer) {

    /**
     * Returns an IndexWriter instance that can be used to add documents to
     * the underlying index or to perform various other modifying operations.
     * If the index is currently locked this method will try for the next
     * two seconds to create the IndexWriter, otherwise it will
     * throw an error.
     * @param {Boolean} create True to create the index (overwriting an
     * existing index), false to append to an existing index. Defaults to false
     * @param {Boolean} autoCommit Enables or disables auto commit (defaults to
     * false)
     * @returns An IndexWriter instance.
     * @type org.apache.lucene.index.IndexWriter
     */
    this.getWriter = function(create, autoCommit) {
        try {
            return new Packages.org.apache.lucene.index.IndexWriter(directory,
                               (autoCommit === true), analyzer, (create === true));
        } catch (e) {
            throw "Failed to get IndexWriter due to active lock (Thread "
                  + java.lang.Thread.currentThread().getId() + ")";
        }
    };

    /**
     * Returns an IndexReader instance. Due to locking issues an
     * IndexModifier should be used for deleting documents.
     * @returns An IndexReader instance
     * @type org.apache.lucene.index.IndexReader
     */
    this.getReader = function() {
        return Packages.org.apache.lucene.index.IndexReader.open(directory);
    };

    /**
     * Returns the directory the underlying Lucene index is located at.
     * @returns The directory of this index
     * @type org.apache.lucene.store.Directory
     */
    this.getDirectory = function() {
        return directory;
    };

    /**
     * Returns the analyzer used within this index.
     * @returns The analyzer used within this index.
     * @type org.apache.lucene.analysis.Analyer
     */
    this.getAnalyzer = function() {
        return analyzer;
    };
    
    /**
     * Returns a searcher for querying this index.
     * @returns A searcher useable for querying the index.
     * @type helma.Search.Searcher
     */
    this.getSearcher = function() {
        return new helma.Search.Searcher(this);
    };
    
    /** @ignore */
    this.toString = function() {
        return ("[Lucene Index " + directory + "]");
    };

    return this;
};

/**
 * Merges the indexes passed as argument into this one.
 * @param {org.apache.lucene.store.Directory} dir At least one
 * index director to add to this index.
 */
helma.Search.Index.prototype.addIndexes = function(dir /* [, dir[, dir...] */) {
    var dirs = java.lang.reflect.Array.newInstance(
                    Packages.org.apache.lucene.store.Directory,
  	                arguments.length);
    for (var i=0;i<arguments.length;i++) {
        dirs[i] = arguments[i];
    }
    try {
        var writer = this.getWriter();
        writer.addIndexes(dirs);
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return;
};

/**
 * Creates a new index. This will delete any existing index
 * files in the directory of this index.
 * @returns True if creating the index was successful, false otherwise
 * @type Boolean
 */
helma.Search.Index.prototype.create = function() {
    var dir = this.getDirectory();
    if (dir instanceof Packages.org.apache.lucene.store.FSDirectory) {
        // FIXME: IndexWriter is supposed to remove files
        // if instantiated with create == true, but for some reason
        // it doesn't, so instead use FSDirectory.getDirectory()
        // for deletion and then re-create the index segments file
        Packages.org.apache.lucene.store.FSDirectory.getDirectory(dir.getFile(), true);
    }
    try {
        var writer = this.getWriter(true);
        return true;
    } catch (e) {
        app.logger.warn("Unable to create the index " + this.getDirectory());
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return false;
};

/**
 * Returns the number of documents in this index.
 * @returns The number of documents in this index.
 * @type Number
 */
helma.Search.Index.prototype.size = function() {
    try {
        var reader = this.getReader();
        return reader.numDocs();
    } finally {
        if (reader != null) {
            reader.close();
        }
    }
    return;
};

/**
 * Returns the number of documents with the given field name and value.
 * @param {String} fieldName The name of the field
 * @param {String} fieldValue The value of the field
 * @returns The number of documents
 * @type Number
 */
helma.Search.Index.prototype.count = function(fieldName, fieldValue) {
    try {
        var reader = this.getReader();
        var term = new Packages.org.apache.lucene.index.Term(fieldName, fieldValue);
        return reader.docFreq(term);
    } finally {
        if (reader !== null) {
            reader.close();
        }
    }
};

/**
 * Optimizes the underlying index.
 */
helma.Search.Index.prototype.optimize = function() {
    try {
        var writer = this.getWriter();
        writer.optimize();
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return;
};

/**
 * Returns an array containing all field names in this index.
 * @returns An array with the field names in this index.
 * @type Array
 */
helma.Search.Index.prototype.getFieldNames = function() {
    try {
        var reader = this.getReader();
        var coll = reader.getFieldNames().toArray();
        // convert java array into javascript array
        var result = [];
        for (var i in coll) {
            result.push(coll[i]);
        }
        return result;
    } finally {
        if (reader != null) {
            reader.close();
        }
    }
    return;
};

/**
 * Checks if the index is locked.
 * @returns True if the underlying index is locked,
 * false otherwise.
 * @type Boolean
 */
helma.Search.Index.prototype.isLocked = function() {
    return Packages.org.apache.lucene.index.IndexReader.isLocked(this.getDirectory());
};

/**
 * Unlocks the index. Use this with caution, as it removes
 * any active locks in the Lucene index, which might lead
 * to index corruption.
 */
helma.Search.Index.prototype.unlock = function() {
    Packages.org.apache.lucene.index.IndexReader.unlock(this.getDirectory());
    return;
};

/**
 * Closes the underlying index directory for future operations.
 */
helma.Search.Index.prototype.close = function() {
    this.getDirectory().close();
    return;
};

/**
 * Adds a document to the index.
 * @param {helma.Search.Document} doc The document to add to the index.
 */
helma.Search.Index.prototype.addDocument = function(doc) {
    try {
        var writer = this.getWriter();
        writer.addDocument(doc.getDocument());
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return;
};

/**
 * Adds all documents in the passed collection to this index.
 * @param {java.util.Hashtable|java.util.Vector|Array} docs
 * The documents to add to the index.
 */
helma.Search.Index.prototype.addDocuments = function(docs, mergeFactor) {
    try {
        var writer = this.getWriter();
        if (mergeFactor) {
            writer.setMergeFactor(mergeFactor);
        }
        if (docs instanceof java.util.Hashtable || docs instanceof java.util.Vector) {
            var e = docs.elements();
            while (e.hasMoreElements()) {
                writer.addDocument(e.nextElement().getDocument());
            }
        } else if (docs instanceof Array) {
            for (var i=0; i<docs.length; i+=1) {
                writer.addDocument(docs[i].getDocument());
            }
        }
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return;
};

/**
 * Remove all documents from the index whose field-value
 * with the given name matches the passed value argument.
 * @param {String} fieldName The name of the field
 * @param {String} fieldValue The value of the field.
 */
helma.Search.Index.prototype.removeDocument = function(fieldName, fieldValue) {
    try {
        var writer = this.getWriter();
        var term = new Packages.org.apache.lucene.index.Term(fieldName,
                           helma.Search.Document.Field.valueToString(fieldValue));
        writer.deleteDocuments(term);
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return;
};

/**
 * Removes all documents whose field with the given name matches
 * the values passed as argument.
 * @param {String} fieldName The name of the field
 * @param {java.util.Hashtable|java.util.Vector|Array} values
 * The values that define the documents that should be removed from
 * the index.
 */
helma.Search.Index.prototype.removeDocuments = function(fieldName, values) {
    try {
        var writer = this.getWriter();
        var term, value;
        if (values instanceof java.util.Hashtable ||
                values instanceof java.util.Vector) {
            var e = values.elements();
            while (e.hasMoreElements()) {
                value = e.nextElement();
                term = new Packages.org.apache.lucene.index.Term(fieldName,
                               helma.Search.Document.Field.valueToString(value));
                writer.deleteDocuments(term);
            }
        } else if (values instanceof Array) {
            for (var i=0;i<values.length;i++) {
                value = values[i];
                term = new Packages.org.apache.lucene.index.Term(fieldName,
                               helma.Search.Document.Field.valueToString(value));
                writer.deleteDocuments(term);
            }
        }
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return;
};

/**
 * Updates the index with the document passed as argument. In contrast
 * to addDocument() this removes any existing objects whose fieldName
 * matches the one of the document object. Eg. if the document object
 * has a field "Id" with the value "123", all document objects whose
 * fieldName "Id" matches "123" will be removed from the index before.
 * @param {helma.Search.Document} docObj Document object to add to index.
 * @param {String} fieldName The name of the identifier field.
 */
helma.Search.Index.prototype.updateDocument = function(docObj, fieldName) {
    try {
        var writer = this.getWriter();
        var doc = docObj.getDocument();
        var term = new Packages.org.apache.lucene.index.Term(fieldName, doc.get(fieldName));
        writer.updateDocument(term, doc);
    } finally {
        if (writer != null) {
            writer.close();
        }
    }
    return;
};

/**
 * Creates a new instance of helma.Search.Seacher
 * @class This class provides basic functionality for
 * searching an index.
 * @param {helma.Search.Index} index The index to search in.
 * @returns A newly created index searcher
 * @constructor
 */
helma.Search.Searcher = function(index) {
    var searcher = new Packages.org.apache.lucene.search.IndexSearcher(index.getDirectory());

    /**
     * The search results.
     * @type helma.Search.HitCollection
     */
    this.hits = null;

    /**
     * A vector with SortField instances, if any have been defined.
     * @type java.util.Vector
     * @see #sortBy
     */
    this.sortFields = null;

    /**
     * Returns the wrapped IndexSearcher instance.
     * @type org.apache.lucene.search.IndexSearcher
     */
    this.getSearcher = function() {
        return searcher;
    };

    return this;
};

/** @ignore */
helma.Search.Searcher.prototype.toString = function() {
    return "[Index Searcher]";
};

/**
 * Searches an index using the query passed as argument.
 * The resulting collection of hits is stored in the property "hits"
 * of this Searcher instance. Don't forget to close the searcher
 * when finished processing its hits.
 * @param {helma.Search.Query} query The query to use for searching
 * @param {helma.Search.QueryFilter} filter An optional query filter
 * for filtering the results.
 * @returns The number of hits.
 * @type Number
 */
helma.Search.Searcher.prototype.search = function(query, filter) {
    var pkg = Packages.org.apache.lucene.search;
    var searcher = this.getSearcher();
    var hits;
    if (query instanceof helma.Search.Query) {
       query = query.getQuery();
    }
    if (filter != null && filter instanceof helma.Search.QueryFilter) {
       filter = filter.getFilter();
    }
    if (this.sortFields != null && this.sortFields.size() > 0) {
        var arr = java.lang.reflect.Array.newInstance(pkg.SortField, this.sortFields.size());
        var sort = pkg.Sort(this.sortFields.toArray(arr));
        if (filter) {
            hits = searcher.search(query, filter, sort);
        } else {
            hits = searcher.search(query, sort);
        }
    } else if (filter) {
        hits = searcher.search(query, filter);
    } else {
        hits = searcher.search(query);
    }
    this.hits = new helma.Search.HitCollection(hits);
    return this.hits.length();
};

/**
 * Sets a field as result sorting field. This method can be called
 * with a different number of arguments:
 * sortBy(fieldName)
 * sortBy(fieldName, type)
 * sortBy(fieldName, reverse)
 * sortBy(fieldName, type, reverse)
 * @param {String} fieldName The name of the field in the index by which
 * the search result should be ordered.
 * @param {String} type The type of the field defined by argument fieldName.
 * Valid arguments are "string", "float", "int", "score", "doc", "auto", "custom".
 * Default is "auto". See http://lucene.apache.org/java/docs/api/org/apache/lucene/search/SortField.html
 * for an explanation.
 */
helma.Search.Searcher.prototype.sortBy = function(fieldName /** type, reverse */) {
    var pkg = Packages.org.apache.lucene.search;
    if (!this.sortFields)
        this.sortFields = new java.util.Vector();
    var f = fieldName;
    var t = pkg.SortField.AUTO;
    var r = false;
    if (arguments.length == 3) {
        t = pkg.SortField[arguments[1].toUpperCase()];
        r = arguments[2];
    } else if (arguments.length == 2) {
        if (arguments[1].constructor == Boolean) {
            r = arguments[1];
        } else {
            t = pkg.SortField[arguments[1].toUpperCase()];
        }
    }
    this.sortFields.add(new pkg.SortField(f, t, r));
    return;
};

/**
 * Closes the wrapped IndexSearcher instance.
 */
helma.Search.Searcher.prototype.close = function() {
    var s = this.getSearcher();
    if (s != null) {
        s.close();
    }
    return;
};

/**
 * Creates a new instance of helma.Search.HitCollection
 * @class This class provides Helma-like methods for accessing
 * a collection of search hits.
 * @param {org.lucene.search.Hits} hits The hit collection returned
 * by lucene.
 * @constructor
 */
helma.Search.HitCollection = function(hits) {
    /**
     * Silently converts the hit at the given index position into
     * an instance of helma.Search.Document.
     * @param {Number} idx The index position of the hit
     * @returns The document object at the given index position
     * @type helma.Search.Document
     */
    this.get = function(idx) {
        var doc = new helma.Search.Document(hits.doc(idx));
        doc.id = hits.id(idx);
        doc.score = hits.score(idx);
        doc.rank = idx +1;
        return doc;
    };

    /**
     * Returns the number of hits in this collection.
     * @returns The number of hits.
     * @type Number
     */
    this.size = function() {
        return (hits != null) ? hits.length() : 0;
    };

    /**
     * Returns the number of hits in this collection.
     * This method is deprecated, use {@link #size} instead.
     * @returns The number of hits.
     * @type Number
     * @deprecated
     * @see #size
     */
    this.length = function() {
        return this.size();
    };
    
    /**
     * Executes a provided function once per hit.
     * @param {Function} fun Function to execute for each element
     * @param {Object} context Object to use as "this" when executing callback.
     * @see https://developer.mozilla.org/En/Core_JavaScript_1.5_Reference/Objects/Array/ForEach
     */
    this.forEach = function(func, context) {
        if (typeof func != "function") {
            throw new TypeError();
        }
        var len = this.size();
        for (var i = 0; i < len; i += 1) {
            var hit = this.get(i);
            if (hit) {
                func.call(context, hit, i, hits);
            }
        }
        return;
    };
    
    return this;
};

/** @ignore */
helma.Search.HitCollection.prototype.toString = function() {
   return "[HitCollection]";
};

/***********************************************
 ***** Q U E R Y   C O N S T R U C T O R S *****
 ***********************************************/

/**
 * Creates a new instance of helma.Search.Query
 * @class Base class for the various query constructors. Don't
 * call this directly, as it provides just basic methods used
 * in all of its extends.
 * @constructor
 */
helma.Search.Query = function() {
    /**
     * The wrapped query instance
     * @type org.apache.lucene.search.Query
     * @private
     */
    this.query = null;

    return this;
};

/**
 * Returns the wrapped Lucene Query object.
 * @returns The wrapped query object
 * @type org.apache.lucene.search.Query
 */
helma.Search.Query.prototype.getQuery = function() {
    return this.query;
};

/** @ignore */
helma.Search.Query.prototype.toString = function(field) {
    return "[" + this.getQuery().toString(field) + "]";
};

/**
 * Returns the boost factor of this query.
 * @type Number
 */
helma.Search.Query.prototype.getBoost = function() {
    return this.getQuery().getBoost();
};

/**
 * Sets the boost factor of this query clause to
 * the given number. Documents matching this query
 * will have their score multiplied with the given
 * factor
 * @param {Number} fact The factor to multiply the score
 * of matching documents with.
 */
helma.Search.Query.prototype.setBoost = function(fact) {
    this.getQuery().setBoost(fact);
    return;
};

/**
 * Creates a new instance of helma.Search.TermQuery
 * @class This class represents a simple Term Query.
 * @param {String} field The name of the field
 * @param {String} str The value of the field
 * @constructor
 * @extends helma.Search.Query
 */
helma.Search.TermQuery = function(field, str) {
    var t = new Packages.org.apache.lucene.index.Term(field, str);
    /**
     * Contains the wrapped TermQuery instance
     * @type org.apache.lucene.search.TermQuery
     */
    this.query = new Packages.org.apache.lucene.search.TermQuery(t);
    return this;
};
helma.Search.TermQuery.prototype = new helma.Search.Query;

/**
 * Creates a new instance of helma.Search.BooleanQuery
 * @class This class represents a Boolean Query, providing
 * various methods for combining other Query instances using boolean operators.
 * @param String name of the field
 * @param String query string
 * @returns Object BooleanQuery object
 * @constructor
 * @extends helma.Search.Query
 */
helma.Search.BooleanQuery = function(field, str, clause, analyzer) {
    /**
     * Contains the wrapped BooleanQuery instance
     * @type org.apache.lucene.search.BooleanQuery
     */
    this.query = new Packages.org.apache.lucene.search.BooleanQuery();

    /**
     * Main constructor body
     */
    if (field && str) {
        this.addTerm.apply(this, arguments);
    }

    return this;
};
helma.Search.BooleanQuery.prototype = new helma.Search.Query;

/**
 * Adds a term to the wrapped query object. This method can be called
 * with two, three or four arguments, eg.:
 * <pre>addTerm("fieldname", "querystring")
 * addTerm("fieldname", "querystring", "and")
 * addTerm("fieldname", "querystring", helma.Search.getAnalyzer("de"))
 * addTerm("fieldname", "querystring", "not", helma.Search.getAnalyzer("simple"))</pre>
 * @param {String|Array} field Either a String or an Array containing Strings
 * that determine the index field(s) to match
 * @param {String} str Query string to match
 * @param {String} clause Boolean clause ("or", "not" or "and", default is "and")
 * @param {org.apache.lucene.analysis.Analyzer} analyzer An analyzer to use
 */
helma.Search.BooleanQuery.prototype.addTerm = function(field, str, clause, analyzer) {
    var pkg = Packages.org.apache.lucene;
    if (arguments.length == 3 && arguments[2] instanceof pkg.analysis.Analyzer) {
        analyzer = arguments[2];
        clause = "or";
    }
    if (!analyzer) {
        analyzer = helma.Search.getAnalyzer();
    }

    var fields = (field instanceof Array) ? field : [field];
    var parser = new pkg.queryParser.MultiFieldQueryParser(fields, analyzer);
    this.addQuery(parser.parse(str), clause);
    return;
};

/**
 * Adds an additional query clause to this query.
 * @param {helma.Search.Query} q The query to add
 * @param {String} clause Boolean clause ("or", "not", or "and", default is "and")
 */
helma.Search.BooleanQuery.prototype.addQuery = function(q, clause) {
    var pkg = Packages.org.apache.lucene;
    var booleanClause;
    if (q instanceof helma.Search.Query) {
        q = q.getQuery();
    }
    switch (clause) {
        case "and":
            booleanClause = pkg.search.BooleanClause.Occur.MUST;
            break;
        case "not":
            booleanClause = pkg.search.BooleanClause.Occur.MUST_NOT;
            break;
        default:
            booleanClause = pkg.search.BooleanClause.Occur.SHOULD;
            break;
    }
    this.getQuery().add(q, booleanClause);
    return;
};

/**
 * Constructs a new helma.Search.PhraseQuery instance that wraps
 * a Lucene Phrase Query object.
 * @class Instances of this class represent a phrase query.
 * @param {String} field The name of the field
 * @param {String} str The phrase query string
 * @returns A newly created PhraseQuery instance
 * @constructor
 * @extends helma.Search.Query
 */
helma.Search.PhraseQuery = function(field, str) {
    /**
     * Contains the wrapped PhraseQuery instance
     * @type org.apache.lucene.search.PhraseQuery
     */
    this.query = new Packages.org.apache.lucene.search.PhraseQuery();

    /**
     * add a term to the end of the phrase query
     */
    this.addTerm = function(field, str) {
        var t = new Packages.org.apache.lucene.index.Term(field, str);
        this.query.add(t);
        return;
    };

    if (field && str)
        this.addTerm(field, str);
    delete this.base;
    return this;
};
helma.Search.PhraseQuery.prototype = new helma.Search.Query;

/**
 * Constructs a new helma.Search.RangeQuery instance.
 * @class Instances of this class represent a range
 * query, wrapping a Lucene RangeQuery instance.
 * @param {String} field The name of the field
 * @param {String} from The minimum value to match (can be null)
 * @param {String} to The maximum value to match (can be null)
 * @param {Boolean} inclusive If true the given min/max values are included
 * @returns A newly created RangeQuery instance
 * @constructor
 * @extends helma.Search.Query
 */
helma.Search.RangeQuery = function(field, from, to, inclusive) {
    if (!field)
        throw "Missing field name in RangeQuery()";
    if (arguments.length < 4)
        inclusive = true;
    var t1 = from ? new Packages.org.apache.lucene.index.Term(field, from) : null;
    var t2 = to ? new Packages.org.apache.lucene.index.Term(field, to) : null;
    /**
     * Contains the wrapped RangeQuery instance
     * @type org.apache.lucene.search.RangeQuery
     */
    this.query = new Packages.org.apache.lucene.search.RangeQuery(t1, t2, inclusive);
    return this;
};
helma.Search.RangeQuery.prototype = new helma.Search.Query;

/**
 * Constructs a new helma.Search.FuzzyQuery instance.
 * @class Instances of this class represent a fuzzy query
 * @param {String} field The name of the field
 * @param {String} str The query string to match
 * @returns A newly created FuzzyQuery instance
 * @constructor
 * @extends helma.Search.Query
 */
helma.Search.FuzzyQuery = function(field, str) {
    var t = new Packages.org.apache.lucene.index.Term(field, str);
    /**
     * Contains the wrapped FuzzyQuery instance
     * @type org.apache.lucene.search.FuzzyQuery
     */
    this.query = new Packages.org.apache.lucene.search.FuzzyQuery(t);
    return this;
};
helma.Search.FuzzyQuery.prototype = new helma.Search.Query;

/**
 * Constructs a new helma.Search.PrefixQuery instance.
 * @class Instances of this class represent a prefix query
 * @param {String} field The name of the field
 * @param {String} str The query string to match
 * @returns A newly created PrefixQuery instance
 * @constructor
 * @extends helma.Search.Query
 */
helma.Search.PrefixQuery = function(field, str) {
    var t = new Packages.org.apache.lucene.index.Term(field, str);
    /**
     * Contains the wrapped PrefixQuery instance
     * @type org.apache.lucene.search.PrefixQuery
     */
    this.query = new Packages.org.apache.lucene.search.PrefixQuery(t);
    return this;
};
helma.Search.PrefixQuery.prototype = new helma.Search.Query;

/**
 * Constructs a new helma.Search.WildcardQuery instance.
 * @class Instances of this class represent a wildcard query.
 * @param {String} field The name of the field
 * @param {String} str The query string to match
 * @returns A newly created WildcardQuery instance
 * @constructor
 * @extends helma.Search.Query
 */
helma.Search.WildcardQuery = function(field, str) {
    var t = new Packages.org.apache.lucene.index.Term(field, str);
    /**
     * Contains the wrapped WildcardQuery instance
     * @type org.apache.lucene.search.WildcardQuery
     */
    this.query = new Packages.org.apache.lucene.search.WildcardQuery(t);
    return this;
};
helma.Search.WildcardQuery.prototype = new helma.Search.Query;



/***************************
 ***** D O C U M E N T *****
 ***************************/


/**
 * Creates a new instance of helma.Search.Document.
 * @class Instances of this class represent a single
 * index document. This class provides various methods for
 * adding content to such documents.
 * @param {org.apache.lucene.document.Document} document Optional Lucene Document object
 * that should be wrapped by this Document instance.
 * @constructor
 */
helma.Search.Document = function(document) {
    if (!document) {
        document = new Packages.org.apache.lucene.document.Document();
    }

    /**
     * Returns the wrapped Lucene Document object
     * @returns The wrapped Document object
     * @type org.apache.lucene.document.Document
     */
    this.getDocument = function() {
        return document;
    };

    return this;
};

/**
 * Adds a field to this document.
 * @param {String|helma.Search.Document.Field} name The name of the field, or
 * an instance of {@link helma.Search.Document.Field}, in which case the other
 * arguments are ignored.
 * @param {String} value The value of the field
 * @param {Object} options Optional object containing the following properties
 * (each of them is optional too):
 * <ul>
 * <li><code>store</code> (String) Defines whether and how the value is stored
 * in the field. Accepted values are "no", "yes" and "compress" (defaults to "yes")</li>
 * <li><code>index</code> (String) Defines whether and how the value is indexed
 * in the field. Accepted values are "no", "tokenized", "unTokenized" and
 * "noNorms" (defaults to "tokenized")</li>
 * <li><code>termVector</code> (String) Defines if and how the fiels should have
 * term vectors. Accepted values are "no", "yes", "withOffsets", "withPositions"
 * and "withPositionsAndOffsets" (defaults to "no")</li>
 * </ul>
 */
helma.Search.Document.prototype.addField = function(name, value, options) {
    if (!name) {
        throw "helma.Search: missing arguments to Document.addField";
    } else if (arguments.length == 1) {
        if (arguments[0] instanceof Packages.org.apache.lucene.document.Field) {
            this.getDocument().add(arguments[0]);
        } else if (arguments[0] instanceof helma.Search.Document.Field) {
            this.getDocument().add(arguments[0].getField());
        }
        return;
    }

    // for backwards compatibility only
    if (options != null) {
        if (typeof(options.store) === "boolean") {
            options.store = (options.store === true) ? "yes" : "no";
        }
        if (typeof(options.index) === "boolean") {
            if (options.index === true) {
                options.index = (options.tokenize === true) ? "tokenized" : "unTokenized";
            } else {
                options.index = "no";
            }
            delete options.tokenize;
        }
    }
    this.addField(new helma.Search.Document.Field(name, value, options));
    return;
};

/**
 * Returns a single document field.
 * @param {String} name The name of the field in this document object.
 * @returns The field with the given name
 * @type helma.Search.Document.Field
 */
helma.Search.Document.prototype.getField = function(name) {
    var f = this.getDocument().getField(name);
    if (f != null) {
        return new helma.Search.Document.Field(f);
    }
    return null;
};

/**
 * Returns the fields of a document object. If a name is passed as argument,
 * this method returns only the fields with the given name
 * @param {String} name Optional name of the fields to return
 * @returns An array containing all fields in this document object.
 * @type Array
 */
helma.Search.Document.prototype.getFields = function(name) {
    var fields;
    if (name != null) {
        fields = this.getDocument().getFields(name);
    } else {
        fields = this.getDocument().getFields().toArray();
    }
    var size = fields.length;
    var result = new Array(size);
    for (var i=0; i<size; i+=1) {
        result[i] = new helma.Search.Document.Field(fields[i]);
    }
    return result;
};

/**
 * Removes all fields with the given name from this document
 * @param {String} name The name of the field(s) to remove
 */
helma.Search.Document.prototype.removeField = function(name) {
    this.getDocument().removeFields(name);
    return;
};

/**
 * Returns the boost factor of a document.
 * @returns The boost factor of a document
 * @type Number
 */
helma.Search.Document.prototype.getBoost = function() {
    return this.getDocument().getBoost();
};

/**
 * Sets the boost factor of a document.
 * @param {Number} boost The boost factor of the document
 */
helma.Search.Document.prototype.setBoost = function(boost) {
    this.getDocument().setBoost(boost);
    return;
};

/** @ignore */
helma.Search.Document.prototype.toString = function() {
    return "[Document]";
};


/*********************
 ***** F I E L D *****
 *********************/


/**
 * Creates a new Field instance
 * @class Instances of this class represent a single field
 * @param {Object} name The name of the field
 * @param {Object} value The value of the field
 * @param {Object} options Optional object containing the following properties
 * (each of them is optional too):
 * <ul>
 * <li><code>store</code> (String) Defines whether and how the value is stored
 * in the field. Accepted values are "no", "yes" and "compress" (defaults to "yes")</li>
 * <li><code>index</code> (String) Defines whether and how the value is indexed
 * in the field. Accepted values are "no", "tokenized", "unTokenized" and
 * "noNorms" (defaults to "tokenized")</li>
 * <li><code>termVector</code> (String) Defines if and how the fiels should have
 * term vectors. Accepted values are "no", "yes", "withOffsets", "withPositions"
 * and "withPositionsAndOffsets" (defaults to "no")</li>
 * </ul>
 */
helma.Search.Document.Field = function(name, value, options) {
    var field;

    /**
     * Contains the name of the field
     * @type String
     */
    this.name = undefined; // for documentation purposes only
    this.__defineGetter__("name", function() {
        return field.name();
    });

    /**
     * Contains the string value of the field
     * @type String
     */
    this.value = undefined; // for documentation purposes only
    this.__defineGetter__("value", function() {
        return field.stringValue();
    });

    /**
     * Contains the value of the field converted into a date object.
     * @type String
     */
    this.dateValue = undefined; // for documentation purposes only
    this.__defineGetter__("dateValue", function() {
        return Packages.org.apache.lucene.document.DateTools.stringToDate(this.value);
    });

    /**
     * Returns the wrapped field instance
     * @returns The wrapped field
     * @type org.apache.lucene.document.Field
     */
    this.getField = function() {
        return field;
    };

    /**
     * Main constructor body
     */
    if (arguments.length == 1 && arguments[0] instanceof Packages.org.apache.lucene.document.Field) {
        // calling the constructor with a single field argument is
        // only used internally (eg. in Document.getFields())
        field = arguments[0];
    } else {
        var pkg = Packages.org.apache.lucene.document.Field;
        // default options
        var store = pkg.Store.YES;
        var index = pkg.Index.TOKENIZED;
        var termVector = pkg.TermVector.NO;
    
        var opt;
        if (options != null) {
            if (options.store != null) {
                opt = options.store.toUpperCase();
                if (opt == "YES" || opt == "NO" || opt == "COMPRESS") {
                    store = pkg.Store[opt];
                } else {
                    app.logger.warn("helma.Search: unknown field storage option '" +
                                    options.store + "'");
                }
            }
            if (options.index != null) {
                opt = options.index.toUpperCase();
                if (opt == "TOKENIZED" || opt == "NO") {
                    index = pkg.Index[opt];
                } else if (opt == "UNTOKENIZED") {
                    index = pkg.Index.UN_TOKENIZED;
                } else if (opt == "NONORMS") {
                    index = pkg.Index.NO_NORMS;
                } else {
                    app.logger.warn("helma.Search: unknown field indexing option '" +
                                    options.index + "'");
                }
            }
            if (options.termVector != null) {
                opt = options.termVector.toUpperCase();
                if (opt == "NO" || opt == "YES") {
                    termVector = pkg.TermVector[opt];
                } else if (opt == "WITHOFFSETS") {
                    termVector = pkg.TermVector.WITH_OFFSETS;
                } else if (opt == "WITHPOSITIONS") {
                    termVector = pkg.TermVector.WITH_POSITIONS;
                } else if (opt == "WITHPOSITIONSANDOFFSETS") {
                    termVector = pkg.TermVector.WITH_POSITIONS_OFFSETS;
                } else {
                    app.logger.warn("helma.Search: unknown field term vector option '" +
                                    options.termVector + "'");
                }
            }
        }

        // construct the field instance and add it to this document
        field = new Packages.org.apache.lucene.document.Field(
                            name, helma.Search.Document.Field.valueToString(value),
                            store, index, termVector);
    }

    return this;
};

/**
 * Converts the value passed as argument to the appropriate string value. For
 * null values this method returns an empty string.
 * @param {Object} value The value to convert into a string
 * @returns The value converted into a string
 * @type String
 */
helma.Search.Document.Field.valueToString = function(value) {
    var pkg = Packages.org.apache.lucene.document;
    if (value != null) {
        if (value.constructor === Date) {
            return pkg.DateTools.timeToString(value.getTime(), pkg.DateTools.Resolution.MINUTE);
        } else if (value.constructor !== String) {
            return String(value);
        }
        return value;
    }
    return "";
};

/** @ignore */
helma.Search.Document.Field.prototype.toString = function() {
   return "[Field '" + this.name + "' (" + this.getField().toString() + ")]";
};

/**
 * Returns the boost factor of this field.
 * @returns The boost factor of this field
 * @type Number
 */
helma.Search.Document.Field.prototype.getBoost = function() {
    return this.getField().getBoost();
};

/**
 * Sets the boost factor of this field.
 * @param {Number} boost The boost factor of this field
 */
helma.Search.Document.Field.prototype.setBoost = function(boost) {
    this.getField().setBoost(boost);
    app.logger.debug("boost is now: " + this.getField().getBoost());
    return;
};

/**
 * Returns true if this field is indexed
 * @returns True if this field's value is indexed, false otherwise
 * @type Boolean
 */
helma.Search.Document.Field.prototype.isIndexed = function() {
    return this.getField().isIndexed();
};

/**
 * Returns true if this field's value is stored in compressed form in the index
 * @returns True if this field's value is compressed, false otherwise
 * @type Boolean
 */
helma.Search.Document.Field.prototype.isCompressed = function() {
    return this.getField().isCompressed();
};

/**
 * Returns true if this field's value is stored in the index
 * @returns True if this field's value is stored, false otherwise
 * @type Boolean
 */
helma.Search.Document.Field.prototype.isStored = function() {
    return this.getField().isStored();
};

/**
 * Returns true if this field's value is tokenized
 * @returns True if this field's value is tokenized, false otherwise
 * @type Boolean
 */
helma.Search.Document.Field.prototype.isTokenized = function() {
    return this.getField().isTokenized();
};

/**
 * Returns true if this field's term vector is stored in the index
 * @returns True if this field's term vector is stored, false otherwise
 * @type Boolean
 */
helma.Search.Document.Field.prototype.isTermVectorStored = function() {
    return this.getField().isTermVectorStored();
};


helma.lib = "Search";
helma.dontEnum(helma.lib);
for (var i in helma[helma.lib])
    helma[helma.lib].dontEnum(i);
for (var i in helma[helma.lib].prototype)
    helma[helma.lib].prototype.dontEnum(i);
delete helma.lib;