package io.dropwizard.jdbi.args;
import org.skife.jdbi.v2.StatementContext;
import org.skife.jdbi.v2.tweak.Argument;
import org.skife.jdbi.v2.tweak.ArgumentFactory;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.sql.Types;
import java.util.Optional;
public class OptionalArgumentFactory implements ArgumentFactory<Optional<Object>> {
private static class DefaultOptionalArgument implements Argument {
private final Optional<?> value;
private final int nullType;
private DefaultOptionalArgument(Optional<?> value, int nullType) {
this.value = value;
this.nullType = nullType;
}
private DefaultOptionalArgument(Optional<?> value) {
this(value, Types.OTHER);
}
@Override
public void apply(int position,
PreparedStatement statement,
StatementContext ctx) throws SQLException {
if (value.isPresent()) {
statement.setObject(position, value.get());
} else {
statement.setNull(position, nullType);
}
}
}
private static class MsSqlOptionalArgument implements Argument {
private final Optional<?> value;
private MsSqlOptionalArgument(Optional<?> value) {
this.value = value;
}
@Override
public void apply(int position,
PreparedStatement statement,
StatementContext ctx) throws SQLException {
statement.setObject(position, value.orElse(null));
}
}
private final String jdbcDriver;
public OptionalArgumentFactory(String jdbcDriver) {
this.jdbcDriver = jdbcDriver;
}
@Override
public boolean accepts(Class<?> expectedType, Object value, StatementContext ctx) {
return value instanceof Optional;
}
@Override
public Argument build(Class<?> expectedType, Optional<Object> value, StatementContext ctx) {
if ("com.microsoft.sqlserver.jdbc.SQLServerDriver".equals(jdbcDriver)) {
return new MsSqlOptionalArgument(value);
} else if ("oracle.jdbc.OracleDriver".equals(jdbcDriver)) {
return new DefaultOptionalArgument(value, Types.NULL);
}
return new DefaultOptionalArgument(value);
}
}