/* * 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 <http://www.gnu.org/licenses/lgpl-2.1.html>. */ package org.hibernate.sql; import java.util.Iterator; import java.util.Map; import org.hibernate.LockMode; import org.hibernate.LockOptions; import org.hibernate.QueryException; import org.hibernate.dialect.Dialect; import org.hibernate.internal.util.StringHelper; /** * @author Gavin King */ public class ForUpdateFragment { private final StringBuilder aliases = new StringBuilder(); private boolean isNowaitEnabled; private boolean isSkipLockedEnabled; private final Dialect dialect; private LockMode lockMode; private LockOptions lockOptions; public ForUpdateFragment(Dialect dialect) { this.dialect = dialect; } public ForUpdateFragment(Dialect dialect, LockOptions lockOptions, Map<String, String[]> keyColumnNames) throws QueryException { this( dialect ); LockMode upgradeType = null; Iterator iter = lockOptions.getAliasLockIterator(); this.lockOptions = lockOptions; if ( !iter.hasNext()) { // no tables referenced final LockMode lockMode = lockOptions.getLockMode(); if ( LockMode.READ.lessThan( lockMode ) ) { upgradeType = lockMode; this.lockMode = lockMode; } } while ( iter.hasNext() ) { final Map.Entry me = ( Map.Entry ) iter.next(); final LockMode lockMode = ( LockMode ) me.getValue(); if ( LockMode.READ.lessThan( lockMode ) ) { final String tableAlias = ( String ) me.getKey(); if ( dialect.forUpdateOfColumns() ) { String[] keyColumns = keyColumnNames.get( tableAlias ); //use the id column alias if ( keyColumns == null ) { throw new IllegalArgumentException( "alias not found: " + tableAlias ); } keyColumns = StringHelper.qualify( tableAlias, keyColumns ); for ( String keyColumn : keyColumns ) { addTableAlias( keyColumn ); } } else { addTableAlias( tableAlias ); } if ( upgradeType != null && lockMode != upgradeType ) { throw new QueryException( "mixed LockModes" ); } upgradeType = lockMode; } } if ( upgradeType == LockMode.UPGRADE_NOWAIT ) { setNowaitEnabled( true ); } if ( upgradeType == LockMode.UPGRADE_SKIPLOCKED ) { setSkipLockedEnabled( true ); } } public ForUpdateFragment addTableAlias(String alias) { if ( aliases.length() > 0 ) { aliases.append( ", " ); } aliases.append( alias ); return this; } public String toFragmentString() { if ( lockOptions!= null ) { if ( aliases.length() == 0) { return dialect.getForUpdateString( lockOptions ); } else { return dialect.getForUpdateString( aliases.toString(), lockOptions ); } } else if ( aliases.length() == 0) { if ( lockMode != null ) { return dialect.getForUpdateString( lockMode ); } return ""; } // TODO: pass lockmode if(isNowaitEnabled) { return dialect.getForUpdateNowaitString( aliases.toString() ); } else if (isSkipLockedEnabled) { return dialect.getForUpdateSkipLockedString( aliases.toString() ); } else { return dialect.getForUpdateString( aliases.toString() ); } } public ForUpdateFragment setNowaitEnabled(boolean nowait) { isNowaitEnabled = nowait; return this; } public ForUpdateFragment setSkipLockedEnabled(boolean skipLocked) { isSkipLockedEnabled = skipLocked; return this; } }