All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.topbraid.jenax.functions.CurrentThreadFunctionRegistry Maven / Gradle / Ivy

/*
 *  Licensed 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.
 *
 *  See the NOTICE file distributed with this work for additional
 *  information regarding copyright ownership.
 */
package org.topbraid.jenax.functions;

import java.util.Iterator;

import org.apache.jena.query.ARQ;
import org.apache.jena.rdf.model.Model;
import org.apache.jena.sparql.core.BasicPattern;
import org.apache.jena.sparql.engine.ExecutionContext;
import org.apache.jena.sparql.engine.QueryIterator;
import org.apache.jena.sparql.engine.iterator.QueryIterBlockTriples;
import org.apache.jena.sparql.engine.main.StageBuilder;
import org.apache.jena.sparql.engine.main.StageGenerator;
import org.apache.jena.sparql.function.FunctionFactory;
import org.apache.jena.sparql.function.FunctionRegistry;

/**
 * An ARQ FunctionRegistry that can be used to associate functions
 * with Threads, so that additional functions from a given Model can
 * be made visible depending on the SPARQL query thread.
 * 
 * 

Note that this concept only works if ARQ has been set to single * threading, which is done by the static block below.

* *

The contract of this class is very strict to prevent memory leaks: * Users always need to make sure that unregister is called as soon * as a query (block) ends, and to restore any old CurrentThreadFunctions * object that was registered before. So a typical block would be:

* * * Model model = ... a Model with extra SHACL/SPIN functions * CurrentThreadFunctions old = CurrentThreadFunctionRegistry.register(model); * try { * // run SPARQL queries here * } * finally { * CurrentThreadFunctionRegistry.unregister(old); * } * *

In preparation of the above, the application should start up with code * such as

* * * FunctionRegistry oldFR = FunctionRegistry.get(); * CurrentThreadFunctionRegistry threadFR = new CurrentThreadFunctionRegistry(oldFR); * FunctionRegistry.set(ARQ.getContext(), threadFR); * * *

and (for SPIN) do the same for the SPINThreadPropertyFunctionRegistry.

* * @author Holger Knublauch */ public class CurrentThreadFunctionRegistry extends FunctionRegistry { static { // Suppress multi-threading (PatternStage-stuff) StageBuilder.setGenerator(ARQ.getContext(), new StageGenerator() { @Override public QueryIterator execute(BasicPattern pattern, QueryIterator input, ExecutionContext execCxt) { return QueryIterBlockTriples.create(input, pattern, execCxt); } }); } private static ThreadLocal localFunctions = new ThreadLocal<>(); /** * Registers a set of extra SPIN functions from a given Model for the current * Thread. * @param model the Model containing the SPIN functions * @return any old object that was registered for the current Thread, so that * the old value can be restored when done. */ public static CurrentThreadFunctions register(Model model) { CurrentThreadFunctions old = localFunctions.get(); CurrentThreadFunctions neo = new CurrentThreadFunctions(model); localFunctions.set(neo); return old; } /** * Unregisters the current Model for the current Thread. * @param old the old functions that shall be restored or null */ public static void unregister(CurrentThreadFunctions old) { if(old != null) { localFunctions.set(old); } else { localFunctions.remove(); } } public static CurrentThreadFunctions getFunctions() { return localFunctions.get(); } private FunctionRegistry base; public CurrentThreadFunctionRegistry(FunctionRegistry base) { this.base = base; } @Override public FunctionFactory get(String uri) { FunctionFactory b = base.get(uri); if(b != null) { return b; } CurrentThreadFunctions functions = localFunctions.get(); if(functions != null) { FunctionFactory ff = functions.getFunctionFactory(uri); if(ff != null) { return ff; } } return null; } @Override public boolean isRegistered(String uri) { if(base.isRegistered(uri)) { return true; } else { return get(uri) != null; } } @Override public Iterator keys() { // Note: only returns base keys return base.keys(); } @Override public void put(String uri, Class funcClass) { base.put(uri, funcClass); } @Override public void put(String uri, FunctionFactory f) { base.put(uri, f); } @Override public FunctionFactory remove(String uri) { return base.remove(uri); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy