/*
 * 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.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
/** * @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 ) { 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; } }