Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Stored procedure call returns wrong BigDecimal scale. #2559

Open
wants to merge 25 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 23 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
7c5f6df
Stored procedure call returns wrong BigDecimal scale.
Ananya2 Dec 6, 2024
fdbda10
Removed try block
Ananya2 Dec 6, 2024
4ab4e5c
Added try-catch block
Ananya2 Dec 6, 2024
44e3aaf
Updated testJdbc41CallableStatementMethods test case for BigDecimal c…
Ananya2 Dec 6, 2024
0c0b4f1
Added test case to validate the BigDecimal precision.
Ananya2 Dec 9, 2024
3c2256f
Updated stored procedure statement test.
Ananya2 Dec 9, 2024
a09a497
Fixed pipeline failure
Ananya2 Dec 9, 2024
c531274
Added 'warning' level log in case of failure while fetching scale for…
Ananya2 Dec 9, 2024
95c9628
throws SQLServerException
Ananya2 Dec 9, 2024
d4859f1
Throw SQLServerException on failure.
Ananya2 Dec 9, 2024
1506f62
Added test case testRegisterOutParameterWithDecimalException
Ananya2 Dec 9, 2024
82546f7
Fixed pipeline failure
Ananya2 Dec 9, 2024
e58a9b7
Updated SQLServerCallableStatement.java
Ananya2 Dec 9, 2024
5815afa
removed test case.
Ananya2 Dec 10, 2024
66d8aad
Removed unused import.
Ananya2 Dec 10, 2024
9581f6b
Created BigDecimalPrecisionTest.java file
Ananya2 Dec 10, 2024
28e4505
Fix: Throw SQLServerException with detailed message for metadata retr…
Ananya2 Dec 11, 2024
a07e707
Add test for BigDecimal precision failure when fetching scale metadata.
Ananya2 Dec 11, 2024
33e6d00
Update SQLServerCallableStatement.java
Ananya2 Dec 11, 2024
222dddc
Update BigDecimalPrecisionTest.java
Ananya2 Dec 11, 2024
67064f5
Drop stored procedure if it exists in init()
Ananya2 Dec 16, 2024
6379d57
Update BigDecimalPrecisionTest.java
Ananya2 Dec 16, 2024
ed0ff76
Added detailed comments to the `registerOutParameter` method to expla…
Ananya2 Dec 20, 2024
d6e4e2b
updated as per review comments
Ananya2 Jan 7, 2025
cb7d65a
fixed build failure
Ananya2 Jan 7, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
import java.sql.SQLXML;
import java.sql.Time;
import java.sql.Timestamp;
import java.sql.ParameterMetaData;
import java.text.MessageFormat;
import java.time.LocalDateTime;
import java.util.Calendar;
Expand Down Expand Up @@ -150,6 +151,21 @@
case microsoft.sql.Types.DATETIMEOFFSET:
param.setOutScale(7);
break;
case java.sql.Types.DECIMAL:
// Dynamically handle the scale for DECIMAL output parameters.
// The scale for the DECIMAL type is fetched from the ParameterMetaData.
// This provides flexibility to automatically apply the correct scale as per the database metadata.
ParameterMetaData parameterMetaData = this.getParameterMetaData();
if (parameterMetaData != null) {
try {
// Fetch scale from metadata for DECIMAL type
int scale = parameterMetaData.getScale(index);
param.setOutScale(scale);
} catch (SQLException e) {
loggerExternal.warning("Failed to fetch scale for DECIMAL type parameter at index " + index + ": " + e.getMessage());

Check warning on line 165 in src/main/java/com/microsoft/sqlserver/jdbc/SQLServerCallableStatement.java

View check run for this annotation

Codecov / codecov/patch

src/main/java/com/microsoft/sqlserver/jdbc/SQLServerCallableStatement.java#L164-L165

Added lines #L164 - L165 were not covered by tests
Ananya2 marked this conversation as resolved.
Show resolved Hide resolved
}
}
break;
default:
break;
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
package com.microsoft.sqlserver.jdbc.unit.statement;

import static org.junit.jupiter.api.Assertions.assertEquals;

import java.math.BigDecimal;
import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Types;

import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Tag;
import org.junit.jupiter.api.Test;

import com.microsoft.sqlserver.jdbc.RandomUtil;
import com.microsoft.sqlserver.testframework.AbstractSQLGenerator;
import com.microsoft.sqlserver.testframework.AbstractTest;

