Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance. Project price only 1 $
You can buy this project and download/modify it how often you want.
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [http://neo4j.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.cypher.internal.logical.plans
import org.neo4j.cypher.internal.util.attribution.Id
import org.neo4j.cypher.internal.util.attribution.IdGen
/**
* Triadic selection is used to solve a common query pattern:
* MATCH (a)-->(b)-->(c) WHERE NOT (a)-->(c)
*
* If this query can be solved by starting from (a) and expand first (a)-->(b)
* and expanding (b)-->(c), we can replace the filter with a triadic selection
* that runs the (a)-->(b) as its left hand side, caching the results for use in
* filtering the results of its right hand side which is the (b)-->(c) expands.
* The filtering is based on the pattern expression predicate. The classical
* example is the friend of a friend that is not already a friend, as shown above,
* but this works for other cases too, like fof that is a friend.
*
* Since the two expands are done by sub-plans, they can be substantially more
* complex than single expands. However, what patterns actually get here need to
* be identified by the triadic selection finder.
*
* In effect the triadic selection interprets the predicate pattern in:
* MATCH (){-->(build)}{-->(target)}
* WHERE NOT ()-->()
*
* as the predicate:
*
* WHERE () NOT IN Set(, for )
*
* With a plan that looks like:
*
* +TriadicSelection (c) NOT IN (b)
* | \
* | + (b)-->(c)
* | |
* | +Argument (b)
* |
* + (a)-->(b)
* |
* + (a)
*/
case class TriadicSelection(override val left: LogicalPlan,
override val right: LogicalPlan,
positivePredicate: Boolean,
sourceId: String,
seenId: String,
targetId: String
)(implicit idGen: IdGen)
extends LogicalBinaryPlan(idGen) with ApplyPlan {
override def withLhs(newLHS: LogicalPlan)(idGen: IdGen): LogicalBinaryPlan = copy(left = newLHS)(idGen)
override def withRhs(newRHS: LogicalPlan)(idGen: IdGen): LogicalBinaryPlan = copy(right = newRHS)(idGen)
override val availableSymbols: Set[String] = left.availableSymbols ++ right.availableSymbols
}
// TriadicBuild and TriadicFilter are used by Pipelined to perform the same logic as TriadicSelection.
// 'triadicSelectionId' is used to link corresponding Build and Filter plans.
case class TriadicBuild(override val source: LogicalPlan,
sourceId: String,
seenId: String,
triadicSelectionId: Some[Id]) // wrapped in Some because Id is a value class and doesn't play well with rewriting
(implicit idGen: IdGen)
extends LogicalUnaryPlan(idGen) {
override def withLhs(newLHS: LogicalPlan)(idGen: IdGen): LogicalUnaryPlan = copy(source = newLHS)(idGen)
override def availableSymbols: Set[String] = source.availableSymbols
}
case class TriadicFilter(override val source: LogicalPlan,
positivePredicate: Boolean,
sourceId: String,
targetId: String,
triadicSelectionId: Some[Id]) // wrapped in Some because Id is a value class and doesn't play well with rewriting
(implicit idGen: IdGen)
extends LogicalUnaryPlan(idGen) {
override def withLhs(newLHS: LogicalPlan)(idGen: IdGen): LogicalUnaryPlan = copy(source = newLHS)(idGen)
override def availableSymbols: Set[String] = source.availableSymbols
}