org.hibernate.hql.internal.classic.ClauseParser Maven / Gradle / Ivy
/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* Copyright (c) 2008, Red Hat Middleware LLC or third-party contributors as
* indicated by the @author tags or express copyright attribution
* statements applied by the authors. All third-party contributions are
* distributed under license by Red Hat Middleware LLC.
*
* This copyrighted material is made available to anyone wishing to use, modify,
* copy, or redistribute it subject to the terms and conditions of the GNU
* Lesser General Public License, as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
* for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this distribution; if not, write to:
* Free Software Foundation, Inc.
* 51 Franklin Street, Fifth Floor
* Boston, MA 02110-1301 USA
*
*/
package org.hibernate.hql.internal.classic;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import org.hibernate.QueryException;
/**
* Parses the Hibernate query into its constituent clauses.
*/
public class ClauseParser implements Parser {
private Parser child;
private List selectTokens;
private boolean cacheSelectTokens = false;
private boolean byExpected = false;
private int parenCount = 0;
public void token(String token, QueryTranslatorImpl q) throws QueryException {
String lcToken = token.toLowerCase();
if ( "(".equals( token ) ) {
parenCount++;
}
else if ( ")".equals( token ) ) {
parenCount--;
}
if ( byExpected && !lcToken.equals( "by" ) ) {
throw new QueryException( "BY expected after GROUP or ORDER: " + token );
}
boolean isClauseStart = parenCount == 0; //ignore subselect keywords
if ( isClauseStart ) {
if ( lcToken.equals( "select" ) ) {
selectTokens = new ArrayList();
cacheSelectTokens = true;
}
else if ( lcToken.equals( "from" ) ) {
child = new FromParser();
child.start( q );
cacheSelectTokens = false;
}
else if ( lcToken.equals( "where" ) ) {
endChild( q );
child = new WhereParser();
child.start( q );
}
else if ( lcToken.equals( "order" ) ) {
endChild( q );
child = new OrderByParser();
byExpected = true;
}
else if ( lcToken.equals( "having" ) ) {
endChild( q );
child = new HavingParser();
child.start( q );
}
else if ( lcToken.equals( "group" ) ) {
endChild( q );
child = new GroupByParser();
byExpected = true;
}
else if ( lcToken.equals( "by" ) ) {
if ( !byExpected ) throw new QueryException( "GROUP or ORDER expected before BY" );
child.start( q );
byExpected = false;
}
else {
isClauseStart = false;
}
}
if ( !isClauseStart ) {
if ( cacheSelectTokens ) {
selectTokens.add( token );
}
else {
if ( child == null ) {
throw new QueryException( "query must begin with SELECT or FROM: " + token );
}
else {
child.token( token, q );
}
}
}
}
private void endChild(QueryTranslatorImpl q) throws QueryException {
if ( child == null ) {
//null child could occur for no from clause in a filter
cacheSelectTokens = false;
}
else {
child.end( q );
}
}
public void start(QueryTranslatorImpl q) {
}
public void end(QueryTranslatorImpl q) throws QueryException {
endChild( q );
if ( selectTokens != null ) {
child = new SelectParser();
child.start( q );
Iterator iter = selectTokens.iterator();
while ( iter.hasNext() ) {
token( ( String ) iter.next(), q );
}
child.end( q );
}
byExpected = false;
parenCount = 0;
cacheSelectTokens = false;
}
}