org.hibernate.hql.internal.QuerySplitter Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of hibernate-core Show documentation
Show all versions of hibernate-core Show documentation
JPMS Module-Info's for a few of the Jakarta Libraries just until they add them in themselves
/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
* See the lgpl.txt file in the root directory or .
*/
package org.hibernate.hql.internal;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.Locale;
import java.util.Set;
import org.hibernate.MappingException;
import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.hql.internal.classic.ParserHelper;
import org.hibernate.internal.CoreLogging;
import org.hibernate.internal.CoreMessageLogger;
import org.hibernate.internal.util.StringHelper;
/**
* Provides query splitting methods, which were originally in QueryTranslator.
*
* @author josh
*/
public final class QuerySplitter {
private static final CoreMessageLogger LOG = CoreLogging.messageLogger( QuerySplitter.class );
private static final Set BEFORE_CLASS_TOKENS = new HashSet();
private static final Set NOT_AFTER_CLASS_TOKENS = new HashSet();
static {
BEFORE_CLASS_TOKENS.add( "from" );
BEFORE_CLASS_TOKENS.add( "delete" );
BEFORE_CLASS_TOKENS.add( "update" );
//beforeClassTokens.add("new"); DEFINITELY DON'T HAVE THIS!!
BEFORE_CLASS_TOKENS.add( "," );
BEFORE_CLASS_TOKENS.add( "join" );
NOT_AFTER_CLASS_TOKENS.add( "in" );
//notAfterClassTokens.add(",");
NOT_AFTER_CLASS_TOKENS.add( "from" );
NOT_AFTER_CLASS_TOKENS.add( ")" );
}
/**
* Private empty constructor.
* (or else checkstyle says: 'warning: Utility classes should not have a public or default constructor.')
*/
private QuerySplitter() {
}
/**
* Handle Hibernate "implicit" polymorphism, by translating the query string into
* several "concrete" queries against mapped classes.
*/
public static String[] concreteQueries(String query, SessionFactoryImplementor factory) throws MappingException {
//scan the query string for class names appearing in the from clause and replace
//with all persistent implementors of the class/interface, returning multiple
//query strings (make sure we don't pick up a class in the select clause!)
//TODO: this is one of the ugliest and most fragile pieces of code in Hibernate....
String[] tokens = StringHelper.split( StringHelper.WHITESPACE + "(),", query, true );
if ( tokens.length == 0 ) {
// just especially for the trivial collection filter
return new String[] { query };
}
ArrayList placeholders = new ArrayList();
ArrayList replacements = new ArrayList();
StringBuilder templateQuery = new StringBuilder( 40 );
int start = getStartingPositionFor( tokens, templateQuery );
int count = 0;
String next;
String last = tokens[start - 1].toLowerCase(Locale.ROOT);
boolean inQuote = false;
for ( int i = start; i < tokens.length; i++ ) {
String token = tokens[i];
if ( ParserHelper.isWhitespace( token ) ) {
templateQuery.append( token );
continue;
}
else if ( isQuoteCharacter( token) ) {
inQuote = !inQuote;
templateQuery.append( token );
continue;
}
else if ( isTokenStartWithAQuoteCharacter( token ) ) {
if ( !isTokenEndWithAQuoteCharacter( token ) ) {
inQuote = true;
}
templateQuery.append( token );
continue;
}
else if ( isTokenEndWithAQuoteCharacter( token ) ) {
inQuote = false;
templateQuery.append( token );
continue;
}
else if ( inQuote ) {
templateQuery.append( token );
continue;
}
next = nextNonWhite( tokens, i ).toLowerCase(Locale.ROOT);
boolean process = isJavaIdentifier( token )
&& isPossiblyClassName( last, next );
last = token.toLowerCase(Locale.ROOT);
if ( process ) {
String importedClassName = getImportedClass( token, factory );
if ( importedClassName != null ) {
String[] implementors = factory.getImplementors( importedClassName );
token = "$clazz" + count++ + "$";
if ( implementors != null ) {
placeholders.add( token );
replacements.add( implementors );
}
}
}
templateQuery.append( token );
}
String[] results = StringHelper.multiply(
templateQuery.toString(),
placeholders.iterator(),
replacements.iterator()
);
if ( results.length == 0 ) {
LOG.noPersistentClassesFound( query );
}
return results;
}
private static boolean isQuoteCharacter(String token) {
return "'".equals( token ) || "\"".equals( token );
}
private static boolean isTokenStartWithAQuoteCharacter(String token) {
return token.startsWith( "'" ) || token.startsWith( "\"" );
}
private static boolean isTokenEndWithAQuoteCharacter(String token) {
return token.endsWith( "'" ) || token.endsWith( "\"" );
}
private static String nextNonWhite(String[] tokens, int start) {
for ( int i = start + 1; i < tokens.length; i++ ) {
if ( !ParserHelper.isWhitespace( tokens[i] ) ) {
return tokens[i];
}
}
return tokens[tokens.length - 1];
}
private static int getStartingPositionFor(String[] tokens, StringBuilder templateQuery) {
templateQuery.append( tokens[0] );
if ( !"select".equals( tokens[0].toLowerCase(Locale.ROOT) ) ) {
return 1;
}
// select-range is terminated by declaration of "from"
for ( int i = 1; i < tokens.length; i++ ) {
if ( "from".equals( tokens[i].toLowerCase(Locale.ROOT) ) ) {
return i;
}
templateQuery.append( tokens[i] );
}
return tokens.length;
}
private static boolean isPossiblyClassName(String last, String next) {
return "class".equals( last )
|| ( BEFORE_CLASS_TOKENS.contains( last ) && !NOT_AFTER_CLASS_TOKENS.contains( next ) );
}
private static boolean isJavaIdentifier(String token) {
return Character.isJavaIdentifierStart( token.charAt( 0 ) );
}
public static String getImportedClass(String name, SessionFactoryImplementor factory) {
return factory.getMetamodel().getImportedClassName( name );
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy