LCOV - code coverage report
Current view: top level - clients/odbc/driver - ODBCError.c (source / functions) Hit Total Coverage
Test: coverage.info Lines: 54 65 83.1 %
Date: 2024-04-26 00:35:57 Functions: 8 9 88.9 %

          Line data    Source code
       1             : /*
       2             :  * SPDX-License-Identifier: MPL-2.0
       3             :  *
       4             :  * This Source Code Form is subject to the terms of the Mozilla Public
       5             :  * License, v. 2.0.  If a copy of the MPL was not distributed with this
       6             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       7             :  *
       8             :  * Copyright 2024 MonetDB Foundation;
       9             :  * Copyright August 2008 - 2023 MonetDB B.V.;
      10             :  * Copyright 1997 - July 2008 CWI.
      11             :  */
      12             : 
      13             : /*
      14             :  * This code was created by Peter Harvey (mostly during Christmas 98/99).
      15             :  * This code is LGPL. Please ensure that this message remains in future
      16             :  * distributions and uses of this code (thats about all I get out of it).
      17             :  * - Peter Harvey pharvey@codebydesign.com
      18             :  *
      19             :  * This file has been modified for the MonetDB project.  See the file
      20             :  * Copyright in this directory for more information.
      21             :  */
      22             : 
      23             : /**********************************************
      24             :  * ODBCError.c
      25             :  *
      26             :  * Description:
      27             :  * This file contains the functions which operate on
      28             :  * ODBC error structures/objects (see ODBCError.h)
      29             :  *
      30             :  * Author: Martin van Dinther, Sjoerd Mullender
      31             :  * Date  : 30 aug 2002
      32             :  *
      33             :  **********************************************/
      34             : 
      35             : #include "ODBCGlobal.h"
      36             : #include "ODBCError.h"
      37             : 
      38             : struct tODBCError {
      39             :         char sqlState[SQL_SQLSTATE_SIZE + 1];   /* +1 for the string terminator */
      40             :         char *message;          /* pointer to the allocated error msg */
      41             :         int nativeErrorCode;
      42             : 
      43             :         struct tODBCError *next;        /* pointer to the next Error object or NULL */
      44             : };
      45             : 
      46             : const char ODBCErrorMsgPrefix[] = "[MonetDB][ODBC Driver " MONETDB_VERSION "]";
      47             : const int ODBCErrorMsgPrefixLength = (int) sizeof(ODBCErrorMsgPrefix) - 1;
      48             : 
      49             : /*
      50             :  * Local utility function which returns the standard ODBC/ISO error
      51             :  * message text for a given ISO SQLState code.
      52             :  * When no message could be found for a given SQLState a msg is
      53             :  * printed to stderr to warn that the programmer has forgotten to
      54             :  * add the message for the SQLState code.
      55             :  *
      56             :  * Precondition: SQLState is a valid string (non null, 5 chars long).
      57             :  * Postcondition: returns a valid pointer to a string (which may be empty).
      58             :  */
      59             : static struct SQLStateMsg {
      60             :         const char *SQLState;
      61             :         const char *SQLMsg;
      62             : } SQLStateMsg[] = {
      63             :         {"01000", "General warning"},
      64             :         {"01001", "Cursor operation conflict"},
      65             :         {"01002", "Disconnect error"},
      66             :         {"01003", "NULL value eliminated in set function"},
      67             :         {"01004", "String data, right truncated"},
      68             :         {"01006", "Privilege not revoked"},
      69             :         {"01007", "Privilege not granted"},
      70             :         {"01S00", "Invalid connection string attribute"},
      71             :         {"01S01", "Error in row"},
      72             :         {"01S02", "Option value changed"},
      73             :         {"01S06", "Attempt to fetch before the result set returned the first "
      74             :                   "rowset"},
      75             :         {"01S07", "Fractional truncation"},
      76             :         {"01S08", "Error saving file DSN"},
      77             :         {"01S09", "Invalid keyword"},
      78             :         {"07002", "COUNT field incorrect"},
      79             :         {"07005", "Prepared statement not a cursor-specification"},
      80             :         {"07006", "Restricted data type attribute violation"},
      81             :         {"07007", "Restricted parameter value violation"},
      82             :         {"07009", "Invalid descriptor index"},
      83             :         {"07S01", "Invalid use of default parameter"},
      84             :         {"08001", "Client unable to establish connection"},
      85             :         {"08002", "Connection name in use"},
      86             :         {"08003", "Connection not open"},
      87             :         {"08004", "Server rejected the connection"},
      88             :         {"08007", "Connection failure during transaction"},
      89             :         {"08S01", "Communication link failure"},
      90             :         {"0A000", "Feature not supported"},
      91             :         {"21S01", "Insert value list does not match column list"},
      92             :         {"21S02", "Degree of derived table does not match column list"},
      93             :         {"22001", "String data, right truncated"},
      94             :         {"22002", "Indicator variable required but not supplied"},
      95             :         {"22003", "Numeric value out of range"},
      96             :         {"22007", "Invalid datetime format"},
      97             :         {"22008", "Datetime field overflow"},
      98             :         {"22012", "Division by zero"},
      99             :         {"22015", "Interval field overflow"},
     100             :         {"22018", "Invalid character value for cast specification"},
     101             :         {"22019", "Invalid escape character"},
     102             :         {"22025", "Invalid escape sequence"},
     103             :         {"22026", "String data, length mismatch"},
     104             :         {"23000", "Integrity constraint violation"},
     105             :         {"24000", "Invalid cursor state"},
     106             :         {"25000", "Invalid transaction state"},
     107             :         {"25S01", "Transaction state unknown"},
     108             :         {"25S02", "Transaction is still active"},
     109             :         {"25S03", "Transaction is rolled back"},
     110             :         {"28000", "Invalid authorization specification"},
     111             :         {"34000", "Invalid cursor name"},
     112             :         {"3C000", "Duplicate cursor name"},
     113             :         {"3D000", "Invalid catalog name"},
     114             :         {"3F000", "Invalid schema name"},
     115             :         {"40001", "Serialization failure"},
     116             :         {"40002", "Integrity constraint violation"},
     117             :         {"40003", "Statement completion unknown"},
     118             :         {"42000", "Syntax error or access violation"},
     119             :         {"42S01", "Base table or view already exists"},
     120             :         {"42S02", "Base table or view not found"},
     121             :         {"42S11", "Index already exists"},
     122             :         {"42S12", "Index not found"},
     123             :         {"42S21", "Column already exists"},
     124             :         {"42S22", "Column not found"},
     125             :         {"44000", "WITH CHECK OPTION violation"},
     126             :         {"HY000", "General error"},
     127             :         {"HY001", "Memory allocation error"},
     128             :         {"HY003", "Invalid application buffer type"},
     129             :         {"HY004", "Invalid SQL data type"},
     130             :         {"HY007", "Associated statement is not prepared"},
     131             :         {"HY008", "Operation canceled"},
     132             :         {"HY009", "Invalid argument value"},
     133             :         {"HY010", "Function sequence error"},
     134             :         {"HY011", "Attribute cannot be set now"},
     135             :         {"HY012", "Invalid transaction operation code"},
     136             :         {"HY013", "Memory management error"},
     137             :         {"HY014", "Limit on the number of handles exceeded"},
     138             :         {"HY015", "No cursor name available"},
     139             :         {"HY016", "Cannot modify an implementation row descriptor"},
     140             :         {"HY017", "Invalid use of an automatically allocated descriptor "
     141             :                   "handle"},
     142             :         {"HY018", "Server declined cancel request"},
     143             :         {"HY019", "Non-character and non-binary data sent in pieces"},
     144             :         {"HY020", "Attempt to concatenate a null value"},
     145             :         {"HY021", "Inconsistent descriptor information"},
     146             :         {"HY024", "Invalid attribute value"},
     147             :         {"HY090", "Invalid string or buffer length"},
     148             :         {"HY091", "Invalid descriptor field identifier"},
     149             :         {"HY092", "Invalid attribute/option identifier"},
     150             :         {"HY095", "Function type out of range"},
     151             :         {"HY096", "Information type out of range"},
     152             :         {"HY097", "Column type out of range"},
     153             :         {"HY098", "Scope type out of range"},
     154             :         {"HY099", "Nullable type out of range"},
     155             :         {"HY100", "Uniqueness option type out of range"},
     156             :         {"HY101", "Accuracy option type out of range"},
     157             :         {"HY103", "Invalid retrieval code"},
     158             :         {"HY104", "Invalid precision or scale value"},
     159             :         {"HY105", "Invalid parameter type"},
     160             :         {"HY106", "Fetch type out of range"},
     161             :         {"HY107", "Row value out of range"},
     162             :         {"HY109", "Invalid cursor position"},
     163             :         {"HY110", "Invalid driver completion"},
     164             :         {"HY111", "Invalid bookmark value"},
     165             :         {"HY114", "Driver does not support connection-level asynchronous "
     166             :                   "function execution"},
     167             :         {"HY115", "SQLEndTran is not allowed for an environment that contains "
     168             :                   "a connection with asynchronous function execution enabled"},
     169             :         {"HY117", "Connection is suspended due to unknown transaction state.  "
     170             :                   "Only disconnect and read-only functions are allowed."},
     171             :         {"HY121", "Cursor Library and Driver-Aware Pooling cannot be enabled "
     172             :                   "at the same time"},
     173             :         {"HYC00", "Optional feature not implemented"},
     174             :         {"HYT00", "Timeout expired"},
     175             :         {"HYT01", "Connection timeout expired"},
     176             :         {"IM001", "Driver does not support this function"},
     177             :         {"IM002", "Data source not found and no default driver specified"},
     178             :         {"IM003", "Specified driver could not be connected to"},
     179             :         {"IM004", "Driver's SQLAllocHandle on SQL_HANDLE_ENV failed"},
     180             :         {"IM005", "Driver's SQLAllocHandle on SQL_HANDLE_DBC failed"},
     181             :         {"IM006", "Driver's SQLSetConnectAttr failed"},
     182             :         {"IM007", "No data source or driver specified; dialog prohibited"},
     183             :         {"IM008", "Dialog failed"},
     184             :         {"IM009", "Unable to connect to translation DLL"},
     185             :         {"IM010", "Data source name too long"},
     186             :         {"IM011", "Driver name too long"},
     187             :         {"IM012", "DRIVER keyword syntax error"},
     188             :         {"IM014", "The specified DSN contains an architecture mismatch "
     189             :                   "between the Driver and Application"},
     190             :         {"IM015", "Driver's SQLConnect on SQL_HANDLE_DBC_INFO_HANDLE failed"},
     191             :         {"IM017", "Polling is disabled in asynchronous notification mode"},
     192             :         {"IM018", "SQLCompleteAsync has not been called to complete the "
     193             :                   "previous asynchronous operation on this handle."},
     194             :         {"S1118", "Driver does not support asynchronous notification"},
     195             :         {0, 0}
     196             : };
     197             : 
     198             : #ifndef ODBCDEBUG
     199             : static
     200             : #endif
     201             : const char *
     202          15 : getStandardSQLStateMsg(const char *SQLState)
     203             : {
     204          15 :         struct SQLStateMsg *p;
     205             : 
     206          15 :         assert(SQLState);
     207             : 
     208         135 :         for (p = SQLStateMsg; p->SQLState; p++)
     209         135 :                 if (strncmp(p->SQLState, SQLState, 5) == 0)
     210          15 :                         return p->SQLMsg;
     211             : 
     212             :         /* Present a msg to notify the system administrator/programmer */
     213           0 :         fprintf(stderr,
     214             :                 "\nMonetDB, ODBC Driver, ODBCError.c: "
     215             :                 "No message defined for SQLState: %.5s. "
     216             :                 "Please report this error.\n", SQLState);
     217             : 
     218           0 :         return SQLState;        /* always return a string */
     219             : }
     220             : 
     221             : 
     222             : static ODBCError malloc_error = {
     223             :         "HY001",
     224             :         NULL,
     225             :         0,
     226             :         NULL,
     227             : };
     228             : 
     229             : /*
     230             :  * Creates a new allocated ODBCError object, initializes it and
     231             :  * adds copies of the SQLstate, msg and nativeErrorCode to the object.
     232             :  *
     233             :  * Precondition: none
     234             :  * Postcondition: returns a new ODBCError object
     235             :  */
     236             : ODBCError *
     237          64 : newODBCError(const char *SQLState, const char *msg, int nativeCode)
     238             : {
     239          64 :         ODBCError *error = (ODBCError *) malloc(sizeof(ODBCError));
     240             : 
     241          64 :         if (error == NULL) {
     242             :                 /* malloc failure, override anything given to us */
     243             :                 return &malloc_error;
     244             :         }
     245             : 
     246          64 :         *error = (ODBCError) {
     247             :                 .nativeErrorCode = nativeCode,
     248             :         };
     249             : 
     250          64 :         if (SQLState) {
     251          64 :                 strcpy_len(error->sqlState, SQLState, sizeof(error->sqlState));
     252             :         }
     253             : 
     254          64 :         if (msg) {
     255           1 :                 size_t len;
     256             : 
     257           1 :                 error->message = strdup(msg);
     258           1 :                 if (error->message == NULL) {
     259           0 :                         free(error);
     260           0 :                         return &malloc_error;
     261             :                 }
     262             : 
     263             :                 /* remove trailing newlines */
     264           1 :                 len = strlen(error->message);
     265           2 :                 while (len > 0 && error->message[len - 1] == '\n') {
     266           1 :                         error->message[--len] = 0;
     267             :                 }
     268             :         }
     269             : 
     270             :         return error;
     271             : }
     272             : 
     273             : 
     274             : 
     275             : /*
     276             :  * Get the SQL State code string.
     277             :  *
     278             :  * Precondition: error must be valid
     279             :  * Returns: a none NULL string pointer, intended for reading only.
     280             :  */
     281             : char *
     282          31 : getSqlState(ODBCError *error)
     283             : {
     284          31 :         assert(error);
     285          31 :         return error->sqlState;
     286             : }
     287             : 
     288             : 
     289             : /*
     290             :  * Get the Message string.
     291             :  *
     292             :  * Precondition: error must be valid
     293             :  * Returns: a string pointer, intended for reading only, which can be NULL !!.
     294             :  */
     295             : char *
     296          31 : getMessage(ODBCError *error)
     297             : {
     298          31 :         assert(error);
     299             : 
     300             :         /* check special case */
     301          31 :         if (error->message == NULL) {
     302             :                 /* No error message was set, use the default error msg
     303             :                    for the set sqlState (if a msg is available) */
     304          15 :                 const char *StandardSQLStateMsg = getStandardSQLStateMsg(error->sqlState);
     305             : 
     306          15 :                 assert(StandardSQLStateMsg);
     307             :                 /* use this message instead of the NULL */
     308          15 :                 error->message = strdup(StandardSQLStateMsg);
     309             :         }
     310             : 
     311          31 :         return error->message;
     312             : }
     313             : 
     314             : 
     315             : /*
     316             :  * Get the native error code value.
     317             :  *
     318             :  * Precondition: error must be valid
     319             :  * Returns: an int value representing the native (MonetDB) error code.
     320             :  */
     321             : int
     322          31 : getNativeErrorCode(ODBCError *error)
     323             : {
     324          31 :         assert(error);
     325          31 :         return error->nativeErrorCode;
     326             : }
     327             : 
     328             : 
     329             : /*
     330             :  * Get the pointer to the recNumber'th (starting at 1) ODBCError
     331             :  * object or NULL when there no next object.
     332             :  *
     333             :  * Precondition: error must be valid or NULL
     334             :  * Returns: the pointer to the next ODBCError object or NULL when
     335             :  * the record does not exist.
     336             :  */
     337             : ODBCError *
     338          47 : getErrorRec(ODBCError *error, int recNumber)
     339             : {
     340          63 :         while (error && --recNumber > 0)
     341          16 :                 error = error->next;
     342          47 :         return error;
     343             : }
     344             : 
     345             : int
     346           0 : getErrorRecCount(ODBCError *error)
     347             : {
     348           0 :         int n = 0;
     349             : 
     350           0 :         while (error) {
     351           0 :                 error = error->next;
     352           0 :                 n++;
     353             :         }
     354           0 :         return n;
     355             : }
     356             : 
     357             : /*
     358             :  * Appends a valid ODBCError object 'this' to the end of the list
     359             :  * of a valid ODBCError object 'head'.
     360             :  *
     361             :  * Precondition: both head and this must be valid (non NULL)
     362             :  */
     363             : void
     364          64 : appendODBCError(ODBCError **head, ODBCError *err)
     365             : {
     366          64 :         assert(head);
     367          64 :         assert(err);
     368             : 
     369          65 :         while (*head)
     370           1 :                 head = &(*head)->next;
     371          64 :         *head = err;
     372          64 :         err->next = NULL;    /* just to be sure */
     373          64 : }
     374             : 
     375             : 
     376             : #if 0                           /* unused */
     377             : /*
     378             :  * Prepends a valid ODBCError object 'err' to the front of the list
     379             :  * of a valid ODBCError object 'head' and return the new head.
     380             :  *
     381             :  * Precondition: both head and err must be valid (non NULL)
     382             :  * Returns: the new head (which is the same as the prepended 'err').
     383             :  */
     384             : void
     385             : prependODBCError(ODBCError **head, ODBCError *err)
     386             : {
     387             :         assert(head);
     388             :         assert(err);
     389             :         assert(err->next == NULL);
     390             : 
     391             :         err->next = *head;
     392             :         *head = err;
     393             : }
     394             : #endif
     395             : 
     396             : 
     397             : /*
     398             :  * Frees the ODBCError object including its linked ODBCError objects.
     399             :  *
     400             :  * Precondition: none (error may be NULL)
     401             :  */
     402             : void
     403         140 : deleteODBCErrorList(ODBCError **error)
     404             : {
     405         140 :         ODBCError *cur;
     406             : 
     407         204 :         while (*error) {
     408          64 :                 cur = *error;
     409          64 :                 *error = cur->next;
     410          64 :                 if (cur->message)
     411          16 :                         free(cur->message);
     412          64 :                 if (cur != &malloc_error)
     413          64 :                         free(cur);
     414             :                 else
     415           0 :                         cur->next = NULL;
     416             :         }
     417         140 : }

Generated by: LCOV version 1.14