/* * (C) Copyright 2008 Nuxeo SAS (http://nuxeo.com/) and contributors. * * All rights reserved. This program and the accompanying materials * are made available under the terms of the GNU Lesser General Public License * (LGPL) version 2.1 which accompanies this distribution, and is available at * http://www.gnu.org/licenses/lgpl.html * * 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. * * Contributors: * Nuxeo - initial API and implementation * * $Id$ */ package org.nuxeo.ecm.platform.ui.web.restAPI; import java.net.URLEncoder; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.dom4j.Element; import org.dom4j.Namespace; import org.dom4j.QName; import org.dom4j.dom.DOMDocument; import org.dom4j.dom.DOMDocumentFactory; import org.nuxeo.ecm.core.api.CoreInstance; import org.nuxeo.ecm.core.api.CoreSession; import org.nuxeo.ecm.core.api.DocumentModel; import org.nuxeo.ecm.core.api.DocumentModelList; import org.nuxeo.ecm.platform.ui.web.tag.fn.DocumentModelFunctions; import org.nuxeo.ecm.platform.ui.web.util.BaseURL; import org.restlet.data.CharacterSet; import org.restlet.data.MediaType; import org.restlet.data.Request; import org.restlet.data.Response; import org.restlet.resource.Representation; import org.restlet.resource.StringRepresentation; /** * Basic OpenSearch REST fulltext search implementation using the RSS 2.0 * results format. * <p> * TODO: make it possible to change the page size and navigate to next results * pages using additional query parameters. See http://opensearch.org for * official specifications. * <p> * TODO: use a OPENSEARCH stateless query model to be able to override the * currently hardcoded request pattern. * <p> * TODO: add OpenSearch XML description snippet in the default theme so that * Firefox can autodetect the service URL. * * @author Olivier Grisel */ public class OpenSearchRestlet extends BaseNuxeoRestlet { private static final Log log = LogFactory.getLog(OpenSearchRestlet.class); public static final String RSS_TAG = "rss"; public static final String CHANNEL_TAG = "channel"; public static final String TITLE_TAG = "title"; public static final String DESCRIPTION_TAG = "description"; public static final String LINK_TAG = "link"; public static final String ITEM_TAG = "item"; public static final String QUERY = "SELECT * FROM Document WHERE ecm:fulltext LIKE '%s' ORDER BY dc:modified DESC"; public static final int MAX = 10; public static final Namespace OPENSEARCH_NS = new Namespace("opensearch", "http://a9.com/-/spec/opensearch/1.1/"); public static final Namespace ATOM_NS = new Namespace("atom", "http://www.w3.org/2005/Atom"); @Override public void handle(Request req, Response res) { try (CoreSession session = CoreInstance.openCoreSession(null, getUserPrincipal(req))) { // read the search term passed as the 'q' request parameter String keywords = getQueryParamValue(req, "q", " "); // perform the search on the fulltext index and wrap the results as // a DocumentModelList with the 10 first matching results ordered by // modification time String query = String.format(QUERY, keywords); DocumentModelList documents = session.query(query, null, MAX, 0, true); // build the RSS 2.0 response document holding the results DOMDocumentFactory domFactory = new DOMDocumentFactory(); DOMDocument resultDocument = (DOMDocument) domFactory.createDocument(); // rss root tag Element rssElement = resultDocument.addElement(RSS_TAG); rssElement.addAttribute("version", "2.0"); rssElement.addNamespace(OPENSEARCH_NS.getPrefix(), OPENSEARCH_NS.getURI()); rssElement.addNamespace(ATOM_NS.getPrefix(), ATOM_NS.getURI()); // channel with OpenSearch metadata Element channelElement = rssElement.addElement(CHANNEL_TAG); channelElement.addElement(TITLE_TAG).setText( "Nuxeo EP OpenSearch channel for " + keywords); channelElement.addElement("link").setText( BaseURL.getBaseURL(getHttpRequest(req)) + "restAPI/opensearch?q=" + URLEncoder.encode(keywords, "UTF-8")); channelElement.addElement(new QName("totalResults", OPENSEARCH_NS)).setText( Long.toString(documents.totalSize())); channelElement.addElement(new QName("startIndex", OPENSEARCH_NS)).setText( "0"); channelElement.addElement(new QName("itemsPerPage", OPENSEARCH_NS)).setText( Integer.toString(documents.size())); Element queryElement = channelElement.addElement(new QName("Query", OPENSEARCH_NS)); queryElement.addAttribute("role", "request"); queryElement.addAttribute("searchTerms", keywords); queryElement.addAttribute("startPage", Integer.toString(1)); // document items String baseUrl = BaseURL.getBaseURL(getHttpRequest(req)); for (DocumentModel doc : documents) { Element itemElement = channelElement.addElement(ITEM_TAG); Element titleElement = itemElement.addElement(TITLE_TAG); String title = doc.getTitle(); if (title != null) { titleElement.setText(title); } Element descriptionElement = itemElement.addElement(DESCRIPTION_TAG); String description = doc.getProperty("dublincore:description").getValue( String.class); if (description != null) { descriptionElement.setText(description); } Element linkElement = itemElement.addElement("link"); linkElement.setText(baseUrl + DocumentModelFunctions.documentUrl(doc)); } Representation rep = new StringRepresentation(resultDocument.asXML(), MediaType.APPLICATION_XML); rep.setCharacterSet(CharacterSet.UTF_8); res.setEntity(rep); } catch (Exception e) { handleError(res, e); } } }