org.opencb.opencga.catalog.db.mongodb.CatalogMongoProjectDBAdaptor.java Source code

Java tutorial

Introduction

Here is the source code for org.opencb.opencga.catalog.db.mongodb.CatalogMongoProjectDBAdaptor.java

Source

/*
 * Copyright 2015 OpenCB
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.opencb.opencga.catalog.db.mongodb;

import com.mongodb.client.MongoCursor;
import com.mongodb.client.model.Aggregates;
import com.mongodb.client.model.Filters;
import com.mongodb.client.model.Projections;
import com.mongodb.client.model.Updates;
import com.mongodb.client.result.UpdateResult;
import org.apache.commons.lang.NotImplementedException;
import org.bson.BsonDocument;
import org.bson.BsonString;
import org.bson.Document;
import org.bson.conversions.Bson;
import org.opencb.commons.datastore.core.ObjectMap;
import org.opencb.commons.datastore.core.Query;
import org.opencb.commons.datastore.core.QueryOptions;
import org.opencb.commons.datastore.core.QueryResult;
import org.opencb.commons.datastore.mongodb.MongoDBCollection;
import org.opencb.opencga.catalog.db.api.CatalogDBIterator;
import org.opencb.opencga.catalog.db.api.CatalogProjectDBAdaptor;
import org.opencb.opencga.catalog.db.api.CatalogStudyDBAdaptor;
import org.opencb.opencga.catalog.db.mongodb.converters.ProjectConverter;
import org.opencb.opencga.catalog.exceptions.CatalogDBException;
import org.opencb.opencga.catalog.models.*;
import org.opencb.opencga.core.common.TimeUtils;
import org.slf4j.LoggerFactory;

import java.util.*;
import java.util.function.Consumer;
import java.util.stream.Collectors;

import static org.opencb.opencga.catalog.db.mongodb.CatalogMongoDBUtils.*;

/**
 * Created by imedina on 08/01/16.
 */
public class CatalogMongoProjectDBAdaptor extends CatalogMongoDBAdaptor implements CatalogProjectDBAdaptor {

    private final MongoDBCollection userCollection;
    private ProjectConverter projectConverter;

    public CatalogMongoProjectDBAdaptor(MongoDBCollection userCollection,
            CatalogMongoDBAdaptorFactory dbAdaptorFactory) {
        super(LoggerFactory.getLogger(CatalogMongoProjectDBAdaptor.class));
        this.dbAdaptorFactory = dbAdaptorFactory;
        this.userCollection = userCollection;
        this.projectConverter = new ProjectConverter();
    }

    @Override
    public boolean projectExists(long projectId) {
        QueryResult<Long> count = userCollection.count(new Document("projects.id", projectId));
        return count.getResult().get(0) != 0;
    }

    @Override
    public QueryResult<Project> createProject(String userId, Project project, QueryOptions options)
            throws CatalogDBException {
        long startTime = startQuery();

        List<Study> studies = project.getStudies();
        if (studies == null) {
            studies = Collections.emptyList();
        }
        project.setStudies(Collections.<Study>emptyList());

        // Check if project.alias already exists.
        //        DBObject countQuery = BasicDBObjectBuilder
        //                .start("id", userId)
        //                .append("projects.alias", project.getAlias())
        //                .get();
        Bson countQuery = Filters.and(Filters.eq("id", userId), Filters.eq("projects.alias", project.getAlias()));
        QueryResult<Long> count = userCollection.count(countQuery);
        if (count.getResult().get(0) != 0) {
            throw new CatalogDBException(
                    "Project {alias:\"" + project.getAlias() + "\"} already exists in this user");
        }
        //        if(getProjectId(userId, project.getAlias()) >= 0){
        //            throw new CatalogManagerException( "Project {alias:\"" + project.getAlias() + "\"} already exists in this user");
        //        }

        //Generate json
        //        int projectId = CatalogMongoDBUtils.getNewAutoIncrementId(metaCollection);
        long projectId = dbAdaptorFactory.getCatalogMetaDBAdaptor().getNewAutoIncrementId();
        project.setId(projectId);
        //        DBObject query = new BasicDBObject("id", userId);
        //        query.put("projects.alias", new BasicDBObject("$ne", project.getAlias()));
        Bson query = Filters.and(Filters.eq("id", userId), Filters.ne("projects.alias", project.getAlias()));

        Document projectDocument = projectConverter.convertToStorageType(project);
        //        DBObject update = new BasicDBObject("$push", new BasicDBObject("projects", projectDBObject));
        Bson update = Updates.push("projects", projectDocument);

        //Update object
        //        QueryResult<WriteResult> queryResult = userCollection.update(query, update, null);
        QueryResult<UpdateResult> queryResult = userCollection.update(query, update, null);

        if (queryResult.getResult().get(0).getModifiedCount() == 0) { // Check if the project has been inserted
            throw new CatalogDBException(
                    "Project {alias:\"" + project.getAlias() + "\"} already exists in this user");
        }

        String errorMsg = "";
        for (Study study : studies) {
            String studyErrorMsg = dbAdaptorFactory.getCatalogStudyDBAdaptor()
                    .createStudy(project.getId(), study, options).getErrorMsg();
            if (studyErrorMsg != null && !studyErrorMsg.isEmpty()) {
                errorMsg += ", " + study.getAlias() + ":" + studyErrorMsg;
            }
        }
        List<Project> result = getProject(project.getId(), null).getResult();
        return endQuery("Create Project", startTime, result, errorMsg, null);
    }

