blob: 8ae5509971ebcf266b722a2516d791834fcc720b [file] [log] [blame]
/**
* 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.sparql.core ;
import static org.apache.jena.atlas.iterator.Iter.take ;
import static org.apache.jena.sparql.core.GraphView.createDefaultGraph;
import static org.apache.jena.sparql.core.GraphView.createNamedGraph;
import java.util.Iterator ;
import java.util.List ;
import org.apache.jena.graph.Graph ;
import org.apache.jena.graph.Node ;
import org.apache.jena.graph.Triple ;
import org.apache.jena.sparql.SystemARQ ;
import org.apache.jena.util.iterator.ExtendedIterator ;
/** Connect a DatasetGraph to a DatasetChanges monitor.
* Any add or delete to the DatasetGraph is notified to the
* monitoring object with a {@link QuadAction} to indicate
* the change made.
*/
public class DatasetGraphMonitor extends DatasetGraphWrapper
{
/** Whether to see if a quad action will change the dataset - test before add for existence, test before delete for absence */
private boolean CheckFirst = true ;
/** Whether to record a no-op (maybe as a comment) */
private boolean RecordNoAction = true ;
/** Where to send the notifications */
private final DatasetChanges monitor ;
/**
* Create a DatasetGraph wrapper that monitors the dataset for changes (add or delete quads).
* Use this DatasetGraph for all operations in order to record changes.
* Note whether additions of deletions cause an actual change to the dataset or not.
* @param dsg The DatasetGraph to monitor
* @param monitor The handler for a change
*
* @see DatasetChanges
* @see QuadAction
*/
public DatasetGraphMonitor(DatasetGraph dsg, DatasetChanges monitor)
{
super(dsg) ;
this.monitor = monitor ;
}
/**
* Create a DatasetGraph wrapper that monitors the dataset for changes (add or delete quads).
* Use this DatasetGraph for all operations in order to record changes.
* @param dsg The DatasetGraph to monitor
* @param monitor The handler for a change
* @param recordOnlyIfRealChange
* If true, check to see if the change would have an effect (e.g. add is a new quad).
* If false, log changes as ADD/DELETE regardless of whether the dataset actually changes.
*
* @see DatasetChanges
* @see QuadAction
*/
public DatasetGraphMonitor(DatasetGraph dsg, DatasetChanges monitor, boolean recordOnlyIfRealChange)
{
super(dsg) ;
CheckFirst = recordOnlyIfRealChange ;
this.monitor = monitor ;
}
/** Return the monitor */
public DatasetChanges getMonitor() { return monitor ; }
/** Return the monitored DatasetGraph */
public DatasetGraph monitored() { return getWrapped() ; }
@Override public void add(Quad quad)
{
if ( CheckFirst && contains(quad) )
{
if ( RecordNoAction )
record(QuadAction.NO_ADD, quad.getGraph(), quad.getSubject(), quad.getPredicate(), quad.getObject()) ;
return ;
}
add$(quad) ;
}
@Override public void add(Node g, Node s, Node p, Node o)
{
if ( CheckFirst && contains(g,s,p,o) )
{
if ( RecordNoAction )
record(QuadAction.NO_ADD,g,s,p,o) ;
return ;
}
add$(g,s,p,o) ;
}
private void add$(Node g, Node s, Node p, Node o)
{
super.add(g,s,p,o) ;
record(QuadAction.ADD,g,s,p,o) ;
}
private void add$(Quad quad)
{
super.add(quad) ;
record(QuadAction.ADD, quad.getGraph(), quad.getSubject(), quad.getPredicate(), quad.getObject()) ;
}
@Override public void delete(Quad quad)
{
if ( CheckFirst && ! contains(quad) )
{
if ( RecordNoAction )
record(QuadAction.NO_DELETE, quad.getGraph(), quad.getSubject(), quad.getPredicate(), quad.getObject()) ;
return ;
}
delete$(quad) ;
}
@Override public void delete(Node g, Node s, Node p, Node o)
{
if ( CheckFirst && ! contains(g,s,p,o) )
{
if ( RecordNoAction )
record(QuadAction.NO_DELETE, g,s,p,o) ;
return ;
}
delete$(g,s,p,o) ;
}
private void delete$(Quad quad)
{
super.delete(quad) ;
record(QuadAction.DELETE, quad.getGraph(), quad.getSubject(), quad.getPredicate(), quad.getObject()) ;
}
private void delete$(Node g, Node s, Node p, Node o)
{
super.delete(g,s,p,o) ;
record(QuadAction.DELETE,g,s,p,o) ;
}
private static int SLICE = 1000 ;
@Override
public void deleteAny(Node g, Node s, Node p, Node o)
{
while (true)
{
Iterator<Quad> iter = find(g, s, p, o) ;
// Materialize - stops possible ConcurrentModificationExceptions
List<Quad> some = take(iter, SLICE) ;
for (Quad q : some)
delete$(q) ;
if (some.size() < SLICE) break ;
}
}
@Override public void addGraph(Node gn, Graph g)
{
// Convert to quads.
//super.addGraph(gn, g) ;
ExtendedIterator<Triple> iter = g.find(Node.ANY, Node.ANY, Node.ANY) ;
for ( ; iter.hasNext(); )
{
Triple t = iter.next() ;
add(gn, t.getSubject(), t.getPredicate(), t.getObject()) ;
}
}
@Override public void removeGraph(Node gn)
{
//super.removeGraph(gn) ;
deleteAny(gn, Node.ANY, Node.ANY, Node.ANY) ;
}
private void record(QuadAction action, Node g, Node s, Node p, Node o)
{
monitor.change(action, g, s, p, o) ;
}
@Override
public void sync() {
SystemARQ.syncObject(monitor) ;
super.sync() ;
}
@Override
public Graph getDefaultGraph() {
return createDefaultGraph(this);
}
@Override
public Graph getGraph(Node graphNode) {
return createNamedGraph(this, graphNode);
}
}