org.apache.jena.propertytable.graph.QueryIterPropertyTableRow Maven / Gradle / Ivy
/*
* 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.
*/
package org.apache.jena.propertytable.graph;
import java.util.ArrayList;
import java.util.List;
import org.apache.jena.graph.Node ;
import org.apache.jena.graph.Triple ;
import org.apache.jena.propertytable.PropertyTable;
import org.apache.jena.propertytable.Row;
import org.apache.jena.sparql.ARQInternalErrorException ;
import org.apache.jena.sparql.core.BasicPattern ;
import org.apache.jena.sparql.core.Var ;
import org.apache.jena.sparql.engine.ExecutionContext ;
import org.apache.jena.sparql.engine.QueryIterator ;
import org.apache.jena.sparql.engine.binding.Binding ;
import org.apache.jena.sparql.engine.binding.BindingFactory ;
import org.apache.jena.sparql.engine.binding.BindingMap ;
import org.apache.jena.sparql.engine.iterator.QueryIter ;
import org.apache.jena.sparql.engine.iterator.QueryIterRepeatApply ;
import org.apache.jena.util.iterator.ClosableIterator ;
import org.apache.jena.util.iterator.ExtendedIterator ;
import org.apache.jena.util.iterator.NiceIterator ;
import org.apache.jena.util.iterator.WrappedIterator ;
/**
* If the triple size within a BasicePattern is greater than 1 (i.e. at least 2 triples), it's turned into a row querying.
*
*/
public class QueryIterPropertyTableRow extends QueryIterRepeatApply{
private final BasicPattern pattern ;
public QueryIterPropertyTableRow( QueryIterator input,
BasicPattern pattern ,
ExecutionContext cxt)
{
super(input, cxt) ;
this.pattern = pattern ;
}
@Override
protected QueryIterator nextStage(Binding binding)
{
return new RowMapper(binding, pattern, getExecContext()) ;
}
static int countMapper = 0 ;
static class RowMapper extends QueryIter
{
private PropertyTable table;
private BasicPattern pattern;
private Binding binding ;
private ClosableIterator graphIter ;
private Binding slot = null ;
private boolean finished = false ;
private volatile boolean cancelled = false ;
RowMapper(Binding binding, BasicPattern pattern, ExecutionContext cxt)
{
super(cxt) ;
GraphPropertyTable graph = (GraphPropertyTable)cxt.getActiveGraph() ;
this.pattern = substitute(pattern, binding);
this.binding = binding ;
BasicPattern pattern2 = tripleNode(pattern);
ExtendedIterator iter = graph.propertyTableBaseFind( new RowMatch( pattern2) );
if ( false )
{
// Materialize the results now. Debugging only.
List x = iter.toList() ;
this.graphIter = WrappedIterator.create(x.iterator()) ;
iter.close();
}
else
// Stream.
this.graphIter = iter ;
}
private static Node tripleNode(Node node)
{
if ( node.isVariable() )
return Node.ANY ;
return node ;
}
private static BasicPattern tripleNode(BasicPattern pattern)
{
List triples = new ArrayList();
for (Triple triple: pattern){
triples.add( tripleNode(triple) );
}
return BasicPattern.wrap(triples);
}
private static Triple tripleNode(Triple triple){
Node s = tripleNode(triple.getSubject()) ;
Node p = tripleNode(triple.getPredicate()) ;
Node o = tripleNode(triple.getObject()) ;
return Triple.create(s, p, o);
}
private static Node substitute(Node node, Binding binding)
{
if ( Var.isVar(node) )
{
Node x = binding.get(Var.alloc(node)) ;
if ( x != null )
return x ;
}
return node ;
}
private static Triple substitute(Triple triple, Binding binding){
Node s = substitute(triple.getSubject(), binding) ;
Node p = substitute(triple.getPredicate(), binding) ;
Node o = substitute(triple.getObject(), binding) ;
return Triple.create(s, p, o);
}
private static BasicPattern substitute(BasicPattern pattern , Binding binding)
{
List triples = new ArrayList();
for (Triple triple: pattern){
triples.add( substitute(triple,binding) );
}
return BasicPattern.wrap(triples);
}
private Binding mapper(Row r)
{
BindingMap results = BindingFactory.create(binding) ;
if ( ! insert(pattern, r, results) )
return null ;
return results ;
}
private static boolean insert(BasicPattern input, Row output, BindingMap results)
{
for (Triple triple: input){
if (! insert(triple, output, results) ){
return false;
}
}
return true;
}
private static boolean insert(Triple input, Row output, BindingMap results){
if ( ! insert(input.getSubject(), output.getRowKey(), results) )
return false ;
// if ( ! insert(input.getPredicate(), output.get, results) )
// return false ;
if ( ! insert(input.getObject(), output.getValue( input.getPredicate() ), results) )
return false ;
return true;
}
private static boolean insert(Node inputNode, Node outputNode, BindingMap results)
{
if ( ! Var.isVar(inputNode) )
return true ;
Var v = Var.alloc(inputNode) ;
Node x = results.get(v) ;
if ( x != null )
return outputNode.equals(x) ;
results.add(v, outputNode) ;
return true ;
}
@Override
protected boolean hasNextBinding()
{
if ( finished ) return false ;
if ( slot != null ) return true ;
if ( cancelled )
{
graphIter.close() ;
finished = true ;
return false ;
}
while(graphIter.hasNext() && slot == null )
{
Row r = graphIter.next() ;
slot = mapper(r) ;
}
if ( slot == null )
finished = true ;
return slot != null ;
}
@Override
protected Binding moveToNextBinding()
{
if ( ! hasNextBinding() )
throw new ARQInternalErrorException() ;
Binding r = slot ;
slot = null ;
return r ;
}
@Override
protected void closeIterator()
{
if ( graphIter != null )
NiceIterator.close(graphIter) ;
graphIter = null ;
}
@Override
protected void requestCancel()
{
// The QueryIteratorBase machinary will do the real work.
cancelled = true ;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy