/* * eXist Open Source Native XML Database * Copyright (C) 2014 The eXist Project * http://exist-db.org * * This program 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 * of the License, or (at your option) any later version. * * This program 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 */ package org.exist.mongodb.xquery.mongodb.collection; import com.mongodb.BasicDBObject; import com.mongodb.DB; import com.mongodb.DBCollection; import com.mongodb.MongoClient; import com.mongodb.MongoCommandException; import com.mongodb.MongoException; import com.mongodb.WriteResult; import com.mongodb.util.JSON; import com.mongodb.util.JSONParseException; import org.exist.dom.QName; import static org.exist.mongodb.shared.FunctionDefinitions.PARAMETER_COLLECTION; import static org.exist.mongodb.shared.FunctionDefinitions.PARAMETER_DATABASE; import static org.exist.mongodb.shared.FunctionDefinitions.PARAMETER_MONGODB_CLIENT; import org.exist.mongodb.shared.MongodbClientStore; import org.exist.mongodb.xquery.MongodbModule; import org.exist.xquery.BasicFunction; import org.exist.xquery.Cardinality; import org.exist.xquery.FunctionSignature; import org.exist.xquery.XPathException; import org.exist.xquery.XQueryContext; import org.exist.xquery.value.FunctionParameterSequenceType; import org.exist.xquery.value.FunctionReturnSequenceType; import org.exist.xquery.value.Sequence; import org.exist.xquery.value.SequenceType; import org.exist.xquery.value.StringValue; import org.exist.xquery.value.Type; /** * Functions to save a document in mongodb * * @author Dannes Wessels */ public class Update extends BasicFunction { private static final String UPDATE = "update"; public static final String PARAM_CRITERIA = "criteria"; public static final String DESCR_CRITERIA = "The selection criteria for the update"; public static final FunctionParameterSequenceType PARAMETER_CRITERIA = new FunctionParameterSequenceType(PARAM_CRITERIA, Type.STRING, Cardinality.ONE, DESCR_CRITERIA); public static final String PARAM_MODIFICATION = "modification"; public static final String DESCR_MODIFICATION = "The modifications to apply"; public static final FunctionParameterSequenceType PARAMETER_MODIFICATION = new FunctionParameterSequenceType(PARAM_MODIFICATION, Type.STRING, Cardinality.ONE, DESCR_MODIFICATION); public static final String PARAM_UPSERT = "upsert"; public static final String DESCR_UPSERT = "When true, inserts a document if no document matches the update query criteria"; public static final FunctionParameterSequenceType PARAMETER_UPSERT = new FunctionParameterSequenceType(PARAM_UPSERT, Type.BOOLEAN, Cardinality.ONE, DESCR_UPSERT); public static final String PARAM_MULTI = "multi"; public static final String DESCR_MULTI = "When true, updates all documents in the collection that match the update query criteria, otherwise only updates one"; public static final FunctionParameterSequenceType PARAMETER_MULTI = new FunctionParameterSequenceType(PARAM_MULTI, Type.BOOLEAN, Cardinality.ONE, DESCR_MULTI); public final static FunctionSignature signatures[] = { new FunctionSignature( new QName(UPDATE, MongodbModule.NAMESPACE_URI, MongodbModule.PREFIX), "Modify an existing document or documents in collection. By default the method updates a single document.", new SequenceType[]{ PARAMETER_MONGODB_CLIENT, PARAMETER_DATABASE, PARAMETER_COLLECTION, PARAMETER_CRITERIA, PARAMETER_MODIFICATION}, new FunctionReturnSequenceType(Type.STRING, Cardinality.ONE, "The write result, JSON formatted") ), new FunctionSignature( new QName(UPDATE, MongodbModule.NAMESPACE_URI, MongodbModule.PREFIX), "Modify an existing document or documents in collection.", new SequenceType[]{ PARAMETER_MONGODB_CLIENT, PARAMETER_DATABASE, PARAMETER_COLLECTION, PARAMETER_CRITERIA, PARAMETER_MODIFICATION, PARAMETER_UPSERT, PARAMETER_MULTI}, new FunctionReturnSequenceType(Type.STRING, Cardinality.ONE, "The write result, JSON formatted") ), }; public Update(XQueryContext context, FunctionSignature signature) { super(context, signature); } @Override public Sequence eval(Sequence[] args, Sequence contextSequence) throws XPathException { try { // Verify clientid and get client String mongodbClientId = args[0].itemAt(0).getStringValue(); MongodbClientStore.getInstance().validate(mongodbClientId); MongoClient client = MongodbClientStore.getInstance().get(mongodbClientId); // Get parameters String dbname = args[1].itemAt(0).getStringValue(); String collection = args[2].itemAt(0).getStringValue(); // Get database DB db = client.getDB(dbname); DBCollection dbcol = db.getCollection(collection); // Get data BasicDBObject criterium = (BasicDBObject) JSON.parse(args[3].itemAt(0).getStringValue()); BasicDBObject modification = (BasicDBObject) JSON.parse(args[4].itemAt(0).getStringValue()); Boolean upsert = (args.length >= 6) ? args[5].itemAt(0).toJavaObject(Boolean.class) : null; Boolean multi = (args.length >= 7) ? args[6].itemAt(0).toJavaObject(Boolean.class) : null; // Execute update WriteResult update = (upsert == null) ? dbcol.update(criterium, modification) : dbcol.update(criterium, modification, upsert, multi); return new StringValue(update.toString()); } catch (MongoCommandException ex){ // TODO return as value? LOG.error(ex.getMessage(), ex); throw new XPathException(this, MongodbModule.MONG0005, ex.getMessage()); } catch (JSONParseException ex) { String msg = "Invalid JSON data: " + ex.getMessage(); LOG.error(msg); throw new XPathException(this, MongodbModule.MONG0004, msg); } catch (XPathException ex) { LOG.error(ex.getMessage(), ex); throw new XPathException(this, ex.getMessage(), ex); } catch (MongoException ex) { LOG.error(ex.getMessage(), ex); throw new XPathException(this, MongodbModule.MONG0002, ex.getMessage()); } catch (Throwable t) { LOG.error(t.getMessage(), t); throw new XPathException(this, MongodbModule.MONG0003, t.getMessage()); } } }