/* * * * Copyright 2014 Orient Technologies LTD (info(at)orientechnologies.com) * * * * 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. * * * * For more information: http://www.orientechnologies.com * */ package com.orientechnologies.orient.graph.sql; import com.orientechnologies.common.exception.OException; import com.orientechnologies.common.types.OModifiableBoolean; import com.orientechnologies.orient.core.command.OCommandDistributedReplicateRequest; import com.orientechnologies.orient.core.command.OCommandExecutor; import com.orientechnologies.orient.core.command.OCommandManager; import com.orientechnologies.orient.core.command.OCommandRequest; import com.orientechnologies.orient.core.command.OCommandRequestInternal; import com.orientechnologies.orient.core.command.OCommandRequestText; import com.orientechnologies.orient.core.command.OCommandResultListener; import com.orientechnologies.orient.core.config.OGlobalConfiguration; import com.orientechnologies.orient.core.db.document.ODatabaseDocument; import com.orientechnologies.orient.core.db.record.OIdentifiable; import com.orientechnologies.orient.core.exception.OCommandExecutionException; import com.orientechnologies.orient.core.id.ORecordId; import com.orientechnologies.orient.core.metadata.OMetadataInternal; import com.orientechnologies.orient.core.metadata.schema.OClass; import com.orientechnologies.orient.core.metadata.security.ORole; import com.orientechnologies.orient.core.record.ORecord; import com.orientechnologies.orient.core.record.impl.ODocument; import com.orientechnologies.orient.core.serialization.serializer.OStringSerializerHelper; import com.orientechnologies.orient.core.sql.OCommandExecutorSQLAbstract; import com.orientechnologies.orient.core.sql.OCommandSQLParsingException; import com.orientechnologies.orient.core.sql.query.OSQLAsynchQuery; import com.tinkerpop.blueprints.impls.orient.OrientBaseGraph; import com.tinkerpop.blueprints.impls.orient.OrientGraph; import com.tinkerpop.blueprints.impls.orient.OrientVertex; import com.tinkerpop.blueprints.impls.orient.OrientVertexType; import java.util.ArrayList; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Set; import java.util.concurrent.atomic.AtomicReference; /** * SQL DELETE VERTEX command. * * @author Luca Garulli */ public class OCommandExecutorSQLDeleteVertex extends OCommandExecutorSQLAbstract implements OCommandDistributedReplicateRequest, OCommandResultListener { public static final String NAME = "DELETE VERTEX"; private static final String KEYWORD_BATCH = "BATCH"; private ORecordId rid; private int removed = 0; private ODatabaseDocument database; private OCommandRequest query; private String returning = "COUNT"; private List<ORecord> allDeletedRecords; private AtomicReference<OrientBaseGraph> currentGraph = new AtomicReference<OrientBaseGraph>(); private OModifiableBoolean shutdownFlag = new OModifiableBoolean(); private boolean txAlreadyBegun; private int batch = 100; @SuppressWarnings("unchecked") public OCommandExecutorSQLDeleteVertex parse(final OCommandRequest iRequest) { final OCommandRequestText textRequest = (OCommandRequestText) iRequest; String queryText = textRequest.getText(); String originalQuery = queryText; try { // System.out.println("NEW PARSER FROM: " + queryText); queryText = preParse(queryText, iRequest); // System.out.println("NEW PARSER TO: " + queryText); textRequest.setText(queryText); database = getDatabase(); init((OCommandRequestText) iRequest); parserRequiredKeyword("DELETE"); parserRequiredKeyword("VERTEX"); OClass clazz = null; String where = null; int limit = -1; String word = parseOptionalWord(true); while (word != null) { if (word.startsWith("#")) { rid = new ORecordId(word); } else if (word.equalsIgnoreCase("from")) { final StringBuilder q = new StringBuilder(); final int newPos = OStringSerializerHelper.getEmbedded(parserText, parserGetCurrentPosition(), -1, q); query = database.command(new OSQLAsynchQuery<ODocument>(q.toString(), this)); parserSetCurrentPosition(newPos); } else if (word.equals(KEYWORD_WHERE)) { if (clazz == null) // ASSIGN DEFAULT CLASS clazz = ((OMetadataInternal) database.getMetadata()).getImmutableSchemaSnapshot().getClass(OrientVertexType.CLASS_NAME); where = parserGetCurrentPosition() > -1 ? " " + parserText.substring(parserGetPreviousPosition()) : ""; query = database.command(new OSQLAsynchQuery<ODocument>("select from `" + clazz.getName() + "`" + where, this)); break; } else if (word.equals(KEYWORD_LIMIT)) { word = parseOptionalWord(true); try { limit = Integer.parseInt(word); } catch (Exception e) { throw OException.wrapException(new OCommandSQLParsingException("Invalid LIMIT: " + word), e); } } else if (word.equals(KEYWORD_RETURN)) { returning = parseReturn(); } else if (word.equals(KEYWORD_BATCH)) { word = parserNextWord(true); if (word != null) batch = Integer.parseInt(word); } else if (word.length() > 0) { // GET/CHECK CLASS NAME clazz = ((OMetadataInternal) database.getMetadata()).getImmutableSchemaSnapshot().getClass(word); if (clazz == null) throw new OCommandSQLParsingException("Class '" + word + "' was not found"); } word = parseOptionalWord(true); if (parserIsEnded()) break; } if (where == null) where = ""; else where = " WHERE " + where; if (query == null && rid == null) { StringBuilder queryString = new StringBuilder(); queryString.append("select from `"); if (clazz == null) { queryString.append(OrientVertexType.CLASS_NAME); } else { queryString.append(clazz.getName()); } queryString.append("`"); queryString.append(where); if (limit > -1) { queryString.append(" LIMIT ").append(limit); } query = database.command(new OSQLAsynchQuery<ODocument>(queryString.toString(), this)); } } finally { textRequest.setText(originalQuery); } return this; } /** * Execute the command and return the ODocument object created. */ public Object execute(final Map<Object, Object> iArgs) { if (rid == null && query == null) throw new OCommandExecutionException("Cannot execute the command because it has not been parsed yet"); if (!returning.equalsIgnoreCase("COUNT")) allDeletedRecords = new ArrayList<ORecord>(); txAlreadyBegun = getDatabase().getTransaction().isActive(); if (rid != null) { // REMOVE PUNCTUAL RID OGraphCommandExecutorSQLFactory.runInConfiguredTxMode(new OGraphCommandExecutorSQLFactory.GraphCallBack<Object>() { @Override public Object call(OrientBaseGraph graph) { final OrientVertex v = graph.getVertex(rid); if (v != null) { v.remove(); removed = 1; } return null; } }); // CLOSE PENDING TX end(); } else if (query != null) { // TARGET IS A CLASS + OPTIONAL CONDITION OGraphCommandExecutorSQLFactory.runInConfiguredTxMode(new OGraphCommandExecutorSQLFactory.GraphCallBack<OrientGraph>() { @Override public OrientGraph call(final OrientBaseGraph iGraph) { // TARGET IS A CLASS + OPTIONAL CONDITION currentGraph.set(iGraph); query.setContext(getContext()); query.execute(iArgs); return null; } }); } else throw new OCommandExecutionException("Invalid target"); if (returning.equalsIgnoreCase("COUNT")) // RETURNS ONLY THE COUNT return removed; else // RETURNS ALL THE DELETED RECORDS return allDeletedRecords; } /** * Delete the current vertex. */ public boolean result(final Object iRecord) { final OIdentifiable id = (OIdentifiable) iRecord; if (id.getIdentity().isValid()) { final ODocument record = id.getRecord(); final OrientBaseGraph g = currentGraph.get(); final OrientVertex v = g.getVertex(record); if (v != null) { v.remove(); if (!txAlreadyBegun && batch > 0 && removed % batch == 0) { if (g instanceof OrientGraph) { g.commit(); ((OrientGraph) g).begin(); } } if (returning.equalsIgnoreCase("BEFORE")) allDeletedRecords.add(record); removed++; } } return true; } @Override public long getDistributedTimeout() { return OGlobalConfiguration.DISTRIBUTED_COMMAND_TASK_SYNCH_TIMEOUT.getValueAsLong(); } @Override public String getSyntax() { return "DELETE VERTEX <rid>|<class>|FROM <query> [WHERE <conditions>] [LIMIT <max-records>] [RETURN <COUNT|BEFORE>]> [BATCH <batch-size>]"; } @Override public void end() { final OrientBaseGraph g = currentGraph.get(); if (g != null) { if (!txAlreadyBegun) { g.commit(); if (shutdownFlag.getValue()) g.shutdown(false); } } } @Override public int getSecurityOperationType() { return ORole.PERMISSION_DELETE; } /** * Parses the returning keyword if found. */ protected String parseReturn() throws OCommandSQLParsingException { final String returning = parserNextWord(true); if (!returning.equalsIgnoreCase("COUNT") && !returning.equalsIgnoreCase("BEFORE")) throwParsingException("Invalid " + KEYWORD_RETURN + " value set to '" + returning + "' but it should be COUNT (default), BEFORE. Example: " + KEYWORD_RETURN + " BEFORE"); return returning; } @Override public QUORUM_TYPE getQuorumType() { return QUORUM_TYPE.WRITE; } public DISTRIBUTED_RESULT_MGMT getDistributedResultManagement() { return getDistributedExecutionMode() == DISTRIBUTED_EXECUTION_MODE.LOCAL ? DISTRIBUTED_RESULT_MGMT.CHECK_FOR_EQUALS : DISTRIBUTED_RESULT_MGMT.MERGE; } @Override public Set<String> getInvolvedClusters() { final HashSet<String> result = new HashSet<String>(); if (rid != null) result.add(database.getClusterNameById(rid.getClusterId())); else if (query != null) { final OCommandExecutor executor = OCommandManager.instance().getExecutor((OCommandRequestInternal) query); // COPY THE CONTEXT FROM THE REQUEST executor.setContext(context); executor.parse(query); return executor.getInvolvedClusters(); } return result; } @Override public Object getResult() { return null; } public OCommandDistributedReplicateRequest.DISTRIBUTED_EXECUTION_MODE getDistributedExecutionMode() { return DISTRIBUTED_EXECUTION_MODE.LOCAL; } /** * setLimit() for DELETE VERTEX is ignored. Please use LIMIT keyword in the SQL statement */ public <RET extends OCommandExecutor> RET setLimit(final int iLimit) { //do nothing return (RET) this; } }