All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.arcadedb.query.sql.function.graph.SQLFunctionShortestPath Maven / Gradle / Ivy

There is a newer version: 24.11.1
Show newest version
/*
 * Copyright © 2021-present Arcade Data Ltd ([email protected])
 *
 * 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.
 *
 * SPDX-FileCopyrightText: 2021-present Arcade Data Ltd ([email protected])
 * SPDX-License-Identifier: Apache-2.0
 */
package com.arcadedb.query.sql.function.graph;

import com.arcadedb.database.Document;
import com.arcadedb.database.Identifiable;
import com.arcadedb.database.RID;
import com.arcadedb.exception.CommandExecutionException;
import com.arcadedb.graph.Edge;
import com.arcadedb.graph.EdgeToVertexIterable;
import com.arcadedb.graph.Vertex;
import com.arcadedb.query.sql.executor.CommandContext;
import com.arcadedb.query.sql.executor.MultiValue;
import com.arcadedb.query.sql.executor.Result;
import com.arcadedb.query.sql.function.math.SQLFunctionMathAbstract;
import com.arcadedb.utility.MultiIterator;
import com.arcadedb.utility.Pair;

import java.util.*;

/**
 * Shortest path algorithm to find the shortest path from one node to another node in a directed graph.
 *
 * @author Luca Garulli (l.garulli--(at)--gmail.com)
 */
public class SQLFunctionShortestPath extends SQLFunctionMathAbstract {
  public static final String NAME            = "shortestPath";
  public static final String PARAM_MAX_DEPTH = "maxDepth";

  public SQLFunctionShortestPath() {
    super(NAME);
  }

  private static class OShortestPathContext {
    Vertex           sourceVertex;
    Vertex           destinationVertex;
    Vertex.DIRECTION directionLeft  = Vertex.DIRECTION.BOTH;
    Vertex.DIRECTION directionRight = Vertex.DIRECTION.BOTH;

    String   edgeType;
    String[] edgeTypeParam;

    ArrayDeque queueLeft  = new ArrayDeque<>();
    ArrayDeque queueRight = new ArrayDeque<>();

    final Set leftVisited  = new HashSet();
    final Set rightVisited = new HashSet();

    final Map previouses = new HashMap();
    final Map nexts      = new HashMap();

    Vertex current;
    Vertex currentRight;
    public Integer maxDepth;
    /**
     * option that decides whether or not to return the edge information
     */
    public Boolean edge;
  }