public class BigDecimalPrecisionTest extends AbstractTest {

String procName1 = AbstractSQLGenerator.escapeIdentifier(RandomUtil.getIdentifier("test_bigdecimal_3"));
String procName2 = AbstractSQLGenerator.escapeIdentifier(RandomUtil.getIdentifier("test_bigdecimal_5"));

@BeforeEach
public void init() throws SQLException {
try (Connection connection = getConnection()) {
String dropProcedureSQL = "DROP PROCEDURE IF EXISTS " + procName1 + ", " + procName2;
try (Statement stmt = connection.createStatement()) {
stmt.execute(dropProcedureSQL);
}

String createProcedureSQL1 = "CREATE PROCEDURE " + procName1 + "\n" +
" @big_decimal_type decimal(15, 3),\n" +
" @big_decimal_type_o decimal(15, 3) OUTPUT\n" +
"AS\n" +
Ananya2 marked this conversation as resolved.
Show resolved Hide resolved
"BEGIN\n" +
" SET @big_decimal_type_o = @big_decimal_type;\n" +
"END;";
String createProcedureSQL2 = "CREATE PROCEDURE " + procName2 + "\n" +
" @big_decimal_type decimal(15, 5),\n" +
" @big_decimal_type_o decimal(15, 5) OUTPUT\n" +
"AS\n" +
"BEGIN\n" +
" SET @big_decimal_type_o = @big_decimal_type;\n" +
"END;";
try (Statement stmt = connection.createStatement()) {
stmt.execute(createProcedureSQL1);
stmt.execute(createProcedureSQL2);
}
}
}

@AfterEach
public void terminate() throws SQLException {
Ananya2 marked this conversation as resolved.
Show resolved Hide resolved
try (Connection connection = getConnection()) {
try (Statement stmt = connection.createStatement()) {
String dropProcedureSQL = "DROP PROCEDURE IF EXISTS " + procName1 + ", " + procName2;
stmt.execute(dropProcedureSQL);
}
}
}

@Test
@Tag("BigDecimal")
public void testBigDecimalPrecision() throws SQLException {
try (Connection connection = getConnection()) {
// Test for DECIMAL(15, 3)
String callSQL1 = "{call " + procName1 + "(100.241, ?)}";
try (CallableStatement call = connection.prepareCall(callSQL1)) {
call.registerOutParameter(1, Types.DECIMAL);
call.execute();
BigDecimal actual1 = call.getBigDecimal(1);
assertEquals(new BigDecimal("100.241"), actual1);
}

// Test for DECIMAL(15, 5)
String callSQL2 = "{call " + procName2 + "(100.24112, ?)}";
try (CallableStatement call = connection.prepareCall(callSQL2)) {
call.registerOutParameter(1, Types.DECIMAL);
call.execute();
BigDecimal actual2 = call.getBigDecimal(1);
assertEquals(new BigDecimal("100.24112"), actual2);
}
}
}

@BeforeAll
public static void setupTests() throws Exception {
setConnection();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -1195,15 +1195,14 @@ public void testJdbc41CallableStatementMethods() throws Exception {
assertEquals("2017-05-19 10:47:15.1234567 +02:00",
cstmt.getObject("col14Value", microsoft.sql.DateTimeOffset.class).toString());

// BigDecimal#equals considers the number of decimal places (OutParams always return 4 decimal
// digits rounded up)
assertEquals(0, cstmt.getObject(15, BigDecimal.class).compareTo(new BigDecimal("0.1235")));
// BigDecimal#equals considers the number of decimal places (OutParams always return full precision as specified in the DB schema)
assertEquals(0, cstmt.getObject(15, BigDecimal.class).compareTo(new BigDecimal("0.123456789")));
assertEquals(0,
cstmt.getObject("col15Value", BigDecimal.class).compareTo(new BigDecimal("0.1235")));
cstmt.getObject("col15Value", BigDecimal.class).compareTo(new BigDecimal("0.123456789")));

assertEquals(0, cstmt.getObject(16, BigDecimal.class).compareTo(new BigDecimal("0.1235")));
assertEquals(0, cstmt.getObject(16, BigDecimal.class).compareTo(new BigDecimal("0.1234567890123456789012345678901234567")));
assertEquals(0,
cstmt.getObject("col16Value", BigDecimal.class).compareTo(new BigDecimal("0.1235")));
cstmt.getObject("col16Value", BigDecimal.class).compareTo(new BigDecimal("0.1234567890123456789012345678901234567")));
}
}
}
Expand Down
Loading