From b4afdfb93567f8ffaa268ed63115bb9672a106aa Mon Sep 17 00:00:00 2001 From: blumf Date: Mon, 4 Jul 2016 11:41:26 +0100 Subject: [PATCH 1/4] Support for raw ODBC DSN connections. Usage: `env:connect{ dsn = [[...ODBC DNS...]] }` Also allows named parameters in general i.e. `env:connect{ source = "SomeODBCSource", user = "dbuser", password = "dbpass", }` --- src/ls_odbc.c | 128 +++++++++++++++++++++++++++++++++++++++++++++----- src/luasql.c | 30 ++++++++++++ src/luasql.h | 3 ++ 3 files changed, 148 insertions(+), 13 deletions(-) diff --git a/src/ls_odbc.c b/src/ls_odbc.c index 1b93d6c..af452e4 100644 --- a/src/ls_odbc.c +++ b/src/ls_odbc.c @@ -576,7 +576,6 @@ static int conn_setautocommit (lua_State *L) { return pass(L); } - /* ** Create a new Connection object and push it on top of the stack. */ @@ -601,32 +600,135 @@ static int create_connection (lua_State *L, int o, env_data *env, SQLHDBC hdbc) return 1; } +/* +** Uses a DSN string to connect to a ODBC source dynamically +** Lua Input: { +** dsn = , +** } +** Lua Returns: +** connection object if successfull +** nil and error message otherwise. +*/ +static int env_table_connect_DSN (lua_State *L) +{ + env_data *env = getenvironment (L); + SQLCHAR *sourcename = (SQLCHAR *)luasql_table_optstring(L, 2, "dsn", NULL); + + SQLHDBC hdbc; + SQLCHAR sqlOutBuf[4097]; + SQLSMALLINT sqlOutLen; + SQLRETURN ret; + + ret = SQLSetEnvAttr (env->henv, SQL_ATTR_ODBC_VERSION, (void *)SQL_OV_ODBC3, 0); + if (error(ret)) { + return luasql_faildirect (L, "error setting SQL version."); + } + + /* tries to allocate connection handle */ + ret = SQLAllocHandle (hDBC, env->henv, &hdbc); + if (error(ret)) { + return luasql_faildirect (L, "connection allocation error."); + } + + /* tries to connect handle */ + ret = SQLDriverConnect (hdbc, NULL, sourcename, SQL_NTS, sqlOutBuf, 4096, + &sqlOutLen, SQL_DRIVER_NOPROMPT); + if (error(ret)) { + ret = fail(L, hDBC, hdbc); + SQLFreeHandle(hDBC, hdbc); + return ret; + } + + /* success, return connection object */ + ret = create_connection (L, 1, env, hdbc); + if(ret == 1) { + /* Add the sqlOutBuf string to the results, for diagnostics */ + lua_pushlstring(L, (char *)sqlOutBuf, sqlOutLen); + return 2; + } + + return ret; +} /* -** Creates and returns a connection object -** Lua Input: source [, user [, pass]] +** Reforms old connection style to new one +** Lua Input: source, [user, [pass]] ** source: data source ** user, pass: data source authentication information ** Lua Returns: +** new connection details table +*/ +static void env_connect_fix_old (lua_State *L) +{ + static const char *const opt_names[] = { + "source", + "user", + "password", + NULL + }; + int i, t = lua_gettop(L)-1; + + lua_newtable(L); + lua_insert(L, 2); + + for(i=0; opt_names[i] != NULL && i, +** user = , +** password = , +** or +** dsn = , +** } +** Lua Returns: ** connection object if successfull ** nil and error message otherwise. */ -static int env_connect (lua_State *L) { - env_data *env = (env_data *) getenvironment (L); - SQLCHAR *sourcename = (SQLCHAR*)luaL_checkstring (L, 2); - SQLCHAR *username = (SQLCHAR*)luaL_optstring (L, 3, NULL); - SQLCHAR *password = (SQLCHAR*)luaL_optstring (L, 4, NULL); - SQLHDBC hdbc; - SQLRETURN ret; +static int env_connect (lua_State *L) +{ + env_data *env = getenvironment (L); + SQLCHAR *sourcename = NULL; + SQLCHAR *username = NULL; + SQLCHAR *password = NULL; + SQLHDBC hdbc = NULL; + SQLRETURN ret = 0; + + if(lua_gettop(L) < 2) { + return luasql_faildirect(L, "No connection details provided"); + } + + if(!lua_istable(L, 2)) { + env_connect_fix_old(L); + } + + /* check for the custom DSN connection string */ + if(luasql_table_optstring(L, 2, "dsn", NULL) != NULL) { + return env_table_connect_DSN(L); + } + + /* get the standard connection details */ + sourcename = (SQLCHAR *)luasql_table_optstring(L, 2, "source", NULL); + username = (SQLCHAR *)luasql_table_optstring(L, 2, "user", NULL); + password = (SQLCHAR *)luasql_table_optstring(L, 2, "password", NULL); /* tries to allocate connection handle */ ret = SQLAllocHandle (hDBC, env->henv, &hdbc); - if (error(ret)) + if (error(ret)) { return luasql_faildirect (L, "connection allocation error."); + } /* tries to connect handle */ - ret = SQLConnect (hdbc, sourcename, SQL_NTS, - username, SQL_NTS, password, SQL_NTS); + ret = SQLConnect (hdbc, sourcename, SQL_NTS, username, SQL_NTS, password, + SQL_NTS); if (error(ret)) { ret = fail(L, hDBC, hdbc); SQLFreeHandle(hDBC, hdbc); diff --git a/src/luasql.c b/src/luasql.c index caf9721..de17eba 100644 --- a/src/luasql.c +++ b/src/luasql.c @@ -131,3 +131,33 @@ LUASQL_API void luasql_set_info (lua_State *L) { lua_pushliteral (L, "LuaSQL 2.3.0"); lua_settable (L, -3); } + +/* +** Pulls an optional string value from the table at idx +*/ +LUASQL_API const char* luasql_table_optstring(lua_State *L, int idx, const char* name, const char* def) { + const char* res = NULL; + + lua_pushstring(L, name); + lua_gettable(L, idx); + + res = lua_tostring(L, -1); + lua_pop(L, 1); + + return (res != NULL) ? res : def; +} + +/* +** Pulls an optional number value from the table at idx +*/ +LUASQL_API lua_Number luasql_table_optnumber(lua_State *L, int idx, const char* name, lua_Number def) { + lua_Number res = 0; + + lua_pushstring(L, name); + lua_gettable(L, idx); + + res = lua_tonumber(L, -1); + lua_pop(L, 1); + + return lua_isnumber(L, -1) ? res : def; +} diff --git a/src/luasql.h b/src/luasql.h index 345bf57..acc17f7 100644 --- a/src/luasql.h +++ b/src/luasql.h @@ -30,6 +30,9 @@ LUASQL_API int luasql_createmeta (lua_State *L, const char *name, const luaL_Reg LUASQL_API void luasql_setmeta (lua_State *L, const char *name); LUASQL_API void luasql_set_info (lua_State *L); +LUASQL_API const char* luasql_table_optstring(lua_State *L, int idx, const char* name, const char* def); +LUASQL_API lua_Number luasql_table_optnumber(lua_State *L, int idx, const char* name, lua_Number def); + #if !defined LUA_VERSION_NUM || LUA_VERSION_NUM==501 void luaL_setfuncs (lua_State *L, const luaL_Reg *l, int nup); #endif From 61af46b14acb4c196e4d8f0de417abbeb226f291 Mon Sep 17 00:00:00 2001 From: blumf Date: Mon, 4 Jul 2016 14:05:08 +0100 Subject: [PATCH 2/4] BUG: Referencing value after it's popped off stack --- src/luasql.c | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/src/luasql.c b/src/luasql.c index de17eba..d87ce08 100644 --- a/src/luasql.c +++ b/src/luasql.c @@ -151,13 +151,15 @@ LUASQL_API const char* luasql_table_optstring(lua_State *L, int idx, const char* ** Pulls an optional number value from the table at idx */ LUASQL_API lua_Number luasql_table_optnumber(lua_State *L, int idx, const char* name, lua_Number def) { - lua_Number res = 0; + lua_Number res = def; lua_pushstring(L, name); lua_gettable(L, idx); - res = lua_tonumber(L, -1); + if(lua_isnumber(L, -1)) { + res = lua_tonumber(L, -1); + } lua_pop(L, 1); - return lua_isnumber(L, -1) ? res : def; + return res; } From 3772f3877df5121cfd7394280649d53d839eb241 Mon Sep 17 00:00:00 2001 From: blumf Date: Mon, 4 Jul 2016 17:08:22 +0100 Subject: [PATCH 3/4] Name changed for clarity --- src/ls_odbc.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/ls_odbc.c b/src/ls_odbc.c index af452e4..7b1f2cd 100644 --- a/src/ls_odbc.c +++ b/src/ls_odbc.c @@ -658,7 +658,7 @@ static int env_table_connect_DSN (lua_State *L) ** Lua Returns: ** new connection details table */ -static void env_connect_fix_old (lua_State *L) +static void env_connect_convert_args_to_table (lua_State *L) { static const char *const opt_names[] = { "source", @@ -707,7 +707,7 @@ static int env_connect (lua_State *L) } if(!lua_istable(L, 2)) { - env_connect_fix_old(L); + env_connect_convert_args_to_table(L); } /* check for the custom DSN connection string */ From cf4dbf5084f47fa6cb332192587855f1b40fbf46 Mon Sep 17 00:00:00 2001 From: blumf Date: Tue, 5 Jul 2016 11:23:43 +0100 Subject: [PATCH 4/4] Removed unused function --- src/luasql.c | 39 +++++++++++---------------------------- 1 file changed, 11 insertions(+), 28 deletions(-) diff --git a/src/luasql.c b/src/luasql.c index d87ce08..306ad2a 100644 --- a/src/luasql.c +++ b/src/luasql.c @@ -135,31 +135,14 @@ LUASQL_API void luasql_set_info (lua_State *L) { /* ** Pulls an optional string value from the table at idx */ -LUASQL_API const char* luasql_table_optstring(lua_State *L, int idx, const char* name, const char* def) { - const char* res = NULL; - - lua_pushstring(L, name); - lua_gettable(L, idx); - - res = lua_tostring(L, -1); - lua_pop(L, 1); - - return (res != NULL) ? res : def; -} - -/* -** Pulls an optional number value from the table at idx -*/ -LUASQL_API lua_Number luasql_table_optnumber(lua_State *L, int idx, const char* name, lua_Number def) { - lua_Number res = def; - - lua_pushstring(L, name); - lua_gettable(L, idx); - - if(lua_isnumber(L, -1)) { - res = lua_tonumber(L, -1); - } - lua_pop(L, 1); - - return res; -} +LUASQL_API const char* luasql_table_optstring(lua_State *L, int idx, const char* name, const char* def) { + const char* res = NULL; + + lua_pushstring(L, name); + lua_gettable(L, idx); + + res = lua_tostring(L, -1); + lua_pop(L, 1); + + return (res != NULL) ? res : def; +}