  public List execute(final Object iThis, final Identifiable iCurrentRecord, final Object iCurrentResult, final Object[] iParams,
      final CommandContext iContext) {

    final OShortestPathContext context = new OShortestPathContext();

    Object source = iParams[0];
    if (MultiValue.isMultiValue(source)) {
      if (MultiValue.getSize(source) > 1)
        throw new IllegalArgumentException("Only one sourceVertex is allowed");
      source = MultiValue.getFirstValue(source);
      if (source instanceof Result && ((Result) source).isElement()) {
        source = ((Result) source).getElement().get();
      }
    }

    //    source = record.get((String) source);

    if (source instanceof Identifiable) {
      final Document elem = (Document) ((Identifiable) source).getRecord();
      if (!(elem instanceof Vertex))
        throw new IllegalArgumentException("The sourceVertex must be a vertex record");

      context.sourceVertex = (Vertex) elem;
    } else {
      throw new IllegalArgumentException("The sourceVertex must be a vertex record");
    }

    Object dest = iParams[1];
    if (MultiValue.isMultiValue(dest)) {
      if (MultiValue.getSize(dest) > 1)
        throw new IllegalArgumentException("Only one destinationVertex is allowed");
      dest = MultiValue.getFirstValue(dest);
      if (dest instanceof Result && ((Result) dest).isElement()) {
        dest = ((Result) dest).getElement().get();
      }
    }

    //    dest = record.get((String) dest);

    if (dest instanceof Identifiable) {
      final Document elem = (Document) ((Identifiable) dest).getRecord();
      if (!(elem instanceof Vertex))
        throw new IllegalArgumentException("The destinationVertex must be a vertex record");

      context.destinationVertex = (Vertex) elem;
    } else {
      throw new IllegalArgumentException("The destinationVertex must be a vertex record");
    }

    if (context.sourceVertex.equals(context.destinationVertex)) {
      final List result = new ArrayList(1);
      result.add(context.destinationVertex.getIdentity());
      return result;
    }

    if (iParams.length > 2 && iParams[2] != null) {
      context.directionLeft = Vertex.DIRECTION.valueOf(iParams[2].toString().toUpperCase(Locale.ENGLISH));
    }
    if (context.directionLeft == Vertex.DIRECTION.OUT) {
      context.directionRight = Vertex.DIRECTION.IN;
    } else if (context.directionLeft == Vertex.DIRECTION.IN) {
      context.directionRight = Vertex.DIRECTION.OUT;
    }

    context.edgeType = null;
    if (iParams.length > 3) {
      context.edgeType = iParams[3] == null ? null : "" + iParams[3];
    }
    context.edgeTypeParam = null;
    if (iParams.length > 3 && iParams[3] != null) {
      if (iParams[3] instanceof List) {
        final List list = (List) iParams[3];
        context.edgeTypeParam = list.toArray(new String[list.size()]);
      } else
        context.edgeTypeParam = new String[] { context.edgeType };
    }

    if (iParams.length > 4) {
      bindAdditionalParams(iParams[4], context);
    }

    context.queueLeft.add(context.sourceVertex);
    context.leftVisited.add(context.sourceVertex.getIdentity());

    context.queueRight.add(context.destinationVertex);
    context.rightVisited.add(context.destinationVertex.getIdentity());

    int depth = 1;
    while (true) {
      if (context.maxDepth != null && context.maxDepth <= depth) {
        break;
      }
      if (context.queueLeft.isEmpty() || context.queueRight.isEmpty())
        break;

      if (Thread.interrupted())
        throw new CommandExecutionException("The shortestPath() function has been interrupted");

      List neighborIdentity;

      if (context.queueLeft.size() <= context.queueRight.size()) {
        // START EVALUATING FROM LEFT
        neighborIdentity = walkLeft(context);
        if (neighborIdentity != null)
          return neighborIdentity;
        depth++;
        if (context.maxDepth != null && context.maxDepth <= depth) {
          break;
        }

        if (context.queueLeft.isEmpty())
          break;

        neighborIdentity = walkRight(context);
        if (neighborIdentity != null)
          return neighborIdentity;

      } else {
        // START EVALUATING FROM RIGHT
        neighborIdentity = walkRight(context);
        if (neighborIdentity != null)
          return neighborIdentity;

        depth++;
        if (context.maxDepth != null && context.maxDepth <= depth) {
          break;
        }

        if (context.queueRight.isEmpty())
          break;

        neighborIdentity = walkLeft(context);
        if (neighborIdentity != null)
          return neighborIdentity;
      }

      depth++;
    }
    return new ArrayList();

  }

  private void bindAdditionalParams(final Object additionalParams, final OShortestPathContext context) {
    if (additionalParams == null)
      return;

    Map mapParams = null;
    if (additionalParams instanceof Map)
      mapParams = (Map) additionalParams;
    else if (additionalParams instanceof Identifiable)
      mapParams = ((Document) ((Identifiable) additionalParams).getRecord()).toMap();

    if (mapParams != null) {
      context.maxDepth = integer(mapParams.get("maxDepth"));
      final Boolean withEdge = toBoolean(mapParams.get("edge"));
      context.edge = Boolean.TRUE.equals(withEdge) ? Boolean.TRUE : Boolean.FALSE;
    }
  }

  private Integer integer(final Object fromObject) {
    if (fromObject == null)
      return null;

    if (fromObject instanceof Number)
      return ((Number) fromObject).intValue();

    if (fromObject instanceof String) {
      try {
        return Integer.parseInt(fromObject.toString());
      } catch (final NumberFormatException ignore) {
      }
    }
    return null;
  }

