org.mariadb.jdbc.CallableParameterMetaData Maven / Gradle / Ivy
Show all versions of mariadb-java-client Show documentation
/*
*
* MariaDB Client for Java
*
* Copyright (c) 2012-2014 Monty Program Ab.
* Copyright (c) 2015-2020 MariaDB Corporation Ab.
*
* This library 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 library 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 library; if not, write to Monty Program Ab [email protected].
*
* This particular MariaDB Client for Java file is work
* derived from a Drizzle-JDBC. Drizzle-JDBC file which is covered by subject to
* the following copyright and notice provisions:
*
* Copyright (c) 2009-2011, Marcus Eriksson
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice, this list
* of conditions and the following disclaimer.
*
* 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.
*
* Neither the name of the driver nor the names of its contributors may not be
* used to endorse or promote products derived from this software without specific
* prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS 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 THE COPYRIGHT HOLDER OR 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.
*
*/
package org.mariadb.jdbc;
import java.math.BigDecimal;
import java.sql.*;
import java.util.BitSet;
import java.util.Locale;
public class CallableParameterMetaData implements ParameterMetaData {
private final ResultSet rs;
private final int parameterCount;
private final boolean isFunction;
public CallableParameterMetaData(ResultSet rs, boolean isFunction) throws SQLException {
this.rs = rs;
int count = 0;
while (rs.next()) count++;
this.parameterCount = count;
this.isFunction = isFunction;
}
/**
* Retrieves the number of parameters in the PreparedStatement
object for which this
* ParameterMetaData
object contains information.
*
* @return the number of parameters
* @since 1.4
*/
@Override
public int getParameterCount() {
return parameterCount;
}
/**
* Retrieves whether null values are allowed in the designated parameter.
*
* @param index the first parameter is 1, the second is 2, ...
* @return the nullability status of the given parameter; one of
* ParameterMetaData.parameterNoNulls
, ParameterMetaData.parameterNullable
* , or ParameterMetaData.parameterNullableUnknown
* @throws SQLException if a database access error occurs
* @since 1.4
*/
@Override
public int isNullable(int index) throws SQLException {
setIndex(index);
return ParameterMetaData.parameterNullableUnknown;
}
private void setIndex(int index) throws SQLException {
if (index < 1 || index > parameterCount) {
throw new SQLException("invalid parameter index " + index);
}
rs.absolute(index);
}
/**
* Retrieves whether values for the designated parameter can be signed numbers.
*
* @param index the first parameter is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @throws SQLException if a database access error occurs
* @since 1.4
*/
@Override
public boolean isSigned(int index) throws SQLException {
setIndex(index);
String paramDetail = rs.getString("DTD_IDENTIFIER");
return !paramDetail.contains(" unsigned");
}
/**
* Retrieves the designated parameter's specified column size.
*
* The returned value represents the maximum column size for the given parameter. For numeric
* data, this is the maximum precision. For character data, this is the length in characters. For
* datetime datatypes, this is the length in characters of the String representation (assuming the
* maximum allowed precision of the fractional seconds component). For binary data, this is the
* length in bytes. For the ROWID datatype, this is the length in bytes. 0 is returned for data
* types where the column size is not applicable.
*
* @param index the first parameter is 1, the second is 2, ...
* @return precision
* @throws SQLException if a database access error occurs
* @since 1.4
*/
@Override
public int getPrecision(int index) throws SQLException {
setIndex(index);
int characterMaxLength = rs.getInt("CHARACTER_MAXIMUM_LENGTH");
int numericPrecision = rs.getInt("NUMERIC_PRECISION");
return (numericPrecision > 0) ? numericPrecision : characterMaxLength;
}
/**
* Retrieves the designated parameter's number of digits to right of the decimal point. 0 is
* returned for data types where the scale is not applicable.
*
* @param index the first parameter is 1, the second is 2, ...
* @return scale
* @throws SQLException if a database access error occurs
* @since 1.4
*/
@Override
public int getScale(int index) throws SQLException {
setIndex(index);
return rs.getInt("NUMERIC_SCALE");
}
public String getParameterName(int index) throws SQLException {
setIndex(index);
return rs.getString("PARAMETER_NAME");
}
/**
* Retrieves the designated parameter's SQL type.
*
* @param index the first parameter is 1, the second is 2, ...
* @return SQL type from java.sql.Types
* @throws SQLException if a database access error occurs
* @see Types
* @since 1.4
*/
@Override
public int getParameterType(int index) throws SQLException {
setIndex(index);
String str = rs.getString("DATA_TYPE").toUpperCase(Locale.ROOT);
switch (str) {
case "BIT":
return Types.BIT;
case "TINYINT":
return Types.TINYINT;
case "SMALLINT":
case "YEAR":
return Types.SMALLINT;
case "MEDIUMINT":
case "INT":
case "INT24":
case "INTEGER":
return Types.INTEGER;
case "LONG":
case "BIGINT":
return Types.BIGINT;
case "REAL":
case "DOUBLE":
return Types.DOUBLE;
case "FLOAT":
return Types.FLOAT;
case "DECIMAL":
return Types.DECIMAL;
case "CHAR":
return Types.CHAR;
case "VARCHAR":
case "ENUM":
case "TINYTEXT":
case "SET":
return Types.VARCHAR;
case "DATE":
return Types.DATE;
case "TIME":
return Types.TIME;
case "TIMESTAMP":
case "DATETIME":
return Types.TIMESTAMP;
case "BINARY":
return Types.BINARY;
case "VARBINARY":
return Types.VARBINARY;
case "TINYBLOB":
case "BLOB":
case "MEDIUMBLOB":
case "LONGBLOB":
case "GEOMETRY":
return Types.BLOB;
case "TEXT":
case "MEDIUMTEXT":
case "LONGTEXT":
return Types.CLOB;
default:
return Types.OTHER;
}
}
/**
* Retrieves the designated parameter's database-specific type name.
*
* @param index the first parameter is 1, the second is 2, ...
* @return type the name used by the database. If the parameter type is a user-defined type, then
* a fully-qualified type name is returned.
* @throws SQLException if a database access error occurs
* @since 1.4
*/
@Override
public String getParameterTypeName(int index) throws SQLException {
setIndex(index);
return rs.getString("DATA_TYPE").toUpperCase(Locale.ROOT);
}
/**
* Retrieves the fully-qualified name of the Java class whose instances should be passed to the
* method PreparedStatement.setObject
.
*
* @param index the first parameter is 1, the second is 2, ...
* @return the fully-qualified name of the class in the Java programming language that would be
* used by the method PreparedStatement.setObject
to set the value in the
* specified parameter. This is the class name used for custom mapping.
* @throws SQLException if a database access error occurs
* @since 1.4
*/
@Override
public String getParameterClassName(int index) throws SQLException {
setIndex(index);
String str = rs.getString("DATA_TYPE").toUpperCase(Locale.ROOT);
switch (str) {
case "BIT":
return BitSet.class.getName();
case "TINYINT":
return byte.class.getName();
case "SMALLINT":
case "YEAR":
return short.class.getName();
case "MEDIUMINT":
case "INT":
case "INTEGER":
return int.class.getName();
case "BINARY":
case "SET":
case "GEOMETRY":
case "VARBINARY":
case "TINYBLOB":
return byte[].class.getName();
case "BIGINT":
return long.class.getName();
case "FLOAT":
return float.class.getName();
case "DECIMAL":
return BigDecimal.class.getName();
case "REAL":
case "DOUBLE":
return double.class.getName();
case "CHAR":
case "VARCHAR":
case "ENUM":
case "TINYTEXT":
return String.class.getName();
case "TEXT":
case "MEDIUMTEXT":
case "LONGTEXT":
return Clob.class.getName();
case "DATE":
return Date.class.getName();
case "TIME":
return Time.class.getName();
case "TIMESTAMP":
case "DATETIME":
return Timestamp.class.getName();
case "BLOB":
case "MEDIUMBLOB":
case "LONGBLOB":
return Blob.class.getName();
default:
return Object.class.getName();
}
}
/**
* Retrieves the designated parameter's mode.
*
* @param index the first parameter is 1, the second is 2, ...
* @return mode of the parameter; one of ParameterMetaData.parameterModeIn
,
* ParameterMetaData.parameterModeOut
, or ParameterMetaData.parameterModeInOut
*
ParameterMetaData.parameterModeUnknown
.
* @throws SQLException if a database access error occurs
* @since 1.4
*/
@Override
public int getParameterMode(int index) throws SQLException {
setIndex(index);
if (isFunction) return ParameterMetaData.parameterModeOut;
String str = rs.getString("PARAMETER_MODE");
switch (str) {
case "IN":
return ParameterMetaData.parameterModeIn;
case "OUT":
return ParameterMetaData.parameterModeOut;
case "INOUT":
return ParameterMetaData.parameterModeInOut;
default:
return ParameterMetaData.parameterModeUnknown;
}
}
/**
* Returns an object that implements the given interface to allow access to non-standard methods,
* or standard methods not exposed by the proxy.
*
*
If the receiver implements the interface then the result is the receiver or a proxy for the
* receiver. If the receiver is a wrapper and the wrapped object implements the interface then the
* result is the wrapped object or a proxy for the wrapped object. Otherwise return the the result
* of calling unwrap
recursively on the wrapped object or a proxy for that result. If
* the receiver is not a wrapper and does not implement the interface, then an SQLException
*
is thrown.
*
* @param iface A Class defining an interface that the result must implement.
* @return an object that implements the interface. May be a proxy for the actual implementing
* object.
* @throws SQLException If no object found that implements the interface
* @since 1.6
*/
@Override
public T unwrap(Class iface) throws SQLException {
if (isWrapperFor(iface)) {
return iface.cast(this);
}
throw new SQLException("The receiver is not a wrapper for " + iface.getName());
}
/**
* Returns true if this either implements the interface argument or is directly or indirectly a
* wrapper for an object that does. Returns false otherwise. If this implements the interface then
* return true, else if this is a wrapper then return the result of recursively calling
* isWrapperFor
on the wrapped object. If this does not implement the interface and is not
* a wrapper, return false. This method should be implemented as a low-cost operation compared to
* unwrap
so that callers can use this method to avoid expensive unwrap
* calls that may fail. If this method returns true then calling unwrap
with the same
* argument should succeed.
*
* @param iface a Class defining an interface.
* @return true if this implements the interface or directly or indirectly wraps an object that
* does.
* @throws SQLException if an error occurs while determining whether this is a wrapper for an
* object with the given interface.
* @since 1.6
*/
@Override
public boolean isWrapperFor(Class> iface) throws SQLException {
return iface.isInstance(this);
}
}