DefaultParameterHandlerTest.java
/*
* Copyright 2009-2025 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.ibatis.scripting.defaults;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyInt;
import static org.mockito.Mockito.doThrow;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import java.sql.ParameterMetaData;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.sql.Types;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import org.apache.ibatis.builder.StaticSqlSource;
import org.apache.ibatis.domain.blog.Author;
import org.apache.ibatis.domain.blog.Section;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.mapping.ParameterMapping;
import org.apache.ibatis.mapping.ResultMap;
import org.apache.ibatis.mapping.ResultMapping;
import org.apache.ibatis.mapping.SqlCommandType;
import org.apache.ibatis.reflection.DefaultReflectorFactory;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.reflection.ReflectorFactory;
import org.apache.ibatis.reflection.factory.DefaultObjectFactory;
import org.apache.ibatis.reflection.factory.ObjectFactory;
import org.apache.ibatis.reflection.wrapper.DefaultObjectWrapperFactory;
import org.apache.ibatis.reflection.wrapper.ObjectWrapperFactory;
import org.apache.ibatis.scripting.xmltags.XMLLanguageDriver;
import org.apache.ibatis.session.Configuration;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.TypeException;
import org.apache.ibatis.type.TypeHandler;
import org.apache.ibatis.type.TypeHandlerRegistry;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
/**
* DefaultParameterHandlerTest
*
* @author Ryan Lamore
*/
class DefaultParameterHandlerTest {
@Test
void setParametersThrowsProperException() throws SQLException {
final MappedStatement mappedStatement = getMappedStatement();
final Object parameterObject = null;
final BoundSql boundSql = mock(BoundSql.class);
TypeHandler<Object> typeHandler = mock(TypeHandler.class);
doThrow(new SQLException("foo")).when(typeHandler).setParameter(any(PreparedStatement.class), anyInt(), any(),
any(JdbcType.class));
ParameterMapping parameterMapping = new ParameterMapping.Builder(mappedStatement.getConfiguration(), "prop",
typeHandler).build();
List<ParameterMapping> parameterMappings = List.of(parameterMapping);
when(boundSql.getParameterMappings()).thenReturn(parameterMappings);
DefaultParameterHandler defaultParameterHandler = new DefaultParameterHandler(mappedStatement, parameterObject,
boundSql);
PreparedStatement ps = mock(PreparedStatement.class);
when(ps.getParameterMetaData()).thenReturn(mock(ParameterMetaData.class));
try {
defaultParameterHandler.setParameters(ps);
Assertions.fail("Should have thrown TypeException");
} catch (Exception e) {
Assertions.assertTrue(e instanceof TypeException, "expected TypeException");
Assertions.assertTrue(e.getMessage().contains("mapping: ParameterMapping"));
}
}
MappedStatement getMappedStatement() {
final Configuration config = new Configuration();
final TypeHandlerRegistry registry = config.getTypeHandlerRegistry();
return new MappedStatement.Builder(config, "testSelect", new StaticSqlSource(config, "some select statement"),
SqlCommandType.SELECT).resultMaps(new ArrayList<ResultMap>() {
private static final long serialVersionUID = 1L;
{
add(new ResultMap.Builder(config, "testMap", HashMap.class, new ArrayList<ResultMapping>() {
private static final long serialVersionUID = 1L;
{
add(new ResultMapping.Builder(config, "cOlUmN1", "CoLuMn1", registry.getTypeHandler(Integer.class))
.build());
}
}).build());
}
}).build();
}
@Test
void parameterObjectGetPropertyValueWithAdditionalParameter() throws SQLException {
Configuration config = new Configuration();
TypeHandlerRegistry registry = config.getTypeHandlerRegistry();
MappedStatement mappedStatement = new MappedStatement.Builder(config, "testSelect",
new StaticSqlSource(config, "some select statement"), SqlCommandType.SELECT).build();
Object parameterObject = 1;
BoundSql boundSql = new BoundSql(config, "some select statement", new ArrayList<ParameterMapping>() {
private static final long serialVersionUID = 1L;
{
add(new ParameterMapping.Builder(config, "id", registry.getTypeHandler(int.class)).build());
}
}, parameterObject) {
{
setAdditionalParameter("id", 2);
}
};
DefaultParameterHandler defaultParameterHandler = new DefaultParameterHandler(mappedStatement, parameterObject,
boundSql);
PreparedStatement ps = mock(PreparedStatement.class);
ParameterMetaData pmd = mock(ParameterMetaData.class);
when(pmd.getParameterType(1)).thenReturn(Types.INTEGER);
when(ps.getParameterMetaData()).thenReturn(pmd);
defaultParameterHandler.setParameters(ps);
verify(ps).setInt(1, 2);
}
@Test
void parameterObjectGetPropertyValueWithNull() throws SQLException {
Configuration config = new Configuration();
TypeHandlerRegistry registry = config.getTypeHandlerRegistry();
MappedStatement mappedStatement = new MappedStatement.Builder(config, "testSelect",
new StaticSqlSource(config, "some select statement"), SqlCommandType.SELECT).build();
Object parameterObject = null;
BoundSql boundSql = new BoundSql(config, "some select statement", new ArrayList<ParameterMapping>() {
private static final long serialVersionUID = 1L;
{
add(new ParameterMapping.Builder(config, "id", registry.getTypeHandler(int.class)).build());
}
}, parameterObject);
DefaultParameterHandler defaultParameterHandler = new DefaultParameterHandler(mappedStatement, parameterObject,
boundSql);
PreparedStatement ps = mock(PreparedStatement.class);
ParameterMetaData pmd = mock(ParameterMetaData.class);
when(pmd.getParameterType(1)).thenReturn(Types.INTEGER);
when(ps.getParameterMetaData()).thenReturn(pmd);
defaultParameterHandler.setParameters(ps);
verify(ps).setNull(1, config.getJdbcTypeForNull().TYPE_CODE);
}
@Test
void parameterObjectGetPropertyValueWithTypeHandler() throws SQLException {
Configuration config = new Configuration();
TypeHandlerRegistry registry = config.getTypeHandlerRegistry();
MappedStatement mappedStatement = new MappedStatement.Builder(config, "testSelect",
new StaticSqlSource(config, "some select statement"), SqlCommandType.SELECT).build();
Object parameterObject = 1;
BoundSql boundSql = new BoundSql(config, "some select statement", new ArrayList<ParameterMapping>() {
private static final long serialVersionUID = 1L;
{
add(new ParameterMapping.Builder(config, "id", registry.getTypeHandler(int.class)).build());
}
}, parameterObject);
DefaultParameterHandler defaultParameterHandler = new DefaultParameterHandler(mappedStatement, parameterObject,
boundSql);
PreparedStatement ps = mock(PreparedStatement.class);
ParameterMetaData pmd = mock(ParameterMetaData.class);
when(pmd.getParameterType(1)).thenReturn(Types.INTEGER);
when(ps.getParameterMetaData()).thenReturn(pmd);
defaultParameterHandler.setParameters(ps);
verify(ps).setInt(1, (Integer) parameterObject);
}
@Test
void parameterObjectGetPropertyValueWithMetaObject() throws SQLException {
Configuration config = new Configuration();
TypeHandlerRegistry registry = config.getTypeHandlerRegistry();
MappedStatement mappedStatement = new MappedStatement.Builder(config, "testSelect",
new StaticSqlSource(config, "some select statement"), SqlCommandType.SELECT).build();
Author parameterObject = new Author(-1, "cbegin", "******", "cbegin@nowhere.com", "N/A", Section.NEWS);
BoundSql boundSql = new BoundSql(config, "some select statement", new ArrayList<ParameterMapping>() {
private static final long serialVersionUID = 1L;
{
add(new ParameterMapping.Builder(config, "id", registry.getTypeHandler(int.class)).build());
add(new ParameterMapping.Builder(config, "username", registry.getTypeHandler(String.class)).build());
add(new ParameterMapping.Builder(config, "password", registry.getTypeHandler(String.class)).build());
add(new ParameterMapping.Builder(config, "email", registry.getTypeHandler(String.class)).build());
add(new ParameterMapping.Builder(config, "bio", registry.getTypeHandler(String.class))
.jdbcType(JdbcType.VARCHAR).build());
add(new ParameterMapping.Builder(config, "favouriteSection", registry.getTypeHandler(Section.class))
.jdbcType(JdbcType.VARCHAR).build());
}
}, parameterObject);
DefaultParameterHandler defaultParameterHandler = new DefaultParameterHandler(mappedStatement, parameterObject,
boundSql);
PreparedStatement ps = mock(PreparedStatement.class);
ParameterMetaData pmd = mock(ParameterMetaData.class);
when(pmd.getParameterType(1)).thenReturn(Types.INTEGER);
when(ps.getParameterMetaData()).thenReturn(pmd);
defaultParameterHandler.setParameters(ps);
verify(ps).setInt(1, parameterObject.getId());
verify(ps).setString(2, parameterObject.getUsername());
verify(ps).setString(3, parameterObject.getPassword());
verify(ps).setString(4, parameterObject.getEmail());
verify(ps).setString(5, parameterObject.getBio());
verify(ps).setObject(6, parameterObject.getFavouriteSection().name(), JdbcType.VARCHAR.TYPE_CODE);
}
@Test
void parameterObjectGetPropertyValueWithMetaObjectAndCreateOnce() {
Author parameterObject = mock(Author.class);
Configuration mockConfig = mock(Configuration.class);
final ObjectFactory objectFactory = new DefaultObjectFactory();
final ObjectWrapperFactory objectWrapperFactory = new DefaultObjectWrapperFactory();
final ReflectorFactory reflectorFactory = new DefaultReflectorFactory();
when(mockConfig.getTypeHandlerRegistry()).thenReturn(new TypeHandlerRegistry(mockConfig));
when(mockConfig.getDefaultScriptingLanguageInstance()).thenReturn(new XMLLanguageDriver());
when(mockConfig.newMetaObject(parameterObject))
.thenReturn(MetaObject.forObject(parameterObject, objectFactory, objectWrapperFactory, reflectorFactory));
TypeHandlerRegistry registry = mockConfig.getTypeHandlerRegistry();
MappedStatement mappedStatement = new MappedStatement.Builder(mockConfig, "testSelect",
new StaticSqlSource(mockConfig, "some select statement"), SqlCommandType.SELECT).build();
BoundSql boundSql = new BoundSql(mockConfig, "some select statement", new ArrayList<ParameterMapping>() {
private static final long serialVersionUID = 1L;
{
add(new ParameterMapping.Builder(mockConfig, "id", registry.getTypeHandler(int.class))
.jdbcType(JdbcType.INTEGER).build());
add(new ParameterMapping.Builder(mockConfig, "username", registry.getTypeHandler(String.class))
.jdbcType(JdbcType.VARCHAR).build());
add(new ParameterMapping.Builder(mockConfig, "password", registry.getTypeHandler(String.class))
.jdbcType(JdbcType.VARCHAR).build());
add(new ParameterMapping.Builder(mockConfig, "email", registry.getTypeHandler(String.class))
.jdbcType(JdbcType.VARCHAR).build());
add(new ParameterMapping.Builder(mockConfig, "bio", registry.getTypeHandler(String.class))
.jdbcType(JdbcType.VARCHAR).build());
add(new ParameterMapping.Builder(mockConfig, "favouriteSection", registry.getTypeHandler(Section.class))
.jdbcType(JdbcType.VARCHAR).build());
}
}, parameterObject);
DefaultParameterHandler defaultParameterHandler = new DefaultParameterHandler(mappedStatement, parameterObject,
boundSql);
PreparedStatement ps = mock(PreparedStatement.class);
defaultParameterHandler.setParameters(ps);
verify(parameterObject).getId();
verify(parameterObject).getUsername();
verify(parameterObject).getPassword();
verify(parameterObject).getEmail();
verify(parameterObject).getBio();
verify(parameterObject).getFavouriteSection();
verify(mockConfig).newMetaObject(parameterObject);
}
}