    @Override
    public QueryResult<Project> getProject(long projectId, QueryOptions options) throws CatalogDBException {
        checkProjectId(projectId);
        return get(new Query(QueryParams.ID.key(), projectId).append(QueryParams.STATUS_NAME.key(),
                "!=" + Status.DELETED), options);
        //
        //        long startTime = startQuery();
        //        Bson query = Filters.eq("projects.id", projectId);
        //        Bson projection = Projections.elemMatch("projects", Filters.eq("id", projectId));
        //
        //        QueryResult<Document> result = userCollection.find(query, projection, options);
        //
        //        User user = parseUser(result);
        //
        //        if (user == null || user.getProjects().isEmpty()) {
        //            throw CatalogDBException.idNotFound("Project", projectId);
        //        }
        //        // Fixme: Check the code below
        //        List<Project> projects = user.getProjects();
        //        joinFields(projects.get(0), options);
        //
        //        return endQuery("Get project", startTime, projects);
    }

    /**
     * At the moment it does not clean external references to itself.
     */
    //    @Override
    //    public QueryResult<Integer> deleteProject(int projectId) throws CatalogDBException {
    //        long startTime = startQuery();
    //        DBObject query = new BasicDBObject("projects.id", projectId);
    //        DBObject pull = new BasicDBObject("$pull",
    //                new BasicDBObject("projects",
    //                        new BasicDBObject("id", projectId)));
    //
    //        QueryResult<WriteResult> update = userCollection.update(query, pull, null);
    //        List<Integer> deletes = new LinkedList<>();
    //        if (update.getResult().get(0).getN() == 0) {
    //            throw CatalogDBException.idNotFound("Project", projectId);
    //        } else {
    //            deletes.add(update.getResult().get(0).getN());
    //            return endQuery("delete project", startTime, deletes);
    //        }
    //    }
    @Override
    public QueryResult<Project> getAllProjects(String userId, QueryOptions options) throws CatalogDBException {
        long startTime = startQuery();
        Query query = new Query(QueryParams.USER_ID.key(), userId);
        return endQuery("User projects list", startTime, get(query, options).getResult());
    }

    /*
     * db.user.update(
     * {
     * "projects.id" : projectId,
     * "projects.alias" : {
     * $ne : newAlias
     * }
     * },
     * {
     * $set:{
     * "projects.$.alias":newAlias
     * }
     * })
     */
    @Override
    public QueryResult renameProjectAlias(long projectId, String newProjectAlias) throws CatalogDBException {
        long startTime = startQuery();
        //        String projectOwner = getProjectOwner(projectId);
        //
        //        int collisionProjectId = getProjectId(projectOwner, newProjectAlias);
        //        if (collisionProjectId != -1) {
        //            throw new CatalogManagerException("Couldn't rename project alias, alias already used in the same user");
        //        }

        QueryResult<Project> projectResult = getProject(projectId, null); // if projectId doesn't exist, an exception is raised
        Project project = projectResult.getResult().get(0);

        //String oldAlias = project.getAlias();
        project.setAlias(newProjectAlias);

        /*
        DBObject query = BasicDBObjectBuilder
            .start("projects.id", projectId)
            .append("projects.alias", new BasicDBObject("$ne", newProjectAlias))    // check that any other project in the user has
            // the new name
            .get();
        DBObject update = new BasicDBObject("$set",
            new BasicDBObject("projects.$.alias", newProjectAlias));
        */
        Bson query = Filters.and(Filters.eq("projects.id", projectId),
                Filters.ne("projects.alias", newProjectAlias));
        Bson update = Updates.set("projects.$.alias", newProjectAlias);

        QueryResult<UpdateResult> result = userCollection.update(query, update, null);
        if (result.getResult().get(0).getModifiedCount() == 0) { //Check if the the study has been inserted
            throw new CatalogDBException("Project {alias:\"" + newProjectAlias + "\"} already exists");
        }
        return endQuery("rename project alias", startTime, result);
    }

    //    @Deprecated
    //    @Override
    //    public QueryResult<Project> modifyProject(long projectId, ObjectMap parameters) throws CatalogDBException {
    //        long startTime = startQuery();
    //
    //        if (!projectExists(projectId)) {
    //            throw CatalogDBException.idNotFound("Project", projectId);
    //        }
    //        //BasicDBObject projectParameters = new BasicDBObject();
    //        Bson projectParameters = new Document();
    //
    //        String[] acceptedParams = {"name", "creationDate", "description", "organization", "status", "lastModified"};
    //        for (String s : acceptedParams) {
    //            if (parameters.containsKey(s)) {
    //                ((Document) projectParameters).put("projects.$." + s, parameters.getString(s));
    //            }
    //        }
    //        String[] acceptedIntParams = {"diskQuota", "diskUsage"};
    //        for (String s : acceptedIntParams) {
    //            if (parameters.containsKey(s)) {
    //                int anInt = parameters.getInt(s, Integer.MIN_VALUE);
    //                if (anInt != Integer.MIN_VALUE) {
    //                    ((Document) projectParameters).put(s, anInt);
    //                }
    //            }
    //        }
    //        Map<String, Object> attributes = parameters.getMap("attributes");
    //        if (attributes != null) {
    //            for (Map.Entry<String, Object> entry : attributes.entrySet()) {
    //                ((Document) projectParameters).put("projects.$.attributes." + entry.getKey(), entry.getValue());
    //            }
    ////            projectParameters.put("projects.$.attributes", attributes);
    //        }
    //
    //        if (!((Document) projectParameters).isEmpty()) {
    //            Bson query = Filters.eq("projects.id", projectId);
    //            Bson updates = new Document("$set", projectParameters);
    //            // Fixme: Updates
    //                    /*
    //            BasicDBObject query = new BasicDBObject("projects.id", projectId);
    //            BasicDBObject updates = new BasicDBObject("$set", projectParameters);
    //            */
    //            QueryResult<UpdateResult> updateResult = userCollection.update(query, updates, null);
    //            if (updateResult.getResult().get(0).getModifiedCount() == 0) {
    //                throw CatalogDBException.idNotFound("Project", projectId);
    //            }
    //        }
    //        /*
    //        if (!projectParameters.isEmpty()) {
    //            BasicDBObject query = new BasicDBObject("projects.id", projectId);
    //            BasicDBObject updates = new BasicDBObject("$set", projectParameters);
    //            QueryResult<WriteResult> updateResult = userCollection.update(query, updates, null);
    //            if (updateResult.getResult().get(0).getN() == 0) {
    //                throw CatalogDBException.idNotFound("Project", projectId);
    //            }
    //        }
    //        */
    //        return endQuery("Modify project", startTime, getProject(projectId, null));
    //    }

    @Override
    public long getProjectId(String userId, String projectAlias) throws CatalogDBException {
        QueryResult<Document> queryResult = userCollection.find(
                new BsonDocument("projects.alias", new BsonString(projectAlias)).append("id",
                        new BsonString(userId)),
                Projections.fields(Projections.include("projects.id"),
                        Projections.elemMatch("projects", Filters.eq("alias", projectAlias))),
                null);
        /*
                QueryResult<DBObject> queryResult = userCollection.find(
            BasicDBObjectBuilder
                    .start("projects.alias", projectAlias)
                    .append("id", userId).get(),
            BasicDBObjectBuilder.start("projects.id", true)
                    .append("projects", new BasicDBObject("$elemMatch", new BasicDBObject("alias", projectAlias))).get(),
            null
                );*/
        User user = parseUser(queryResult);
        if (user == null || user.getProjects().isEmpty()) {
            return -1;
        } else {
            return user.getProjects().get(0).getId();
        }
    }

