org.seadpdt.ROServices.java Source code

Java tutorial

Introduction

Here is the source code for org.seadpdt.ROServices.java

Source

/*
 *
 * Copyright 2015 The Trustees of Indiana University, 2015 University of Michigan
 *
 * 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.
 *
 *
 * @author myersjd@umich.edu
 * @author smccaula@indiana.edu
 */

package org.seadpdt;

import com.mongodb.BasicDBObject;
import com.mongodb.BasicDBObjectBuilder;
import com.mongodb.DB;
import com.mongodb.DBObject;
import com.mongodb.client.FindIterable;
import com.mongodb.client.MongoCollection;
import com.mongodb.client.MongoCursor;
import com.mongodb.client.MongoDatabase;
import com.mongodb.client.model.Filters;
import com.mongodb.client.result.DeleteResult;
import com.mongodb.client.result.UpdateResult;
import com.mongodb.gridfs.GridFS;
import com.mongodb.gridfs.GridFSDBFile;
import com.mongodb.gridfs.GridFSInputFile;
import com.mongodb.util.JSON;
import com.sun.jersey.api.client.ClientResponse;

import org.bson.Document;
import org.bson.types.BasicBSONList;
import org.bson.types.ObjectId;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import org.seadpdt.people.Profile;
import org.seadpdt.people.Provider;
import org.seadpdt.util.Constants;
import org.seadpdt.util.MongoDB;

import javax.ws.rs.*;
import javax.ws.rs.core.CacheControl;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.sql.Date;
import java.text.DateFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.regex.Pattern;

@Path("/researchobjects")
public class ROServices {

    private MongoDatabase db = null;
    private MongoDatabase metaDb = null;
    private DB oreDb = null;
    private MongoCollection<Document> publicationsCollection = null;
    private MongoCollection<Document> repositoriesCollection = null;
    private MongoCollection<Document> oreMapCollection = null;
    private GridFS oreMapBucket = null;
    private MongoCollection<Document> fgdcCollection = null;
    private CacheControl control = new CacheControl();

    public ROServices() {
        db = MongoDB.getServicesDB();
        metaDb = MongoDB.geMetaGenDB();
        oreDb = MongoDB.geOreDB();
        publicationsCollection = db.getCollection(MongoDB.researchObjects);

        repositoriesCollection = db.getCollection(MongoDB.repositories);
        oreMapCollection = metaDb.getCollection(MongoDB.oreMap);
        oreMapBucket = new GridFS(oreDb, MongoDB.oreMap);
        fgdcCollection = metaDb.getCollection(MongoDB.fgdc);
        control.setNoCache(true);
    }

    @POST
    @Path("/")
    @Consumes(MediaType.APPLICATION_JSON)
    @Produces(MediaType.APPLICATION_JSON)
    public Response startROPublicationProcess(String publicationRequestString,
            @QueryParam("requestUrl") String requestURL, @QueryParam("oreId") String oreId)
            throws URISyntaxException {
        String messageString = "";
        Document request = Document.parse(publicationRequestString);
        Document content = (Document) request.get("Aggregation");
        if (content == null) {
            messageString += "Missing Aggregation ";
        }
        Document preferences = (Document) request.get("Preferences");
        if (preferences == null) {
            messageString += "Missing Preferences ";
        }
        Object repository = request.get("Repository");
        if (repository == null) {
            messageString += "Missing Respository ";
        } else {
            FindIterable<Document> iter = repositoriesCollection.find(new Document("orgidentifier", repository));
            if (iter.first() == null) {
                messageString += "Unknown Repository: " + repository + " ";
            }

        }

        if (messageString.equals("")) {
            // Get organization from profile(s)
            // Add to base document
            Object creatorObject = content.get("Creator");
            String ID = (String) content.get("Identifier");
            BasicBSONList affiliations = new BasicBSONList();
            if (creatorObject != null) {
                if (creatorObject instanceof ArrayList) {
                    Iterator<String> iter = ((ArrayList<String>) creatorObject).iterator();

                    while (iter.hasNext()) {
                        String creator = iter.next();
                        List<String> orgs = getOrganizationforPerson(creator);
                        if (!orgs.isEmpty()) {
                            affiliations.addAll(orgs);
                        }
                    }

                } else {
                    // BasicDBObject - single value
                    List<String> orgs = getOrganizationforPerson((String) creatorObject);
                    if (!orgs.isEmpty()) {
                        affiliations.addAll(orgs);
                    }
                }
            }

            request.append("Affiliations", affiliations);

            // Add first status message

            List<DBObject> statusreports = new ArrayList<DBObject>();
            DBObject status = BasicDBObjectBuilder.start()
                    .add("date", DateFormat.getDateTimeInstance().format(new Date(System.currentTimeMillis())))
                    .add("reporter", Constants.serviceName).add("stage", "Receipt Acknowledged")
                    .add("message", "request recorded and processing will begin").get();
            statusreports.add(status);
            request.append("Status", statusreports);
            // Create initial status message - add
            // Add timestamp
            // Generate ID - by calling Workflow?
            // Add doc, return 201

            String newMapURL = requestURL + "/" + ID + "/oremap";
            URI uri = new URI(newMapURL);
            uri = uri.normalize();
            content.put("@id", uri.toString() + "#aggregation");
            content.put("authoratativeMap", oreId);

            publicationsCollection.insertOne(request);
            URI resource = null;
            try {
                resource = new URI("./" + ID);
            } catch (URISyntaxException e) {
                // Should not happen given simple ids
                e.printStackTrace();
            }
            return Response.created(resource).entity(new Document("identifier", ID)).build();
        } else {
            return Response.status(ClientResponse.Status.BAD_REQUEST).entity(messageString).build();
        }
    }

    @GET
    @Path("/")
    @Produces(MediaType.APPLICATION_JSON)
    public Response getROsList(@QueryParam("Purpose") final String purpose) {
        FindIterable<Document> iter;
        if (purpose != null && purpose.equals("Production")) {
            iter = publicationsCollection.find(Filters.ne("Preferences.Purpose", "Testing-Only"));
        } else if (purpose != null && purpose.equals("Testing-Only")) {
            iter = publicationsCollection.find(Filters.eq("Preferences.Purpose", purpose));
        } else if (purpose != null) {
            return Response.status(ClientResponse.Status.BAD_REQUEST)
                    .entity(new JSONObject()
                            .put("Error", "'" + purpose + "' is not an acceptable value for 'Purpose'").toString())
                    .build();
        } else {
            iter = publicationsCollection.find();
        }
        iter.projection(new Document("Status", 1).append("Repository", 1).append("Aggregation.Identifier", 1)
                .append("Aggregation.Title", 1).append("_id", 0));
        MongoCursor<Document> cursor = iter.iterator();
        JSONArray array = new JSONArray();
        while (cursor.hasNext()) {
            array.put(JSON.parse(cursor.next().toJson()));
        }
        return Response.ok(array.toString()).cacheControl(control).build();
    }

    @GET
    @Path("/new/")
    @Produces(MediaType.APPLICATION_JSON)
    public Response getNewROsList(@QueryParam("Purpose") final String purpose) {
        //Find ROs that have a status not from the services and don't include them :-)
        Document reporterRule = new Document("$ne", Constants.serviceName);
        Document reporter = new Document("reporter", reporterRule);
        Document elem = new Document("$elemMatch", reporter);
        Document not = new Document("$not", elem);
        Document match = new Document("Status", not);

        FindIterable<Document> iter;
        if (purpose != null && purpose.equals("Production")) {
            iter = publicationsCollection
                    .find(Filters.and(match, Filters.ne("Preferences.Purpose", "Testing-Only")));
        } else if (purpose != null && purpose.equals("Testing-Only")) {
            iter = publicationsCollection.find(Filters.and(match, Filters.eq("Preferences.Purpose", purpose)));
        } else if (purpose != null) {
            return Response.status(ClientResponse.Status.BAD_REQUEST)
                    .entity(new JSONObject()
                            .put("Error", "'" + purpose + "' is not an acceptable value for 'Purpose'").toString())
                    .build();
        } else {
            iter = publicationsCollection.find(match);
        }
        iter.projection(new Document("Status", 1).append("Repository", 1).append("Aggregation.Identifier", 1)
                .append("Aggregation.Title", 1).append("_id", 0));
        MongoCursor<Document> cursor = iter.iterator();
        JSONArray array = new JSONArray();
        while (cursor.hasNext()) {
            array.put(JSON.parse(cursor.next().toJson()));
        }
        return Response.ok(array.toString()).cacheControl(control).build();
    }

    @GET
    @Path("/{id}")
    @Produces(MediaType.APPLICATION_JSON)
    public Response getROProfile(@PathParam("id") String id) {

        FindIterable<Document> iter = publicationsCollection.find(new Document("Aggregation.Identifier", id));
        if (iter == null || iter.first() == null) {

            FindIterable<Document> altIter = publicationsCollection
                    .find(new Document("Aggregation." + Constants.alternateOf, id));
            if (altIter != null && altIter.first() != null) {
                Document aggDocument = (Document) altIter.first().get("Aggregation");
                String movedTo = (String) aggDocument.get("Identifier");
                return Response.status(ClientResponse.Status.MOVED_PERMANENTLY).header("Location", movedTo)
                        .entity(new JSONObject().put("Error", "The document has moved to " + movedTo).toString())
                        .build();
            }

            return Response.status(ClientResponse.Status.NOT_FOUND)
                    .entity(new JSONObject().put("Error", "Cannot find RO with id " + id).toString()).build();
        }

        Document document = iter.first();
        // Internal meaning only - strip from exported doc
        document.remove("_id");
        Document aggDocument = (Document) document.get("Aggregation");
        aggDocument.remove("authoratativeMap");

        return Response.ok(document.toJson()).cacheControl(control).build();
    }

    @POST
    @Path("/{id}/status")
    @Consumes(MediaType.APPLICATION_JSON)
    public Response setROStatus(@PathParam("id") String id, String state) {
        try {
            Document statusUpdateDocument = Document.parse(state);
            statusUpdateDocument.append("date",
                    DateFormat.getDateTimeInstance().format(new Date(System.currentTimeMillis())));
            UpdateResult ur = publicationsCollection.updateOne(new Document("Aggregation.Identifier", id),
                    new BasicDBObject("$push", new BasicDBObject("Status", statusUpdateDocument)));
            if (ur.wasAcknowledged()) {
                return Response.status(ClientResponse.Status.OK).build();

            } else {
                return Response.status(ClientResponse.Status.NOT_FOUND).build();

            }
        } catch (org.bson.BsonInvalidOperationException e) {
            return Response.status(ClientResponse.Status.BAD_REQUEST).build();
        }
    }

    @GET
    @Path("/{id}/status")
    @Produces(MediaType.APPLICATION_JSON)
    public Response getROStatus(@PathParam("id") String id) {
        FindIterable<Document> iter = publicationsCollection.find(new Document("Aggregation.Identifier", id));
        iter.projection(new Document("Status", 1).append("_id", 0));

        Document document = iter.first();
        document.remove("_id");
        return Response.ok(document.toJson()).cacheControl(control).build();
    }

