LCOV - code coverage report
Current view: top level - src/odbc/unittests - transaction.c (source / functions) Hit Total Coverage
Test: FreeTDS coverage Lines: 43 46 93.5 %
Date: 2025-01-18 11:50:39 Functions: 2 2 100.0 %

          Line data    Source code
       1             : #include "common.h"
       2             : #include <freetds/bool.h>
       3             : 
       4             : static int
       5          16 : Test(bool discard_test)
       6             : {
       7          16 :         ODBC_BUF *odbc_buf = NULL;
       8             :         SQLINTEGER out_buf;
       9             :         SQLLEN out_len;
      10             :         SQLLEN rows;
      11          16 :         int retcode = 0;
      12             :         SQLTCHAR buf[512];
      13             :         SQLTCHAR sqlstate[6];
      14             : 
      15          16 :         const char *createErrorProcedure = "CREATE PROCEDURE testerror AS\n"
      16             :                 "SELECT value FROM TestTransaction\n" "SELECT value / (value-value) FROM TestTransaction\n";
      17             : 
      18             :         /* select after insert is required to test data discarding */
      19             :         char createProcedure[512];
      20             : 
      21          16 :         sprintf(createProcedure,
      22             :                 "CREATE PROCEDURE testinsert @value INT AS\n"
      23             :                 "INSERT INTO TestTransaction VALUES ( @value )\n%s", discard_test ? "SELECT * FROM TestTransaction\n" : "");
      24             : 
      25             :         /* create stored proc */
      26          16 :         odbc_command_with_result(odbc_stmt, "DROP PROCEDURE testinsert");
      27             : 
      28          16 :         odbc_command(createProcedure);
      29             : 
      30             :         /* create stored proc that generates an error */
      31          16 :         odbc_command_with_result(odbc_stmt, "DROP PROCEDURE testerror");
      32             : 
      33          16 :         odbc_command(createErrorProcedure);
      34             : 
      35             :         /* Start transaction */
      36          16 :         CHKSetConnectAttr(SQL_ATTR_AUTOCOMMIT, (void *) SQL_AUTOCOMMIT_OFF, 0, "S");
      37             : 
      38             :         /* Insert a value */
      39          16 :         odbc_command("EXEC testinsert 1");
      40             : 
      41             :         /* we should be able to read row count */
      42          16 :         CHKRowCount(&rows, "S");
      43             : 
      44             :         /* Commit transaction */
      45          16 :         CHKEndTran(SQL_HANDLE_DBC, odbc_conn, SQL_COMMIT, "S");
      46             : 
      47          16 :         SQLCloseCursor(odbc_stmt);
      48             : 
      49             :         /* Start transaction */
      50          16 :         CHKSetConnectAttr(SQL_ATTR_AUTOCOMMIT, (void *) SQL_AUTOCOMMIT_OFF, 0, "S");
      51             : 
      52             :         /* Insert another value */
      53          16 :         odbc_command("EXEC testinsert 2");
      54             : 
      55             :         /* Roll back transaction */
      56          16 :         CHKEndTran(SQL_HANDLE_DBC, odbc_conn, SQL_ROLLBACK, "S");
      57             : 
      58             :         /* TODO test row inserted */
      59             : 
      60          16 :         CHKSetConnectAttr(SQL_ATTR_AUTOCOMMIT, (void *) SQL_AUTOCOMMIT_ON, 0, "S");
      61             : 
      62             :         /* generate an error */
      63          16 :         odbc_command("EXEC testerror");
      64          16 :         CHKBindCol(1, SQL_C_SLONG, &out_buf, sizeof(out_buf), &out_len, "S");
      65             : 
      66          56 :         while (CHKFetch("SNo") == SQL_SUCCESS) {
      67          24 :                 printf("\t%ld\n", (long int) out_buf);
      68          24 :                 if (out_buf != 1) {
      69           0 :                         fprintf(stderr, "error: expected to select 1 got %ld\n", (long int) out_buf);
      70           0 :                         retcode = 1;
      71           0 :                         goto cleanup;
      72             :                 }
      73             :         }
      74             : 
      75          16 :         CHKMoreResults("E");
      76             : 
      77          16 :         CHKGetDiagRec(SQL_HANDLE_STMT, odbc_stmt, 1, sqlstate, NULL, buf, TDS_VECTOR_SIZE(buf), NULL, "SI");
      78          16 :         printf("err=%s\n", C(buf));
      79             : 
      80          16 :         CHKMoreResults("No");
      81             : 
      82          16 :       cleanup:
      83             :         /* drop table */
      84          16 :         odbc_command_with_result(odbc_stmt, "DROP PROCEDURE testinsert");
      85          16 :         odbc_command_with_result(odbc_stmt, "DROP PROCEDURE testerror");
      86             : 
      87          16 :         ODBC_FREE();
      88          16 :         return retcode;
      89             : }
      90             : 
      91             : int
      92           8 : main(void)
      93             : {
      94           8 :         int retcode = 0;
      95             : 
      96           8 :         odbc_connect();
      97             : 
      98             :         /* create table */
      99           8 :         odbc_command_with_result(odbc_stmt, "DROP TABLE TestTransaction");
     100           8 :         odbc_command("CREATE TABLE TestTransaction ( value INT )");
     101             : 
     102             :         if (!retcode)
     103           8 :                 retcode = Test(true);
     104           8 :         if (!retcode)
     105           8 :                 retcode = Test(false);
     106             : 
     107             :         /* drop table */
     108           8 :         odbc_command_with_result(odbc_stmt, "DROP TABLE TestTransaction");
     109             : 
     110           8 :         odbc_disconnect();
     111             : 
     112           8 :         printf("Done.\n");
     113             :         return retcode;
     114             : }

Generated by: LCOV version 1.13