    @Override
    public String getProjectOwnerId(long projectId) throws CatalogDBException {
        //        DBObject query = new BasicDBObject("projects.id", projectId);
        Bson query = Filters.eq("projects.id", projectId);

        //        DBObject projection = new BasicDBObject("id", "true");
        Bson projection = Projections.include("id");

        //        QueryResult<DBObject> result = userCollection.find(query, projection, null);
        QueryResult<Document> result = userCollection.find(query, projection, null);

        if (result.getResult().isEmpty()) {
            throw CatalogDBException.idNotFound("Project", projectId);
        } else {
            return result.getResult().get(0).get("id").toString();
        }
    }

    /*
    @Deprecated
    public AclEntry getFullProjectAcl(int projectId, String userId) throws CatalogDBException {
    QueryResult<Project> project = getProject(projectId, null);
    if (project.getNumResults() != 0) {
        List<AclEntry> acl = project.getResult().get(0).getAcl();
        for (AclEntry acl1 : acl) {
            if (userId.equals(acl1.getUserId())) {
                return acl1;
            }
        }
    }
    return null;
    }
    */

    /*
     * db.user.aggregate(
     * {"$match": {"projects.id": 2}},
     * {"$project": {"projects.acl":1, "projects.id":1}},
     * {"$unwind": "$projects"},
     * {"$match": {"projects.id": 2}},
     * {"$unwind": "$projects.acl"},
     * {"$match": {"projects.acl.userId": "jmmut"}}).pretty()
     */
    @Override
    public QueryResult<AclEntry> getProjectAcl(long projectId, String userId) throws CatalogDBException {
        long startTime = startQuery();
        /*
        DBObject match1 = new BasicDBObject("$match", new BasicDBObject("projects.id", projectId));
        DBObject project = new BasicDBObject("$project", BasicDBObjectBuilder
            .start("_id", false)
            .append("projects.acl", true)
            .append("projects.id", true).get());
        DBObject unwind1 = new BasicDBObject("$unwind", "$projects");
        DBObject match2 = new BasicDBObject("$match", new BasicDBObject("projects.id", projectId));
        DBObject unwind2 = new BasicDBObject("$unwind", "$projects.acl");
        DBObject match3 = new BasicDBObject("$match", new BasicDBObject("projects.acl.userId", userId));
            
        */
        Bson match1 = Aggregates.match(Filters.eq("projects.id", projectId));
        //        Bson project = Projections.fields(Projections.excludeId(), Projections.include("projects.acl", "projects.id"));
        Bson project = Aggregates.project(
                Projections.fields(Projections.excludeId(), Projections.include("projects.acl", "projects.id")));
        Bson unwind1 = Aggregates.unwind("$projects");
        Bson match2 = Aggregates.match(Filters.eq("projects.id", projectId));
        Bson unwind2 = Aggregates.unwind("$projects.acl");
        Bson match3 = Aggregates.match(Filters.eq("projects.acl.userId", userId));

        List<Bson> operations = new LinkedList<>();
        operations.add(match1);
        operations.add(project);
        operations.add(unwind1);
        operations.add(match2);
        operations.add(unwind2);
        operations.add(match3);

        QueryResult aggregate = userCollection.aggregate(operations, null);
        List<AclEntry> acls = new LinkedList<>();
        if (aggregate.getNumResults() != 0) {
            //            DBObject aclObject = (DBObject) ((DBObject) ((DBObject) aggregate.getResult().get(0)).get("projects")).get("acl");
            Document aclObject = (Document) ((Document) ((Document) aggregate.getResult().get(0)).get("projects"))
                    .get("acl");
            AclEntry acl = parseObject(aclObject, AclEntry.class);
            acls.add(acl);
        }
        return endQuery("get project ACL", startTime, acls);
    }

    @Override
    public QueryResult setProjectAcl(long projectId, AclEntry newAcl) throws CatalogDBException {
        long startTime = startQuery();
        String userId = newAcl.getUserId();
        if (!dbAdaptorFactory.getCatalogUserDBAdaptor().userExists(userId)) {
            throw new CatalogDBException("Can not set ACL to non-existent user: " + userId);
        }

        Document newAclObject = getMongoDBDocument(newAcl, "ACL");
        //DBObject newAclObject = getDbObject(newAcl, "ACL");

        List<AclEntry> projectAcls = getProjectAcl(projectId, userId).getResult();
        Bson query = new Document("projects.id", projectId);
        Bson push = Updates.push("projects.$.acl", newAclObject);
        if (!projectAcls.isEmpty()) { // ensure that there is no acl for that user in that project. pull
            Bson pull = Updates.pull("projects.$.acl", Filters.eq("userId", userId));
            userCollection.update(query, pull, null);
        }
        /*
        DBObject query = new BasicDBObject("projects.id", projectId);
        BasicDBObject push = new BasicDBObject("$push", new BasicDBObject("projects.$.acl", newAclObject));
        if (!projectAcls.isEmpty()) {  // ensure that there is no acl for that user in that project. pull
        DBObject pull = new BasicDBObject("$pull", new BasicDBObject("projects.$.acl", new BasicDBObject("userId", userId)));
        userCollection.update(query, pull, null);
        }
        */
        //Put study
        QueryResult pushResult = userCollection.update(query, push, null);
        return endQuery("Set project acl", startTime, pushResult);
    }

    @Override
    public QueryResult<Long> count(Query query) throws CatalogDBException {
        Bson bson = parseQuery(query);
        return userCollection.count(bson);
    }

    @Override
    public QueryResult distinct(Query query, String field) throws CatalogDBException {
        Bson bson = parseQuery(query);
        return userCollection.distinct(field, bson);
    }

    @Override
    public QueryResult stats(Query query) {
        return null;
    }

    @Override
    public QueryResult<Project> get(Query query, QueryOptions options) throws CatalogDBException {
        long startTime = startQuery();
        if (!query.containsKey(QueryParams.STATUS_NAME.key())) {
            query.append(QueryParams.STATUS_NAME.key(), "!=" + Status.TRASHED + ";!=" + Status.DELETED);
        }
        List<Bson> aggregates = new ArrayList<>();

        aggregates.add(Aggregates.unwind("$projects"));
        aggregates.add(Aggregates.match(parseQuery(query)));

        // Check include
        if (options != null && options.get(QueryOptions.INCLUDE) != null) {
            List<String> includeList = new ArrayList<>();
            List<String> optionsAsStringList = options.getAsStringList(QueryOptions.INCLUDE);
            includeList.addAll(optionsAsStringList.stream().collect(Collectors.toList()));
            if (!includeList.contains(QueryParams.ID.key())) {
                includeList.add(QueryParams.ID.key());
            }

            // Check if they start with projects.
            for (int i = 0; i < includeList.size(); i++) {
                if (!includeList.get(i).startsWith("projects.")) {
                    String param = "projects." + includeList.get(i);
                    includeList.set(i, param);
                }
            }
            if (includeList.size() > 0) {
                aggregates.add(Aggregates.project(Projections.include(includeList)));
            }
        }

        QueryResult<Project> projectQueryResult = userCollection.aggregate(aggregates, projectConverter, options);

        if (options == null || !options.containsKey(QueryOptions.EXCLUDE)
                || !options.getAsStringList(QueryOptions.EXCLUDE).contains("projects.studies")) {
            for (Project project : projectQueryResult.getResult()) {
                Query studyQuery = new Query(CatalogStudyDBAdaptor.QueryParams.PROJECT_ID.key(), project.getId());
                try {
                    QueryResult<Study> studyQueryResult = dbAdaptorFactory.getCatalogStudyDBAdaptor()
                            .get(studyQuery, options);
                    project.setStudies(studyQueryResult.getResult());
                } catch (CatalogDBException e) {
                    e.printStackTrace();
                }
            }
        }

        return endQuery("Get project", startTime, projectQueryResult.getResult());
    }

    @Override
    public QueryResult nativeGet(Query query, QueryOptions options) throws CatalogDBException {
        if (!query.containsKey(QueryParams.STATUS_NAME.key())) {
            query.append(QueryParams.STATUS_NAME.key(), "!=" + Status.TRASHED + ";!=" + Status.DELETED);
        }
        List<Bson> aggregates = new ArrayList<>();
        aggregates.add(Aggregates.match(parseQuery(query)));
        aggregates.add(Aggregates.unwind("$projects"));

        // Check include & excludes
        if (options != null && options.get(QueryOptions.INCLUDE) != null) {
            List<String> includeList = new ArrayList<>();
            List<String> optionsAsStringList = options.getAsStringList(QueryOptions.INCLUDE);
            includeList.addAll(optionsAsStringList.stream().collect(Collectors.toList()));

            if (includeList.size() > 0) {
                aggregates.add(Aggregates.project(Projections.include(includeList)));
            }
        }

        QueryResult<Document> projectQueryResult = userCollection.aggregate(aggregates, options);
        ArrayList<Document> returnedProjectList = new ArrayList<>();

        for (Document user : projectQueryResult.getResult()) {
            Document project = (Document) user.get("projects");
            Query studyQuery = new Query(CatalogStudyDBAdaptor.QueryParams.PROJECT_ID.key(), project.get("id"));
            QueryResult studyQueryResult = dbAdaptorFactory.getCatalogStudyDBAdaptor().nativeGet(studyQuery,
                    options);
            project.remove("studies");
            project.append("studies", studyQueryResult.getResult());
            returnedProjectList.add(project);
        }

        projectQueryResult.setResult(returnedProjectList);

        return projectQueryResult;

    }

    @Override
    public QueryResult<Long> update(Query query, ObjectMap parameters) throws CatalogDBException {
        long startTime = startQuery();

        Bson projectParameters = new Document();

        String[] acceptedParams = { QueryParams.NAME.key(), QueryParams.CREATION_DATE.key(),
                QueryParams.DESCRIPTION.key(), QueryParams.ORGANIZATION.key(), QueryParams.LAST_MODIFIED.key(), };
        for (String s : acceptedParams) {
            if (parameters.containsKey(s)) {
                ((Document) projectParameters).put("projects.$." + s, parameters.getString(s));
            }
        }
        String[] acceptedIntParams = { QueryParams.DISK_USAGE.key() };
        for (String s : acceptedIntParams) {
            if (parameters.containsKey(s)) {
                int anInt = parameters.getInt(s, Integer.MIN_VALUE);
                if (anInt != Integer.MIN_VALUE) {
                    ((Document) projectParameters).put(s, anInt);
                }
            }
        }
        Map<String, Object> attributes = parameters.getMap(QueryParams.ATTRIBUTES.key());
        if (attributes != null) {
            for (Map.Entry<String, Object> entry : attributes.entrySet()) {
                ((Document) projectParameters).put("projects.$.attributes." + entry.getKey(), entry.getValue());
            }
            //            projectParameters.put("projects.$.attributes", attributes);
        }

        if (parameters.containsKey(QueryParams.STATUS_NAME.key())) {
            ((Document) projectParameters).put("projects.$." + QueryParams.STATUS_NAME.key(),
                    parameters.get(QueryParams.STATUS_NAME.key()));
            ((Document) projectParameters).put("projects.$." + QueryParams.STATUS_DATE.key(), TimeUtils.getTime());
        }

        QueryResult<UpdateResult> updateResult = new QueryResult<>();
        if (!((Document) projectParameters).isEmpty()) {
            Bson bsonQuery = parseQuery(query);
            Bson updates = new Document("$set", projectParameters);
            // Fixme: Updates
            /*
            BasicDBObject query = new BasicDBObject("projects.id", projectId);
            BasicDBObject updates = new BasicDBObject("$set", projectParameters);
            */
            updateResult = userCollection.update(bsonQuery, updates, null);
        }
        /*
        if (!projectParameters.isEmpty()) {
        BasicDBObject query = new BasicDBObject("projects.id", projectId);
        BasicDBObject updates = new BasicDBObject("$set", projectParameters);
        QueryResult<WriteResult> updateResult = userCollection.update(query, updates, null);
        if (updateResult.getResult().get(0).getN() == 0) {
            throw CatalogDBException.idNotFound("Project", projectId);
        }
        }
        */
        return endQuery("Update project", startTime,
                Collections.singletonList(updateResult.first().getModifiedCount()));
    }

