/*
* 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.expr;
import java.nio.charset.StandardCharsets ;
import java.security.MessageDigest ;
import java.security.NoSuchAlgorithmException ;
import org.apache.jena.atlas.lib.Bytes ;
import org.apache.jena.datatypes.xsd.XSDDatatype ;
import org.apache.jena.graph.Node ;
import org.apache.jena.sparql.ARQInternalErrorException ;
public abstract class ExprDigest extends ExprFunction1
{
private final String digestName ;
private MessageDigest digestCache ;
public ExprDigest(Expr expr, String symbol, String digestName)
{
super(expr, symbol) ;
this.digestName = digestName ;
try
{
digestCache = MessageDigest.getInstance(digestName) ;
} catch (NoSuchAlgorithmException e)
{
throw new ARQInternalErrorException("Digest not provided in this Java system: "+digestName) ;
}
}
// Each digest expr function is unique
// Pool needs to be static and per-digest-type
// private Pool<MessageDigest> pool = PoolSync.create(new PoolBase<MessageDigest>()) ;
// private MessageDigest getDigest()
// {
// MessageDigest md = pool.get() ;
// if ( md == null )
// {
// synchronized (pool)
// {
// md = pool.get() ;
// if ( md == null )
// {
// md = createDigest() ;
// pool.put(md) ;
// }
// }
// }
// return md ;
// }
private MessageDigest getDigest()
{
if ( digestCache != null )
{
MessageDigest digest2 = null ;
try {
digest2 = (MessageDigest)digestCache.clone() ;
return digest2 ;
} catch (CloneNotSupportedException ex)
{
// Can't clone - remove cache copy.
digestCache = null ;
}
}
return createDigest() ;
}
private MessageDigest createDigest()
{
try { return MessageDigest.getInstance(digestName) ; }
catch (Exception ex2) { throw new ARQInternalErrorException(ex2) ; }
}
NodeValue lastSeen = null ;
NodeValue lastCalc = null ;
@Override
public NodeValue eval(NodeValue v)
{
if ( lastSeen != null && lastSeen.equals(v) )
return lastCalc ;
Node n = v.asNode() ;
if ( ! n.isLiteral() )
throw new ExprEvalException("Not a literal: "+v) ;
if ( n.getLiteralLanguage() != null && ! n.getLiteralLanguage().equals("") )
throw new ExprEvalException("Can't make a digest of an RDF term with a language tag") ;
// Literal, no language tag.
if ( n.getLiteralDatatype() != null && ! XSDDatatype.XSDstring.equals(n.getLiteralDatatype()) )
throw new ExprEvalException("Not a simple literal nor an XSD string") ;
try {
MessageDigest digest = getDigest() ;
String x = n.getLiteralLexicalForm() ;
byte b[] = x.getBytes(StandardCharsets.UTF_8) ;
byte d[] = digest.digest(b) ;
String y = Bytes.asHexLC(d) ;
NodeValue result = NodeValue.makeString(y) ;
// Cache
lastSeen = v ;
lastCalc = result ;
return result ;
} catch (Exception ex2) { throw new ARQInternalErrorException(ex2) ; }
}
}