    @DELETE
    @Path("/{id}")
    public Response rescindROPublicationRequest(@PathParam("id") String id) {

        // First remove map
        FindIterable<Document> iter = publicationsCollection.find(new Document("Aggregation.Identifier", id));
        Document document = iter.first();

        if (document != null) {
            boolean processing = false;
            ArrayList Statuses = (ArrayList) document.get("Status");
            for (Object status : Statuses) {
                Document statusObj = (Document) status;
                String stage = statusObj.get("stage").toString();
                if (stage.equalsIgnoreCase("Success") || stage.equalsIgnoreCase("Pending")) {
                    processing = true;
                    break;
                }
            }
            Document preferences = (Document) document.get("Preferences");
            String purpose = null;
            if (preferences != null) {
                purpose = (String) preferences.get("Purpose");
            }
            if (Constants.deleteTestRO) {
                if ((purpose == null || !purpose.equalsIgnoreCase("Testing-Only")) && processing) {
                    // if server is configured to delete test ROs then still don't delete ROs that are not flagged as "Testing-Only" and in processing/deposited stage
                    return Response.status(ClientResponse.Status.BAD_REQUEST).entity(
                            "Cannot revoke the request since the repository is either processing or has deposited the requested RO")
                            .build();
                }
            } else {
                if (processing) {
                    // if server is configured not to delete test ROs then don't delete all ROs in processing/deposit stage
                    return Response.status(ClientResponse.Status.BAD_REQUEST).entity(
                            "Cannot revoke the request since the repository is either processing or has deposited the requested RO")
                            .build();
                }
            }
        }

        iter.projection(new Document("Aggregation", 1).append("_id", 0));

        document = iter.first();
        if (document == null) {
            return Response.status(javax.ws.rs.core.Response.Status.NOT_FOUND)
                    .entity("RO with ID " + id + " does not exist").build();
        }

        ObjectId mapId = new ObjectId(((Document) document.get("Aggregation")).get("authoratativeMap").toString());
        oreMapBucket.remove(mapId);
        /*DeleteResult mapDeleteResult = oreMapCollection.deleteOne(new Document(
        "describes.Identifier", id));*/
        //if (mapDeleteResult.getDeletedCount() != 1) {
        // Report error
        //System.out.println("Could not find map corresponding to " + id);
        //}

        DeleteResult dr = publicationsCollection.deleteOne(new Document("Aggregation.Identifier", id));
        if (dr.getDeletedCount() == 1) {
            return Response.status(ClientResponse.Status.OK).entity("RO Successfully Deleted").build();
        } else {
            return Response.status(ClientResponse.Status.NOT_FOUND).entity("RO with ID " + id + " does not exist")
                    .build();
        }
    }

    @DELETE
    @Path("/{id}/override")
    public Response DeleteOverrideRO(@PathParam("id") String id) {

        // First remove map
        FindIterable<Document> iter = publicationsCollection.find(new Document("Aggregation.Identifier", id));
        iter.projection(new Document("Aggregation", 1).append("_id", 0));

        Document document = iter.first();
        if (document == null) {
            return Response.status(javax.ws.rs.core.Response.Status.NOT_FOUND)
                    .entity("RO with ID " + id + " does not exist").build();
        }

        ObjectId mapId = new ObjectId(((Document) document.get("Aggregation")).get("authoratativeMap").toString());
        oreMapBucket.remove(mapId);
        DeleteResult dr = publicationsCollection.deleteOne(new Document("Aggregation.Identifier", id));
        if (dr.getDeletedCount() == 1) {
            return Response.status(ClientResponse.Status.OK).entity("RO Successfully Deleted").build();
        } else {
            return Response.status(ClientResponse.Status.NOT_FOUND).entity("RO with ID " + id + " does not exist")
                    .build();
        }
    }

    @POST
    @Path("/oremap")
    @Consumes(MediaType.APPLICATION_JSON)
    @Produces(MediaType.APPLICATION_JSON)
    public Response addOreMap(@QueryParam("objectId") String id, String oreMapString) {
        ObjectId objectId = new ObjectId(id);
        String fileName = "ore-file";
        ByteArrayInputStream inputStream = new ByteArrayInputStream(oreMapString.getBytes());
        GridFSInputFile gfsFile = oreMapBucket.createFile(inputStream, fileName, true);
        gfsFile.setId(objectId);
        gfsFile.save();
        return Response.ok().build();
    }