    @Override
    public QueryResult<Project> update(long id, ObjectMap parameters) throws CatalogDBException {
        long startTime = startQuery();
        checkProjectId(id);
        QueryResult<Long> update = update(new Query(QueryParams.ID.key(), id), parameters);
        if (update.getNumTotalResults() != 1) {
            throw new CatalogDBException("Could not update project with id " + id);
        }
        return endQuery("Update project", startTime, getProject(id, null));
    }

    @Override
    public QueryResult<Project> delete(long id, QueryOptions queryOptions) throws CatalogDBException {
        long startTime = startQuery();

        checkProjectId(id);
        // Check the project is active
        Query query = new Query(QueryParams.ID.key(), id).append(QueryParams.STATUS_NAME.key(), Status.READY);
        if (count(query).first() == 0) {
            query.put(QueryParams.STATUS_NAME.key(), Status.TRASHED + "," + Status.DELETED);
            QueryOptions options = new QueryOptions(QueryOptions.INCLUDE, QueryParams.STATUS_NAME.key());
            Project project = get(query, options).first();
            throw new CatalogDBException("The project {" + id + "} was already " + project.getStatus().getName());
        }

        // If we don't find the force parameter, we check first if the user does not have an active project.
        if (!queryOptions.containsKey(FORCE) || !queryOptions.getBoolean(FORCE)) {
            checkCanDelete(id);
        }

        if (queryOptions.containsKey(FORCE) && queryOptions.getBoolean(FORCE)) {
            // Delete the active studies (if any)
            query = new Query(CatalogStudyDBAdaptor.QueryParams.PROJECT_ID.key(), id);
            dbAdaptorFactory.getCatalogStudyDBAdaptor().delete(query, queryOptions);
        }

        // Change the status of the project to deleted
        setStatus(id, Status.TRASHED);

        query = new Query(QueryParams.ID.key(), id).append(QueryParams.STATUS_NAME.key(), Status.TRASHED);

        return endQuery("Delete project", startTime, get(query, queryOptions));
    }

    @Override
    public QueryResult<Long> delete(Query query, QueryOptions queryOptions) throws CatalogDBException {
        long startTime = startQuery();
        query.append(QueryParams.STATUS_NAME.key(), Status.READY);
        QueryResult<Project> projectQueryResult = get(query,
                new QueryOptions(MongoDBCollection.INCLUDE, QueryParams.ID.key()));
        for (Project project : projectQueryResult.getResult()) {
            delete(project.getId(), queryOptions);
        }
        return endQuery("Delete project", startTime,
                Collections.singletonList(projectQueryResult.getNumTotalResults()));
    }

    QueryResult<Long> setStatus(Query query, String status) throws CatalogDBException {
        return update(query, new ObjectMap(QueryParams.STATUS_NAME.key(), status));
    }

    private QueryResult<Project> setStatus(long projectId, String status) throws CatalogDBException {
        return update(projectId, new ObjectMap(QueryParams.STATUS_NAME.key(), status));
    }

    @Override
    public QueryResult<Project> remove(long id, QueryOptions queryOptions) throws CatalogDBException {
        throw new UnsupportedOperationException("Remove not yet implemented.");
    }

    @Override
    public QueryResult<Long> remove(Query query, QueryOptions queryOptions) throws CatalogDBException {
        throw new UnsupportedOperationException("Remove not yet implemented.");
    }

    @Override
    public QueryResult<Long> restore(Query query, QueryOptions queryOptions) throws CatalogDBException {
        long startTime = startQuery();
        query.put(QueryParams.STATUS_NAME.key(), Status.TRASHED);
        return endQuery("Restore projects", startTime, setStatus(query, Status.READY));
    }

    @Override
    public QueryResult<Project> restore(long id, QueryOptions queryOptions) throws CatalogDBException {
        long startTime = startQuery();

        checkProjectId(id);
        // Check if the cohort is active
        Query query = new Query(QueryParams.ID.key(), id).append(QueryParams.STATUS_NAME.key(), Status.TRASHED);
        if (count(query).first() == 0) {
            throw new CatalogDBException("The project {" + id + "} is not deleted");
        }

        // Change the status of the cohort to deleted
        setStatus(id, Status.READY);
        query = new Query(QueryParams.ID.key(), id);

        return endQuery("Restore project", startTime, get(query, null));
    }

