/* * Copyright (C) 2009 eXo Platform SAS. * * This is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This software 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 software; if not, write to the Free * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA, or see the FSF site: http://www.fsf.org. */ package org.exoplatform.services.jcr.impl.storage.jdbc.db; import org.exoplatform.services.jcr.datamodel.NodeData; import org.exoplatform.services.jcr.datamodel.PropertyData; import org.exoplatform.services.jcr.datamodel.ValueData; import org.exoplatform.services.jcr.impl.Constants; import org.exoplatform.services.jcr.impl.dataflow.ValueDataUtil; import org.exoplatform.services.jcr.impl.storage.jdbc.JDBCDataContainerConfig; import org.exoplatform.services.jcr.impl.storage.jdbc.JDBCStorageConnection; import java.io.IOException; import java.io.InputStream; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Types; import java.util.List; import javax.jcr.RepositoryException; /** * Single database connection implementation. * * Created by The eXo Platform SAS 27.04.2006 * * @author <a href="mailto:peter.nedonosko@exoplatform.com.ua">Peter * Nedonosko</a> * @version $Id: SingleDbJDBCConnection.java 20950 2008-10-06 14:23:07Z * pnedonosko $ */ public class SingleDbJDBCConnection extends JDBCStorageConnection { /** * Singledatabase JDBC Connection constructor. * * @param dbConnection * JDBC connection, should be opened before * @param readOnly * boolean if true the dbConnection was marked as READ-ONLY. * @param containerConfig * Workspace Storage Container configuration */ public SingleDbJDBCConnection(Connection dbConnection, boolean readOnly, JDBCDataContainerConfig containerConfig) throws SQLException { super(dbConnection, readOnly, containerConfig); } /** * {@inheritDoc} */ @Override protected String getInternalId(final String identifier) { return this.containerConfig.containerName + identifier; } /** * {@inheritDoc} */ @Override protected String getIdentifier(final String internalId) { if (internalId == null) { return null; } return internalId.substring(this.containerConfig.containerName.length()); } /** * {@inheritDoc} */ @Override protected void prepareQueries() throws SQLException { FIND_ITEM_BY_ID = "select * from JCR_SITEM where ID=?"; FIND_ITEM_BY_NAME = "select * from JCR_SITEM" + " where CONTAINER_NAME=? and PARENT_ID=? and NAME=? and I_INDEX=? order by I_CLASS, VERSION DESC"; FIND_PROPERTY_BY_NAME = "select V.DATA" + " from JCR_SITEM I, JCR_SVALUE V" + " where I.I_CLASS=2 and I.CONTAINER_NAME=? and I.PARENT_ID=? and I.NAME=? and" + " I.ID=V.PROPERTY_ID order by V.ORDER_NUM"; FIND_REFERENCES = "select P.ID, P.PARENT_ID, P.VERSION, P.P_TYPE, P.P_MULTIVALUED, P.NAME" + " from JCR_SREF R, JCR_SITEM P" + " where R.NODE_ID=? and P.CONTAINER_NAME=? and P.ID=R.PROPERTY_ID and P.I_CLASS=2"; FIND_VALUES_BY_PROPERTYID = "select PROPERTY_ID, ORDER_NUM, DATA, STORAGE_DESC from JCR_SVALUE where PROPERTY_ID=? order by ORDER_NUM"; FIND_NODES_BY_PARENTID = "select * from JCR_SITEM" + " where I_CLASS=1 and CONTAINER_NAME=? and PARENT_ID=?" + " order by N_ORDER_NUM"; FIND_LAST_ORDER_NUMBER_BY_PARENTID = "select count(*), max(N_ORDER_NUM) from JCR_SITEM where I_CLASS=1 and CONTAINER_NAME=? and PARENT_ID=?"; FIND_NODES_COUNT_BY_PARENTID = "select count(ID) from JCR_SITEM" + " where I_CLASS=1 and CONTAINER_NAME=? and PARENT_ID=?"; FIND_PROPERTIES_BY_PARENTID = "select * from JCR_SITEM" + " where I_CLASS=2 and CONTAINER_NAME=? and PARENT_ID=?" + " order by ID"; FIND_MAX_PROPERTY_VERSIONS = "select max(VERSION) FROM JCR_SITEM WHERE PARENT_ID=? and CONTAINER_NAME=? and NAME=? and I_INDEX=? and I_CLASS=2"; INSERT_NODE = "insert into JCR_SITEM(ID, PARENT_ID, NAME, CONTAINER_NAME, VERSION, I_CLASS, I_INDEX, N_ORDER_NUM) VALUES(?,?,?,?,?," + I_CLASS_NODE + ",?,?)"; INSERT_PROPERTY = "insert into JCR_SITEM(ID, PARENT_ID, NAME, CONTAINER_NAME, VERSION, I_CLASS, I_INDEX, P_TYPE, P_MULTIVALUED) " + "VALUES(?,?,?,?,?," + I_CLASS_PROPERTY + ",?,?,?)"; INSERT_VALUE = "insert into JCR_SVALUE(DATA, ORDER_NUM, PROPERTY_ID, STORAGE_DESC) VALUES(?,?,?,?)"; INSERT_REF = "insert into JCR_SREF(NODE_ID, PROPERTY_ID, ORDER_NUM) VALUES(?,?,?)"; RENAME_NODE = "update JCR_SITEM set PARENT_ID=?, NAME=?, VERSION=?, I_INDEX=?, N_ORDER_NUM=? where ID=?"; UPDATE_NODE = "update JCR_SITEM set VERSION=?, I_INDEX=?, N_ORDER_NUM=? where ID=?"; UPDATE_PROPERTY = "update JCR_SITEM set VERSION=?, P_TYPE=? where ID=?"; DELETE_ITEM = "delete from JCR_SITEM where ID=?"; DELETE_VALUE = "delete from JCR_SVALUE where PROPERTY_ID=?"; DELETE_REF = "delete from JCR_SREF where PROPERTY_ID=?"; FIND_NODES_COUNT = "select count(*) from JCR_SITEM I where I.I_CLASS=1 and I.CONTAINER_NAME=?"; FIND_WORKSPACE_DATA_SIZE = "select sum(length(DATA)) from JCR_SITEM I, JCR_SVALUE V where I.I_CLASS=2 and I.CONTAINER_NAME=?" + " and I.ID=V.PROPERTY_ID"; FIND_WORKSPACE_PROPERTIES_ON_VALUE_STORAGE = "select V.PROPERTY_ID, V.STORAGE_DESC, V.ORDER_NUM from JCR_SITEM I, JCR_SVALUE V" + " where I.I_CLASS=2 and I.CONTAINER_NAME=? and I.ID=V.PROPERTY_ID and V.STORAGE_DESC is not null"; FIND_NODE_DATA_SIZE = "select sum(length(DATA)) from JCR_SITEM I, JCR_SVALUE V where I.PARENT_ID=? and I.I_CLASS=2" + " and I.CONTAINER_NAME=? and I.ID=V.PROPERTY_ID"; FIND_NODE_PROPERTIES_ON_VALUE_STORAGE = "select V.PROPERTY_ID, V.STORAGE_DESC, V.ORDER_NUM from JCR_SITEM I, JCR_SVALUE V" + " where I.PARENT_ID=? and I.I_CLASS=2 and I.CONTAINER_NAME=? and I.ID=V.PROPERTY_ID" + " and V.STORAGE_DESC is not null"; FIND_VALUE_STORAGE_DESC_AND_SIZE = "select length(DATA), STORAGE_DESC from JCR_SVALUE where PROPERTY_ID=?"; } /** * {@inheritDoc} */ @Override protected int addNodeRecord(NodeData data) throws SQLException { if (insertNode == null) { insertNode = dbConnection.prepareStatement(INSERT_NODE); } else { insertNode.clearParameters(); } insertNode.setString(1, getInternalId(data.getIdentifier())); // if root then parent identifier equals space string insertNode.setString(2, data.getParentIdentifier() == null ? Constants.ROOT_PARENT_UUID : getInternalId(data .getParentIdentifier())); insertNode.setString(3, data.getQPath().getName().getAsString()); insertNode.setString(4, this.containerConfig.containerName); insertNode.setInt(5, data.getPersistedVersion()); insertNode.setInt(6, data.getQPath().getIndex()); insertNode.setInt(7, data.getOrderNumber()); return insertNode.executeUpdate(); } /** * {@inheritDoc} */ @Override protected int addPropertyRecord(PropertyData data) throws SQLException { if (insertProperty == null) { insertProperty = dbConnection.prepareStatement(INSERT_PROPERTY); } else { insertProperty.clearParameters(); } insertProperty.setString(1, getInternalId(data.getIdentifier())); insertProperty.setString(2, getInternalId(data.getParentIdentifier())); insertProperty.setString(3, data.getQPath().getName().getAsString()); insertProperty.setString(4, this.containerConfig.containerName); insertProperty.setInt(5, data.getPersistedVersion()); insertProperty.setInt(6, data.getQPath().getIndex()); insertProperty.setInt(7, data.getType()); insertProperty.setBoolean(8, data.isMultiValued()); return insertProperty.executeUpdate(); } /** * {@inheritDoc} */ @Override protected int addReference(PropertyData data) throws SQLException, IOException { if (insertReference == null) { insertReference = dbConnection.prepareStatement(INSERT_REF); } else { insertReference.clearParameters(); } List<ValueData> values = data.getValues(); int added = 0; for (int i = 0; i < values.size(); i++) { ValueData vdata = values.get(i); String refNodeIdentifier; try { refNodeIdentifier = ValueDataUtil.getString(vdata); } catch (RepositoryException e) { throw new IOException(e.getMessage(), e); } insertReference.setString(1, getInternalId(refNodeIdentifier)); insertReference.setString(2, getInternalId(data.getIdentifier())); insertReference.setInt(3, i); added += insertReference.executeUpdate(); } return added; } /** * {@inheritDoc} */ @Override protected int deleteReference(String propertyCid) throws SQLException { if (deleteReference == null) { deleteReference = dbConnection.prepareStatement(DELETE_REF); } else { deleteReference.clearParameters(); } deleteReference.setString(1, propertyCid); return deleteReference.executeUpdate(); } /** * {@inheritDoc} */ @Override protected int deleteItemByIdentifier(String cid) throws SQLException { if (deleteItem == null) { deleteItem = dbConnection.prepareStatement(DELETE_ITEM); } else { deleteItem.clearParameters(); } deleteItem.setString(1, cid); return deleteItem.executeUpdate(); } /** * {@inheritDoc} */ @Override protected ResultSet findChildNodesByParentIdentifier(String parentCid) throws SQLException { if (findNodesByParentId == null) { findNodesByParentId = dbConnection.prepareStatement(FIND_NODES_BY_PARENTID); } else { findNodesByParentId.clearParameters(); } findNodesByParentId.setString(1, this.containerConfig.containerName); findNodesByParentId.setString(2, parentCid); return findNodesByParentId.executeQuery(); } /** * {@inheritDoc} */ @Override protected ResultSet findLastOrderNumberByParentIdentifier(String parentIdentifier) throws SQLException { if (findLastOrderNumberByParentId == null) { findLastOrderNumberByParentId = dbConnection.prepareStatement(FIND_LAST_ORDER_NUMBER_BY_PARENTID); } else { findLastOrderNumberByParentId.clearParameters(); } findLastOrderNumberByParentId.setString(1, this.containerConfig.containerName); findLastOrderNumberByParentId.setString(2, parentIdentifier); return findLastOrderNumberByParentId.executeQuery(); } /** * {@inheritDoc} */ @Override protected ResultSet findChildNodesCountByParentIdentifier(String parentCid) throws SQLException { if (findNodesCountByParentId == null) { findNodesCountByParentId = dbConnection.prepareStatement(FIND_NODES_COUNT_BY_PARENTID); } else { findNodesCountByParentId.clearParameters(); } findNodesCountByParentId.setString(1, this.containerConfig.containerName); findNodesCountByParentId.setString(2, parentCid); return findNodesCountByParentId.executeQuery(); } /** * {@inheritDoc} */ @Override protected ResultSet findChildPropertiesByParentIdentifier(String parentCid) throws SQLException { if (findPropertiesByParentId == null) { findPropertiesByParentId = dbConnection.prepareStatement(FIND_PROPERTIES_BY_PARENTID); } else { findPropertiesByParentId.clearParameters(); } findPropertiesByParentId.setString(1, this.containerConfig.containerName); findPropertiesByParentId.setString(2, parentCid); return findPropertiesByParentId.executeQuery(); } /** * {@inheritDoc} */ @Override protected ResultSet findItemByName(String parentId, String name, int index) throws SQLException { if (findItemByName == null) { findItemByName = dbConnection.prepareStatement(FIND_ITEM_BY_NAME); } else { findItemByName.clearParameters(); } findItemByName.setString(1, this.containerConfig.containerName); findItemByName.setString(2, parentId); findItemByName.setString(3, name); findItemByName.setInt(4, index); return findItemByName.executeQuery(); } /** * {@inheritDoc} */ @Override protected ResultSet findPropertyByName(String parentCid, String name) throws SQLException { if (findPropertyByName == null) { findPropertyByName = dbConnection.prepareStatement(FIND_PROPERTY_BY_NAME); } else { findPropertyByName.clearParameters(); } findPropertyByName.setString(1, this.containerConfig.containerName); findPropertyByName.setString(2, parentCid); findPropertyByName.setString(3, name); return findPropertyByName.executeQuery(); } /** * {@inheritDoc} */ @Override protected ResultSet findItemByIdentifier(String cid) throws SQLException { if (findItemById == null) { findItemById = dbConnection.prepareStatement(FIND_ITEM_BY_ID); } else { findItemById.clearParameters(); } findItemById.setString(1, cid); return findItemById.executeQuery(); } /** * {@inheritDoc} */ @Override protected ResultSet findReferences(String cid) throws SQLException { if (findReferences == null) { findReferences = dbConnection.prepareStatement(FIND_REFERENCES); } else { findReferences.clearParameters(); } findReferences.setString(1, cid); findReferences.setString(2, this.containerConfig.containerName); return findReferences.executeQuery(); } /** * {@inheritDoc} */ @Override protected int updateNodeByIdentifier(int version, int index, int orderNumb, String cid) throws SQLException { if (updateNode == null) { updateNode = dbConnection.prepareStatement(UPDATE_NODE); } else { updateNode.clearParameters(); } updateNode.setInt(1, version); updateNode.setInt(2, index); updateNode.setInt(3, orderNumb); updateNode.setString(4, cid); return updateNode.executeUpdate(); } /** * {@inheritDoc} */ @Override protected int updatePropertyByIdentifier(int version, int type, String cid) throws SQLException { if (updateProperty == null) { updateProperty = dbConnection.prepareStatement(UPDATE_PROPERTY); } else { updateProperty.clearParameters(); } updateProperty.setInt(1, version); updateProperty.setInt(2, type); updateProperty.setString(3, cid); return updateProperty.executeUpdate(); } /** * {@inheritDoc} */ protected ResultSet findChildNodesByParentIdentifier(String parentCid, int fromOrderNum, int offset, int limit) throws SQLException { throw new UnsupportedOperationException("findChildNodesByParentIdentifier is not supported for old queries"); } // -------- values processing ------------ /** * {@inheritDoc} */ @Override protected int addValueData(String cid, int orderNumber, InputStream stream, int streamLength, String storageDesc) throws SQLException { if (insertValue == null) { insertValue = dbConnection.prepareStatement(INSERT_VALUE); } else { insertValue.clearParameters(); } if (stream == null) { // [PN] store vd reference to external storage etc. insertValue.setNull(1, Types.BINARY); insertValue.setString(4, storageDesc); } else { insertValue.setBinaryStream(1, stream, streamLength); insertValue.setNull(4, Types.VARCHAR); } insertValue.setInt(2, orderNumber); insertValue.setString(3, cid); return insertValue.executeUpdate(); } /** * {@inheritDoc} */ @Override protected int deleteValueData(String cid) throws SQLException { if (deleteValue == null) { deleteValue = dbConnection.prepareStatement(DELETE_VALUE); } else { deleteValue.clearParameters(); } deleteValue.setString(1, cid); return deleteValue.executeUpdate(); } /** * {@inheritDoc} */ @Override protected ResultSet findValuesByPropertyId(String cid) throws SQLException { if (findValuesByPropertyId == null) { findValuesByPropertyId = dbConnection.prepareStatement(FIND_VALUES_BY_PROPERTYID); } else { findValuesByPropertyId.clearParameters(); } findValuesByPropertyId.setString(1, cid); return findValuesByPropertyId.executeQuery(); } /** * {@inheritDoc} */ @Override protected int renameNode(NodeData data) throws SQLException { if (renameNode == null) { renameNode = dbConnection.prepareStatement(RENAME_NODE); } else { renameNode.clearParameters(); } renameNode.setString(1, data.getParentIdentifier() == null ? Constants.ROOT_PARENT_UUID : getInternalId(data .getParentIdentifier())); renameNode.setString(2, data.getQPath().getName().getAsString()); renameNode.setInt(3, data.getPersistedVersion()); renameNode.setInt(4, data.getQPath().getIndex()); renameNode.setInt(5, data.getOrderNumber()); renameNode.setString(6, getInternalId(data.getIdentifier())); return renameNode.executeUpdate(); } /** * {@inheritDoc} */ @Override protected ResultSet findNodesAndProperties(String lastNodeId, int offset, int limit) throws SQLException { throw new UnsupportedOperationException( "The method findNodesAndProperties is not supported for this type of connection use the complex queries instead"); } /** * {@inheritDoc} */ protected void deleteLockProperties() throws SQLException { PreparedStatement removeValuesStatement = null; PreparedStatement removeItemsStatement = null; try { removeValuesStatement = dbConnection .prepareStatement("DELETE FROM JCR_SVALUE WHERE PROPERTY_ID " + "IN (SELECT ID FROM JCR_SITEM WHERE CONTAINER_NAME = ? AND " + "(NAME = '[http://www.jcp.org/jcr/1.0]lockIsDeep' OR NAME = '[http://www.jcp.org/jcr/1.0]lockOwner'))"); removeValuesStatement.setString(1, this.containerConfig.containerName); removeItemsStatement = dbConnection.prepareStatement("DELETE FROM JCR_SITEM WHERE CONTAINER_NAME = ? AND " + "(NAME = '[http://www.jcp.org/jcr/1.0]lockIsDeep' OR NAME = '[http://www.jcp.org/jcr/1.0]lockOwner')"); removeItemsStatement.setString(1, this.containerConfig.containerName); removeValuesStatement.executeUpdate(); removeItemsStatement.executeUpdate(); } finally { if (removeValuesStatement != null) { try { removeValuesStatement.close(); } catch (SQLException e) { LOG.error("Can't close statement", e); } } if (removeItemsStatement != null) { try { removeItemsStatement.close(); } catch (SQLException e) { LOG.error("Can't close statement", e); } } } } /** * {@inheritDoc} */ @Override protected ResultSet findNodesCount() throws SQLException { if (findNodesCount == null) { findNodesCount = dbConnection.prepareStatement(FIND_NODES_COUNT); } else { findNodesCount.clearParameters(); } findNodesCount.setString(1, this.containerConfig.containerName); return findNodesCount.executeQuery(); } /** * {@inheritDoc} */ protected ResultSet findMaxPropertyVersion(String parentId, String name, int index) throws SQLException { if (findMaxPropertyVersions == null) { findMaxPropertyVersions = dbConnection.prepareStatement(FIND_MAX_PROPERTY_VERSIONS); } findMaxPropertyVersions.setString(1, getInternalId(parentId)); findMaxPropertyVersions.setString(2, containerConfig.containerName); findMaxPropertyVersions.setString(3, name); findMaxPropertyVersions.setInt(4, index); return findMaxPropertyVersions.executeQuery(); } /** * {@inheritDoc} */ protected ResultSet findWorkspaceDataSize() throws SQLException { if (findWorkspaceDataSize == null) { findWorkspaceDataSize = dbConnection.prepareStatement(FIND_WORKSPACE_DATA_SIZE); } findWorkspaceDataSize.setString(1, containerConfig.containerName); return findWorkspaceDataSize.executeQuery(); } /** * {@inheritDoc} */ protected ResultSet findNodeDataSize(String parentId) throws SQLException { if (findNodeDataSize == null) { findNodeDataSize = dbConnection.prepareStatement(FIND_NODE_DATA_SIZE); } findNodeDataSize.setString(1, parentId); findNodeDataSize.setString(2, containerConfig.containerName); return findNodeDataSize.executeQuery(); } /** * {@inheritDoc} */ protected ResultSet findWorkspacePropertiesOnValueStorage() throws SQLException { if (findWorkspacePropertiesOnValueStorage == null) { findWorkspacePropertiesOnValueStorage = dbConnection.prepareStatement(FIND_WORKSPACE_PROPERTIES_ON_VALUE_STORAGE); } findWorkspacePropertiesOnValueStorage.setString(1, containerConfig.containerName); return findWorkspacePropertiesOnValueStorage.executeQuery(); } /** * {@inheritDoc} */ protected ResultSet findNodePropertiesOnValueStorage(String parentId) throws SQLException { if (findNodePropertiesOnValueStorage == null) { findNodePropertiesOnValueStorage = dbConnection.prepareStatement(FIND_NODE_PROPERTIES_ON_VALUE_STORAGE); } findNodePropertiesOnValueStorage.setString(1, parentId); findNodePropertiesOnValueStorage.setString(2, containerConfig.containerName); return findNodePropertiesOnValueStorage.executeQuery(); } /** * {@inheritDoc} */ protected ResultSet findValueStorageDescAndSize(String cid) throws SQLException { if (findValueStorageDescAndSize == null) { findValueStorageDescAndSize = dbConnection.prepareStatement(FIND_VALUE_STORAGE_DESC_AND_SIZE); } findValueStorageDescAndSize.setString(1, cid); return findValueStorageDescAndSize.executeQuery(); } }