    @GET
    @Path("/{id}/oremap")
    @Produces(MediaType.APPLICATION_JSON)
    public Response getROOREMap(@PathParam("id") String id) throws JSONException, IOException {

        FindIterable<Document> iter = publicationsCollection.find(new Document("Aggregation.Identifier", id));

        if (iter == null || iter.first() == null) {
            return Response.status(javax.ws.rs.core.Response.Status.NOT_FOUND)
                    .entity(new JSONObject().put("Error", "Cannot find RO with id " + id).toString()).build();
        }

        Document document = iter.first();
        ObjectId mapId = new ObjectId(((Document) document.get("Aggregation")).get("authoratativeMap").toString());

        //FindIterable<Document> oreIter = oreMapCollection.find(new Document("_id", mapId));
        //Document map = oreIter.first();
        GridFSDBFile dbFile = oreMapBucket.findOne(mapId);
        if (dbFile == null) {
            return Response.status(javax.ws.rs.core.Response.Status.NOT_FOUND)
                    .entity(new JSONObject().put("Error", "Cannot find ORE with id " + id).toString()).build();
        }

        ByteArrayOutputStream bos = new ByteArrayOutputStream();
        dbFile.writeTo(bos);
        Document map = Document.parse(bos.toString());

        //Internal meaning only
        map.remove("_id");
        //document.remove("_id");

        return Response.ok(map.toJson()).build();
    }

    @DELETE
    @Path("/{id}/oremap")
    public Response deleteOreByDocumentId(@PathParam("id") String id) {
        oreMapBucket.remove(new ObjectId(id));
        return Response.status(ClientResponse.Status.OK).build();
    }

    @POST
    @Path("/{id}/fgdc")
    @Consumes(MediaType.APPLICATION_XML)
    @Produces(MediaType.APPLICATION_XML)
    public Response addFgdc(String fgdcString, @PathParam("id") String id) {
        fgdcCollection.deleteMany(new Document("@id", id));
        Document document = new Document();
        document.put("@id", id);
        document.put("metadata", fgdcString);
        fgdcCollection.insertOne(document);
        return Response.ok().build();
    }

    @GET
    @Path("/{id}/fgdc")
    @Produces(MediaType.APPLICATION_XML)
    public Response getFgdc(@PathParam("id") String id) {
        FindIterable<Document> iter = fgdcCollection.find(new Document("@id", id));
        if (iter != null && iter.first() != null) {
            return Response.ok(iter.first().get("metadata").toString()).build();
        } else {
            return Response.status(ClientResponse.Status.NOT_FOUND).build();
        }
    }

    //If pid resolves to a published research object, return that RO ID
    @GET
    @Path("/pid/{pid}")
    @Produces(MediaType.APPLICATION_JSON)
    public Response getRoOfPID(@PathParam("pid") String pid) {

        BasicDBObject statusObj = new BasicDBObject();
        statusObj.put("stage", "Success");
        if (pid.contains("http://doi.org/") || pid.contains("http://dx.doi.org/") || pid.matches("^doi:.*")) {
            String pidQuery = pid.replace("http://doi.org/", "").replace("http://dx.doi.org/", "")
                    .replaceAll("^doi:", "").replaceAll("^/+", "").replaceAll("/+$", "");
            Pattern pattern = Pattern.compile("(http://doi.org/|http://dx.doi.org/|doi:)" + pidQuery);
            statusObj.put("message", pattern);
        } else {
            String pidQuery = pid.replaceAll("^/+", "").replaceAll("/+$", "");
            Pattern pattern = Pattern.compile(pidQuery);
            statusObj.put("message", pattern);
        }
        BasicDBObject elemMatch = new BasicDBObject();
        elemMatch.put("$elemMatch", statusObj);
        BasicDBObject query = new BasicDBObject("Status", elemMatch);

        FindIterable<Document> iter = publicationsCollection.find(query);
        iter.projection(new Document("Aggregation.Identifier", 1).append("_id", 0));
        if (iter != null && iter.first() != null) {
            return Response.ok("{\"roId\" : \""
                    + ((Document) iter.first().get("Aggregation")).get("Identifier").toString() + "\" }").build();
        } else {
            return Response.status(ClientResponse.Status.NOT_FOUND).build();
        }
    }