  /**
   * @return
   *
   * @author Thomas Young ([email protected])
   */
  private Boolean toBoolean(final Object fromObject) {
    if (fromObject == null)
      return null;

    if (fromObject instanceof Boolean)
      return (Boolean) fromObject;

    if (fromObject instanceof String) {
      try {
        return Boolean.parseBoolean(fromObject.toString());
      } catch (final NumberFormatException ignore) {
      }
    }
    return null;
  }

  /**
   * get adjacent vertices and edges
   *
   * @param srcVertex
   * @param direction
   * @param types
   *
   * @return
   *
   * @author Thomas Young ([email protected])
   */
  private Pair, Iterable> getVerticesAndEdges(final Vertex srcVertex, final Vertex.DIRECTION direction, final String... types) {
    if (direction == Vertex.DIRECTION.BOTH) {
      final MultiIterator vertexIterator = new MultiIterator<>();
      final MultiIterator edgeIterator = new MultiIterator<>();
      final Pair, Iterable> pair1 = getVerticesAndEdges(srcVertex, Vertex.DIRECTION.OUT, types);
      final Pair, Iterable> pair2 = getVerticesAndEdges(srcVertex, Vertex.DIRECTION.IN, types);
      vertexIterator.addIterator(pair1.getFirst());
      vertexIterator.addIterator(pair2.getFirst());
      edgeIterator.addIterator(pair1.getSecond());
      edgeIterator.addIterator(pair2.getSecond());
      return new Pair<>(vertexIterator, edgeIterator);
    } else {
      final Iterable edges1 = srcVertex.getEdges(direction, types);
      final Iterable edges2 = srcVertex.getEdges(direction, types);
      return new Pair<>(new EdgeToVertexIterable(edges1, direction), edges2);
    }
  }

  /**
   * get adjacent vertices and edges
   *
   * @param srcVertex
   * @param direction
   *
   * @return
   *
   * @author Thomas Young ([email protected])
   */
  private Pair, Iterable> getVerticesAndEdges(final Vertex srcVertex, final Vertex.DIRECTION direction) {
    return getVerticesAndEdges(srcVertex, direction, (String[]) null);
  }

  public String getSyntax() {
    return "shortestPath(, , [, [  ]])";
  }

  protected List walkLeft(final SQLFunctionShortestPath.OShortestPathContext context) {
    final ArrayDeque nextLevelQueue = new ArrayDeque<>();
    if (!Boolean.TRUE.equals(context.edge)) {
      while (!context.queueLeft.isEmpty()) {
        context.current = context.queueLeft.poll();

        final Iterable neighbors;
        if (context.edgeType == null) {
          neighbors = context.current.getVertices(context.directionLeft);
        } else {
          neighbors = context.current.getVertices(context.directionLeft, context.edgeTypeParam);
        }
        for (final Vertex neighbor : neighbors) {
          final Vertex v = neighbor;
          final RID neighborIdentity = v.getIdentity();

          if (context.rightVisited.contains(neighborIdentity)) {
            context.previouses.put(neighborIdentity, context.current.getIdentity());
            return computePath(context.previouses, context.nexts, neighborIdentity);
          }
          if (!context.leftVisited.contains(neighborIdentity)) {
            context.previouses.put(neighborIdentity, context.current.getIdentity());

            nextLevelQueue.offer(v);
            context.leftVisited.add(neighborIdentity);
          }

        }
      }
    } else {
      while (!context.queueLeft.isEmpty()) {
        context.current = context.queueLeft.poll();

        final Pair, Iterable> neighbors;
        if (context.edgeType == null) {
          neighbors = getVerticesAndEdges(context.current, context.directionLeft);
        } else {
          neighbors = getVerticesAndEdges(context.current, context.directionLeft, context.edgeTypeParam);
        }
        final Iterator vertexIterator = neighbors.getFirst().iterator();
        final Iterator edgeIterator = neighbors.getSecond().iterator();
        while (vertexIterator.hasNext() && edgeIterator.hasNext()) {
          final Vertex v = vertexIterator.next();
          final RID neighborVertexIdentity = v.getIdentity();
          final RID neighborEdgeIdentity = edgeIterator.next().getIdentity();

          if (context.rightVisited.contains(neighborVertexIdentity)) {
            context.previouses.put(neighborVertexIdentity, neighborEdgeIdentity);
            context.previouses.put(neighborEdgeIdentity, context.current.getIdentity());
            return computePath(context.previouses, context.nexts, neighborVertexIdentity);
          }
          if (!context.leftVisited.contains(neighborVertexIdentity)) {
            context.previouses.put(neighborVertexIdentity, neighborEdgeIdentity);
            context.previouses.put(neighborEdgeIdentity, context.current.getIdentity());

            nextLevelQueue.offer(v);
            context.leftVisited.add(neighborVertexIdentity);
          }
        }
      }
    }
    context.queueLeft = nextLevelQueue;
    return null;
  }

