Publicidad

Resultados 1 al 1 de 1
  1. #1
    [IG] [email protected] Avatar de Comunity
    Ingreso
    25 ago, 13
    Mensajes
    76
    País
    Gracias
    24
    52 gracias en 18 Posts

    Post Desarrollo MuOnline(GXSoft) Modulo 7 – Añadiendo conexión con SQL

    Modulo 7 – Añadiendo conexión con SQL


    Bueno en el último modulo al final prometi mostrar como instalar el SQL 2005 y si fue asi. Preferi hacerlo como videotutorial para mejor entendimiento de todos, al contrario de muchos videotutoriales que andan por ahí, yo narre la instalación explicando paso a paso. Sin más que decir siga el video de la instalación de SQL en el siguiente link.

    (Video en Portugues)
    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]


    Protocolos

    Vamos a comenzar la programacion y entender cada parte del código responsable de los protocolos de comunicación entre cliente> Servidor/Servidor >Cliente. En primer lugar, voy a explicar algunos puntos importantes que deben hacer la diferencia a la hora de elegir el protocolo que se utilizará, y tendrá que elegir.

    En el GameServer estamos desarrollando (1.00.90) su protocolo nativo es el JAPONES (JPN), por lo que si desea utilizar este protocolo, esta parte del tutorial puede saltarla porque, enseñare cómo cambiar un protocolo de comunicación entre GS> MAIN.

    Primero abra su proyecto Season4.sln, si usted ah estado siguiendome en el desarrollo, es decir, en el módulo 6, entonces debe haber creado el archivo Protocolo.cpp, si no lo tiene, creelo.

    Como se explicó anteriormente, hay que añadir los includes en la "cabecera" del archivo .cpp, entonces vamos a hacerlo:

    Código:
    #include "stdafx.h"
    #include "Includes.h"
    En el cuerpo del archivo debemos crear las funciones que entraran en el protocolo de comunicación:

    Código:
    bool ProtocolCore(BYTE protoNum,LPBYTE aRecv,int aLen,int aIndex,DWORD Encrypt,DWORD Serial)
     
    {
    OBJECTSTRUCT *gObj = (OBJECTSTRUCT*)OBJECT_POINTER(aIndex);
    OBJECTSTRUCT *ObjTable = (OBJECTSTRUCT*)OBJECT_BASE;
     
    BYTE ProtocolType = aRecv[0];
    switch(ProtocolType)
    {
    case 0xC1:
    switch(BYTE(protoNum))
    {
    case 0x11:
    {
    protoNum = 0xD7;
    aRecv[2] = 0xD7;
    }break;
    }
    }
    switch(protoNum)
    {
    case 0x00:
    {
    }break;
    case 0x01:
    {
    }break;
    case 0x03:
    {
    }break;
    case 0x26:
    {
    }break;
    case 0x30:
    {
    }break;
     
    case 0x1E:
    {
    }break;
    case 0xBC:
    {
    }break;
    case 0x95:
    {
    }break;
    }
    DataRecv(protoNum,aRecv,aLen,aIndex,Encrypt,Serial);
    return true;
    }
    Podemos ver que el código anterior, tiene una serie de case (casos), estos casos son donde insertamos las diversas funciones que vamos a utilizar durante el desarrollo, cada case es responsable de una acción, y estos cases son otra cosa que los paquetes recibida del cliente, y transmitida de otra manera al servidor. Por desgracia, no puedo entrar en detalles más específicos sobre el protocolo porque yo no sé lo suficiente, y no quiero información errónea para ti. Pero asegúrese de que todo lo que sé, voy a tratar de explicarlo de la mejor forma para todos.

    Al crear el código anterior en su proyecto, usted pudo notar que la línea:

    DataRecv (protoNum, aRecv, aLen, aIndex, Encrypt, Serial);

    está subrayada en rojo, significa que no hemos encontrado una referencia a esta funcion llamada DataRecv, que tiene 6 parámetros. Esta línea corresponde, es decir, modifica una función ya existente en el GameServer, así que tenemos que encontrar esta función en nuestro GameServer, y definirla en nuestro proyecto, informando tambien el offset donde se encontraba esta función en el GS. Para encontrar estos offsets tienen varias formas, una de ellas es sabiendo el lenguaje ASM (Assambly) interpretando un código y pasarlo al proyecto en C ++, pero hay métodos más fáciles, como por ejemplo tomar un GameServer donde ya existe esta función y comparar las funciones, hice un videotutorial que explica cómo hacer una búsqueda de offsets:

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    Bueno vamos a continuar, encontro el offset de DataRecv (o ProtocolCore), que es el offset: 0x004368E0, después hay que definir la función de acuerdo a su función original:

    Código:
     #define DataRecv ((void (*) (BYTE, PBYTE, DWORD DWORD, ...)) 0x004368E0)
    Ahora vamos a crear los paquetes, voy a proporcionar una lista de los paquetes que capturé, faltan algunos, pero en nuestro caso necesitamos solo tener conocimientos de los paquetes JPN y PHI, entonces no hay problema si falta alguna.

    Código:
    Packet JPN 
    0x1D 
    0xD6 
    0xDC 
    0xD7 
     
    Packet PHI 
    0xDD 
    0xDF 
    0xD6 
    0×11
    Lo que básicamente tenemos que hacer en el protocolo es: Cuando el cliente envíe un paquete PHI este lea e interprete JPN, y cuando el servidor envía un paquete JPN este lo convierta a PHI y envíe. Hay varias maneras de hacer esto, entonces vamos a crear el siguiente código dentro del archivo Protocolo.cpp al final del archivo:

    Código:
    BYTE ProtocolServer (BYTE Tipo)
    {
        switch(Tipo)
        {
        case 0x1D:
            return 0xDD;
     
        case 0xD6:
            return 0xDF;
     
        case 0xDC:
            return 0xD6;
     
        case 0xD7:
            return 0x11;
        }
        return Tipo;
    }
     
    BYTE ProtocolClient (BYTE Tipo)
    {
        switch(Tipo)
        {
        case 0xDD:
            return 0x1D;
     
        case 0xDF:
            return 0xD6;
     
        case 0xD6:
            return 0xDC;
     
        case 0x11:
            return 0xD7;
        }
        return Tipo;
    }
    El código anterior, sólo hizo dos funciones de condición, una para el cliente>Servidor y otra para el servidor>cliente, pero no tienen ningún efecto todavía.

    Vamos ahora a hacer el hook de estas dos funciones para que estas mismas puedas ser leidas al tener una solicitud o enviar datos entre cliente/servidor.

    Vamos a crear una función al final del archivo Protocolo.cpp con el nombre CambiarProtocolo, así:

    Código:
    void CambiarProtocolo()
    {
        SetByte(0x004B21C6,0x33);
        SetByte(0x004B21C7,0xD2);
        SetByte(0x004B21C8,0xE9);
        SetByte(0x004B21C9,0xC8);
        SetByte(0x004B21CA,0x09);
        SetByte(0x004B21CB,0x00);
        SetByte(0x004B21CC,0x00);
        SetByte(0x004B21CD,0x90);
        SetByte(0x004B21CE,0x90);
     
        SetByte(0x004B2B95,0x8A);
        SetByte(0x004B2B96,0x51);
        SetByte(0x004B2B97,0x02);
        SetByte(0x004B2B98,0x52);
        SetByte(0x004B2B99,0xFF);
        SetByte(0x004B2B9A,0x15);
        SetByte(0x004B2B9B,0x04);
        SetByte(0x004B2B9C,0x28);
        SetByte(0x004B2B9D,0x65);
        SetByte(0x004B2B9E,0x00);
        SetByte(0x004B2B9F,0x88);
        SetByte(0x004B2BA0,0x45);
        SetByte(0x004B2BA1,0xE4);
        SetByte(0x004B2BA2,0x8B);
        SetByte(0x004B2BA3,0x45);
        SetByte(0x004B2BA4,0xF0);
        SetByte(0x004B2BA5,0xE9);
        SetByte(0x004B2BA6,0x25);
        SetByte(0x004B2BA7,0xF6);
        SetByte(0x004B2BA8,0xFF);
        SetByte(0x004B2BA9,0xFF);
     
        SetByte(0x004C256B,0x33);
        SetByte(0x004C256C,0xC9);
        SetByte(0x004C256D,0x8A);
        SetByte(0x004C256E,0x4D);
        SetByte(0x004C256F,0x0C);
        SetByte(0x004C2570,0x51);
        SetByte(0x004C2571,0xFF);
        SetByte(0x004C2572,0x15);
        SetByte(0x004C2573,0x00);
        SetByte(0x004C2574,0x28);
        SetByte(0x004C2575,0x65);
        SetByte(0x004C2576,0x00);
        SetByte(0x004C2577,0x8B);
        SetByte(0x004C2578,0x75);
        SetByte(0x004C2579,0x08);
        SetByte(0x004C257A,0x88);
        SetByte(0x004C257B,0x46);
        SetByte(0x004C257C,0x02);
        SetByte(0x004C257D,0x5F);
        SetByte(0x004C257E,0x5E);
        SetByte(0x004C257F,0x5B);
        SetByte(0x004C2580,0x8B);
        SetByte(0x004C2581,0xE5);
        SetByte(0x004C2582,0x5D);
        SetByte(0x004C2583,0xC3);
     
        _asm
        {
     
            Mov Dword Ptr Ds:[00652800H],Offset ProtocolServer
                Mov Dword Ptr Ds:[00652804H],Offset ProtocolClient
        }
    }
    Bueno estos offsets que son seguidos por SetByte, no son más que fixs, el gameserver estamos trabajando, y cualquier otro gs 1.00.87 (desde el original), debemos hacer estas correcciones al protocolo para trabajar, esto no es nada más que el cambio de ubicación de los archivos y otras cosas. Tienes que hacer esto también en el GS_CS por lo tanto, ya has visto la mitad de camino otorgandoles la mitad de los offsets del GS normal. Sí viste bien, tenemos una declaración en ASM (assambly) para hacer un hook de las funciones que acabamos de hacer con anterioridad.

    Estas instrucciones no vienen mucho al caso, ya que es básicamente insertar el código en el GS cuando se solicita por el paquete.

    Con todo esto, ya tenemos el protocolo del servidor cambiado a PHI, pero es posible que note que no empezamos las funciones que creamos, entonces vamos a abrir el archivo Season4.cpp. El último cambio que debería haber hecho, fue la inclusión de hookthis como una función de ProtocolCore, bueno, después de este hookthis, vamos a añadir el siguiente código:

    CambiarProtocolo ();

    el archivo se verá así:

    Código:
    #include "StdAfx.h"
    #include "Includes.h"
     
    extern "C"__declspec(dllexport) void Init()
    {
        if(VirtualProtect(LPVOID(0x401000),3543191,PAGE_EXECUTE_READWRITE,&OldProtect))
        {
            HookThis((DWORD)&ProtocolCore, 0x004038BE);
            CambiarProtocolo();
        }
    }
    Pero es posible que note que la declaración CambiarProtocolo (); está subrayada en rojo ... Debido a que es este error? Simple, creamos la función, pero se olvidó de ponerlo en la biblioteca de nuestro proyecto, así que vamos a hacer esto, abra el archivo Includes.h y al final de este archivo colocamos la instrucción: void CambiarProtocolo();

    Vuelva a Season4.cpp y podra ver que la llamada para CambiarProtocolo se normalizo, compilamos nuestra DLL para ver si no hay error, vaya al Build>Build Solution. 100% compilado si el código es correcto.

    SQL Server + C++

    Como había prometido abordaré la conexión con la base de datos, para que podamos hacer que los códigos interactúen con la base de datos.

    Primero vamos a crear una nueva carpeta en el proyecto llamada SQL y dentro de esta carpeta los archivos:
    cSQL.cpp
    cSQL.h
    SQL.h

    Abrir el archivo SQL.h y agrega el siguiente código dentro:

    Código:
    //-----------------------------------------------------------------------------
    // File:            sql.h
    //
    // Copyright:       Copyright (c) Microsoft Corporation
    //
    // Contents:        This is the the main include for ODBC Core functions.
    //
    // Comments:        preconditions: #include "windows.h"
    //
    //-----------------------------------------------------------------------------
     
    #ifndef __SQL
    #define __SQL
     
    /*
    * ODBCVER  ODBC version number (0x0351).   To exclude
    *          definitions introduced in version 3.5 (or above)
    *          #define ODBCVER 0x0300 before #including <sql.h>
    */
    #ifndef ODBCVER
    #define ODBCVER 0x0351
    #endif
     
    #ifndef __SQLTYPES
    #include "sqltypes.h"
    #endif
     
    #ifdef __cplusplus
    extern "C" {            /* Assume C declarations for C++   */
    #endif  /* __cplusplus */
     
    /* special length/indicator values */
    #define SQL_NULL_DATA             (-1)
    #define SQL_DATA_AT_EXEC          (-2)
     
    /* return values from functions */
    #define SQL_SUCCESS                0
    #define SQL_SUCCESS_WITH_INFO      1
    #if (ODBCVER >= 0x0300)
    #define SQL_NO_DATA              100
    #endif
    #define SQL_ERROR                 (-1)
    #define SQL_INVALID_HANDLE        (-2)
     
    #define SQL_STILL_EXECUTING        2
    #define SQL_NEED_DATA             99
     
    /* test for SQL_SUCCESS or SQL_SUCCESS_WITH_INFO */
    #define SQL_SUCCEEDED(rc) (((rc)&(~1))==0)
     
    /* flags for null-terminated string */
    #define SQL_NTS                   (-3)
    #define SQL_NTSL                  (-3L)
     
    /* maximum message length */
    #define SQL_MAX_MESSAGE_LENGTH   512
     
    /* date/time length constants */
    #if (ODBCVER >= 0x0300)
    #define SQL_DATE_LEN           10
    #define SQL_TIME_LEN            8  /* add P+1 if precision is nonzero */
    #define SQL_TIMESTAMP_LEN      19  /* add P+1 if precision is nonzero */
    #endif
     
    /* handle type identifiers */
    #if (ODBCVER >= 0x0300)
    #define SQL_HANDLE_ENV             1
    #define SQL_HANDLE_DBC             2
    #define SQL_HANDLE_STMT            3
    #define SQL_HANDLE_DESC            4
    #endif
     
    /* environment attribute */
    #if (ODBCVER >= 0x0300)
    #define SQL_ATTR_OUTPUT_NTS    10001
    #endif
     
    /* connection attributes */
    #if (ODBCVER >= 0x0300)
    #define SQL_ATTR_AUTO_IPD      10001
    #define SQL_ATTR_METADATA_ID   10014
    #endif  /* ODBCVER >= 0x0300 */
     
    /* statement attributes */
    #if (ODBCVER >= 0x0300)
    #define SQL_ATTR_APP_ROW_DESC       10010
    #define SQL_ATTR_APP_PARAM_DESC     10011
    #define SQL_ATTR_IMP_ROW_DESC       10012
    #define SQL_ATTR_IMP_PARAM_DESC     10013
    #define SQL_ATTR_CURSOR_SCROLLABLE  (-1)
    #define SQL_ATTR_CURSOR_SENSITIVITY (-2)
    #endif
     
    /* SQL_ATTR_CURSOR_SCROLLABLE values */
    #if (ODBCVER >= 0x0300)
    #define SQL_NONSCROLLABLE           0
    #define SQL_SCROLLABLE              1
    #endif  /* ODBCVER >= 0x0300 */
     
    /* identifiers of fields in the SQL descriptor */
    #if (ODBCVER >= 0x0300)
    #define SQL_DESC_COUNT                  1001
    #define SQL_DESC_TYPE                   1002
    #define SQL_DESC_LENGTH                 1003
    #define SQL_DESC_OCTET_LENGTH_PTR       1004
    #define SQL_DESC_PRECISION              1005
    #define SQL_DESC_SCALE                  1006
    #define SQL_DESC_DATETIME_INTERVAL_CODE 1007
    #define SQL_DESC_NULLABLE               1008
    #define SQL_DESC_INDICATOR_PTR          1009
    #define SQL_DESC_DATA_PTR               1010
    #define SQL_DESC_NAME                   1011
    #define SQL_DESC_UNNAMED                1012
    #define SQL_DESC_OCTET_LENGTH           1013
    #define SQL_DESC_ALLOC_TYPE             1099
    #endif
     
    /* identifiers of fields in the diagnostics area */
    #if (ODBCVER >= 0x0300)
    #define SQL_DIAG_RETURNCODE        1
    #define SQL_DIAG_NUMBER            2
    #define SQL_DIAG_ROW_COUNT         3
    #define SQL_DIAG_SQLSTATE          4
    #define SQL_DIAG_NATIVE            5
    #define SQL_DIAG_MESSAGE_TEXT      6
    #define SQL_DIAG_DYNAMIC_FUNCTION  7
    #define SQL_DIAG_CLASS_ORIGIN      8
    #define SQL_DIAG_SUBCLASS_ORIGIN   9
    #define SQL_DIAG_CONNECTION_NAME  10
    #define SQL_DIAG_SERVER_NAME      11
    #define SQL_DIAG_DYNAMIC_FUNCTION_CODE 12
    #endif
     
    /* dynamic function codes */
    #if (ODBCVER >= 0x0300)
    #define SQL_DIAG_ALTER_DOMAIN           3
    #define SQL_DIAG_ALTER_TABLE            4
    #define SQL_DIAG_CALL                   7
    #define SQL_DIAG_CREATE_ASSERTION       6
    #define SQL_DIAG_CREATE_CHARACTER_SET   8
    #define SQL_DIAG_CREATE_COLLATION       10
    #define SQL_DIAG_CREATE_DOMAIN          23
    #define SQL_DIAG_CREATE_INDEX          (-1)
    #define SQL_DIAG_CREATE_SCHEMA          64
    #define SQL_DIAG_CREATE_TABLE          77
    #define SQL_DIAG_CREATE_TRANSLATION     79
    #define SQL_DIAG_CREATE_VIEW           84
    #define SQL_DIAG_DELETE_WHERE          19
    #define SQL_DIAG_DROP_ASSERTION         24
    #define SQL_DIAG_DROP_CHARACTER_SET     25
    #define SQL_DIAG_DROP_COLLATION         26
    #define SQL_DIAG_DROP_DOMAIN            27
    #define SQL_DIAG_DROP_INDEX            (-2)
    #define SQL_DIAG_DROP_SCHEMA            31
    #define SQL_DIAG_DROP_TABLE            32
    #define SQL_DIAG_DROP_TRANSLATION      33
    #define SQL_DIAG_DROP_VIEW             36
    #define SQL_DIAG_DYNAMIC_DELETE_CURSOR 38
    #define SQL_DIAG_DYNAMIC_UPDATE_CURSOR 81
    #define SQL_DIAG_GRANT                 48
    #define SQL_DIAG_INSERT                50
    #define SQL_DIAG_REVOKE                59
    #define SQL_DIAG_SELECT_CURSOR         85
    #define SQL_DIAG_UNKNOWN_STATEMENT      0
    #define SQL_DIAG_UPDATE_WHERE          82
    #endif  /* ODBCVER >= 0x0300 */
     
    /* SQL data type codes */
    #define SQL_UNKNOWN_TYPE    0
    #define SQL_CHAR            1
    #define SQL_NUMERIC         2
    #define SQL_DECIMAL         3
    #define SQL_INTEGER         4
    #define SQL_SMALLINT        5
    #define SQL_FLOAT           6
    #define SQL_REAL            7
    #define SQL_DOUBLE          8
    #if (ODBCVER >= 0x0300)
    #define SQL_DATETIME        9
    #endif
    #define SQL_VARCHAR        12
     
    /* One-parameter shortcuts for date/time data types */
    #if (ODBCVER >= 0x0300)
    #define SQL_TYPE_DATE      91
    #define SQL_TYPE_TIME      92
    #define SQL_TYPE_TIMESTAMP 93
    #endif
     
    /* Statement attribute values for cursor sensitivity */
    #if (ODBCVER >= 0x0300)
    #define SQL_UNSPECIFIED     0
    #define SQL_INSENSITIVE     1
    #define SQL_SENSITIVE       2
    #endif
     
    /* GetTypeInfo() request for all data types */
    #define SQL_ALL_TYPES       0
     
    /* Default conversion code for SQLBindCol(), SQLBindParam() and SQLGetData() */
    #if (ODBCVER >= 0x0300)
    #define SQL_DEFAULT        99
    #endif
     
    /* SQLSQLLEN GetData() code indicating that the application row descriptor
     * specifies the data type
     */
    #if (ODBCVER >= 0x0300)
    #define SQL_ARD_TYPE      (-99)
    #endif
     
    /* SQL date/time type subcodes */
    #if (ODBCVER >= 0x0300)
    #define SQL_CODE_DATE       1
    #define SQL_CODE_TIME       2
    #define SQL_CODE_TIMESTAMP  3
    #endif
     
    /* CLI option values */
    #if (ODBCVER >= 0x0300)
    #define SQL_FALSE           0
    #define SQL_TRUE            1
    #endif
     
    /* values of NULLABLE field in descriptor */
    #define SQL_NO_NULLS        0
    #define SQL_NULLABLE        1
     
    /* Value returned by SQLGetTypeInfo() to denote that it is
     * not known whether or not a data type supports null values.
     */
    #define SQL_NULLABLE_UNKNOWN  2
     
    /* Values returned by SQLGetTypeInfo() to show WHERE clause
     * supported
     */
    #if (ODBCVER >= 0x0300)
    #define SQL_PRED_NONE     0
    #define SQL_PRED_CHAR     1
    #define SQL_PRED_BASIC    2
    #endif
     
    /* values of UNNAMED field in descriptor */
    #if (ODBCVER >= 0x0300)
    #define SQL_NAMED           0
    #define SQL_UNNAMED         1
    #endif
     
    /* values of ALLOC_TYPE field in descriptor */
    #if (ODBCVER >= 0x0300)
    #define SQL_DESC_ALLOC_AUTO 1
    #define SQL_DESC_ALLOC_USER 2
    #endif
     
    /* FreeStmt() options */
    #define SQL_CLOSE           0
    #define SQL_DROP            1
    #define SQL_UNBIND          2
    #define SQL_RESET_PARAMS    3
     
    /* Codes used for FetchOrientation in SQLFetchScroll(),
       and in SQLDataSources()
    */
    #define SQL_FETCH_NEXT      1
    #define SQL_FETCH_FIRST     2
     
    /* Other codes used for FetchOrientation in SQLFetchScroll() */
    #define SQL_FETCH_LAST      3
    #define SQL_FETCH_PRIOR     4
    #define SQL_FETCH_ABSOLUTE  5
    #define SQL_FETCH_RELATIVE  6
     
    /* SQLEndTran() options */
    #define SQL_COMMIT          0
    #define SQL_ROLLBACK        1
     
    /* null handles returned by SQLAllocHandle() */
    #define SQL_NULL_HENV       0
    #define SQL_NULL_HDBC       0
    #define SQL_NULL_HSTMT      0
    #if (ODBCVER >= 0x0300)
    #define SQL_NULL_HDESC      0
    #endif
     
    /* null handle used in place of parent handle when allocating HENV */
    #if (ODBCVER >= 0x0300)
    #define SQL_NULL_HANDLE     0L
    #endif
     
    /* Values that may appear in the result set of SQLSpecialColumns() */
    #define SQL_SCOPE_CURROW    0
    #define SQL_SCOPE_TRANSACTION 1
    #define SQL_SCOPE_SESSION   2
     
    #define SQL_PC_UNKNOWN      0
    #if (ODBCVER >= 0x0300)
    #define SQL_PC_NON_PSEUDO   1
    #endif
    #define SQL_PC_PSEUDO       2
     
    /* Reserved value for the IdentifierType argument of SQLSpecialColumns() */
    #if (ODBCVER >= 0x0300)
    #define SQL_ROW_IDENTIFIER  1
    #endif
     
    /* Reserved values for UNIQUE argument of SQLStatistics() */
    #define SQL_INDEX_UNIQUE    0
    #define SQL_INDEX_ALL       1
     
    /* Values that may appear in the result set of SQLStatistics() */
    #define SQL_INDEX_CLUSTERED 1
    #define SQL_INDEX_HASHED    2
    #define SQL_INDEX_OTHER     3
     
    /* SQLGetFunctions() values to identify ODBC APIs */
    #define SQL_API_SQLALLOCCONNECT         1
    #define SQL_API_SQLALLOCENV             2
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLALLOCHANDLE       1001
    #endif
    #define SQL_API_SQLALLOCSTMT            3
    #define SQL_API_SQLBINDCOL              4
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLBINDPARAM         1002
    #endif
    #define SQL_API_SQLCANCEL               5
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLCLOSECURSOR       1003
    #define SQL_API_SQLCOLATTRIBUTE         6
    #endif
    #define SQL_API_SQLCOLUMNS             40
    #define SQL_API_SQLCONNECT              7
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLCOPYDESC          1004
    #endif
    #define SQL_API_SQLDATASOURCES         57
    #define SQL_API_SQLDESCRIBECOL          8
    #define SQL_API_SQLDISCONNECT           9
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLENDTRAN           1005
    #endif
    #define SQL_API_SQLERROR               10
    #define SQL_API_SQLEXECDIRECT          11
    #define SQL_API_SQLEXECUTE             12
    #define SQL_API_SQLFETCH               13
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLFETCHSCROLL       1021
    #endif
    #define SQL_API_SQLFREECONNECT         14
    #define SQL_API_SQLFREEENV             15
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLFREEHANDLE        1006
    #endif
    #define SQL_API_SQLFREESTMT            16
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLGETCONNECTATTR    1007
    #endif
    #define SQL_API_SQLGETCONNECTOPTION    42
    #define SQL_API_SQLGETCURSORNAME       17
    #define SQL_API_SQLGETDATA             43
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLGETDESCFIELD      1008
    #define SQL_API_SQLGETDESCREC        1009
    #define SQL_API_SQLGETDIAGFIELD      1010
    #define SQL_API_SQLGETDIAGREC        1011
    #define SQL_API_SQLGETENVATTR        1012
    #endif
    #define SQL_API_SQLGETFUNCTIONS        44
    #define SQL_API_SQLGETINFO             45
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLGETSTMTATTR       1014
    #endif
    #define SQL_API_SQLGETSTMTOPTION       46
    #define SQL_API_SQLGETTYPEINFO         47
    #define SQL_API_SQLNUMRESULTCOLS       18
    #define SQL_API_SQLPARAMDATA           48
    #define SQL_API_SQLPREPARE             19
    #define SQL_API_SQLPUTDATA             49
    #define SQL_API_SQLROWCOUNT            20
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLSETCONNECTATTR    1016
    #endif
    #define SQL_API_SQLSETCONNECTOPTION    50
    #define SQL_API_SQLSETCURSORNAME       21
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLSETDESCFIELD      1017
    #define SQL_API_SQLSETDESCREC        1018
    #define SQL_API_SQLSETENVATTR        1019
    #endif
    #define SQL_API_SQLSETPARAM            22
    #if (ODBCVER >= 0x0300)
    #define SQL_API_SQLSETSTMTATTR       1020
    #endif
    #define SQL_API_SQLSETSTMTOPTION       51
    #define SQL_API_SQLSPECIALCOLUMNS      52
    #define SQL_API_SQLSTATISTICS          53
    #define SQL_API_SQLTABLES              54
    #define SQL_API_SQLTRANSACT            23
     
    /* Information requested by SQLGetInfo() */
    #if (ODBCVER >= 0x0300)
    #define SQL_MAX_DRIVER_CONNECTIONS           0
    #define SQL_MAXIMUM_DRIVER_CONNECTIONS      SQL_MAX_DRIVER_CONNECTIONS
    #define SQL_MAX_CONCURRENT_ACTIVITIES        1
    #define SQL_MAXIMUM_CONCURRENT_ACTIVITIES   SQL_MAX_CONCURRENT_ACTIVITIES
    #endif
    #define SQL_DATA_SOURCE_NAME                 2
    #define SQL_FETCH_DIRECTION                  8
    #define SQL_SERVER_NAME                     13
    #define SQL_SEARCH_PATTERN_ESCAPE           14
    #define SQL_DBMS_NAME                       17
    #define SQL_DBMS_VER                        18
    #define SQL_ACCESSIBLE_TABLES               19
    #define SQL_ACCESSIBLE_PROCEDURES           20
    #define SQL_CURSOR_COMMIT_BEHAVIOR          23
    #define SQL_DATA_SOURCE_READ_ONLY           25
    #define SQL_DEFAULT_TXN_ISOLATION           26
    #define SQL_IDENTIFIER_CASE                 28
    #define SQL_IDENTIFIER_QUOTE_CHAR           29
    #define SQL_MAX_COLUMN_NAME_LEN             30
    #define SQL_MAXIMUM_COLUMN_NAME_LENGTH      SQL_MAX_COLUMN_NAME_LEN
    #define SQL_MAX_CURSOR_NAME_LEN             31
    #define SQL_MAXIMUM_CURSOR_NAME_LENGTH      SQL_MAX_CURSOR_NAME_LEN
    #define SQL_MAX_SCHEMA_NAME_LEN             32
    #define SQL_MAXIMUM_SCHEMA_NAME_LENGTH      SQL_MAX_SCHEMA_NAME_LEN
    #define SQL_MAX_CATALOG_NAME_LEN            34
    #define SQL_MAXIMUM_CATALOG_NAME_LENGTH     SQL_MAX_CATALOG_NAME_LEN
    #define SQL_MAX_TABLE_NAME_LEN              35
    #define SQL_SCROLL_CONCURRENCY              43
    #define SQL_TXN_CAPABLE                     46
    #define SQL_TRANSACTION_CAPABLE             SQL_TXN_CAPABLE
    #define SQL_USER_NAME                       47
    #define SQL_TXN_ISOLATION_OPTION            72
    #define SQL_TRANSACTION_ISOLATION_OPTION    SQL_TXN_ISOLATION_OPTION
    #define SQL_INTEGRITY                       73
    #define SQL_GETDATA_EXTENSIONS              81
    #define SQL_NULL_COLLATION                  85
    #define SQL_ALTER_TABLE                     86
    #define SQL_ORDER_BY_COLUMNS_IN_SELECT      90
    #define SQL_SPECIAL_CHARACTERS              94
    #define SQL_MAX_COLUMNS_IN_GROUP_BY         97
    #define SQL_MAXIMUM_COLUMNS_IN_GROUP_BY     SQL_MAX_COLUMNS_IN_GROUP_BY
    #define SQL_MAX_COLUMNS_IN_INDEX            98
    #define SQL_MAXIMUM_COLUMNS_IN_INDEX        SQL_MAX_COLUMNS_IN_INDEX
    #define SQL_MAX_COLUMNS_IN_ORDER_BY         99
    #define SQL_MAXIMUM_COLUMNS_IN_ORDER_BY     SQL_MAX_COLUMNS_IN_ORDER_BY
    #define SQL_MAX_COLUMNS_IN_SELECT          100
    #define SQL_MAXIMUM_COLUMNS_IN_SELECT      SQL_MAX_COLUMNS_IN_SELECT
    #define SQL_MAX_COLUMNS_IN_TABLE           101
    #define SQL_MAX_INDEX_SIZE                 102
    #define SQL_MAXIMUM_INDEX_SIZE             SQL_MAX_INDEX_SIZE
    #define SQL_MAX_ROW_SIZE                   104
    #define SQL_MAXIMUM_ROW_SIZE               SQL_MAX_ROW_SIZE
    #define SQL_MAX_STATEMENT_LEN              105
    #define SQL_MAXIMUM_STATEMENT_LENGTH       SQL_MAX_STATEMENT_LEN
    #define SQL_MAX_TABLES_IN_SELECT           106
    #define SQL_MAXIMUM_TABLES_IN_SELECT       SQL_MAX_TABLES_IN_SELECT
    #define SQL_MAX_USER_NAME_LEN              107
    #define SQL_MAXIMUM_USER_NAME_LENGTH       SQL_MAX_USER_NAME_LEN
    #if (ODBCVER >= 0x0300)
    #define SQL_OJ_CAPABILITIES                115
    #define SQL_OUTER_JOIN_CAPABILITIES        SQL_OJ_CAPABILITIES
    #endif /* ODBCVER >= 0x0300 */
     
    #if (ODBCVER >= 0x0300)
    #define SQL_XOPEN_CLI_YEAR               10000
    #define SQL_CURSOR_SENSITIVITY           10001
    #define SQL_DESCRIBE_PARAMETER           10002
    #define SQL_CATALOG_NAME                 10003
    #define SQL_COLLATION_SEQ                10004
    #define SQL_MAX_IDENTIFIER_LEN           10005
    #define SQL_MAXIMUM_IDENTIFIER_LENGTH    SQL_MAX_IDENTIFIER_LEN
    #endif /* ODBCVER >= 0x0300 */
     
    /* SQL_ALTER_TABLE bitmasks */
    #if (ODBCVER >= 0x0200)
    #define SQL_AT_ADD_COLUMN                       0x00000001L
    #define SQL_AT_DROP_COLUMN                      0x00000002L
    #endif /* ODBCVER >= 0x0200 */
     
    #if (ODBCVER >= 0x0300)
    #define SQL_AT_ADD_CONSTRAINT                   0x00000008L
     
    /* The following bitmasks are ODBC extensions and defined in sqlext.h
    *#define    SQL_AT_COLUMN_SINGLE                    0x00000020L
    *#define    SQL_AT_ADD_COLUMN_DEFAULT               0x00000040L
    *#define    SQL_AT_ADD_COLUMN_COLLATION             0x00000080L
    *#define    SQL_AT_SET_COLUMN_DEFAULT               0x00000100L
    *#define    SQL_AT_DROP_COLUMN_DEFAULT              0x00000200L
    *#define    SQL_AT_DROP_COLUMN_CASCADE              0x00000400L
    *#define    SQL_AT_DROP_COLUMN_RESTRICT             0x00000800L
    *#define SQL_AT_ADD_TABLE_CONSTRAINT                0x00001000L
    *#define SQL_AT_DROP_TABLE_CONSTRAINT_CASCADE       0x00002000L
    *#define SQL_AT_DROP_TABLE_CONSTRAINT_RESTRICT      0x00004000L
    *#define SQL_AT_CONSTRAINT_NAME_DEFINITION          0x00008000L
    *#define SQL_AT_CONSTRAINT_INITIALLY_DEFERRED       0x00010000L
    *#define SQL_AT_CONSTRAINT_INITIALLY_IMMEDIATE      0x00020000L
    *#define SQL_AT_CONSTRAINT_DEFERRABLE               0x00040000L
    *#define SQL_AT_CONSTRAINT_NON_DEFERRABLE           0x00080000L
    */
    #endif  /* ODBCVER >= 0x0300 */
     
    /* SQL_ASYNC_MODE values */
    #if (ODBCVER >= 0x0300)
    #define SQL_AM_NONE                         0
    #define SQL_AM_CONNECTION                   1
    #define SQL_AM_STATEMENT                    2
    #endif
     
    /* SQL_CURSOR_COMMIT_BEHAVIOR values */
    #define SQL_CB_DELETE                       0
    #define SQL_CB_CLOSE                        1
    #define SQL_CB_PRESERVE                     2
     
    /* SQL_FETCH_DIRECTION bitmasks */
    #define SQL_FD_FETCH_NEXT                   0x00000001L
    #define SQL_FD_FETCH_FIRST                  0x00000002L
    #define SQL_FD_FETCH_LAST                   0x00000004L
    #define SQL_FD_FETCH_PRIOR                  0x00000008L
    #define SQL_FD_FETCH_ABSOLUTE               0x00000010L
    #define SQL_FD_FETCH_RELATIVE               0x00000020L
     
    /* SQL_GETDATA_EXTENSIONS bitmasks */
    #define SQL_GD_ANY_COLUMN                   0x00000001L
    #define SQL_GD_ANY_ORDER                    0x00000002L
     
    /* SQL_IDENTIFIER_CASE values */
    #define SQL_IC_UPPER                        1
    #define SQL_IC_LOWER                        2
    #define SQL_IC_SENSITIVE                    3
    #define SQL_IC_MIXED                        4
     
    /* SQL_OJ_CAPABILITIES bitmasks */
    /* NB: this means 'outer join', not what  you may be thinking */
     
    #if (ODBCVER >= 0x0201)
    #define SQL_OJ_LEFT                         0x00000001L
    #define SQL_OJ_RIGHT                        0x00000002L
    #define SQL_OJ_FULL                         0x00000004L
    #define SQL_OJ_NESTED                       0x00000008L
    #define SQL_OJ_NOT_ORDERED                  0x00000010L
    #define SQL_OJ_INNER                        0x00000020L
    #define SQL_OJ_ALL_COMPARISON_OPS           0x00000040L
    #endif
     
    /* SQL_SCROLL_CONCURRENCY bitmasks */
    #define SQL_SCCO_READ_ONLY                  0x00000001L
    #define SQL_SCCO_LOCK                       0x00000002L
    #define SQL_SCCO_OPT_ROWVER                 0x00000004L
    #define SQL_SCCO_OPT_VALUES                 0x00000008L
     
    /* SQL_TXN_CAPABLE values */
    #define SQL_TC_NONE                         0
    #define SQL_TC_DML                          1
    #define SQL_TC_ALL                          2
    #define SQL_TC_DDL_COMMIT                   3
    #define SQL_TC_DDL_IGNORE                   4
     
    /* SQL_TXN_ISOLATION_OPTION bitmasks */
    #define SQL_TXN_READ_UNCOMMITTED            0x00000001L
    #define SQL_TRANSACTION_READ_UNCOMMITTED    SQL_TXN_READ_UNCOMMITTED
    #define SQL_TXN_READ_COMMITTED              0x00000002L
    #define SQL_TRANSACTION_READ_COMMITTED      SQL_TXN_READ_COMMITTED
    #define SQL_TXN_REPEATABLE_READ             0x00000004L
    #define SQL_TRANSACTION_REPEATABLE_READ     SQL_TXN_REPEATABLE_READ
    #define SQL_TXN_SERIALIZABLE                0x00000008L
    #define SQL_TRANSACTION_SERIALIZABLE        SQL_TXN_SERIALIZABLE
     
    /* SQL_NULL_COLLATION values */
    #define SQL_NC_HIGH                         0
    #define SQL_NC_LOW                          1
     
    #ifndef RC_INVOKED
     
    SQLRETURN  SQL_API SQLAllocConnect(SQLHENV EnvironmentHandle,
               SQLHDBC *ConnectionHandle);
     
    SQLRETURN  SQL_API SQLAllocEnv(SQLHENV *EnvironmentHandle);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLAllocHandle(SQLSMALLINT HandleType,
               SQLHANDLE InputHandle, SQLHANDLE *OutputHandle);
    #endif
     
    SQLRETURN  SQL_API SQLAllocStmt(SQLHDBC ConnectionHandle,
               SQLHSTMT *StatementHandle);
     
    SQLRETURN  SQL_API SQLBindCol(SQLHSTMT StatementHandle,
               SQLUSMALLINT ColumnNumber, SQLSMALLINT TargetType,
               SQLPOINTER TargetValue, SQLLEN BufferLength,
               SQLLEN *StrLen_or_Ind);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLBindParam(SQLHSTMT StatementHandle,
               SQLUSMALLINT ParameterNumber, SQLSMALLINT ValueType,
               SQLSMALLINT ParameterType, SQLULEN LengthPrecision,
               SQLSMALLINT ParameterScale, SQLPOINTER ParameterValue,
               SQLLEN *StrLen_or_Ind);
    #endif
     
    SQLRETURN  SQL_API SQLCancel(SQLHSTMT StatementHandle);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLCloseCursor(SQLHSTMT StatementHandle);
     
    #ifdef _WIN64
    SQLRETURN  SQL_API SQLColAttribute (SQLHSTMT StatementHandle,
               SQLUSMALLINT ColumnNumber, SQLUSMALLINT FieldIdentifier,
               SQLPOINTER CharacterAttribute, SQLSMALLINT BufferLength,
               SQLSMALLINT *StringLength, SQLLEN *NumericAttribute);
    #else
    SQLRETURN  SQL_API SQLColAttribute (SQLHSTMT StatementHandle,
               SQLUSMALLINT ColumnNumber, SQLUSMALLINT FieldIdentifier,
               SQLPOINTER CharacterAttribute, SQLSMALLINT BufferLength,
               SQLSMALLINT *StringLength, SQLPOINTER NumericAttribute);
    #endif
    #endif
     
    SQLRETURN  SQL_API SQLColumns(SQLHSTMT StatementHandle,
               SQLCHAR *CatalogName, SQLSMALLINT NameLength1,
               SQLCHAR *SchemaName, SQLSMALLINT NameLength2,
               SQLCHAR *TableName, SQLSMALLINT NameLength3,
               SQLCHAR *ColumnName, SQLSMALLINT NameLength4);
     
    SQLRETURN  SQL_API SQLConnect(SQLHDBC ConnectionHandle,
               SQLCHAR *ServerName, SQLSMALLINT NameLength1,
               SQLCHAR *UserName, SQLSMALLINT NameLength2,
               SQLCHAR *Authentication, SQLSMALLINT NameLength3);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLCopyDesc(SQLHDESC SourceDescHandle,
               SQLHDESC TargetDescHandle);
    #endif
     
    SQLRETURN  SQL_API SQLDataSources(SQLHENV EnvironmentHandle,
               SQLUSMALLINT Direction, SQLCHAR *ServerName,
               SQLSMALLINT BufferLength1, SQLSMALLINT *NameLength1,
               SQLCHAR *Description, SQLSMALLINT BufferLength2,
               SQLSMALLINT *NameLength2);
     
    SQLRETURN  SQL_API SQLDescribeCol(SQLHSTMT StatementHandle,
               SQLUSMALLINT ColumnNumber, SQLCHAR *ColumnName,
               SQLSMALLINT BufferLength, SQLSMALLINT *NameLength,
               SQLSMALLINT *DataType, SQLULEN *ColumnSize,
               SQLSMALLINT *DecimalDigits, SQLSMALLINT *Nullable);
     
    SQLRETURN  SQL_API SQLDisconnect(SQLHDBC ConnectionHandle);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLEndTran(SQLSMALLINT HandleType, SQLHANDLE Handle,
               SQLSMALLINT CompletionType);
    #endif
     
    SQLRETURN  SQL_API SQLError(SQLHENV EnvironmentHandle,
               SQLHDBC ConnectionHandle, SQLHSTMT StatementHandle,
               SQLCHAR *Sqlstate, SQLINTEGER *NativeError,
               SQLCHAR *MessageText, SQLSMALLINT BufferLength,
               SQLSMALLINT *TextLength);
     
    SQLRETURN  SQL_API SQLExecDirect
    (
        SQLHSTMT StatementHandle,
        __in_ecount_opt(TextLength) SQLCHAR* StatementText,
        SQLINTEGER TextLength
    );
     
    SQLRETURN  SQL_API SQLExecute(SQLHSTMT StatementHandle);
     
    SQLRETURN  SQL_API SQLFetch(SQLHSTMT StatementHandle);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLFetchScroll(SQLHSTMT StatementHandle,
               SQLSMALLINT FetchOrientation, SQLLEN FetchOffset);
    #endif
     
    SQLRETURN  SQL_API SQLFreeConnect(SQLHDBC ConnectionHandle);
     
    SQLRETURN  SQL_API SQLFreeEnv(SQLHENV EnvironmentHandle);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLFreeHandle(SQLSMALLINT HandleType, SQLHANDLE Handle);
    #endif
     
    SQLRETURN  SQL_API SQLFreeStmt(SQLHSTMT StatementHandle,
               SQLUSMALLINT Option);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLGetConnectAttr(SQLHDBC ConnectionHandle,
               SQLINTEGER Attribute, SQLPOINTER Value,
               SQLINTEGER BufferLength, SQLINTEGER *StringLength);
    #endif
     
    SQLRETURN  SQL_API SQLGetConnectOption(SQLHDBC ConnectionHandle,
               SQLUSMALLINT Option, SQLPOINTER Value);
     
    SQLRETURN  SQL_API SQLGetCursorName
    (
        SQLHSTMT StatementHandle,
        __out_ecount_opt(BufferLength) SQLCHAR *CursorName,
        SQLSMALLINT BufferLength,
        SQLSMALLINT *NameLength
    );
     
    SQLRETURN  SQL_API SQLGetData(SQLHSTMT StatementHandle,
               SQLUSMALLINT ColumnNumber, SQLSMALLINT TargetType,
               SQLPOINTER TargetValue, SQLLEN BufferLength,
               SQLLEN *StrLen_or_Ind);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLGetDescField(SQLHDESC DescriptorHandle,
               SQLSMALLINT RecNumber, SQLSMALLINT FieldIdentifier,
               SQLPOINTER Value, SQLINTEGER BufferLength,
               SQLINTEGER *StringLength);
     
    SQLRETURN  SQL_API SQLGetDescRec(SQLHDESC DescriptorHandle,
               SQLSMALLINT RecNumber, SQLCHAR *Name,
               SQLSMALLINT BufferLength, SQLSMALLINT *StringLength,
               SQLSMALLINT *Type, SQLSMALLINT *SubType,
               SQLLEN     *Length, SQLSMALLINT *Precision,
               SQLSMALLINT *Scale, SQLSMALLINT *Nullable);
     
    SQLRETURN  SQL_API SQLGetDiagField(SQLSMALLINT HandleType, SQLHANDLE Handle,
               SQLSMALLINT RecNumber, SQLSMALLINT DiagIdentifier,
               SQLPOINTER DiagInfo, SQLSMALLINT BufferLength,
               SQLSMALLINT *StringLength);
     
    SQLRETURN  SQL_API SQLGetDiagRec
    (
        SQLSMALLINT HandleType,
        SQLHANDLE Handle,
        SQLSMALLINT RecNumber,
        __out_ecount_opt(6) SQLCHAR *Sqlstate,
        SQLINTEGER *NativeError,
        __out_ecount_opt(BufferLength) SQLCHAR* MessageText,
        SQLSMALLINT BufferLength,
        SQLSMALLINT *TextLength
    );
     
    SQLRETURN  SQL_API SQLGetEnvAttr(SQLHENV EnvironmentHandle,
               SQLINTEGER Attribute, SQLPOINTER Value,
               SQLINTEGER BufferLength, SQLINTEGER *StringLength);
    #endif  /* ODBCVER >= 0x0300 */
     
    SQLRETURN  SQL_API SQLGetFunctions(SQLHDBC ConnectionHandle,
               SQLUSMALLINT FunctionId, SQLUSMALLINT *Supported);
     
    SQLRETURN  SQL_API SQLGetInfo(SQLHDBC ConnectionHandle,
               SQLUSMALLINT InfoType, __out_bcount_part(BufferLength, *StringLength) SQLPOINTER InfoValue,
               SQLSMALLINT BufferLength, SQLSMALLINT *StringLength);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLGetStmtAttr(SQLHSTMT StatementHandle,
               SQLINTEGER Attribute, SQLPOINTER Value,
               SQLINTEGER BufferLength, SQLINTEGER *StringLength);
    #endif  /* ODBCVER >= 0x0300 */
     
    SQLRETURN  SQL_API SQLGetStmtOption(SQLHSTMT StatementHandle,
               SQLUSMALLINT Option, SQLPOINTER Value);
     
    SQLRETURN  SQL_API SQLGetTypeInfo(SQLHSTMT StatementHandle,
               SQLSMALLINT DataType);
     
    SQLRETURN  SQL_API SQLNumResultCols(SQLHSTMT StatementHandle,
               SQLSMALLINT *ColumnCount);
     
    SQLRETURN  SQL_API SQLParamData(SQLHSTMT StatementHandle,
               SQLPOINTER *Value);
     
    SQLRETURN  SQL_API SQLPrepare
    (
        SQLHSTMT StatementHandle,
        __in_ecount(TextLength) SQLCHAR* StatementText,
        SQLINTEGER TextLength
    );
     
    SQLRETURN  SQL_API SQLPutData(SQLHSTMT StatementHandle,
               SQLPOINTER Data, SQLLEN StrLen_or_Ind);
     
    SQLRETURN  SQL_API SQLRowCount(__in SQLHSTMT StatementHandle,
                                   __out SQLLEN* RowCount);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLSetConnectAttr(SQLHDBC ConnectionHandle,
               SQLINTEGER Attribute, SQLPOINTER Value,
               SQLINTEGER StringLength);
    #endif /* ODBCVER >= 0x0300 */
     
    SQLRETURN  SQL_API SQLSetConnectOption(SQLHDBC ConnectionHandle,
               SQLUSMALLINT Option, SQLULEN Value);
     
    SQLRETURN  SQL_API SQLSetCursorName
    (
        SQLHSTMT StatementHandle,
        __in_ecount(NameLength) SQLCHAR* CursorName,
        SQLSMALLINT NameLength
    );
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLSetDescField(SQLHDESC DescriptorHandle,
               SQLSMALLINT RecNumber, SQLSMALLINT FieldIdentifier,
               SQLPOINTER Value, SQLINTEGER BufferLength);
     
    SQLRETURN  SQL_API SQLSetDescRec(SQLHDESC DescriptorHandle,
               SQLSMALLINT RecNumber, SQLSMALLINT Type,
               SQLSMALLINT SubType, SQLLEN Length,
               SQLSMALLINT Precision, SQLSMALLINT Scale,
               SQLPOINTER Data, SQLLEN *StringLength,
               SQLLEN *Indicator);
     
    SQLRETURN  SQL_API SQLSetEnvAttr(SQLHENV EnvironmentHandle,
               SQLINTEGER Attribute, SQLPOINTER Value,
               SQLINTEGER StringLength);
    #endif /* ODBCVER >= 0x0300 */
     
    SQLRETURN  SQL_API SQLSetParam(SQLHSTMT StatementHandle,
               SQLUSMALLINT ParameterNumber, SQLSMALLINT ValueType,
               SQLSMALLINT ParameterType, SQLULEN LengthPrecision,
               SQLSMALLINT ParameterScale, SQLPOINTER ParameterValue,
               SQLLEN *StrLen_or_Ind);
     
    #if (ODBCVER >= 0x0300)
    SQLRETURN  SQL_API SQLSetStmtAttr(SQLHSTMT StatementHandle,
               SQLINTEGER Attribute, SQLPOINTER Value,
               SQLINTEGER StringLength);
    #endif
     
    SQLRETURN  SQL_API SQLSetStmtOption(SQLHSTMT StatementHandle,
               SQLUSMALLINT Option, SQLULEN Value);
     
    SQLRETURN  SQL_API SQLSpecialColumns(SQLHSTMT StatementHandle,
               SQLUSMALLINT IdentifierType, SQLCHAR *CatalogName,
               SQLSMALLINT NameLength1, SQLCHAR *SchemaName,
               SQLSMALLINT NameLength2, SQLCHAR *TableName,
               SQLSMALLINT NameLength3, SQLUSMALLINT Scope,
               SQLUSMALLINT Nullable);
     
    SQLRETURN  SQL_API SQLStatistics(SQLHSTMT StatementHandle,
               SQLCHAR *CatalogName, SQLSMALLINT NameLength1,
               SQLCHAR *SchemaName, SQLSMALLINT NameLength2,
               SQLCHAR *TableName, SQLSMALLINT NameLength3,
               SQLUSMALLINT Unique, SQLUSMALLINT Reserved);
     
    SQLRETURN  SQL_API SQLTables(SQLHSTMT StatementHandle,
               SQLCHAR *CatalogName, SQLSMALLINT NameLength1,
               SQLCHAR *SchemaName, SQLSMALLINT NameLength2,
               SQLCHAR *TableName, SQLSMALLINT NameLength3,
               SQLCHAR *TableType, SQLSMALLINT NameLength4);
     
    SQLRETURN  SQL_API SQLTransact(SQLHENV EnvironmentHandle,
               SQLHDBC ConnectionHandle, SQLUSMALLINT CompletionType);
     
    #endif  /* RC_INVOKED */
     
    #ifdef __cplusplus
    }                                    /* End of extern "C" { */
    #endif  /* __cplusplus */
    #endif  /* #ifndef __SQL */
    Este código no lo explicare a fondo línea a línea, ya que es de la documentación de C++, es decir, no fue hecho por mi, es nativo de C++.

    Abra el archivo cSQL.cpp y coloque el siguiente código:

    Código:
    #include "Includes.h"
     
    CSQLEx MySQL;
     
    CSQLEx::CSQLEx()
    {
    }
     
    CSQLEx::~CSQLEx()
    {
        this->DisconnectCustom();
    }
     
    void CSQLEx::Load()
    {
        szDriver="{SQL Server}";
        szUser[11];
        szPassword[50];
        szDatabase[11];
        GetPrivateProfileStringA("Connection","SQLIP","(local)",szServer,22,IniConnect);
        GetPrivateProfileStringA("Connection","SQLIP2","(local)",szServer2,22,IniConnect);
        GetPrivateProfileStringA("Connection","SQLLogin","sa",szUser,11,IniConnect);
        GetPrivateProfileStringA("Connection","SQLPassword","123456",szPassword,50,IniConnect);
        GetPrivateProfileStringA("Connection","SQLDB","MuOnline",szDatabase,11,IniConnect);
     
        if(!this->ConnectCustom())
        {
            MessageBoxA(NULL,"Fallo de conexion!!\nRevisa 'GXGroup//GX_Connect.ini'","SQL Error!!",MB_OK);
            ::ExitProcess(0);
        }
    }
    bool CSQLEx::ConnectCustom()
    {
        if(this->m_bConnected == 1)
        {
            return false;
        }
     
        if(SQL_SUCCEEDED(SQLAllocHandle(SQL_HANDLE_ENV,SQL_NULL_HANDLE,&this->m_SQLEnvironment)) == 0)
        {
            return false;
        }
     
        if(SQL_SUCCEEDED(SQLSetEnvAttr(this->m_SQLEnvironment,SQL_ATTR_ODBC_VERSION,(void *)SQL_OV_ODBC3,0)) == 0)
        {
            this->FreeHandle();
     
            return false;
        }
     
        if(SQL_SUCCEEDED(SQLAllocHandle(SQL_HANDLE_DBC,this->m_SQLEnvironment,&this->m_SQLConnection)) == 0)
        {
            this->FreeHandle();
     
            return false;
        }
     
        SQLSetConnectOption(this->m_SQLConnection,SQL_LOGIN_TIMEOUT,5);
     
        SQLSetConnectOption(this->m_SQLConnection,SQL_CURSOR_TYPE,SQL_CURSOR_STATIC);
     
        SQLCHAR szConStrIn[1024],szConStrOut[1024];
        SQLSMALLINT iConOutSize = 0;
     
        sprintf((char*)szConStrIn,"DRIVER=%s; SERVER=%s; UID=%s; PWD=%s; DATABASE=%s;",szDriver,szServer,szUser,szPassword,szDatabase);
     
        SQLRETURN Connect = SQLDriverConnect(this->m_SQLConnection,NULL,szConStrIn,SQL_NTS,szConStrOut,sizeof(szConStrOut),&iConOutSize,SQL_DRIVER_NOPROMPT);
     
        if(SQL_SUCCEEDED(Connect) == 0)
        {
            this->FreeHandle();
     
            return false;
        }
     
        this->m_bConnected = 1;
     
        return true;
    }
     
    bool CSQLEx::ConnectExt()
    {
        if(this->m_bConnected == 1)
        {
            return false;
        }
     
        if(SQL_SUCCEEDED(SQLAllocHandle(SQL_HANDLE_ENV,SQL_NULL_HANDLE,&this->m_SQLEnvironment)) == 0)
        {
            return false;
        }
     
        if(SQL_SUCCEEDED(SQLSetEnvAttr(this->m_SQLEnvironment,SQL_ATTR_ODBC_VERSION,(void *)SQL_OV_ODBC3,0)) == 0)
        {
            this->FreeHandle();
     
            return false;
        }
     
        if(SQL_SUCCEEDED(SQLAllocHandle(SQL_HANDLE_DBC,this->m_SQLEnvironment,&this->m_SQLConnection)) == 0)
        {
            this->FreeHandle();
     
            return false;
        }
     
        SQLSetConnectOption(this->m_SQLConnection,SQL_LOGIN_TIMEOUT,5);
     
        SQLSetConnectOption(this->m_SQLConnection,SQL_CURSOR_TYPE,SQL_CURSOR_STATIC);
     
        SQLCHAR szConStrIn[1024],szConStrOut[1024];
        SQLSMALLINT iConOutSize = 0;
     
        sprintf((char*)szConStrIn,"DRIVER=%s; SERVER=%s; UID=%s; PWD=%s; DATABASE=%s;",szDriver,szServer,szUser,szPassword,szDatabase);
     
        SQLRETURN Connect = SQLDriverConnect(this->m_SQLConnection,NULL,szConStrIn,SQL_NTS,szConStrOut,sizeof(szConStrOut),&iConOutSize,SQL_DRIVER_NOPROMPT);
     
        if(SQL_SUCCEEDED(Connect) == 0)
        {
            this->FreeHandle();
     
            return false;
        }
     
        this->m_bConnected = 1;
     
        return true;
    }
     
    void CSQLEx::DisconnectCustom()
    {
        if(this->m_bConnected == 1)
        {
            SQLDisconnect(this->m_SQLConnection);
     
            this->FreeHandle();
        }
     
        this->m_SQLConnection = NULL;
        this->m_SQLEnvironment = NULL;
     
        this->m_bConnected = 0;
    }
     
    void CSQLEx::FreeHandle()
    {
        if(this->m_SQLConnection != NULL)
        {
            SQLFreeHandle(SQL_HANDLE_DBC,this->m_SQLConnection);
        }
     
        if(this->m_SQLEnvironment != NULL)
        {
            SQLFreeHandle(SQL_HANDLE_ENV,this->m_SQLEnvironment);
        }
    }
     
    bool CSQLEx::Execute(char* szQuery,...)
    {
        if(this->m_bConnected == 0)
        {
            return false;
        }
     
        if(SQL_SUCCEEDED(SQLAllocHandle(SQL_HANDLE_STMT,this->m_SQLConnection,&this->m_STMT)) == 0)
        {
            return false;
        }
     
        char szTemp[1024];
     
        va_list pArguments;
     
        va_start(pArguments,szQuery);
        vsprintf(szTemp,szQuery,pArguments);
        va_end(pArguments);
     
        if(SQL_SUCCEEDED(SQLPrepare(this->m_STMT,(unsigned char*)szTemp,strlen(szTemp))) == 0)
        {
            return false;
        }
     
        SQLRETURN Execute = SQLExecute(this->m_STMT);
     
        if(Execute == SQL_SUCCESS || Execute == SQL_SUCCESS_WITH_INFO || Execute == SQL_NO_DATA)
        {
            if(SQLRowCount(this->m_STMT,&this->m_RowCount) != SQL_SUCCESS)
            {
                this->m_RowCount = 0;
            }
     
            if(SQLNumResultCols(this->m_STMT,(SQLSMALLINT*)&this->m_ColCount) != SQL_SUCCESS)
            {
                this->m_ColCount = 0;
            }
     
            return true;
        }
     
        return false;
    }
     
    char* CSQLEx::GetLastError()
    {
        char szError[1024];
     
        SQLCHAR     szState[20];
        SQLINTEGER  iNativeError;
        SQLSMALLINT iOutErrorSize;
     
        SQLError(this->m_SQLEnvironment,this->m_SQLConnection,this->m_STMT,szState,&iNativeError,(SQLCHAR*)szError,sizeof(szError),&iOutErrorSize);
     
        return &szError[0];
    }
     
    int CSQLEx::GetRow()
    {
        return this->m_ColCount;
    }
     
    int CSQLEx::GetInt()
    {
        long lSize,lResult=0;
     
        while(true)
        {
            if(SQL_SUCCEEDED(SQLFetch(this->m_STMT)) == 0)
            {
                break;
            }
     
            if(SQL_SUCCEEDED(SQLGetData(this->m_STMT,1,SQL_C_LONG,&lResult,sizeof(long),&lSize)) == 0)
            {
                break;
            }
     
            return lResult;
        }
     
        return 0;
    }
     
    bool CSQLEx::FetchRow(char* out,int len)
    {
        long lSize;
     
        while(true)
        {
            if(SQL_SUCCEEDED(SQLFetch(this->m_STMT)) == 0)
            {
                break;
            }
     
            for(int n=0;n < this->m_ColCount;n++)
            {
                if(SQL_SUCCEEDED(SQLGetData(this->m_STMT,(n+1),SQL_C_CHAR,&out[n],len,&lSize)) == 0)
                {
                    break;
                }
            }
     
            return 1;
        }
     
        return 0;
    }
    int CSQLEx::FindCol(char *name)
    {
        int i;
        for (i=0;i<nCol;++i) {
            if (stricmp(name,(LPCTSTR)ColName[i])==0)
                return i+1;
        }
        return -1;
    }
    void CSQLEx::GetStrInt(int nCol, char *buf)
    {
        if (nCol > this->nCol) {
            //strcpy(buf, "CQUERYNOCOL");
            buf[0] = '\0';
            return;
        }
     
        if (lCol[nCol-1]==SQL_NULL_DATA)
        {
            buf[0] = '\0';
            //lstrcpy(buf,"NULL");
        } else {
            lstrcpy(buf,Col[nCol-1]);
        }
    }
    void CSQLEx::GetStr(char *sCol, char *buf)
    {
        int n;
        n=FindCol(sCol);
        if (n==-1) {
            //lstrcpy(buf,"ERROR:Columna no encontrada");
            buf[0] = '\0';
        } else {
            GetStrInt(n, buf);
        }
    }
    En este archivo tenemos una conexión con la base de datos, localhost, y también remota, así como las funciones de conectar y desconectar la DB

    En este archivo se puede ver que existen estas instrucciones:

    Código:
    GetPrivateProfileStringA("Connection","SQLIP","(local)",szServer,22,IniConnect);
    GetPrivateProfileStringA("Connection","SQLIP2","(local)",szServer2,22,IniConnect);
    GetPrivateProfileStringA("Connection","SQLLogin","sa",szUser,11,IniConnect);
    GetPrivateProfileStringA("Connection","SQLPassword","123456",szPassword,50,IniConnect);
    GetPrivateProfileStringA("Connection","SQLDB","MuOnline",szDatabase,11,IniConnect);
    GetPrivateProfileStringA – Esta function es responsible de buscar un valor en un grupo de valores de un archivo.ini. No voy a explicar por ahora.

    Después tenemos (en orden) los valores:
    SQLIP - IP del servidor
    SQLIP2 - IP2 del servidor
    SQLLogin - Usario de SQL
    SQLPassword - Contraseña de SQL
    SQLDB - Base de datos que se conectará

    Para que lo configure como quiera. Ejemplo en vez de SQLDB > DBSQL (Por ahora recomiendo dejarlo tal cual)

    Ahora abre el archivo cSQL.h y agrega el siguiente código:

    Código:
    #ifndef _SQL_H
    #define _SQL_H
     
    #include "SQL.h"
    #include <SqlExt.h>
    #include "StdAfx.h"
     
    class CSQLEx
    {
    public:
        CSQLEx();
        virtual ~CSQLEx();
     
        void LoadRemote();
        void Load();
        bool ConnectExt();
        bool ConnectCustom();
        void DisconnectCustom();
        bool Execute(char* szQuery,...);
        char* GetLastError();
        void FreeHandle();
     
        enum { MAXCOL=100, BLOBBATCH=10000, CQUERYNULL=-100, CQUERYEOF=-101,CQUERYNOCOL=-102, CQUERYERROR=-103 };
        char Col[MAXCOL][255];
        void GetStr(char *sCol, char *buf);
        int FindCol(char *name);
        void GetStrInt(int nCol, char *buf);
        SQLSMALLINT nCol;
        SQLCHAR ColName[MAXCOL][50];
        SQLINTEGER lCol[MAXCOL];
        SQLRETURN ret;
        SQLHSTMT hStmt;
        int GetRow();
     
        int GetInt();
        bool FetchRow(char* out,int len);
        char*szDriver;
        char szServer[22];
        char szServer2[22];
        char szUser[11];
        char szPassword[50];
        char szDatabase[11];
    private:
        bool m_bConnected;
        HENV m_SQLEnvironment;
        HDBC m_SQLConnection;
        HSTMT m_STMT;
        long m_RowCount;
        long m_ColCount;
    };
     
    extern CSQLEx MySQL;
     
    #endif
    El código anterior, representa todas las definiciones de las funciones creadas al archivo cSQL.cpp. Este código se hizo en classe, ya que no siempre se requiere, que SQL abra una conexión y cierre una conexión solamente, entonces el nivel de procesamiento de baja permite el uso de classes. Como ya se ha mensionado en otros módulos de la estructura de classes y cómo hacerlo, no voy a explicar esto ahora, vaya a los módulos anteriores que explique todo sobre class.

    Antes de construir la DLL, hay que hacer un paso importante que es necesario acostumbrarse... Incluya TODOS los archivos .h que creamos al archivo Includes.h, así que vamos a añadir los archivos que acabamos de crear al archivo Includes.h.

    Al compilar recibirá el siguiente error:
    Código:
    Error 2 Error C2065: 'IniConnect': identificador no declarado C:\Users \usuario \desktop \ Season4 \ Season4 \ csql.cpp 20 1 Season4 
    
    (La ruta puede no ser la misma)

    Este error se debe a que la línea en la que configuramos la ini, es decir, en GetPrivateProfileIntA, este último argumento, equivale a la ubicación + nombre del archivo de configuración. Usted tiene varias opciones para definir el nombre de archivo y la ubicación, pero me parece más organizado y eficaz, definir una variable como un nombre y un valor de cadena, así que vamos a hacer esto:

    Abra el Archivo Includes.h y una vez abierto después de #includes, vamos a añadir un área de definiciones, introduzca el siguiente código:

    Código:
    #define IniConnect "./Connect.ini"
    También voy a explicar cómo armar esta Ini, voy a dejar para más tarde, cuando abarquemos eso.

    Vamos a tratar de compilar de nuevo, y al hacerlo, podemos ver que la compilacion se ha realizado correctamente.

    Terminamos aquí un módulo más. En el siguiente módulo testearemos nuestro servidor por primera vez. Vale la pena mencionar que estudie mucho sobre la creación de funciones y las definiciones de las mismas en el archivo de biblioteca (.h)

    Abrazos a todos y buen estudio, las preguntas están disponibles.



    INDICE:

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]
    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]
    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]

    [Hola Invitado, necesitas registrarte para ver los links , Has click aquí para registrarte !!]
    Última edición por Comunity; 06/01/2015 a las 00:46 Razón: Add Index

  2.    Publicidad



 

Etiquetas para este tema

Marcadores

Permisos de publicación

  • No puedes crear nuevos temas
  • No puedes responder temas
  • No puedes subir archivos adjuntos
  • No puedes editar tus mensajes
  •