From 90fa84bcfec3a8402f58b3a3cf2a0cc48cfda19f Mon Sep 17 00:00:00 2001 From: javiertuya <10879637+javiertuya@users.noreply.github.com> Date: Wed, 3 Jan 2024 20:44:46 +0100 Subject: [PATCH] Review .net ResultSet wrapper --- net/TdRules/Java.Sql/DriverManager.cs | 5 +- net/TdRules/Java.Sql/ResultSet.N.cs | 19 +-- net/TdRules/Java.Sql/SqliteConnection.N.cs | 7 -- .../Java.Sql/SqliteConnectionFactory.N.cs | 13 ++ .../TestSqliteResultSet.cs | 16 +++ .../TestSqlserverResultSet.cs | 118 ++++++++++++++++++ .../TestSqlserverSchemaMetadata.cs | 2 +- .../store/rdb/TestSqlserverResultSet.java | 113 +++++++++++++++++ .../rdb/TestSqlserverSchemaMetadata.java | 2 +- .../store/rdb/sqlite/TestSqliteResultSet.java | 11 ++ 10 files changed, 281 insertions(+), 25 deletions(-) delete mode 100644 net/TdRules/Java.Sql/SqliteConnection.N.cs create mode 100644 net/TdRules/Java.Sql/SqliteConnectionFactory.N.cs create mode 100644 net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb.Sqlite/TestSqliteResultSet.cs create mode 100644 net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverResultSet.cs create mode 100644 tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverResultSet.java create mode 100644 tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/sqlite/TestSqliteResultSet.java diff --git a/net/TdRules/Java.Sql/DriverManager.cs b/net/TdRules/Java.Sql/DriverManager.cs index 3c12562..602bef9 100644 --- a/net/TdRules/Java.Sql/DriverManager.cs +++ b/net/TdRules/Java.Sql/DriverManager.cs @@ -1,5 +1,4 @@ using System.Data.Common; -using System.Data.SqlClient; namespace Java.Sql { @@ -14,12 +13,12 @@ public static Connection GetConnection(string url, string user, string password) { DbConnection nativeConn; if (url.ToLower().StartsWith("server")) - nativeConn = new SqlConnection(url + nativeConn = new System.Data.SqlClient.SqlConnection(url + (string.IsNullOrEmpty(user) ? "" : ";UID=" + user) + (string.IsNullOrEmpty(password) ? "" : ";PWD=" + password) + ";MultipleActiveResultSets=true"); else if (url.ToLower().StartsWith("data source")) - nativeConn = new Microsoft.Data.Sqlite.SqliteConnection(url); + nativeConn = SqliteConnectionFactory.GetDbConnection(url); else throw new SQLException("Unrecognized connection string, it should start with 'Server' (SqlServer) or 'Data Source' (Sqlite), but was " + url); nativeConn.Open(); diff --git a/net/TdRules/Java.Sql/ResultSet.N.cs b/net/TdRules/Java.Sql/ResultSet.N.cs index ceab1bd..36f95de 100644 --- a/net/TdRules/Java.Sql/ResultSet.N.cs +++ b/net/TdRules/Java.Sql/ResultSet.N.cs @@ -102,35 +102,28 @@ public int GetInt(string field) object Value=GetCellValue(field); if (this.LastFieldWasNull) return 0; - return (int)Value; + return Convert.ToInt32(Value); } public int GetInt(int col) { object Value = GetCellValue(col); if (this.LastFieldWasNull) return 0; - return (int)Value; + return Convert.ToInt32(Value); } - public long GetLong(int col) - { - object Value = GetCellValue(col); - if (this.LastFieldWasNull) - return 0; - return (long)Value; - } - public short GetShort(string field) + public long GetLong(string field) { object Value = GetCellValue(field); if (this.LastFieldWasNull) return 0; - return short.Parse(Value.ToString()); //no permite cast short + return Convert.ToInt64(Value); } - public short GetShort(int col) + public long GetLong(int col) { object Value = GetCellValue(col); if (this.LastFieldWasNull) return 0; - return short.Parse(Value.ToString()); + return Convert.ToInt64(Value); } public bool GetBoolean(string field) { diff --git a/net/TdRules/Java.Sql/SqliteConnection.N.cs b/net/TdRules/Java.Sql/SqliteConnection.N.cs deleted file mode 100644 index 71f2a44..0000000 --- a/net/TdRules/Java.Sql/SqliteConnection.N.cs +++ /dev/null @@ -1,7 +0,0 @@ -//This is part of CORE.Net, don't edit outside this solution -namespace Java.Sql -{ - internal class SqliteConnection - { - } -} \ No newline at end of file diff --git a/net/TdRules/Java.Sql/SqliteConnectionFactory.N.cs b/net/TdRules/Java.Sql/SqliteConnectionFactory.N.cs new file mode 100644 index 0000000..0be9e11 --- /dev/null +++ b/net/TdRules/Java.Sql/SqliteConnectionFactory.N.cs @@ -0,0 +1,13 @@ +//This is part of CORE.Net, don't edit outside this solution +using System.Data.Common; + +namespace Java.Sql +{ + internal class SqliteConnectionFactory + { + public static DbConnection GetDbConnection(string url) + { + return new Microsoft.Data.Sqlite.SqliteConnection(url); + } + } +} \ No newline at end of file diff --git a/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb.Sqlite/TestSqliteResultSet.cs b/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb.Sqlite/TestSqliteResultSet.cs new file mode 100644 index 0000000..14caa6d --- /dev/null +++ b/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb.Sqlite/TestSqliteResultSet.cs @@ -0,0 +1,16 @@ +///////////////////////////////////////////////////////////////////////////////////////////// +/////// THIS FILE HAS BEEN AUTOMATICALLY CONVERTED FROM THE JAVA SOURCES. DO NOT EDIT /////// +///////////////////////////////////////////////////////////////////////////////////////////// +using Sharpen; +using Test4giis.Tdrules.Store.Rdb; + +namespace Test4giis.Tdrules.Store.Rdb.Sqlite +{ + public class TestSqliteResultSet : TestSqlserverResultSet + { + public TestSqliteResultSet() + { + this.dbmsname = "sqlite"; + } + } +} diff --git a/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverResultSet.cs b/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverResultSet.cs new file mode 100644 index 0000000..2d3afca --- /dev/null +++ b/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverResultSet.cs @@ -0,0 +1,118 @@ +///////////////////////////////////////////////////////////////////////////////////////////// +/////// THIS FILE HAS BEEN AUTOMATICALLY CONVERTED FROM THE JAVA SOURCES. DO NOT EDIT /////// +///////////////////////////////////////////////////////////////////////////////////////////// +using System; +using Giis.Portable.Util; +using Java.Sql; +using NUnit.Framework; +using Sharpen; + +namespace Test4giis.Tdrules.Store.Rdb +{ + /// Reading basic data types from a ResultSet, check nullability and access by name or column number. + /// + /// Reading basic data types from a ResultSet, check nullability and access by name or column number. + /// Intended to test the .net ResultSet wrapper implementation + /// + public class TestSqlserverResultSet : Base + { + protected internal Connection dbt; + + protected internal ResultSet rs; + + protected internal string sCreate1 = "create table tabrs1 (pk integer primary key not null, " + "tint integer, tchar varchar(16), tdate date, tlong bigint)"; + + protected internal string sSelect1 = "select pk, tint, tchar, tdate, tlong from tabrs1"; + + /// + [NUnit.Framework.SetUp] + public override void SetUp() + { + base.SetUp(); + dbt = GetConnection(TestDbname2); + ExecuteNotThrow(dbt, "drop table tabrs1"); + Execute(dbt, sCreate1); + } + + /// + [NUnit.Framework.TearDown] + public virtual void TearDown() + { + if (rs != null) + { + rs.Close(); + } + dbt.Close(); + } + + /// + [Test] + public virtual void TestReadResultSet() + { + Execute(dbt, "insert into tabrs1 (pk,tint,tchar,tdate,tlong) values(1, 2, null, '2023-01-02', null)"); + Execute(dbt, "insert into tabrs1 (pk,tint,tchar,tdate,tlong) values(2, null, 'chrchr', null, 55)"); + // 1st round: even columns are not null/odd columns are null (except pk), + // read by column number + rs = Query(dbt, sSelect1); + NUnit.Framework.Legacy.ClassicAssert.IsTrue(rs.Next()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(1, rs.GetInt(1)); + int tint = rs.GetInt(2); + NUnit.Framework.Legacy.ClassicAssert.IsFalse(rs.WasNull()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(2, tint); + NUnit.Framework.Legacy.ClassicAssert.AreEqual("2", rs.GetString(2)); + string tchar = rs.GetString(3); + NUnit.Framework.Legacy.ClassicAssert.IsTrue(rs.WasNull()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(null, tchar); + DateTime tdate; + if (!"sqlite".Equals(dbmsname)) + { + // sqlite does not have native Date + tdate = rs.GetDate(4); + NUnit.Framework.Legacy.ClassicAssert.IsFalse(rs.WasNull()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual("2023-01-02", DateString(tdate)); + // normalize Date for .net compatibility + NUnit.Framework.Legacy.ClassicAssert.IsNotNull(rs.GetString(4)); + } + // do not check value because is system/locale dependent + long tlong = rs.GetLong(5); + NUnit.Framework.Legacy.ClassicAssert.IsTrue(rs.WasNull()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(0, tlong); + // 2nd round: alternate nullability, read by column name + NUnit.Framework.Legacy.ClassicAssert.IsTrue(rs.Next()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(2, rs.GetInt("pk")); + tint = rs.GetInt("tint"); + NUnit.Framework.Legacy.ClassicAssert.IsTrue(rs.WasNull()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(0, tint); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(null, rs.GetString("tint")); + tchar = rs.GetString("tchar"); + NUnit.Framework.Legacy.ClassicAssert.IsFalse(rs.WasNull()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual("chrchr", tchar); + if (!"sqlite".Equals(dbmsname)) + { + tdate = rs.GetDate("tdate"); + NUnit.Framework.Legacy.ClassicAssert.IsTrue(rs.WasNull()); + if ("java".Equals(Platform)) + { + NUnit.Framework.Legacy.ClassicAssert.AreEqual(null, tdate); + } + else + { + // on .net, DateTime is no nullable, returns DateTime.MinValue + NUnit.Framework.Legacy.ClassicAssert.AreEqual("0001-01-01", DateString(tdate)); + } + NUnit.Framework.Legacy.ClassicAssert.AreEqual(null, rs.GetString("tdate")); + } + tlong = rs.GetLong("tlong"); + NUnit.Framework.Legacy.ClassicAssert.IsFalse(rs.WasNull()); + NUnit.Framework.Legacy.ClassicAssert.AreEqual(55, tlong); + NUnit.Framework.Legacy.ClassicAssert.AreEqual("55", rs.GetString("tlong")); + // resultset end + NUnit.Framework.Legacy.ClassicAssert.IsFalse(rs.Next()); + } + + private string DateString(DateTime dt) + { + return JavaCs.Substring(JavaCs.GetIsoDate(dt), 0, 10); + } + } +} diff --git a/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverSchemaMetadata.cs b/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverSchemaMetadata.cs index 84352e9..a4306a9 100644 --- a/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverSchemaMetadata.cs +++ b/net/TdRulesTest/Translated/Test4giis.Tdrules.Store.Rdb/TestSqlserverSchemaMetadata.cs @@ -173,7 +173,7 @@ protected internal virtual void AssertMetadata(string metadata, string fileName) FileUtil.FileWrite(TestPathOutput, fileName, metadata); string expected = FileUtil.FileRead(TestPathBenchmark, fileName); VisualAssert va = new VisualAssert(); - va.AssertEquals(expected, metadata); + va.AssertEquals(expected.Replace("\r", string.Empty), metadata.Replace("\r", string.Empty)); } /// List de tablas y/o vistas, discriminado por tipo diff --git a/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverResultSet.java b/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverResultSet.java new file mode 100644 index 0000000..05a1ca8 --- /dev/null +++ b/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverResultSet.java @@ -0,0 +1,113 @@ +package test4giis.tdrules.store.rdb; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertTrue; + +import java.sql.Connection; +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.Date; + +import org.junit.After; +import org.junit.Before; +import org.junit.Test; + +import giis.portable.util.JavaCs; + +/** + * Reading basic data types from a ResultSet, check nullability and access by name or column number. + * Intended to test the .net ResultSet wrapper implementation + */ +public class TestSqlserverResultSet extends Base { + protected Connection dbt; + protected ResultSet rs; + + protected String sCreate1 = "create table tabrs1 (pk integer primary key not null, " + + "tint integer, tchar varchar(16), tdate date, tlong bigint)"; + protected String sSelect1 = "select pk, tint, tchar, tdate, tlong from tabrs1"; + + @Before + public void setUp() throws SQLException { + super.setUp(); + dbt = getConnection(TEST_DBNAME2); + executeNotThrow(dbt, "drop table tabrs1"); + execute(dbt, sCreate1); + } + + @After + public void tearDown() throws SQLException { + if (rs != null) + rs.close(); + dbt.close(); + } + + @Test + public void testReadResultSet() throws SQLException { + execute(dbt, "insert into tabrs1 (pk,tint,tchar,tdate,tlong) values(1, 2, null, '2023-01-02', null)"); + execute(dbt, "insert into tabrs1 (pk,tint,tchar,tdate,tlong) values(2, null, 'chrchr', null, 55)"); + + // 1st round: even columns are not null/odd columns are null (except pk), + // read by column number + rs = query(dbt, sSelect1); + assertTrue(rs.next()); + assertEquals(1, rs.getInt(1)); + + int tint = rs.getInt(2); + assertFalse(rs.wasNull()); + assertEquals(2, tint); + assertEquals("2", rs.getString(2)); + + String tchar = rs.getString(3); + assertTrue(rs.wasNull()); + assertEquals(null, tchar); + + Date tdate; + if (!"sqlite".equals(dbmsname)) { // sqlite does not have native Date + tdate = rs.getDate(4); + assertFalse(rs.wasNull()); + assertEquals("2023-01-02", dateString(tdate)); // normalize Date for .net compatibility + assertNotNull(rs.getString(4)); // do not check value because is system/locale dependent + } + + long tlong = rs.getLong(5); + assertTrue(rs.wasNull()); + assertEquals(0, tlong); + + // 2nd round: alternate nullability, read by column name + assertTrue(rs.next()); + assertEquals(2, rs.getInt("pk")); + + tint = rs.getInt("tint"); + assertTrue(rs.wasNull()); + assertEquals(0, tint); + assertEquals(null, rs.getString("tint")); + + tchar = rs.getString("tchar"); + assertFalse(rs.wasNull()); + assertEquals("chrchr", tchar); + + if (!"sqlite".equals(dbmsname)) { + tdate = rs.getDate("tdate"); + assertTrue(rs.wasNull()); + if ("java".equals(PLATFORM)) + assertEquals(null, tdate); + else // on .net, DateTime is no nullable, returns DateTime.MinValue + assertEquals("0001-01-01", dateString(tdate)); + assertEquals(null, rs.getString("tdate")); + } + + tlong = rs.getLong("tlong"); + assertFalse(rs.wasNull()); + assertEquals(55, tlong); + assertEquals("55", rs.getString("tlong")); + + // resultset end + assertFalse(rs.next()); + } + + private String dateString(Date dt) { + return JavaCs.substring(JavaCs.getIsoDate(dt), 0, 10); + } +} diff --git a/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverSchemaMetadata.java b/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverSchemaMetadata.java index 3d492f5..4df7f47 100644 --- a/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverSchemaMetadata.java +++ b/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/TestSqlserverSchemaMetadata.java @@ -156,7 +156,7 @@ protected void assertMetadata(String metadata, String fileName) { FileUtil.fileWrite(TEST_PATH_OUTPUT, fileName, metadata); String expected = FileUtil.fileRead(TEST_PATH_BENCHMARK, fileName); VisualAssert va = new VisualAssert().setFramework(Framework.JUNIT4); - va.assertEquals(expected, metadata); + va.assertEquals(expected.replace("\r", ""), metadata.replace("\r", "")); } /** diff --git a/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/sqlite/TestSqliteResultSet.java b/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/sqlite/TestSqliteResultSet.java new file mode 100644 index 0000000..dc3bc77 --- /dev/null +++ b/tdrules-store-rdb/src/test/java/test4giis/tdrules/store/rdb/sqlite/TestSqliteResultSet.java @@ -0,0 +1,11 @@ +package test4giis.tdrules.store.rdb.sqlite; + +import test4giis.tdrules.store.rdb.TestSqlserverResultSet; + +public class TestSqliteResultSet extends TestSqlserverResultSet { + + public TestSqliteResultSet() { + this.dbmsname = "sqlite"; + } + +}