    @Override
    public CatalogDBIterator<Project> iterator(Query query, QueryOptions options) throws CatalogDBException {
        Bson bson = parseQuery(query);
        MongoCursor<Document> iterator = userCollection.nativeQuery().find(bson, options).iterator();
        return new CatalogMongoDBIterator<>(iterator, projectConverter);
    }

    @Override
    public CatalogDBIterator nativeIterator(Query query, QueryOptions options) throws CatalogDBException {
        Bson bson = parseQuery(query);
        MongoCursor<Document> iterator = userCollection.nativeQuery().find(bson, options).iterator();
        return new CatalogMongoDBIterator<>(iterator);
    }

    @Override
    public QueryResult rank(Query query, String field, int numResults, boolean asc) {
        throw new NotImplementedException("Rank project is not implemented");
    }

    @Override
    public QueryResult groupBy(Query query, String field, QueryOptions options) {
        throw new NotImplementedException("GroupBy in project is not implemented");
    }

    @Override
    public QueryResult groupBy(Query query, List<String> fields, QueryOptions options) {
        throw new NotImplementedException("GroupBy in project is not implemented");
    }

    @Override
    public void forEach(Query query, Consumer<? super Object> action, QueryOptions options)
            throws CatalogDBException {
        Objects.requireNonNull(action);
        CatalogDBIterator<Project> catalogDBIterator = iterator(query, options);
        while (catalogDBIterator.hasNext()) {
            action.accept(catalogDBIterator.next());
        }
        catalogDBIterator.close();
    }

    private Bson parseQuery(Query query) throws CatalogDBException {
        List<Bson> andBsonList = new ArrayList<>();

        for (Map.Entry<String, Object> entry : query.entrySet()) {
            String key = entry.getKey().split("\\.")[0];
            QueryParams queryParam = QueryParams.getParam(entry.getKey()) != null
                    ? QueryParams.getParam(entry.getKey())
                    : QueryParams.getParam(key);
            try {
                switch (queryParam) {
                case ID:
                    addOrQuery("projects." + queryParam.key(), queryParam.key(), query, queryParam.type(),
                            andBsonList);
                    break;
                case USER_ID:
                    addOrQuery(PRIVATE_ID, queryParam.key(), query, queryParam.type(), andBsonList);
                    break;
                case ATTRIBUTES:
                    addAutoOrQuery("projects." + entry.getKey(), entry.getKey(), query, queryParam.type(),
                            andBsonList);
                    break;
                case BATTRIBUTES:
                    String mongoKey = "projects."
                            + entry.getKey().replace(QueryParams.BATTRIBUTES.key(), QueryParams.ATTRIBUTES.key());
                    addAutoOrQuery(mongoKey, entry.getKey(), query, queryParam.type(), andBsonList);
                    break;
                case NATTRIBUTES:
                    mongoKey = "projects."
                            + entry.getKey().replace(QueryParams.NATTRIBUTES.key(), QueryParams.ATTRIBUTES.key());
                    addAutoOrQuery(mongoKey, entry.getKey(), query, queryParam.type(), andBsonList);
                    break;
                default:
                    addAutoOrQuery("projects." + queryParam.key(), queryParam.key(), query, queryParam.type(),
                            andBsonList);
                    break;
                }
            } catch (Exception e) {
                throw new CatalogDBException(e);
            }
        }

        if (andBsonList.size() > 0) {
            return Filters.and(andBsonList);
        } else {
            return new Document();
        }
    }

    /**
     * Checks whether the projectId has any active study.
     *
     * @param projectId project id.
     * @throws CatalogDBException when the project has active studies. Studies must be deleted first.
     */
    private void checkCanDelete(long projectId) throws CatalogDBException {
        checkProjectId(projectId);
        Query query = new Query(CatalogStudyDBAdaptor.QueryParams.PROJECT_ID.key(), projectId)
                .append(CatalogStudyDBAdaptor.QueryParams.STATUS_NAME.key(), Status.READY);
        Long count = dbAdaptorFactory.getCatalogStudyDBAdaptor().count(query).first();
        if (count > 0) {
            throw new CatalogDBException("The project {" + projectId + "} cannot be deleted. The project has "
                    + count + " studies in use.");
        }
    }
}