    //Deprecate oldRO by newRO. Delete the old RO request and OREMap
    @GET
    @Path("/deprecate/{newRO}/{oldRO}")
    @Produces(MediaType.APPLICATION_JSON)
    public Response deprecateRO(@PathParam("newRO") String newRoId, @PathParam("oldRO") String oldRoId) {

        try {
            Response roObjectResponse = getROProfile(newRoId);
            Object context = new JSONObject(roObjectResponse.getEntity().toString()).get("@context");
            addIfNeeded(context, Constants.alternateOf, Constants.alternateOfIRI);

            UpdateResult urContext = publicationsCollection.updateOne(
                    new Document("Aggregation.Identifier", newRoId),
                    new BasicDBObject("$set", new BasicDBObject("@context", JSON.parse(context.toString()))));
            UpdateResult urAggregation = publicationsCollection.updateOne(
                    new Document("Aggregation.Identifier", newRoId),
                    new BasicDBObject("$set", new BasicDBObject("Aggregation." + Constants.alternateOf, oldRoId)));
            if (urContext.wasAcknowledged() && urAggregation.wasAcknowledged()) {
                DeleteOverrideRO(oldRoId);
                return Response.status(ClientResponse.Status.OK).build();
            } else {
                return Response.status(ClientResponse.Status.NOT_FOUND).build();

            }
        } catch (org.bson.BsonInvalidOperationException e) {
            return Response.status(ClientResponse.Status.BAD_REQUEST).build();
        }
    }

    private void addIfNeeded(Object context, String key, String uri) {
        if (!isInContext(context, key)) {
            addToContext(context, key, uri);
        }
    }

    private boolean addToContext(Object context, String label, String predicate) {
        if (context instanceof JSONArray) {
            // Look for an object in the array to add to
            for (int i = 0; i < ((JSONArray) context).length(); i++) {
                if (addToContext(((JSONArray) context).get(i), label, predicate)) {
                    return true;
                }
            }
        } else if (context instanceof JSONObject) {
            ((JSONObject) context).put(label, predicate);
            return true;
        }
        return false;
    }

    private boolean isInContext(Object context, String label) {
        if (context instanceof JSONArray) {
            for (int i = 0; i < ((JSONArray) context).length(); i++) {
                if (isInContext(((JSONArray) context).get(i), label)) {
                    return true;
                }
            }
        } else if (context instanceof JSONObject) {
            if (((JSONObject) context).has(label)) {
                return true;
            }
        }
        return false;
    }

    //This is a management method used to copy oreMaps from main mongoDB to the GridFS DB
    @PUT
    @Path("/copyoremaps")
    public Response copyOreMaps() {
        FindIterable<Document> iter = oreMapCollection.find();
        MongoCursor<Document> cursor = iter.iterator();
        while (cursor.hasNext()) {
            Document document = cursor.next();
            ObjectId objectId = new ObjectId((String) document.get("_id").toString());
            document.remove("_id");
            String fileName = "ore-file";
            ByteArrayInputStream inputStream = new ByteArrayInputStream(document.toJson().getBytes());
            GridFSInputFile gfsFile = oreMapBucket.createFile(inputStream, fileName, true);
            gfsFile.setId(objectId);
            gfsFile.save();
        }
        return Response.ok().build();
    }

    private List<String> getOrganizationforPerson(String pID) {

        List<String> orgs = new ArrayList<String>();

        Profile profile = Provider.findCanonicalId(pID);

        // If null, no chance that we have a profile...
        if (profile != null) {

            Document profileDoc = PeopleServices.retrieveProfile(profile.getIdentifier());

            // NeverFail
            if (profileDoc == null) {
                // Handle per provider

                PeopleServices.registerPerson("{\"provider\": \"" + profile.getProvider() + "\", \"identifier\":\""
                        + profile.getIdentifier() + "\" }");
            }
            profileDoc = PeopleServices.retrieveProfile(profile.getIdentifier());

            if (profileDoc != null) {
                String currentAffiliations = profileDoc.getString("affiliation");
                if (currentAffiliations != null) {
                    orgs = Arrays.asList(currentAffiliations.split("\\s*,\\s*"));
                }
            }
        }

        return orgs;
    }

}