/*
 *  Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you 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 Jorge Bay Gondra
 */
'use strict';
const gt = require('../process/graph-traversal');
const { TraversalStrategies } = require('../process/traversal-strategy');
/**
 * An "empty" graph object to server only as a reference.
 */
class Graph {
  /**
   * Returns the graph traversal source.
   * @param {Function} [traversalSourceClass] The constructor to use for the {@code GraphTraversalSource} instance.
   * @returns {GraphTraversalSource}
   * @deprecated As of release 3.3.5, replaced by the traversal() anonymous function.
   */
  traversal(traversalSourceClass) {
    const traversalSourceConstructor = traversalSourceClass || gt.GraphTraversalSource;
    return new traversalSourceConstructor(this, new TraversalStrategies());
  }
  toString() {
    return 'graph[]';
  }
}
class Element {
  // properties are stored as list of property objects
  constructor(id, label, properties = []) {
    this.id = id;
    this.label = label;
    this.properties = properties != null ? properties : [];
  }
  /**
   * Compares this instance to another and determines if they can be considered as equal.
   * @param {Element} other
   * @returns {boolean}
   */
  equals(other) {
    return other instanceof Element && this.id === other.id;
  }
}
class Vertex extends Element {
  constructor(id, label, properties = []) {
    super(id, label, properties);
  }
  toString() {
    return `v[${this.id}]`;
  }
}
class Edge extends Element {
  constructor(id, outV, label, inV, properties = []) {
    super(id, label, properties);
    this.outV = outV;
    this.inV = inV;
  }
  toString() {
    const outVId = this.outV ? this.outV.id : '?';
    const inVId = this.inV ? this.inV.id : '?';
    return `e[${this.id}][${outVId}-${this.label}->${inVId}]`;
  }
}
class VertexProperty extends Element {
  constructor(id, label, value, properties = []) {
    super(id, label, properties);
    this.value = value;
    this.key = this.label;
  }
  toString() {
    return `vp[${this.label}->${summarize(this.value)}]`;
  }
}
class Property {
  constructor(key, value) {
    this.key = key;
    this.value = value;
  }
  toString() {
    return `p[${this.key}->${summarize(this.value)}]`;
  }
  equals(other) {
    return other instanceof Property && this.key === other.key && this.value === other.value;
  }
}
class Path {
  /**
   * Represents a walk through a graph as defined by a traversal.
   * @param {Array} labels
   * @param {Array} objects
   * @constructor
   */
  constructor(labels, objects) {
    this.labels = labels;
    this.objects = objects;
  }
  toString() {
    return `path[${(this.objects || []).join(', ')}]`;
  }
  equals(other) {
    if (!(other instanceof Path)) {
      return false;
    }
    if (other === this) {
      return true;
    }
    return areEqual(this.objects, other.objects) && areEqual(this.labels, other.labels);
  }
}
function areEqual(obj1, obj2) {
  if (obj1 === obj2) {
    return true;
  }
  if (typeof obj1.equals === 'function') {
    return obj1.equals(obj2);
  }
  if (Array.isArray(obj1) && Array.isArray(obj2)) {
    if (obj1.length !== obj2.length) {
      return false;
    }
    for (let i = 0; i < obj1.length; i++) {
      if (!areEqual(obj1[i], obj2[i])) {
        return false;
      }
    }
    return true;
  }
  return false;
}
function summarize(value) {
  if (value === null || value === undefined) {
    return value;
  }
  const strValue = value.toString();
  return strValue.length > 20 ? strValue.substr(0, 20) : strValue;
}
module.exports = {
  Edge,
  Graph,
  Path,
  Property,
  Vertex,
  VertexProperty,
};