package io.lumify.web.routes.vertex; import com.google.inject.Inject; import io.lumify.core.config.Configuration; import io.lumify.core.exception.LumifyException; import io.lumify.core.model.SourceInfo; import io.lumify.core.model.audit.AuditAction; import io.lumify.core.model.audit.AuditRepository; import io.lumify.core.model.ontology.OntologyProperty; import io.lumify.core.model.ontology.OntologyRepository; import io.lumify.core.model.properties.LumifyProperties; import io.lumify.core.model.termMention.TermMentionRepository; import io.lumify.core.model.user.UserRepository; import io.lumify.core.model.workQueue.WorkQueueRepository; import io.lumify.core.model.workspace.Workspace; import io.lumify.core.model.workspace.WorkspaceRepository; import io.lumify.core.security.VisibilityTranslator; import io.lumify.core.user.User; import io.lumify.core.util.ClientApiConverter; import io.lumify.core.util.GraphUtil; import io.lumify.core.util.LumifyLogger; import io.lumify.core.util.LumifyLoggerFactory; import io.lumify.miniweb.HandlerChain; import io.lumify.web.BaseRequestHandler; import io.lumify.web.clientapi.model.ClientApiElement; import org.securegraph.*; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class VertexSetProperty extends BaseRequestHandler { private static final LumifyLogger LOGGER = LumifyLoggerFactory.getLogger(VertexSetProperty.class); private final Graph graph; private final OntologyRepository ontologyRepository; private final AuditRepository auditRepository; private final VisibilityTranslator visibilityTranslator; private final WorkspaceRepository workspaceRepository; private final WorkQueueRepository workQueueRepository; private final TermMentionRepository termMentionRepository; @Inject public VertexSetProperty( final OntologyRepository ontologyRepository, final Graph graph, final AuditRepository auditRepository, final VisibilityTranslator visibilityTranslator, final UserRepository userRepository, final Configuration configuration, final WorkspaceRepository workspaceRepository, final WorkQueueRepository workQueueRepository, final TermMentionRepository termMentionRepository ) { super(userRepository, workspaceRepository, configuration); this.ontologyRepository = ontologyRepository; this.graph = graph; this.auditRepository = auditRepository; this.visibilityTranslator = visibilityTranslator; this.workspaceRepository = workspaceRepository; this.workQueueRepository = workQueueRepository; this.termMentionRepository = termMentionRepository; } @Override public void handle(HttpServletRequest request, HttpServletResponse response, HandlerChain chain) throws Exception { final String graphVertexId = getAttributeString(request, "graphVertexId"); final String propertyName = getRequiredParameter(request, "propertyName"); final String propertyKey = getOptionalParameter(request, "propertyKey"); final String valueStr = getOptionalParameter(request, "value"); final String[] valuesStr = getOptionalParameterArray(request, "value[]"); final String visibilitySource = getRequiredParameter(request, "visibilitySource"); final String justificationText = getOptionalParameter(request, "justificationText"); final String sourceInfo = getOptionalParameter(request, "sourceInfo"); final String metadataString = getOptionalParameter(request, "metadata"); User user = getUser(request); String workspaceId = getActiveWorkspaceId(request); Authorizations authorizations = getAuthorizations(request, user); if (valueStr == null && valuesStr == null) { throw new LumifyException("Parameter: 'value' or 'value[]' is required in the request"); } if (!graph.isVisibilityValid(new Visibility(visibilitySource), authorizations)) { LOGGER.warn("%s is not a valid visibility for %s user", visibilitySource, user.getDisplayName()); respondWithBadRequest(response, "visibilitySource", getString(request, "visibility.invalid")); chain.next(request, response); return; } if (propertyName.equals(LumifyProperties.COMMENT.getPropertyName()) && request.getPathInfo().equals("/vertex/property")) { throw new LumifyException("Use /vertex/comment to save comment properties"); } else if (request.getPathInfo().equals("/vertex/comment") && !propertyName.equals(LumifyProperties.COMMENT.getPropertyName())) { throw new LumifyException("Use /vertex/property to save non-comment properties"); } respondWithClientApiObject(response, handle( graphVertexId, propertyName, propertyKey, valueStr, valuesStr, justificationText, sourceInfo, metadataString, visibilitySource, user, workspaceId, authorizations)); } private ClientApiElement handle( String graphVertexId, String propertyName, String propertyKey, String valueStr, String[] valuesStr, String justificationText, String sourceInfoString, String metadataString, String visibilitySource, User user, String workspaceId, Authorizations authorizations) { if (propertyKey == null) { propertyKey = this.graph.getIdGenerator().nextId(); } if (valueStr == null && valuesStr != null && valuesStr.length == 1) { valueStr = valuesStr[0]; } if (valuesStr == null && valueStr != null) { valuesStr = new String[1]; valuesStr[0] = valueStr; } Metadata metadata = GraphUtil.metadataStringToMap(metadataString, this.visibilityTranslator.getDefaultVisibility()); Object value; if (propertyName.equals("http://lumify.io#comment")) { value = valueStr; } else { OntologyProperty property = ontologyRepository.getPropertyByIRI(propertyName); if (property == null) { throw new RuntimeException("Could not find property: " + propertyName); } if (property.hasDependentPropertyIris()) { if (valuesStr == null) { throw new LumifyException("properties with dependent properties must contain a value"); } if (property.getDependentPropertyIris().size() != valuesStr.length) { throw new LumifyException("properties with dependent properties must contain the same number of values. expected " + property.getDependentPropertyIris().size() + " found " + valuesStr.length); } ClientApiElement clientApiElement = null; int valuesIndex = 0; for (String dependentPropertyIri : property.getDependentPropertyIris()) { clientApiElement = handle( graphVertexId, dependentPropertyIri, propertyKey, valuesStr[valuesIndex++], null, justificationText, sourceInfoString, metadataString, visibilitySource, user, workspaceId, authorizations ); } return clientApiElement; } else { if (valuesStr != null && valuesStr.length > 1) { throw new LumifyException("properties without dependent properties must not contain more than one value."); } if (valueStr == null) { throw new LumifyException("properties without dependent properties must have a value"); } try { value = property.convertString(valueStr); } catch (Exception ex) { LOGGER.warn(String.format("Validation error propertyName: %s, valueStr: %s", propertyName, valueStr), ex); throw new LumifyException(ex.getMessage(), ex); } } } Vertex graphVertex = graph.getVertex(graphVertexId, authorizations); SourceInfo sourceInfo = SourceInfo.fromString(sourceInfoString); GraphUtil.VisibilityAndElementMutation<Vertex> setPropertyResult = GraphUtil.setProperty( graph, graphVertex, propertyName, propertyKey, value, metadata, visibilitySource, workspaceId, this.visibilityTranslator, justificationText, sourceInfo, termMentionRepository, user, authorizations); auditRepository.auditVertexElementMutation(AuditAction.UPDATE, setPropertyResult.elementMutation, graphVertex, "", user, setPropertyResult.visibility.getVisibility()); graphVertex = setPropertyResult.elementMutation.save(authorizations); graph.flush(); Workspace workspace = workspaceRepository.findById(workspaceId, user); this.workspaceRepository.updateEntityOnWorkspace(workspace, graphVertex.getId(), null, null, user); this.workQueueRepository.pushGraphPropertyQueue(graphVertex, propertyKey, propertyName, workspaceId, visibilitySource); if (sourceInfo != null) { this.workQueueRepository.pushTextUpdated(sourceInfo.getVertexId()); } return ClientApiConverter.toClientApi(graphVertex, workspaceId, authorizations); } }