/** * Redistribution and use of this software and associated documentation * ("Software"), with or without modification, are permitted provided * that the following conditions are met: * * 1. Redistributions of source code must retain copyright * statements and notices. Redistributions must also contain a * copy of this document. * * 2. Redistributions in binary form must reproduce the * above copyright notice, this list of conditions and the * following disclaimer in the documentation and/or other * materials provided with the distribution. * * 3. The name "Exolab" must not be used to endorse or promote * products derived from this Software without prior written * permission of Intalio, Inc. For written permission, * please contact info@exolab.org. * * 4. Products derived from this Software may not be called "Exolab" * nor may "Exolab" appear in their names without prior written * permission of Intalio, Inc. Exolab is a registered * trademark of Intalio, Inc. * * 5. Due credit should be given to the Exolab Project * (http://www.exolab.org/). * * THIS SOFTWARE IS PROVIDED BY INTALIO, INC. AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT * NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL * INTALIO, INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED * OF THE POSSIBILITY OF SUCH DAMAGE. * * Copyright 2000-2001 (C) Intalio, Inc. All Rights Reserved. * * $Id$ */ package org.exolab.castor.persist; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import org.castor.jdo.util.JDOUtils; import org.exolab.castor.jdo.PersistenceException; import org.exolab.castor.mapping.TypeConvertor; import org.exolab.castor.persist.spi.Identity; import org.exolab.castor.persist.spi.PersistenceFactory; /** * SQLRelationLoader is a quick hack for creating and removing * relation from a many-to-many relation database from ClassMolder. * Eventually, it will be merged into SQLEngine. But, it requires * chaning of the Persistence interface. * * @author <a href="mailto:yip@intalio.com">Thomas Yip</a> */ public class SQLRelationLoader { private String _tableName; private int[] _leftType; private TypeConvertor[] _leftFrom; private TypeConvertor[] _rightFrom; private int[] _rightType; private String[] _left; private String[] _right; /** The SQL statement for selecting the relation from the relation table. */ private String _select; /** The SQL statement to insert the a new relation into the relation table. */ private String _insert; /** The SQL statement to delete an relation from the relation table. */ private String _delete; /** The SQL statement to delete all the relation assoicate with the left side type. */ private String _deleteAll; public SQLRelationLoader(final String table, final String[] key, final int[] keyType, final TypeConvertor[] idTo, final TypeConvertor[] idFrom, final String[] otherKey, final int[] otherKeyType, final TypeConvertor[] ridTo, final TypeConvertor[] ridFrom, final PersistenceFactory factory) { _leftFrom = idFrom; _rightFrom = ridFrom; _tableName = table; _left = key; _right = otherKey; _leftType = keyType; _rightType = otherKeyType; // construct select statement StringBuffer sb = new StringBuffer(); int count = 0; sb.append("SELECT "); for (int i = 0; i < _left.length; i++) { if (i > 0) { sb.append(","); } sb.append(factory.quoteName(_left[i])); count++; } for (int i = 0; i < _right.length; i++) { sb.append(","); sb.append(factory.quoteName(_right[i])); count++; } sb.append(" FROM "); sb.append(factory.quoteName(_tableName)); sb.append(" WHERE "); for (int i = 0; i < _left.length; i++) { if (i > 0) { sb.append(" AND "); } sb.append(factory.quoteName(_left[i])); sb.append("=?"); } for (int i = 0; i < _right.length; i++) { sb.append(" AND "); sb.append(factory.quoteName(_right[i])); sb.append("=?"); } _select = sb.toString(); // construct insert statement sb = new StringBuffer(); count = 0; sb.append("INSERT INTO "); sb.append(factory.quoteName(_tableName)); sb.append(" ("); for (int i = 0; i < _left.length; i++) { if (i > 0) { sb.append(","); } sb.append(factory.quoteName(_left[i])); count++; } for (int i = 0; i < _right.length; i++) { sb.append(","); sb.append(factory.quoteName(_right[i])); count++; } sb.append(") VALUES ("); for (int i = 0; i < count; i++) { if (i > 0) { sb.append(","); } sb.append("?"); } sb.append(")"); _insert = sb.toString(); // construct delete statement sb = new StringBuffer(); count = 0; sb.append("DELETE FROM "); sb.append(factory.quoteName(_tableName)); sb.append(" WHERE "); for (int i = 0; i < _left.length; i++) { if (i > 0) { sb.append(" AND "); } sb.append(factory.quoteName(_left[i])); sb.append("=?"); } for (int i = 0; i < _right.length; i++) { sb.append(" AND "); sb.append(factory.quoteName(_right[i])); sb.append("=?"); } _delete = sb.toString(); // construct delete statement for the left side only sb = new StringBuffer(); count = 0; sb.append("DELETE FROM "); sb.append(factory.quoteName(_tableName)); sb.append(" WHERE "); for (int i = 0; i < _left.length; i++) { if (i > 0) { sb.append(" AND "); } sb.append(factory.quoteName(_left[i])); sb.append("=?"); } _deleteAll = sb.toString(); } private Object idToSQL(final int index, final Object object) { if ((object == null) || (_leftFrom[index] == null)) { return object; } return _leftFrom[index].convert(object); } private Object ridToSQL(final int index, final Object object) { if ((object == null) || (_rightFrom[index] == null)) { return object; } return _rightFrom[index].convert(object); } public void createRelation(final Connection conn, final Identity left, final Identity right) throws PersistenceException { ResultSet rset = null; PreparedStatement selectStatement = null; PreparedStatement insertStatement = null; try { int count = 1; selectStatement = conn.prepareStatement(_select); for (int i = 0; i < left.size(); i++) { selectStatement.setObject(count, idToSQL(i, left.get(i)), _leftType[i]); count++; } for (int i = 0; i < right.size(); i++) { selectStatement.setObject(count, ridToSQL(i, right.get(i)), _rightType[i]); count++; } rset = selectStatement.executeQuery(); count = 1; insertStatement = conn.prepareStatement(_insert); if (!rset.next()) { for (int i = 0; i < left.size(); i++) { insertStatement.setObject(count, idToSQL(i, left.get(i)), _leftType[i]); count++; } for (int i = 0; i < right.size(); i++) { insertStatement.setObject(count, ridToSQL(i, right.get(i)), _rightType[i]); count++; } insertStatement.executeUpdate(); } } catch (SQLException e) { e.printStackTrace(); throw new PersistenceException(e.toString()); } finally { JDOUtils.closeResultSet(rset); JDOUtils.closeStatement(selectStatement); JDOUtils.closeStatement(insertStatement); } } public void deleteRelation(final Connection conn, final Identity left) throws PersistenceException { PreparedStatement stmt = null; try { int count = 1; stmt = conn.prepareStatement(_deleteAll); for (int i = 0; i < left.size(); i++) { stmt.setObject(count, idToSQL(i, left.get(i)), _leftType[i]); count++; } stmt.executeUpdate(); } catch (SQLException e) { e.printStackTrace(); throw new PersistenceException(e.toString()); } finally { JDOUtils.closeStatement(stmt); } } public void deleteRelation(final Connection conn, final Identity left, final Identity right) throws PersistenceException { PreparedStatement stmt = null; try { int count = 1; stmt = conn.prepareStatement(_delete); for (int i = 0; i < left.size(); i++) { stmt.setObject(count, idToSQL(i, left.get(i)), _leftType[i]); count++; } for (int i = 0; i < right.size(); i++) { stmt.setObject(count, ridToSQL(i, right.get(i)), _rightType[i]); count++; } stmt.executeUpdate(); } catch (SQLException e) { e.printStackTrace(); throw new PersistenceException(e.toString()); } finally { JDOUtils.closeStatement(stmt); } } }