Skip to content

Commit

Permalink
refactor: Remove GravsearchQueryOptimisationFeature and simplify (#2909)
Browse files Browse the repository at this point in the history
  • Loading branch information
seakayone authored Nov 2, 2023
1 parent 303ca0f commit a91f6f6
Show file tree
Hide file tree
Showing 3 changed files with 34 additions and 110 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,7 @@ case class Count(inputVariable: QueryVariable, distinct: Boolean, outputVariable

val outputVariable: QueryVariable = QueryVariable(outputVariableName)

val distinctAsStr: String = if (distinct) {
private val distinctAsStr: String = if (distinct) {
"DISTINCT"
} else {
""
Expand All @@ -110,13 +110,6 @@ case class Count(inputVariable: QueryVariable, distinct: Boolean, outputVariable
*/
case class IriRef(iri: SmartIri, propertyPathOperator: Option[Char] = None) extends Entity {

/**
* If this is a knora-api entity IRI, converts it to an internal entity IRI.
*
* @return the equivalent internal entity IRI.
*/
def toInternalEntityIri: IriRef = IriRef(iri.toOntologySchema(InternalSchema))

override def toSparql: String =
if (propertyPathOperator.nonEmpty) {
s"${iri.toSparql}${propertyPathOperator.get}"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,6 @@ import org.knora.webapi.messages.OntologyConstants
import org.knora.webapi.messages.SmartIri
import org.knora.webapi.messages.StringFormatter
import org.knora.webapi.messages.ValuesValidator
import org.knora.webapi.messages.util.search.CompareExpressionOperator
import org.knora.webapi.messages.util.search._
import org.knora.webapi.messages.util.search.gravsearch.GravsearchQueryChecker
import org.knora.webapi.messages.util.search.gravsearch.transformers.SparqlTransformer
Expand Down Expand Up @@ -146,14 +145,7 @@ abstract class AbstractPrequeryGenerator(
* @return the optimised query patterns.
*/
override def optimiseQueryPatterns(patterns: Seq[QueryPattern]): Task[Seq[QueryPattern]] =
ZIO.attempt(
GravsearchQueryOptimisationFactory
.getGravsearchQueryOptimisationFeature(
typeInspectionResult = typeInspectionResult,
querySchema = querySchema
)
.optimiseQueryPatterns(patterns)
)
ZIO.attempt(GravsearchQueryOptimisation.optimiseQueryPatterns(patterns, typeInspectionResult))

/**
* Transforms a [[org.knora.webapi.messages.util.search.FilterPattern]] in a WHERE clause into zero or more statement patterns.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,76 +8,37 @@ package org.knora.webapi.messages.util.search.gravsearch.prequery
import scalax.collection.Graph
import scalax.collection.GraphEdge.DiHyperEdge

import org.knora.webapi.ApiV2Schema
import org.knora.webapi.messages.OntologyConstants
import org.knora.webapi.messages.SmartIri
import org.knora.webapi.messages.util.search._
import org.knora.webapi.messages.util.search.gravsearch.prequery.RemoveEntitiesInferredFromProperty.removeEntitiesInferredFromProperty
import org.knora.webapi.messages.util.search.gravsearch.prequery.RemoveRedundantKnoraApiResource.removeRedundantKnoraApiResource
import org.knora.webapi.messages.util.search.gravsearch.prequery.ReorderPatternsByDependency.reorderPatternsByDependency
import org.knora.webapi.messages.util.search.gravsearch.types.GravsearchTypeInspectionResult
import org.knora.webapi.messages.util.search.gravsearch.types.GravsearchTypeInspectionUtil
import org.knora.webapi.messages.util.search.gravsearch.types.TypeableEntity

/**
* Represents optimisation algorithms that transform Gravsearch input queries.
*
* @param typeInspectionResult the type inspection result.
* @param querySchema the query schema.
*/
abstract class GravsearchQueryOptimisationFeature(
protected val typeInspectionResult: GravsearchTypeInspectionResult,
protected val querySchema: ApiV2Schema
) {

/**
* Performs the optimisation.
*
* @param patterns the query patterns.
* @return the optimised query patterns.
*/
def optimiseQueryPatterns(patterns: Seq[QueryPattern]): Seq[QueryPattern]
}

/**
* A feature factory that constructs Gravsearch query optimisation algorithms.
*/
object GravsearchQueryOptimisationFactory {

/**
* Returns a [[GravsearchQueryOptimisationFeature]] implementing one or more optimisations, depending
* on the feature factory configuration.
*
* @param typeInspectionResult the type inspection result.
* @param querySchema the query schema.
* @return a [[GravsearchQueryOptimisationFeature]] implementing one or more optimisations.
*/
def getGravsearchQueryOptimisationFeature(
typeInspectionResult: GravsearchTypeInspectionResult,
querySchema: ApiV2Schema
): GravsearchQueryOptimisationFeature =
new GravsearchQueryOptimisationFeature(typeInspectionResult, querySchema) {
override def optimiseQueryPatterns(patterns: Seq[QueryPattern]): Seq[QueryPattern] =
new ReorderPatternsByDependencyOptimisationFeature(this.typeInspectionResult, querySchema)
.optimiseQueryPatterns(
new RemoveEntitiesInferredFromPropertyOptimisationFeature(this.typeInspectionResult, querySchema)
.optimiseQueryPatterns(
new RemoveRedundantKnoraApiResourceOptimisationFeature(this.typeInspectionResult, querySchema)
.optimiseQueryPatterns(patterns)
)
)

}
object GravsearchQueryOptimisation {

def optimiseQueryPatterns(
patterns: Seq[QueryPattern],
typeInspectionResult: GravsearchTypeInspectionResult
): Seq[QueryPattern] = {
val removedRedundant = removeRedundantKnoraApiResource(patterns)
val removedEntities = removeEntitiesInferredFromProperty(removedRedundant, typeInspectionResult)
val result = reorderPatternsByDependency(removedEntities)
result
}
}

/**
* Removes a statement with rdf:type knora-api:Resource if there is another rdf:type statement with the same subject
* and a different type.
*
* @param typeInspectionResult the type inspection result.
* @param querySchema the query schema.
*/
class RemoveRedundantKnoraApiResourceOptimisationFeature(
typeInspectionResult: GravsearchTypeInspectionResult,
querySchema: ApiV2Schema
) extends GravsearchQueryOptimisationFeature(typeInspectionResult, querySchema) {
private object RemoveRedundantKnoraApiResource {

/**
* If the specified statement has rdf:type with an IRI as object, returns that IRI, otherwise None.
Expand All @@ -97,7 +58,7 @@ class RemoveRedundantKnoraApiResourceOptimisationFeature(
case _ => None
}

override def optimiseQueryPatterns(patterns: Seq[QueryPattern]): Seq[QueryPattern] = {
def removeRedundantKnoraApiResource(patterns: Seq[QueryPattern]): Seq[QueryPattern] = {
// Make a Set of subjects that have rdf:type statements whose objects are not knora-api:Resource.
val rdfTypesBySubj: Set[Entity] = patterns
.foldLeft(Set.empty[Entity]) { case (acc, queryPattern: QueryPattern) =>
Expand Down Expand Up @@ -141,18 +102,18 @@ class RemoveRedundantKnoraApiResourceOptimisationFeature(
* use with a property that can only be used with that type (unless the property
* statement is in an `OPTIONAL` block).
*/
class RemoveEntitiesInferredFromPropertyOptimisationFeature(
typeInspectionResult: GravsearchTypeInspectionResult,
querySchema: ApiV2Schema
) extends GravsearchQueryOptimisationFeature(typeInspectionResult, querySchema) {
private object RemoveEntitiesInferredFromProperty {

/**
* Performs the optimisation.
*
* @param patterns the query patterns.
* @return the optimised query patterns.
*/
override def optimiseQueryPatterns(patterns: Seq[QueryPattern]): Seq[QueryPattern] = {
def removeEntitiesInferredFromProperty(
patterns: Seq[QueryPattern],
typeInspectionResult: GravsearchTypeInspectionResult
): Seq[QueryPattern] = {

// Collect all entities which are used as subject or object of an OptionalPattern.
val optionalEntities: Seq[TypeableEntity] = patterns.collect { case optionalPattern: OptionalPattern =>
Expand Down Expand Up @@ -224,10 +185,7 @@ class RemoveEntitiesInferredFromPropertyOptimisationFeature(
/**
* Optimises query patterns by reordering them on the basis of dependencies between subjects and objects.
*/
class ReorderPatternsByDependencyOptimisationFeature(
typeInspectionResult: GravsearchTypeInspectionResult,
querySchema: ApiV2Schema
) extends GravsearchQueryOptimisationFeature(typeInspectionResult, querySchema) {
private object ReorderPatternsByDependency {

/**
* Converts a sequence of query patterns into DAG representing dependencies between
Expand Down Expand Up @@ -395,43 +353,24 @@ class ReorderPatternsByDependencyOptimisationFeature(
* @param patterns the query patterns.
* @return the optimised query patterns.
*/
override def optimiseQueryPatterns(patterns: Seq[QueryPattern]): Seq[QueryPattern] = {
// Separate the statement patterns from the other patterns.
def reorderPatternsByDependency(patterns: Seq[QueryPattern]): Seq[QueryPattern] = {
val (statementPatterns: Seq[StatementPattern], otherPatterns: Seq[QueryPattern]) =
patterns.foldLeft((Vector.empty[StatementPattern], Vector.empty[QueryPattern])) {
case ((statementPatternAcc, otherPatternAcc), pattern: QueryPattern) =>
pattern match {
case statementPattern: StatementPattern => (statementPatternAcc :+ statementPattern, otherPatternAcc)
case _ => (statementPatternAcc, otherPatternAcc :+ pattern)
}
patterns.partition {
case _: StatementPattern => true
case _ => false
}

val sortedStatementPatterns: Seq[QueryPattern] = createAndSortGraph(statementPatterns)
val sortedStatementPatterns = createAndSortGraph(statementPatterns)

val sortedOtherPatterns: Seq[QueryPattern] = otherPatterns.map {
// sort statements inside each UnionPattern block
val sortedOtherPatterns = otherPatterns.map {
case unionPattern: UnionPattern =>
val sortedUnionBlocks: Seq[Seq[QueryPattern]] =
unionPattern.blocks.map(block => optimiseQueryPatterns(block))
UnionPattern(blocks = sortedUnionBlocks)

// sort statements inside OptionalPattern
UnionPattern(unionPattern.blocks.map(reorderPatternsByDependency))
case optionalPattern: OptionalPattern =>
val sortedOptionalPatterns: Seq[QueryPattern] = optimiseQueryPatterns(optionalPattern.patterns)
OptionalPattern(patterns = sortedOptionalPatterns)

// sort statements inside MinusPattern
OptionalPattern(reorderPatternsByDependency(optionalPattern.patterns))
case minusPattern: MinusPattern =>
val sortedMinusPatterns: Seq[QueryPattern] = optimiseQueryPatterns(minusPattern.patterns)
MinusPattern(patterns = sortedMinusPatterns)

// sort statements inside FilterNotExistsPattern
MinusPattern(reorderPatternsByDependency(minusPattern.patterns))
case filterNotExistsPattern: FilterNotExistsPattern =>
val sortedFilterNotExistsPatterns: Seq[QueryPattern] =
optimiseQueryPatterns(filterNotExistsPattern.patterns)
FilterNotExistsPattern(patterns = sortedFilterNotExistsPatterns)

// return any other query pattern as it is
FilterNotExistsPattern(reorderPatternsByDependency(filterNotExistsPattern.patterns))
case pattern: QueryPattern => pattern
}

Expand Down

0 comments on commit a91f6f6

Please sign in to comment.