org.modeshape.jdbc.JcrResultSetMetaData Maven / Gradle / Ivy
Show all versions of modeshape-jdbc-local
/*
* ModeShape (http://www.modeshape.org)
* See the COPYRIGHT.txt file distributed with this work for information
* regarding copyright ownership. Some portions may be licensed
* to Red Hat, Inc. under one or more contributor license agreements.
* See the AUTHORS.txt file in the distribution for a full listing of
* individual contributors.
*
* ModeShape is free software. Unless otherwise indicated, all code in ModeShape
* is licensed to you 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.
*
* ModeShape 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.modeshape.jdbc;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import javax.jcr.PropertyType;
import javax.jcr.RepositoryException;
import javax.jcr.nodetype.NodeType;
import javax.jcr.nodetype.PropertyDefinition;
import javax.jcr.query.QueryResult;
import org.modeshape.jdbc.JcrType.DefaultDataTypes;
/**
* This driver's {@link ResultSetMetaData} implementation that obtains the metadata information from the JCR query result and
* (where possible) the column's corresponding property definitions.
*/
public class JcrResultSetMetaData implements ResultSetMetaData {
private final JcrConnection connection;
private final QueryResult results;
private int[] nullable;
protected JcrResultSetMetaData( JcrConnection connection,
QueryResult results ) {
this.connection = connection;
this.results = results;
}
/**
* {@inheritDoc}
*
* All columns come from the same repository (i.e., catalog).
*
*
* @see java.sql.ResultSetMetaData#getCatalogName(int)
*/
@Override
public String getCatalogName( int column ) {
return connection.info().getRepositoryName();
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#getColumnClassName(int)
*/
@Override
public String getColumnClassName( int column ) {
return getJcrType(column).getRepresentationClass().getName();
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#getColumnCount()
*/
@Override
public int getColumnCount() throws SQLException {
try {
return results.getColumnNames().length;
} catch (RepositoryException e) {
throw new SQLException(e.getLocalizedMessage(), e);
}
}
/**
* {@inheritDoc}
*
* This method returns the nominal display size based upon the column's type. Therefore, the value may not reflect the optimal
* display size for any given value.
*
*
* @see java.sql.ResultSetMetaData#getColumnDisplaySize(int)
*/
@Override
public int getColumnDisplaySize( int column ) {
return getJcrType(column).getNominalDisplaySize();
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#getColumnLabel(int)
*/
@Override
public String getColumnLabel( int column ) throws SQLException {
try {
return results.getColumnNames()[column - 1]; // column value is 1-based
} catch (RepositoryException e) {
throw new SQLException(e.getLocalizedMessage(), e);
}
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#getColumnName(int)
*/
@Override
public String getColumnName( int column ) throws SQLException {
return getColumnLabel(column);
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#getColumnType(int)
*/
@Override
public int getColumnType( int column ) {
return getJcrType(column).getJdbcType();
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#getColumnTypeName(int)
*/
@Override
public String getColumnTypeName( int column ) {
return getJcrType(column).getJdbcTypeName();
}
/**
* {@inheritDoc}
*
* This method always returns the nominal display size for the type.
*
*
* @see java.sql.ResultSetMetaData#getPrecision(int)
*/
@Override
public int getPrecision( int column ) {
return getJcrType(column).getNominalDisplaySize();
}
/**
* {@inheritDoc}
*
* This method returns the number of digits behind the decimal point, which is assumed to be 3 if the type is
* {@link PropertyType#DOUBLE} or 0 otherwise.
*
*
* @see java.sql.ResultSetMetaData#getScale(int)
*/
@Override
public int getScale( int column ) {
JcrType typeInfo = getJcrType(column);
if (typeInfo.getJcrType() == PropertyType.DOUBLE) {
return 3; // pulled from thin air
}
return 0;
}
/**
* {@inheritDoc}
*
* This method always returns the workspace name.
*
*
* @see java.sql.ResultSetMetaData#getSchemaName(int)
*/
@Override
public String getSchemaName( int column ) {
return connection.info().getWorkspaceName();
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#getTableName(int)
*/
@Override
public String getTableName( int column ) throws SQLException {
try {
return results.getSelectorNames()[column - 1]; // column value is 1-based
} catch (RepositoryException e) {
throw new SQLException(e.getLocalizedMessage(), e);
}
}
/**
* {@inheritDoc}
*
* This method always returns false, since this JCR property types don't represent auto-incremented values.
*
*
* @see java.sql.ResultSetMetaData#isAutoIncrement(int)
*/
@Override
public boolean isAutoIncrement( int column ) {
return false;
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#isCaseSensitive(int)
*/
@Override
public boolean isCaseSensitive( int column ) {
return getJcrType(column).isCaseSensitive();
}
/**
* {@inheritDoc}
*
* This method always returns false, since no JCR property types (directly) represent currency.
*
*
* @see java.sql.ResultSetMetaData#isCurrency(int)
*/
@Override
public boolean isCurrency( int column ) {
return false;
}
/**
* {@inheritDoc}
*
* This method always returns false, since this JDBC driver does not support writes.
*
*
* @see java.sql.ResultSetMetaData#isDefinitelyWritable(int)
*/
@Override
public boolean isDefinitelyWritable( int column ) {
return false;
}
/**
* {@inheritDoc}
*
* @see java.sql.ResultSetMetaData#isNullable(int)
*/
@Override
public int isNullable( int column ) throws SQLException {
if (nullable == null) {
int length = getColumnCount();
nullable = new int[length];
for (int i = 0; i != length; ++i) {
nullable[i] = -1;
}
} else {
int result = nullable[column - 1];
if (result != -1) {
// Already found this value, so return it ...
return result;
}
}
// Find the node type for the column (given that the column name is the property name and
// the table name is the node type), and determine if the property definition is multi-valued or not mandatory.
String nodeTypeName = getTableName(column);
if (nodeTypeName.length() == 0) {
// There is no table for the column, so therefore we don't know the node type ...
return ResultSetMetaData.columnNullableUnknown;
}
String propertyName = getColumnName(column);
boolean singleProp = false;
boolean singleResidual = false;
boolean multiResidual = false;
NodeType type = connection.nodeType(nodeTypeName);
for (PropertyDefinition defn : type.getPropertyDefinitions()) {
if (defn.getName().equals(propertyName)) {
if (defn.isMultiple() || defn.isMandatory()) {
// We know this IS nullable
return ResultSetMetaData.columnNullable;
}
// Otherwise this is a single-valued property that is mandatory,
// but we can't return columnNotNullable because we may not have found the multi-valued property ...
singleProp = true;
} else if (defn.getName().equals("*")) { // Residual
if (defn.isMultiple() || defn.isMandatory()) multiResidual = true;
else singleResidual = true;
}
}
int result = ResultSetMetaData.columnNullableUnknown;
if (multiResidual) result = ResultSetMetaData.columnNullable;
else if (singleProp || singleResidual) result = ResultSetMetaData.columnNoNulls;
nullable[column - 1] = result;
return result;
}
/**
* {@inheritDoc}
*
* Even though the value may be writable in the JCR repository, this JDBC driver does not support writes. Therefore, this
* method always returns true.
*
*
* @see java.sql.ResultSetMetaData#isReadOnly(int)
*/
@Override
public boolean isReadOnly( int column ) {
return true;
}
/**
* {@inheritDoc}
*
* In JCR-SQL2, every property can be used in a WHERE clause. Therefore, this method always returns true.
*
*
* @see java.sql.ResultSetMetaData#isSearchable(int)
*/
@Override
public boolean isSearchable( int column ) {
return true;
}
/**
* {@inheritDoc}
*
* This method returns true if the column is a {@link PropertyType#DOUBLE}, {@link PropertyType#LONG} or
* {@link PropertyType#DATE}.
*
*
* @see java.sql.ResultSetMetaData#isSigned(int)
*/
@Override
public boolean isSigned( int column ) {
return getJcrType(column).isSigned();
}
/**
* {@inheritDoc}
*
* Even though the value may be writable in the JCR repository, this JDBC driver does not support writes. Therefore, this
* method always returns false.
*
*
* @see java.sql.ResultSetMetaData#isWritable(int)
*/
@Override
public boolean isWritable( int column ) {
return false;
}
/**
* {@inheritDoc}
*
* @see java.sql.Wrapper#isWrapperFor(java.lang.Class)
*/
@Override
public boolean isWrapperFor( Class> iface ) /*throws SQLException*/{
return iface.isInstance(results);
}
/**
* {@inheritDoc}
*
* @see java.sql.Wrapper#unwrap(java.lang.Class)
*/
@Override
public T unwrap( Class iface ) throws SQLException {
if (iface.isInstance(results)) {
return iface.cast(results);
}
throw new SQLException(JdbcLocalI18n.classDoesNotImplementInterface.text(ResultSetMetaData.class.getSimpleName(),
iface.getName()));
}
private JcrType getJcrType( int column ) {
JcrType typeInfo = null;
if (results instanceof org.modeshape.jcr.api.query.QueryResult) {
final String typeName = ((org.modeshape.jcr.api.query.QueryResult)results).getColumnTypes()[column - 1];
typeInfo = JcrType.typeInfo(typeName);
}
/**
* If no type is matched, then default to STRING
*/
return (typeInfo != null ? typeInfo : JcrType.typeInfo(DefaultDataTypes.STRING));
}
}