  protected List walkRight(final SQLFunctionShortestPath.OShortestPathContext context) {
    final ArrayDeque nextLevelQueue = new ArrayDeque<>();
    if (!Boolean.TRUE.equals(context.edge)) {
      while (!context.queueRight.isEmpty()) {
        context.currentRight = context.queueRight.poll();

        final Iterable neighbors;
        if (context.edgeType == null) {
          neighbors = context.currentRight.getVertices(context.directionRight);
        } else {
          neighbors = context.currentRight.getVertices(context.directionRight, context.edgeTypeParam);
        }
        for (final Vertex neighbor : neighbors) {
          final Vertex v = neighbor;
          final RID neighborIdentity = v.getIdentity();

          if (context.leftVisited.contains(neighborIdentity)) {
            context.nexts.put(neighborIdentity, context.currentRight.getIdentity());
            return computePath(context.previouses, context.nexts, neighborIdentity);
          }
          if (!context.rightVisited.contains(neighborIdentity)) {

            context.nexts.put(neighborIdentity, context.currentRight.getIdentity());

            nextLevelQueue.offer(v);
            context.rightVisited.add(neighborIdentity);
          }

        }
      }
    } else {
      while (!context.queueRight.isEmpty()) {
        context.currentRight = context.queueRight.poll();

        final Pair, Iterable> neighbors;
        if (context.edgeType == null) {
          neighbors = getVerticesAndEdges(context.currentRight, context.directionRight);
        } else {
          neighbors = getVerticesAndEdges(context.currentRight, context.directionRight, context.edgeTypeParam);
        }

        final Iterator vertexIterator = neighbors.getFirst().iterator();
        final Iterator edgeIterator = neighbors.getSecond().iterator();
        while (vertexIterator.hasNext() && edgeIterator.hasNext()) {
          final Vertex v = vertexIterator.next();
          final RID neighborVertexIdentity = v.getIdentity();
          final RID neighborEdgeIdentity = edgeIterator.next().getIdentity();

          if (context.leftVisited.contains(neighborVertexIdentity)) {
            context.nexts.put(neighborVertexIdentity, neighborEdgeIdentity);
            context.nexts.put(neighborEdgeIdentity, context.currentRight.getIdentity());
            return computePath(context.previouses, context.nexts, neighborVertexIdentity);
          }
          if (!context.rightVisited.contains(neighborVertexIdentity)) {
            context.nexts.put(neighborVertexIdentity, neighborEdgeIdentity);
            context.nexts.put(neighborEdgeIdentity, context.currentRight.getIdentity());

            nextLevelQueue.offer(v);
            context.rightVisited.add(neighborVertexIdentity);
          }
        }
      }
    }
    context.queueRight = nextLevelQueue;
    return null;
  }

  private List computePath(final Map leftDistances, final Map rightDistances, final RID neighbor) {
    final List result = new ArrayList();

    RID current = neighbor;
    while (current != null) {
      result.add(0, current);
      current = leftDistances.get(current);
    }

    current = neighbor;
    while (current != null) {
      current = rightDistances.get(current);
      if (current != null) {
        result.add(current);
      }
    }

    return result;
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy