LCOV - code coverage report
Current view: top level - clients/odbc/driver - SQLDriverConnect.c (source / functions) Hit Total Coverage
Test: coverage.info Lines: 111 142 78.2 %
Date: 2025-03-24 23:16:36 Functions: 3 3 100.0 %

          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, 2025 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 (that's 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             :  * SQLDriverConnect()
      25             :  * CLI Compliance: ODBC (Microsoft)
      26             :  *
      27             :  * Author: Martin van Dinther, Sjoerd Mullender
      28             :  * Date  : 30 aug 2002
      29             :  *
      30             :  **********************************************************************/
      31             : 
      32             : #include "ODBCGlobal.h"
      33             : #include "ODBCDbc.h"
      34             : #include "ODBCUtil.h"
      35             : #include "ODBCAttrs.h"
      36             : #ifdef HAVE_STRINGS_H
      37             : #include <strings.h>              /* for strcasecmp */
      38             : #else
      39             : #include <string.h>
      40             : #endif
      41             : 
      42             : int
      43         576 : ODBCGetKeyAttr(const SQLCHAR **conn, SQLSMALLINT *nconn, char **key, char **attr)
      44             : {
      45         576 :         const SQLCHAR *p;
      46         576 :         size_t len;
      47             : 
      48         576 :         *key = *attr = NULL;
      49             : 
      50         576 :         p = *conn;
      51         576 :         if (!**conn)
      52             :                 return 0;
      53        1832 :         while (*nconn > 0 && **conn && **conn != '=' && **conn != ';') {
      54        1432 :                 (*conn)++;
      55        1432 :                 (*nconn)--;
      56             :         }
      57         400 :         if (*nconn == 0 || !**conn || **conn == ';')
      58             :                 return 0;
      59         399 :         len = *conn - p;
      60         399 :         *key = (char *) malloc(len + 1);
      61         399 :         if (*key == NULL)
      62             :                 return -1;
      63         399 :         strcpy_len(*key, (char *) p, len + 1);
      64         399 :         (*conn)++;
      65         399 :         (*nconn)--;
      66         399 :         p = *conn;
      67             : 
      68         399 :         if (*nconn > 0 && **conn == '{' && strcasecmp(*key, "DRIVER") == 0) {
      69          10 :                 (*conn)++;
      70          10 :                 (*nconn)--;
      71          10 :                 p++;
      72          80 :                 while (*nconn > 0 && **conn && **conn != '}') {
      73          70 :                         (*conn)++;
      74          70 :                         (*nconn)--;
      75             :                 }
      76          10 :                 len = *conn - p;
      77          10 :                 *attr = (char *) malloc(len + 1);
      78          10 :                 if (*attr == NULL) {
      79           0 :                         free(*key);
      80           0 :                         *key = NULL;
      81           0 :                         return -1;
      82             :                 }
      83          10 :                 strcpy_len(*attr, (char *) p, len + 1);
      84          10 :                 (*conn)++;
      85          10 :                 (*nconn)--;
      86             :                 /* should check that *nconn == 0 || **conn == ';' */
      87             :         } else {
      88        4951 :                 while (*nconn > 0 && **conn && **conn != ';') {
      89        4562 :                         (*conn)++;
      90        4562 :                         (*nconn)--;
      91             :                 }
      92         389 :                 len = *conn - p;
      93         389 :                 *attr = (char *) malloc(len + 1);
      94         389 :                 if (*attr == NULL) {
      95           0 :                         free(*key);
      96           0 :                         *key = NULL;
      97           0 :                         return -1;
      98             :                 }
      99         389 :                 strcpy_len(*attr, (char *) p, len + 1);
     100             :         }
     101         399 :         if (*nconn > 0 && **conn) {
     102         367 :                 (*conn)++;
     103         367 :                 (*nconn)--;
     104             :         }
     105             :         return 1;
     106             : }
     107             : 
     108             : #ifdef ODBCDEBUG
     109             : static char *
     110         169 : translateDriverCompletion(SQLUSMALLINT DriverCompletion)
     111             : {
     112         169 :         switch (DriverCompletion) {
     113             :         case SQL_DRIVER_PROMPT:
     114             :                 return "SQL_DRIVER_PROMPT";
     115           0 :         case SQL_DRIVER_COMPLETE:
     116           0 :                 return "SQL_DRIVER_COMPLETE";
     117           0 :         case SQL_DRIVER_COMPLETE_REQUIRED:
     118           0 :                 return "SQL_DRIVER_COMPLETE_REQUIRED";
     119         169 :         case SQL_DRIVER_NOPROMPT:
     120         169 :                 return "SQL_DRIVER_NOPROMPT";
     121           0 :         default:
     122           0 :                 return "unknown";
     123             :         }
     124             : }
     125             : #endif
     126             : 
     127             : SQLRETURN
     128         169 : MNDBDriverConnect(ODBCDbc *dbc,
     129             :                   SQLHWND WindowHandle,
     130             :                   const SQLCHAR *InConnectionString,
     131             :                   SQLSMALLINT StringLength1,
     132             :                   SQLCHAR *OutConnectionString,
     133             :                   SQLSMALLINT BufferLength,
     134             :                   SQLSMALLINT *StringLength2Ptr,
     135             :                   SQLUSMALLINT DriverCompletion,
     136             :                   int tryOnly)
     137             : {
     138         169 :         (void) WindowHandle;
     139             : 
     140         169 :         SQLRETURN rc = SQL_SUCCESS;
     141         169 :         const char *sqlstate = NULL;
     142         169 :         size_t out_len;
     143         169 :         const char *scratch_no_alloc;
     144             : 
     145             :         // These will be free'd at the end label
     146         169 :         msettings *settings = NULL;
     147         169 :         char *scratch_alloc = NULL;
     148         169 :         char *dsn = NULL;
     149             : 
     150             :         /* check connection state, should not be connected */
     151         169 :         if (dbc->Connected) {
     152           0 :                 sqlstate = "08002";
     153           0 :                 goto failure;
     154             :         }
     155             : 
     156         169 :         fixODBCstring(InConnectionString, StringLength1, SQLSMALLINT,
     157             :                       addDbcError, dbc, return SQL_ERROR);
     158             : 
     159         169 :         settings = msettings_clone(dbc->settings);
     160         169 :         if (!settings)
     161           0 :                 goto failure;
     162             : 
     163             : #ifdef ODBCDEBUG
     164         169 :         ODBCLOG("\"%.*s\" %s\n", StringLength1,
     165             :                 (char *) InConnectionString,
     166             :                 translateDriverCompletion(DriverCompletion));
     167             : #endif
     168             : 
     169             :         /* check input arguments */
     170         169 :         switch (DriverCompletion) {
     171             :         case SQL_DRIVER_PROMPT:
     172             :         case SQL_DRIVER_COMPLETE:
     173             :         case SQL_DRIVER_COMPLETE_REQUIRED:
     174             :         case SQL_DRIVER_NOPROMPT:
     175         169 :                 break;
     176           0 :         default:
     177             :                 /* Invalid attribute/option identifier */
     178           0 :                 sqlstate = "HY092";
     179           0 :                 goto failure;
     180             :         }
     181             : 
     182         169 :         rc = takeFromConnString(dbc, settings, InConnectionString, StringLength1, &dsn);
     183         169 :         if (!SQL_SUCCEEDED(rc))
     184           0 :                 goto end;
     185             : 
     186         169 :         scratch_no_alloc = msettings_validate(settings);
     187         169 :         if (scratch_no_alloc != NULL) {
     188           0 :                 addDbcError(dbc, "HY009", scratch_no_alloc, 0);
     189           0 :                 rc = SQL_ERROR;
     190           0 :                 goto end;
     191             :         }
     192             : 
     193             :         // Build a connect string for the current connection and put it in the out buffer.
     194         330 :         scratch_alloc = buildConnectionString(dsn ? dsn : "DEFAULT", settings);
     195         169 :         if (!scratch_alloc)
     196           0 :                 goto failure;
     197         169 :         out_len = strcpy_len((char*)OutConnectionString, scratch_alloc, BufferLength);
     198         169 :         if (StringLength2Ptr)
     199         169 :                 *StringLength2Ptr = (SQLSMALLINT)out_len;
     200         169 :         if (out_len + 1 > (size_t)BufferLength) {
     201          73 :                 addDbcError(dbc, "01004", NULL, 0);
     202          73 :                 rc = SQL_SUCCESS_WITH_INFO;
     203             :         }
     204             : 
     205         169 :         if (tryOnly) {
     206          73 :                 assert(sqlstate == NULL);
     207          73 :                 goto end;
     208             :         }
     209             : 
     210          96 :         scratch_no_alloc = msetting_string(settings, MP_LOGFILE);
     211          96 :         if (*scratch_no_alloc)
     212           0 :                 setODBCdebug(scratch_no_alloc, false);
     213             : 
     214          96 :         rc = MNDBConnectSettings(dbc, dsn, settings);
     215          96 :         settings = NULL; // do not free now
     216             : 
     217             :         // always go to end, MNDBConnectSettings has already logged any failures
     218          96 :         goto end;
     219             : 
     220           0 : failure:
     221           0 :         if (sqlstate == NULL)
     222           0 :                 sqlstate = "HY001"; // malloc failure
     223             :         rc = SQL_ERROR;
     224             :         // fallthrough
     225         169 : end:
     226         169 :         if (sqlstate != NULL)
     227           0 :                 addDbcError(dbc, sqlstate, NULL, 0);
     228         169 :         msettings_destroy(settings);
     229         169 :         free(scratch_alloc);
     230         169 :         free(dsn);
     231         169 :         return rc;
     232             : }
     233             : 
     234             : SQLRETURN SQL_API
     235             : SQLDriverConnect(SQLHDBC ConnectionHandle,
     236             :                  SQLHWND WindowHandle,
     237             :                  SQLCHAR *InConnectionString,
     238             :                  SQLSMALLINT StringLength1,
     239             :                  SQLCHAR *OutConnectionString,
     240             :                  SQLSMALLINT BufferLength,
     241             :                  SQLSMALLINT *StringLength2Ptr,
     242             :                  SQLUSMALLINT DriverCompletion)
     243             : {
     244          13 :         ODBCDbc *dbc = (ODBCDbc *) ConnectionHandle;
     245             : 
     246             : #ifdef ODBCDEBUG
     247          13 :         ODBCLOG("SQLDriverConnect %p ", ConnectionHandle);
     248             : #endif
     249             : 
     250          13 :         if (!isValidDbc(dbc))
     251             :                 return SQL_INVALID_HANDLE;
     252             : 
     253          13 :         clearDbcErrors(dbc);
     254             : 
     255          13 :         return MNDBDriverConnect(dbc,
     256             :                                  WindowHandle,
     257             :                                  InConnectionString,
     258             :                                  StringLength1,
     259             :                                  OutConnectionString,
     260             :                                  BufferLength,
     261             :                                  StringLength2Ptr,
     262             :                                  DriverCompletion,
     263             :                                  0);
     264             : }
     265             : 
     266             : SQLRETURN SQL_API
     267             : SQLDriverConnectA(SQLHDBC ConnectionHandle,
     268             :                   SQLHWND WindowHandle,
     269             :                   SQLCHAR *InConnectionString,
     270             :                   SQLSMALLINT StringLength1,
     271             :                   SQLCHAR *OutConnectionString,
     272             :                   SQLSMALLINT BufferLength,
     273             :                   SQLSMALLINT *StringLength2Ptr,
     274             :                   SQLUSMALLINT DriverCompletion)
     275             : {
     276           0 :         return SQLDriverConnect(ConnectionHandle,
     277             :                                 WindowHandle,
     278             :                                 InConnectionString,
     279             :                                 StringLength1,
     280             :                                 OutConnectionString,
     281             :                                 BufferLength,
     282             :                                 StringLength2Ptr,
     283             :                                 DriverCompletion);
     284             : }
     285             : 
     286             : SQLRETURN SQL_API
     287             : SQLDriverConnectW(SQLHDBC ConnectionHandle,
     288             :                   SQLHWND WindowHandle,
     289             :                   SQLWCHAR *InConnectionString,
     290             :                   SQLSMALLINT StringLength1,
     291             :                   SQLWCHAR *OutConnectionString,
     292             :                   SQLSMALLINT BufferLength,
     293             :                   SQLSMALLINT *StringLength2Ptr,
     294             :                   SQLUSMALLINT DriverCompletion)
     295             : {
     296          73 :         ODBCDbc *dbc = (ODBCDbc *) ConnectionHandle;
     297          73 :         SQLCHAR *in = NULL, *out;
     298          73 :         SQLSMALLINT n;
     299          73 :         SQLRETURN rc;
     300             : 
     301             : #ifdef ODBCDEBUG
     302          73 :         ODBCLOG("SQLDriverConnectW %p ", ConnectionHandle);
     303             : #endif
     304             : 
     305          73 :         if (!isValidDbc(dbc))
     306             :                 return SQL_INVALID_HANDLE;
     307             : 
     308          73 :         clearDbcErrors(dbc);
     309             : 
     310          73 :         fixWcharIn(InConnectionString, StringLength1, SQLCHAR, in,
     311             :                    addDbcError, dbc, return SQL_ERROR);
     312             : 
     313          73 :         rc = MNDBDriverConnect(dbc, WindowHandle, in, SQL_NTS, NULL, 0, &n,
     314             :                                DriverCompletion, 1);  // 1 = Try Only
     315          73 :         if (!SQL_SUCCEEDED(rc))
     316             :                 return rc;
     317          73 :         clearDbcErrors(dbc);
     318          73 :         n++;                    /* account for NUL byte */
     319          73 :         out = malloc(n);
     320          73 :         if (out == NULL) {
     321             :                 /* Memory allocation error */
     322           0 :                 addDbcError(dbc, "HY001", NULL, 0);
     323           0 :                 return SQL_ERROR;
     324             :         }
     325          73 :         rc = MNDBDriverConnect(dbc, WindowHandle, in, SQL_NTS, out, n, &n,
     326             :                                DriverCompletion, 0);
     327          73 :         if (SQL_SUCCEEDED(rc)) {
     328          72 :                 fixWcharOut(rc, out, n, OutConnectionString, BufferLength,
     329             :                             StringLength2Ptr, 1, addDbcError, dbc);
     330             :         }
     331          73 :         free(out);
     332          73 :         if (in)
     333          73 :                 free(in);
     334             :         return rc;
     335             : }

Generated by: LCOV version 1.14