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

org.topbraid.spin.arq.SPINFunctionsCache Maven / Gradle / Ivy

There is a newer version: 1.4.3
Show newest version
/*
 *  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.spin.arq;

import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Map.Entry;

import org.apache.jena.graph.Node;
import org.apache.jena.query.Dataset;
import org.apache.jena.query.QuerySolution;
import org.apache.jena.rdf.model.Model;
import org.apache.jena.sparql.expr.ExprEvalException;
import org.apache.jena.sparql.expr.NodeValue;

/**
 * A cache that remembers previous calls to SPIN functions marked with spin:cachable.
 * 
 * @author Holger Knublauch
 */
public class SPINFunctionsCache {

	private static SPINFunctionsCache singleton = new SPINFunctionsCache();
	
	public static SPINFunctionsCache get() {
		return singleton;
	}
	
	public static void set(SPINFunctionsCache value) {
		SPINFunctionsCache.singleton = value;
	}
	
	
	private static final int capacity = 10000;
	
	private static class MyCache extends LinkedHashMap {

		MyCache() {
			super(capacity + 1, 1.1f, true);
		}

		@Override
		protected boolean removeEldestEntry(Entry eldest) {
			if(size() > capacity) {
				return true;
			}
			else {
				return false;
			}
		}
	};
	
	private Map cache = Collections.synchronizedMap(new MyCache());

	
	public void clear() {
		cache.clear();
	}
	
	
	public NodeValue execute(SPINARQFunction function, Dataset dataset, Model defaultModel, QuerySolution bindings, Node[] args) {
		Key key = new Key(function.getSPINFunction().getURI(), args);
		Result result = cache.get(key);
		if(result == null) {
			result = new Result();
			try {
				result.nodeValue = function.executeBody(dataset, defaultModel, bindings);
			}
			catch(ExprEvalException ex) {
				result.ex = ex;
			}
			cache.put(key, result);
		}
		if(result.ex != null) {
			throw new ExprEvalException(result.ex.getMessage());
		}
		else {
			return result.nodeValue;
		}
	}
	
	
	private static class Key {
		
		private int hashCode;
		
		private Node[] args;
		
		private String functionURI;
		
		
		Key(String functionURI, Node[] args) {
			this.args = args;
			this.functionURI = functionURI;
			hashCode = functionURI.hashCode();
			for(Node arg : args) {
				if(arg != null) {
					hashCode += arg.hashCode();
				}
			}
		}
		
		
		private boolean argEquals(Node arg1, Node arg2) {
			if(arg1 == null) {
				return arg2 == null;
			}
			else if(arg2 == null) {
				return false;
			}
			else {
				return arg1.equals(arg2);
			}
		}
		
		
		@Override
		public boolean equals(Object obj) {
			
			if(!(obj instanceof Key)) {
				return false;
			}
			
			Key other = (Key) obj;
			if(!functionURI.equals(other.functionURI)) {
				return false;
			}
			
			if(args.length != other.args.length) {
				return false;
			}
			
			for(int i = 0; i < args.length; i++) {
				if(!argEquals(args[i], other.args[i])) {
					return false;
				}
			}
			
			return true;
		}
		
		
		@Override
		public int hashCode() {
			return hashCode;
		}
		
		
		@Override
        public String toString() {
			String str = "<" + functionURI + ">(";
			for(int i = 0; i < args.length; i++) {
				if(i > 0) {
					str += ", ";
				}
				str += args[i];
			}
			return str + ")";
		}
	}
	
	
	private static class Result {
		
		ExprEvalException ex;
		
		NodeValue nodeValue;
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy