home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 5 / 05.iso / a / a025 / 6.ddi / SECRCBS.C < prev    next >
Encoding:
C/C++ Source or Header  |  1992-09-15  |  34.2 KB  |  1,256 lines

  1. // This module defines all of the "callback" functions.  These modules
  2. // are defined here due to the fact that they must be mult-threaded and
  3. // therefore completely reentrant.  They must be linked with the proper
  4. // C Runtime Multi-Threaded Library (LLIBCDLL.LIB).
  5.  
  6. #include    <stdlib.h>
  7. #include    <stdio.h>
  8. #include    <string.h>
  9.  
  10. #define     INCL_NETUSER
  11. #define     INCL_NETGROUP
  12. #define     INCL_NETERRORS
  13. #define     INCL_NETHANDLE
  14. #include    <lan.h>        // LAN Manager header files
  15.  
  16. #define INCL_BASE
  17. #include    <os2.h>
  18. #include    <sqlfront.h>
  19. #include    <sqldb.h>
  20. #include    <srv.h>
  21.  
  22. // Misc. defines
  23. //
  24. #define BUFFLEN        128
  25.  
  26. // Define some user message codes.
  27. //
  28. #define SRV_MAXERROR    20000
  29. #define RPC_UNKNOWN    SRV_MAXERROR + 1
  30. #define COMPUTE_ROW    SRV_MAXERROR + 2
  31. #define REMOTE_MSG        SRV_MAXERROR + 3
  32. #define SEND_FAILED    SRV_MAXERROR + 4
  33. #define BULK_CMD        "insert bulk"
  34.  
  35. // This error value must be sent back to the client in the event
  36. // of a failure to login to the target database.  It is used in the
  37. // init_remote() function below.
  38. //
  39. #define REMOTE_FAIL 4002
  40.  
  41. // Declare the structure to use for our private data area.
  42. // It will be passed to the event handlers in the SRV_PROC structure.
  43. // It contains information to make and use a connection to the remote
  44. // DBMS -- in this case a SQL Server.
  45. //
  46. // A gateway to a non-SQL Server DBMS would contain a different structure.
  47. //
  48. #define    MAXPARAMS    255
  49. typedef struct remote_dbms {
  50.     LOGINREC FAR *login;        // The SQL Server login structure
  51.     DBPROCESS FAR *dbproc;        // The connection to the remote SQL Server
  52.     BOOL bulk_mode;                // Flag indicating "bulk insert" mode
  53.     int retparams[MAXPARAMS];    // tag return parameters
  54. } REMOTE_DBMS;
  55.  
  56. // The remote server name of this gateway
  57. //
  58. DBCHAR *remote_server = NULL;
  59. SRV_PROC *Newsrvproc;            // Latest SRV_PROC allocated.
  60. ULONG init_remote_SEM = 0L;    // Semaphore used init_remote()
  61.  
  62. // Flags declared in SECURE.C for turning functions on or off
  63. //
  64. BOOL fLogEnabled;           // request logging
  65. BOOL fConnectCheck;         // LM connection verification
  66.  
  67. // Prototype dblib API's
  68. //
  69. void dbsetuserdata(DBPROCESS FAR *dbproc, void FAR *data);
  70.  
  71. void FAR *dbgetuserdata(DBPROCESS FAR *dbproc);
  72.  
  73. int dbcolntype(DBPROCESS FAR *dbproc, int column);
  74.  
  75. int attn_handler(SRV_PROC FAR *srvproc);
  76.  
  77. int chk_err(SRV_SERVER FAR *server, 
  78.              SRV_PROC FAR *srvproc, 
  79.              int srverror,
  80.              BYTE severity, 
  81.              BYTE state, 
  82.              int oserrnum, 
  83.              DBCHAR FAR *errtext,
  84.              int errtextlen, 
  85.              DBCHAR FAR *oserrtext, 
  86.              int oserrtextlen);
  87.              
  88. int init_remote(SRV_PROC FAR *srvproc);
  89.  
  90. int init_server(SRV_SERVER FAR *server);
  91.  
  92. int rpc_execute(SRV_PROC FAR *srvproc);
  93.  
  94. int exit_remote(SRV_PROC FAR *srvproc);
  95.  
  96. int lang_execute(SRV_PROC FAR *srvproc);
  97.  
  98. int handle_results(DBPROCESS FAR *rmtproc, SRV_PROC FAR *srvproc);
  99.  
  100. int remotemsgs(DBPROCESS FAR *dbproc, 
  101.                 DBINT msgno, 
  102.                 DBSMALLINT msgstate,
  103.                 DBSMALLINT severity, 
  104.                 char FAR *msgtext);
  105.  
  106. int remoteerr(DBPROCESS FAR *dbproc, 
  107.                int severity, 
  108.                int dberr, 
  109.                int oserr,
  110.                char FAR *dberrstr, 
  111.                char FAR *oserrstr);
  112.  
  113. void set_remote_server_name(char FAR *name);
  114.  
  115. void set_security_options(BOOL logflag, BOOL checkflag);
  116.  
  117. //
  118. // This section defines the "call-back" functions supplied to the gateway
  119. // application.
  120. //
  121. #pragma check_stack( off )  // turn off stack checking
  122.  
  123. //
  124. // SET_REMOTE_SERVER_NAME
  125. //
  126. //       This function sets the name of the remote server.
  127. //
  128. // Parameters:
  129. //       name - Pointer to name of server.
  130. //
  131. // Returns:
  132. //       none
  133. //
  134. void set_remote_server_name(name)
  135. char FAR *name;
  136. {
  137.     remote_server = name;
  138. }
  139.  
  140. // SET_SECURITY_OPTIONS
  141. //    This function sets connection checking and request logging on or off.
  142. //
  143. // Parameters:
  144. //    logflag;    // request logging on or off
  145. //    checkflag;  // LM connection verification on or off
  146. //
  147. // Returns:
  148. //    none
  149. //
  150. void set_security_options(logflag, checkflag)
  151. BOOL logflag;
  152. BOOL checkflag;
  153. {
  154.     fLogEnabled = logflag;
  155.     fConnectCheck = checkflag;
  156. }
  157.  
  158. //
  159. // INIT_SERVER
  160. //       Initialize the server on a SRV_START event.
  161. //       Event handlers for the server are installed.
  162. //
  163. // Parameters:
  164. //       server - Pointer to SRV_SERVER structure
  165. //
  166. // Returns:
  167. //       SRV_CONTINUE
  168. //
  169. int init_server(server)
  170. SRV_SERVER FAR *server;
  171. {
  172.     char log_buffer[256];
  173.  
  174.     // When we get a connection request from a client, we want to
  175.     // call "init_remote()" to make a connection to the remote
  176.     // server.
  177.     //
  178.     srv_handle(server, (DBINT)SRV_CONNECT, init_remote);
  179.  
  180.     // When the client issues a language request, call
  181.     // "lang_execute()" to send the SQL statement to the remote DBMS.
  182.     //
  183.     srv_handle(server, (DBINT)SRV_LANGUAGE, lang_execute);
  184.  
  185.     // When the client issues an RSP, call "rpc_execute()"
  186.     // to send the RSP to the remote DBMS (the SQL Server).
  187.     //
  188.     srv_handle(server, (DBINT)SRV_RPC, rpc_execute);
  189.  
  190.     // When a disconnect request is issued, call "exit_remote()"
  191.     // to close the connection to the remote DBMS.
  192.     //
  193.     srv_handle(server, (DBINT)SRV_DISCONNECT, exit_remote);
  194.  
  195.     // Install the handler that will be called when the
  196.     // gateway receives an attention from one of its
  197.     // clients. An attention gets received anytime a client
  198.     // calls dbcancel().
  199.     //
  200.     srv_handle(server, (DBINT)SRV_ATTENTION, attn_handler);
  201.  
  202.     // Now we'll install the message and error handlers for any
  203.     // messages from the remote DBMS.
  204.     //
  205.     dberrhandle(remoteerr);
  206.     dbmsghandle(remotemsgs);
  207.  
  208.     //    Log Server information to log file
  209.     //
  210.     sprintf(log_buffer, "Server pipe name = %s", 
  211.             srv_sfield(server, SRV_SERVERNAME, (int *)NULL));
  212.  
  213.     srv_log(server, FALSE, log_buffer, SRV_NULLTERM);
  214.     printf("%s\n", log_buffer);
  215.  
  216.     sprintf(log_buffer, "Client connections allowed = %s",
  217.             srv_sfield(server, SRV_CONNECTIONS, (int *)NULL));
  218.  
  219.     srv_log(server, FALSE, log_buffer, SRV_NULLTERM);
  220.     printf("%s\n", log_buffer);
  221.  
  222.     return(SRV_CONTINUE);
  223. }
  224.  
  225. // INIT_REMOTE
  226. //    Event handler for a SRV_CONNECT event.
  227. //    A connection is made to the remote DBMS.
  228. //
  229. //    In this example program, the connection is to a SQL Server.
  230. //    If using a non-SQL Server remote DBMS, the connection code
  231. //    would be different, but would probably still occur here.
  232. //
  233. // Parameters:
  234. //        srvproc - the handle to the client connection that got the SRV_CONNECT.
  235. //
  236. // Returns:
  237. //       SRV_CONTINUE
  238. //
  239. // Side Effects:
  240. //       If the connection to the remote dbms cannot be made, then issue
  241. //       a SRV_DISCONNECT request.
  242. //
  243. //
  244. int init_remote(srvproc)
  245. SRV_PROC FAR *srvproc;
  246. {
  247.     char *string;
  248.     int len;
  249.     REMOTE_DBMS FAR *remote;        // the private data area to keep
  250.                                     // track of the remote DBMS
  251.                                     // connection.
  252.    char achBuffer[BUFFLEN];    // Data buffer
  253.    unsigned short hPipe;       // Handle to named pipe
  254.    unsigned short cbAvail;     // Bytes available from GetInfo
  255.    API_RET_TYPE uReturnCode;   // LAN Manager API return code
  256.    struct handle_info_2 *pHandleInfo2;
  257.    DBINT uCompareResult;       // result of string compare
  258.    char log_buffer[256];
  259.    SRV_SERVER FAR *server;     // pointer to server info for log
  260.  
  261.    // Lan Manager security verification
  262.    //   
  263.    if (fConnectCheck == TRUE) {
  264.        // Query to see who is on the other end of the pipe
  265.        //
  266.        hPipe = (unsigned short)srvproc->srvio.handle;
  267.  
  268.        uReturnCode = NetHandleGetInfo(hPipe,   // Handle to named pipe
  269.            2,              // Level 2
  270.            achBuffer,      // Data returned here
  271.            BUFFLEN,        // Size of buffer, in bytes
  272.            (unsigned short far *)&cbAvail);
  273.  
  274.        if (uReturnCode == NERR_Success) {
  275.            pHandleInfo2 = (struct handle_info_2 *)achBuffer;
  276.            printf("   User of the named pipe is %Fs\n",
  277.                pHandleInfo2->hdli2_username);
  278.            printf("   Login record name is %s\n", srv_pfield(srvproc, SRV_USER,
  279.                                                            (int *)NULL));
  280.        } else {
  281.            printf("  Net HandleGetInfo failed, return code %d", uReturnCode);
  282.            server = SRV_GETSERVER(srvproc);
  283.            srv_log(server, TRUE, "Call to LanMan failed", SRV_NULLTERM);
  284.            return(SRV_DISCONNECT);
  285.        }
  286.  
  287.        // Compare login name with handle user name.  If they are different,
  288.        // reject the login.
  289.        //
  290.        uCompareResult = strcmpi(pHandleInfo2->hdli2_username,
  291.        srv_pfield(srvproc, SRV_USER, (int *)NULL));
  292.        if (uCompareResult != 0) {
  293.            // Send a message to the client that
  294.            // they must connect with their network login.
  295.            //
  296.            srv_sendmsg(srvproc, SRV_MSG_ERROR, (DBINT)REMOTE_FAIL,
  297.                        (DBTINYINT)0, (DBTINYINT)0, NULL, 0, 0,
  298.                        "Login id must match network user id.", SRV_NULLTERM);
  299.  
  300.            srv_senddone(srvproc, SRV_DONE_FINAL | SRV_DONE_ERROR,
  301.                        (DBUSMALLINT)0, (DBINT)0);
  302.  
  303.            sprintf(log_buffer,
  304.                    "LOGIN REJECTED: username = %s, login request = %s",
  305.                    pHandleInfo2->hdli2_username,
  306.                    srv_pfield(srvproc, SRV_USER, (int *)NULL));
  307.  
  308.            server = SRV_GETSERVER(srvproc);
  309.            srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
  310.            return(SRV_DISCONNECT);
  311.        }
  312.    }
  313.  
  314.     // Set Newsrvproc.  This is used if we get an error on
  315.     // the open from DBLIB.  Since there isn't a dbproc,
  316.     // it is clear which srvproc to send the msg back on when the
  317.     // DBLIB error-handler gets called.
  318.     //
  319.     // First lock out any other threads trying to connect using
  320.     // this same function.
  321.     //
  322.     DosSemRequest(&init_remote_SEM, -1L);
  323.  
  324.     Newsrvproc = srvproc;
  325.  
  326.     // Allocate a REMOTE_DBMS information structure.
  327.     //
  328.     remote = (REMOTE_DBMS *)srv_alloc((DBINT)sizeof(*remote));
  329.  
  330.     // Try to open a connection to the remote DBMS.
  331.     //
  332.     if (remote == NULL) {
  333.         // Send a message to the client that
  334.         // the remote connection failed.
  335.         //
  336.         srv_sendmsg(srvproc, 
  337.                     SRV_MSG_ERROR, 
  338.                     (DBINT)REMOTE_FAIL, 
  339.                     (DBTINYINT)0,
  340.                     (DBTINYINT)0, 
  341.                     NULL, 
  342.                     0, 
  343.                     0, 
  344.                     "Login to remote DBMS failed (srv_alloc).",
  345.                     SRV_NULLTERM);
  346.  
  347.         // Now allow other threads to enter this function.
  348.         //
  349.         DosSemClear(&init_remote_SEM);
  350.         return(SRV_DISCONNECT);
  351.     }
  352.  
  353.     // Set "bulk insert" mode flag to false.
  354.     //
  355.     remote->bulk_mode = FALSE;
  356.  
  357.     // Allocate the LOGINREC structure used to make connections to the
  358.     // remote server. Open the connection in the SRV_CONNECT handler.
  359.     //
  360.     remote->login = dblogin();
  361.  
  362.     if (remote->login == NULL) {
  363.         // Send a message to the client that the
  364.         // remote connection failed.
  365.         //
  366.         srv_sendmsg(srvproc, 
  367.                     SRV_MSG_ERROR, 
  368.                     (DBINT)REMOTE_FAIL, 
  369.                     (DBTINYINT)0,
  370.                     (DBTINYINT)0, 
  371.                     NULL, 
  372.                     0, 
  373.                     0, 
  374.                     "Login to remote DBMS failed (dblogin).",
  375.                     SRV_NULLTERM);
  376.  
  377.         // Deallocate the remote structure and set the user data
  378.         // pointer in "srvproc" to null so that the disconnect
  379.         // handler won't try to disconnect from the remote dbms.
  380.         //
  381.         srv_free(remote);
  382.  
  383.         // Now allow other threads to enter this function.
  384.         //
  385.         DosSemClear(&init_remote_SEM);
  386.         return(SRV_DISCONNECT);
  387.     }
  388.     remote->dbproc = (DBPROCESS *)NULL;
  389.  
  390.     // Set the user name, password, and application name for the remote DBMS.
  391.     //
  392.     DBSETLUSER(remote->login, srv_pfield(srvproc, SRV_USER, (int *)NULL));
  393.     DBSETLPWD(remote->login, srv_pfield(srvproc, SRV_PWD, (int *)NULL));
  394.     DBSETLAPP(remote->login, srv_pfield(srvproc, SRV_APPLNAME, (int *)NULL));
  395.     DBSETLNATLANG(remote->login, srv_pfield(srvproc, SRV_NATLANG, (int *)NULL));
  396.  
  397.     // See if client has set Bulk Copy flag
  398.     //
  399.     if (strcmp(srv_pfield(srvproc, SRV_BCPFLAG, (int *)NULL), "TRUE") == 0)
  400.         BCP_SETL(remote->login, TRUE);
  401.     else
  402.         BCP_SETL(remote->login, FALSE);
  403.  
  404.     // If no server name was specified as a parameter to the main program,
  405.     // then assume that the name is coming from the client.
  406.     //
  407.     if (remote_server == NULL || remote_server[0] == '\0') {
  408.         remote_server = srv_pfield(srvproc,    SRV_HOST, (int *)NULL);
  409.     }
  410.  
  411.     // Try to open a connection to the remote DBMS.
  412.     //
  413.     if ((remote->dbproc = dbopen(remote->login, remote_server))
  414.         == (DBPROCESS *)NULL) {
  415.  
  416.         // Send a message to the client that
  417.         // the remote connection failed.
  418.         //
  419.         srv_sendmsg(srvproc, 
  420.                     SRV_MSG_ERROR, 
  421.                     (DBINT)REMOTE_FAIL, 
  422.                     (DBTINYINT)0,
  423.                     (DBTINYINT)0, 
  424.                     NULL, 
  425.                     0, 
  426.                     0, 
  427.                     "Login to remote DBMS failed (dbopen).",
  428.                     SRV_NULLTERM);
  429.  
  430.         // Deallocate the remote structure and set the user data
  431.         // pointer in "srvproc" to null so that the disconnect
  432.         // handler won't try to disconnect from the remote DBMS.
  433.         //
  434.         srv_free(remote);
  435.         srv_setuserdata(srvproc, (BYTE *)NULL);
  436.  
  437.         // Now allow other threads to enter this function.
  438.         //
  439.         DosSemClear(&init_remote_SEM);
  440.         return(SRV_DISCONNECT);
  441.     } 
  442.     else {
  443.         // Connection to the remote DBMS successful.  Save
  444.         // remote data structure in the "srvproc" so it will be
  445.         // available to the other handlers. We'll also map the remote
  446.         // DBMS connection to our "srvproc".
  447.         //
  448.         srv_setuserdata(srvproc, (BYTE *)remote);
  449.         dbsetuserdata(remote->dbproc, (SRV_PROC *)srvproc);
  450.     }
  451.  
  452.     // Now allow other threads to enter this function.
  453.     //
  454.     DosSemClear(&init_remote_SEM);
  455.  
  456.     // Display connect info on console
  457.     //
  458.     string = srv_pfield(srvproc, SRV_CPID, &len);
  459.    string[len] = '\0';
  460.     printf("\nClient process ID: %s\n", string);
  461.  
  462.    string = srv_pfield(srvproc, SRV_USER, &len);
  463.    string[len] = '\0';
  464.    printf("User name: %s\n", string);
  465.  
  466.    string = srv_pfield(srvproc, SRV_APPLNAME, &len);
  467.    string[len] = '\0';
  468.    if (len > 0)
  469.        printf("Application program name: %s\n", string);
  470.  
  471.    string = srv_pfield(srvproc, SRV_RMTSERVER, &len);
  472.    string[len] = '\0';
  473.    if (len > 0)
  474.        printf("Remote Server: %s\n", string);
  475.  
  476.     return(SRV_CONTINUE);
  477. }
  478.  
  479. //
  480. // LANG_EXECUTE
  481. //       Execute a client language request on the remote dbms.
  482. //       Any results from the remote dbms are passed back to the client.
  483. //
  484. //    Parameters:
  485. //       srvproc -  process handle to the current client connection.
  486. //
  487. // Returns:
  488. //       SRV_CONTINUE
  489. //
  490. #define MAXQUERY    (64 * 1024)
  491. int lang_execute(srvproc)
  492. SRV_PROC FAR *srvproc;
  493. {
  494.     REMOTE_DBMS FAR *rmt_dbms;    // the remote database pointer
  495.     DBPROCESS FAR *rmtproc;        // our DBPROCESS pointer
  496.     DBCHAR FAR *query;            // pointer to language buffer
  497.     long query_len;                // length of query
  498.     int status;                    // status of dblib calls
  499.    SRV_SERVER FAR *server;     // pointer to server info for log
  500.    DBCHAR FAR *user;           // userid for log file
  501.  
  502.  
  503.     // Get the remote dbms pointer we saved in the srvproc via
  504.     // srv_setuserdata.
  505.     //
  506.     rmt_dbms = (REMOTE_DBMS *)srv_getuserdata(srvproc);
  507.  
  508.     // Get the pointer to the remote DBPROCESS
  509.     //
  510.     rmtproc = rmt_dbms->dbproc;
  511.  
  512.     // Get the pointer to the client language request command buffer.
  513.     //
  514.     query = srv_langptr(srvproc);
  515.  
  516.    // Log language event
  517.    //
  518.    if (fLogEnabled == TRUE) {
  519.        server = SRV_GETSERVER(srvproc);
  520.        user = srv_pfield(srvproc, SRV_USER, (int *)NULL);
  521.        srv_log(server, TRUE, user, SRV_NULLTERM);
  522.        srv_log(server, FALSE, query, SRV_NULLTERM);
  523.    }
  524.  
  525.     // See if the previous command was a "bulk insert" command
  526.     //
  527.     if (rmt_dbms->bulk_mode) {
  528.  
  529.         // Get length of the SQL command.
  530.         //
  531.         query_len = srv_langlen(srvproc);
  532.  
  533.         // If length of data is zero, then send a zero length buffer
  534.         // to the destination SQL Server.  This is required in order to
  535.         // signal the SQL Server that no more data follows.
  536.         //
  537.         if (query_len == -1L)
  538.             query_len = 0L;
  539.  
  540.         // Place buffer into target SQL server's buffer.  Since
  541.         // the buffer is binary data from the client, we need some
  542.         // method of sending it to the target SQL Server.    We will use
  543.         // dbbcmd(), an undocumented DBLIB API that is used for sending
  544.         // binary data.
  545.         //
  546.         status = dbfcmd(rmtproc, query,    query_len);
  547.  
  548.         rmt_dbms->bulk_mode = FALSE;
  549.     } 
  550.     else {
  551.         // Let's check for "insert bulk" request
  552.         //
  553.         if (srv_langlen(srvproc) > (sizeof(BULK_CMD) - 1))
  554.             if (strnicmp(query, BULK_CMD, (sizeof(BULK_CMD) - 1)) == 0)
  555.                 rmt_dbms->bulk_mode = TRUE;
  556.  
  557.             // Place buffer into target SQL server's buffer.
  558.             //
  559.             status = dbcmd(rmtproc,
  560.             query);
  561.     }
  562.  
  563.     // If previous DBLIB call successful, send command buffer to SQL Server.
  564.     //
  565.     if (status == SUCCEED) {    // if previous DBLIB call successful
  566.         status = dbsqlexec(rmtproc);
  567.     }
  568.     if (!SRV_GOT_ATTENTION(srvproc) && (status == SUCCEED)) {
  569.         //
  570.         // Get the remote dbms results and pass them back to
  571.         // client.
  572.         //
  573.         handle_results(rmtproc, srvproc);
  574.     } 
  575.     else {
  576.         //
  577.         // If an attention event was issued or the dbsqlexec failed,
  578.         // acknowledge with senddone.
  579.         //
  580.         if (DBDEAD(rmtproc)) {
  581.             printf("thread shutting down");
  582.             srv_sendmsg(srvproc, 
  583.                         SRV_MSG_ERROR, 
  584.                         (DBINT)REMOTE_FAIL,
  585.                         (DBTINYINT)23, 
  586.                         (DBTINYINT)17, 
  587.                         NULL, 
  588.                         0, 
  589.                         0,
  590.                            "Remote Server connection no longer active: \
  591.                          thread shutting down",
  592.                         SRV_NULLTERM);
  593.             srv_senddone(srvproc, SRV_DONE_FINAL | SRV_DONE_ERROR,
  594.                          (DBUSMALLINT)0, (DBINT)0);
  595.             srv_event(srvproc, SRV_DISCONNECT, NULL);
  596.         } 
  597.         else {
  598.             srv_senddone(srvproc, SRV_DONE_FINAL, (DBUSMALLINT)0, (DBINT)0);
  599.         }
  600.     }
  601.     return(SRV_CONTINUE);
  602. }
  603.  
  604. //
  605. // RPC_EXECUTE
  606. //       Execute a client RPC.
  607. //
  608. //       Note that in our example, the remote DBMS is an SQL Server.
  609. //       If our remote DBMS was a foreign DBMS, the code to send
  610. //       the query to the remote DBMS and receive remote results
  611. //       would be different.  In this example, we'll just send
  612. //       the rpc along to the SQL Server.
  613. //
  614. // Parameters:
  615. //    srvproc - The Open Server process handle to use to send results to the 
  616. //                   client.
  617. //
  618. // Returns:
  619. //       SRV_CONTINUE
  620. //
  621. int rpc_execute(srvproc)
  622. SRV_PROC FAR *srvproc;
  623. {
  624.     DBPROCESS FAR *rmtproc;    // Our DBPROCESS pointer
  625.     int i;                    // Index variable
  626.     int params;
  627.     int retparam;
  628.     int FAR *paramarray;
  629.  
  630.     rmtproc = ((REMOTE_DBMS *)srv_getuserdata(srvproc))->dbproc;
  631.     paramarray = ((REMOTE_DBMS *)srv_getuserdata(srvproc))->retparams;
  632.  
  633.     // Get the RPC name and i nitialize the RPC to the remote DBMS.
  634.     //
  635.     dbrpcinit(rmtproc, srv_rpcname(srvproc, (int *)NULL),
  636.               (DBUSMALLINT)srv_rpcoptions(srvproc));
  637.  
  638.     // Set up any RPC parameters.
  639.     //
  640.     params = srv_rpcparams(srvproc);
  641.     retparam = 1;
  642.  
  643.     for (i = 1; i <= params; i++) {
  644.  
  645.         dbrpcparam(rmtproc, 
  646.                    srv_paramname(srvproc, i, (int *)NULL),
  647.                    (BYTE)srv_paramstatus(srvproc, i),
  648.                    srv_paramtype(srvproc, i), 
  649.                    srv_parammaxlen(srvproc, i),
  650.                    srv_paramlen(srvproc, i), 
  651.                    srv_paramdata(srvproc, i));
  652.  
  653.         // The current rpc may have three parameters, but only the
  654.         // first and third are return parameters.  This means that
  655.         // dbnumrets() returns two parameters, not three.  The first
  656.         // call to dbretdata() referes to the rpc's first parameter,
  657.         // and the second call to dbretdata() refers to the rpc's
  658.         // third parameter.  To handle this, we map each return
  659.         // parameter to its original parameter so we can later
  660.         // reset the return value of the correct return parameters
  661.         // in "handle_results()".
  662.         //
  663.         if ((BYTE)srv_paramstatus(srvproc, i) & SRV_PARAMRETURN) {
  664.             paramarray[retparam] = i;
  665.             retparam++;
  666.         }
  667.     }
  668.  
  669.     // Send the RPC to the remote DBMS.
  670.     //
  671.     dbrpcsend(rmtproc);
  672.     dbsqlok(rmtproc);
  673.  
  674.     // Now get the remote DBMS results and pass them back to
  675.     // Open Server client.
  676.     //
  677.     handle_results(rmtproc, srvproc);
  678.     return(SRV_CONTINUE);
  679. }
  680.  
  681. //
  682. // HANDLE_RESULTS
  683. //       Once a command has been sent to the remote DBMS by the
  684. //       SRV_LANGUAGE  handler, this routine processes the results
  685. //       and passes them back to the client.
  686. //
  687. // Parameters:
  688. //       rmtproc - The DBPROCESS handle to the remote DBMS.
  689. //       srvproc - The process handle to use to send results to the client.
  690. //
  691. // Returns:
  692. //       SUCCEED or FAIL.
  693. //
  694. int handle_results(rmtproc, srvproc)
  695. DBPROCESS FAR *rmtproc;
  696. SRV_PROC FAR *srvproc;
  697. {
  698.     int cols;                // data columns returned
  699.     int i;                    // index variable
  700.     DBINT rows;                // number of rows sent
  701.     BOOL results_sent;        // number of result sets sent
  702.     BOOL gotcompute;        // COMPUTE row indicator
  703.     int *paramarray;
  704.     RETCODE returnvalue;    // value returned from dblib call
  705.  
  706.     results_sent = FALSE;
  707.     rows = 0L;
  708.     gotcompute = FALSE;
  709.     paramarray = ((REMOTE_DBMS *)srv_getuserdata(srvproc))->retparams;
  710.  
  711.     // Process the results from the remote DBMS.
  712.     // Since a command may consist of multiple commands or a single
  713.     // command that has multiple sets of results, we'll loop through
  714.     // each results set.
  715.     //
  716.     while (TRUE) {
  717.  
  718.         returnvalue = dbresults(rmtproc);
  719.         if (returnvalue == NO_MORE_RESULTS) {
  720.             break;
  721.         }
  722.  
  723.         // Check to see if the client has sent an attention event.  If
  724.         // so, simply discard data from remote server
  725.         //
  726.         if (SRV_GOT_ATTENTION(srvproc)) {
  727.             continue;
  728.         }
  729.  
  730.         //
  731.         // If this is the second time through the loop,
  732.         // send a completion message to the client
  733.         // for the previous results sent.
  734.         //
  735.         if (results_sent == TRUE) {
  736.  
  737.             // If there are some COMPUTE rows, send a message
  738.             // to the client that Data Services Library doesn't yet handle them.
  739.             //
  740.             if (gotcompute == TRUE) {
  741.                 gotcompute = FALSE;
  742.                 srv_sendmsg(srvproc, 
  743.                             SRV_MSG_ERROR, 
  744.                             (DBINT)COMPUTE_ROW,
  745.                             (DBTINYINT)0, 
  746.                             (DBTINYINT)0, 
  747.                             NULL, 
  748.                             0, 
  749.                             0,
  750.                             "Data Services library can't handle COMPUTE rows.", 
  751.                             SRV_NULLTERM);
  752.             }
  753.  
  754.             // If the previous batch was one that may
  755.             // have returned rows, set the DONE status
  756.             // accordingly.
  757.             //
  758.             if (rows > 0) {
  759.                 srv_senddone(srvproc, SRV_DONE_MORE | SRV_DONE_COUNT,
  760.                              (DBUSMALLINT)0, rows);
  761.             } 
  762.             else
  763.                 srv_senddone(srvproc, SRV_DONE_MORE, (DBUSMALLINT)0,
  764.                              (DBINT)0);
  765.         }
  766.  
  767.         // How many data columns are in the row?
  768.         // Non-"select" statements will have 0 columns.
  769.         //
  770.         cols = dbnumcols(rmtproc);
  771.  
  772.         // Build the row description for the client return.
  773.         //
  774.         for (i = 1; i <= cols; i++) {
  775.  
  776.             //    Call "srv_describe()" for each column in the row.
  777.             //
  778.             srv_describe(srvproc, 
  779.                          i, 
  780.                          dbcolname(rmtproc, i), 
  781.                          SRV_NULLTERM,
  782.                          (DBINT)dbcolntype(rmtproc, i), 
  783.                          dbcollen(rmtproc, i),
  784.                          (DBINT)dbcolntype(rmtproc, i), 
  785.                          dbcollen(rmtproc, i),
  786.                          (BYTE *)NULL);
  787.         }
  788.  
  789.         // Send each row from the remote DBMS to the client.
  790.         //
  791.         rows = 0;
  792.         while (TRUE) {
  793.  
  794.             returnvalue = dbnextrow(rmtproc);
  795.             if (returnvalue == NO_MORE_ROWS) {
  796.                 break;
  797.             }
  798.  
  799.             // If it's not a regular row, it's a COMPUTE row.
  800.             // This SQL extension is particular to Sybase
  801.             // TRANSACT-SQL and is not yet supported.
  802.             //
  803.             if (DBROWTYPE(rmtproc) != REG_ROW) {
  804.                 gotcompute = TRUE;
  805.                 continue;
  806.             } 
  807.             else
  808.                 gotcompute = FALSE;
  809.  
  810.             // The row description is built.  Move the
  811.             // rows from the remote server to the client.
  812.             //
  813.             for (i = 1; i <= cols; i++) {
  814.                 srv_setcollen(srvproc, i, (int)dbdatlen(rmtproc, i));
  815.                 srv_setcoldata(srvproc, i, dbdata(rmtproc, i));
  816.             }
  817.  
  818.             // Check to see if the client has issued an attention event.
  819.             // If so, discard data from the remote server.
  820.             //
  821.             if (SRV_GOT_ATTENTION(srvproc)) {
  822.                 continue;
  823.             }
  824.             if (srv_sendrow(srvproc) == SUCCEED)
  825.                 rows++;
  826.         }
  827.         // Check to see if any parameter were returned from
  828.         // the remote DBMS.  If so, pass them through to the
  829.         // client.
  830.         for (i = 1; i <= dbnumrets(rmtproc); i++) {
  831.             //
  832.             // If the return parameters are a result of
  833.             // an rpc, we used srv_paramset() to set the return
  834.             // value.  If the return parameters are not the
  835.             // result of an rpc, we use srv_returnval().
  836.             //
  837.             if (srv_rpcname(srvproc, NULL) != NULL) {
  838.                 //
  839.                 // The current rpc may have three parameters, but
  840.                 // only the first and third are return parameters.
  841.                 // This means that dbnumrets() returns two parameters,
  842.                 // not three.  The first call to dbretdata() refers to
  843.                 // the rpc's first parameter, and the second call to
  844.                 // dbretdata() refers to the rpc's third parameter.
  845.                 // To handle this, we map each return parameter to
  846.                 // its original parameter so we can later reset the
  847.                 // return value of the correct return parameters in
  848.                 // handle_results().
  849.                 //
  850.                 srv_paramset(srvproc, paramarray[i], dbretdata(rmtproc, i),
  851.                              (int)dbretlen(rmtproc, i));
  852.             } 
  853.             else {
  854.                 srv_returnval(srvproc, 
  855.                               dbretname(rmtproc, i), 
  856.                               SRV_NULLTERM,
  857.                               SRV_PARAMRETURN, 
  858.                               dbrettype(rmtproc, i),
  859.                               dbretlen(rmtproc, i), 
  860.                               dbretlen(rmtproc, i),
  861.                               dbretdata(rmtproc, i));
  862.             }
  863.         }
  864.  
  865.         // Check to see if we got a return status code from the
  866.         // remote DBMS.  Pass it through to the client.
  867.         //
  868.         if (dbhasretstat(rmtproc))
  869.             srv_sendstatus(srvproc, dbretstatus(rmtproc));
  870.  
  871.         // If the command was one where count is meaningful, DBCOUNT()
  872.         // will be >= 0 and we will set rows accordingly.
  873.         //
  874.         if (dbnumcols(rmtproc) > 0)
  875.             rows = DBCOUNT(rmtproc) < 0L ? 0L : rows;
  876.         else
  877.             rows = DBCOUNT(rmtproc);
  878.  
  879.         // Set flag so that we will send a completion
  880.         // message for the current set of results.
  881.         //
  882.         results_sent = TRUE;
  883.     }
  884.  
  885.     // If there are some COMPUTE rows, send a message
  886.     // to the client that Open Services Library doesn't handle them yet.
  887.     //
  888.     if (gotcompute == TRUE) {
  889.         gotcompute = FALSE;
  890.         srv_sendmsg(srvproc, 
  891.                     SRV_MSG_ERROR, 
  892.                     (DBINT)COMPUTE_ROW, 
  893.                     (DBTINYINT)0,
  894.                     (DBTINYINT)0, 
  895.                     NULL, 
  896.                     0, 
  897.                     0,
  898.                     "Data Services Library can't yet handle COMPUTE rows.",
  899.                     SRV_NULLTERM);
  900.     }
  901.  
  902.     // Send the final done packet for the execution of the command batch.
  903.     //
  904.     // If the previous batch was one that may
  905.     // have returned rows, set the DONE status
  906.     // accordingly.
  907.     //
  908.     if (rows > 0) {
  909.         srv_senddone(srvproc, SRV_DONE_COUNT | SRV_DONE_FINAL, (DBUSMALLINT)0,
  910.                      rows);
  911.     } 
  912.     else {
  913.         srv_senddone(srvproc, SRV_DONE_FINAL, (DBUSMALLINT)0, (DBINT)0);
  914.     }
  915.     return(SUCCEED);
  916. }
  917.  
  918. // EXIT_REMOTE
  919. //       Handler for SRV_DISCONNECT events.
  920. //       Closes remote DBMS connection if appropriate.
  921. //
  922. //    The code to disconnect from the remote DBMS is SQL Server specific. If
  923. //    using a non-SQL Server database, the disconnect from the  remote database
  924. //    would be different but would probably still occur here.
  925. //
  926. // Parameters:
  927. //       srvproc - the handle to the client connection
  928. //
  929. // Returns:
  930. //       SRV_CONTINUE
  931. //
  932. int exit_remote(srvproc)
  933. SRV_PROC FAR *srvproc;
  934. {
  935.     char *string;
  936.     int len;
  937.     REMOTE_DBMS FAR *remote;    // pointer to target connect structure
  938.  
  939.     remote = (REMOTE_DBMS *)srv_getuserdata(srvproc);
  940.  
  941.     // Is there a REMOTE_DBMS structure to clean-up?
  942.     //
  943.     if (remote != (REMOTE_DBMS *)NULL) {
  944.  
  945.         // Is there a live dbproc?
  946.         //
  947.         if (remote->dbproc != (DBPROCESS *)NULL) {
  948.             dbclose(remote->dbproc);
  949.         }
  950.         dbfreelogin(remote->login);
  951.         srv_free(remote);
  952.     }
  953.  
  954.     // Display info on console
  955.     //
  956.     string = srv_pfield(srvproc, SRV_CPID, &len);
  957.    string[len] = '\0';
  958.       printf("\nClient connection closed, process ID: %s\n", string);
  959.  
  960.     return(SRV_CONTINUE);
  961. }
  962.  
  963. // CHK_ERR
  964. //       Print out errors.
  965. //
  966. // Parameters:
  967. //       server         - pointer to gateway server structure.
  968. //       srvproc      - pointer to client connection structure
  969. //       errornum     - error number.
  970. //       severity     - error severity.
  971. //       state         - error state.
  972. //       oserrnum     - operating system error number, if any.
  973. //       errtext      - the text of the error message.
  974. //       errtextlen   - length of the errtext message
  975. //       oserrtext    - the text of the operating system error message.
  976. //       oserrtextlen - length of the errtext message
  977. //
  978. // Returns:
  979. //       SRV_CONTINUE, SRV_CANCEL, or SRV_EXIT
  980. //
  981.  
  982. int chk_err(server, srvproc, errornum, severity, state, oserrnum, errtext,
  983.             errtextlen, oserrtext, oserrtextlen)
  984. SRV_SERVER FAR *server;
  985. SRV_PROC FAR *srvproc;
  986. int errornum;
  987. BYTE severity;
  988. BYTE state;
  989. int oserrnum;
  990. DBCHAR FAR *errtext;
  991. int errtextlen;
  992. DBCHAR FAR *oserrtext;
  993. int oserrtextlen;
  994. {
  995.     char log_buffer[256];
  996.  
  997.     // Operating system error?
  998.     //
  999.     if (oserrnum != SRV_ENO_OS_ERR) {
  1000.         sprintf(log_buffer, "SERVER OS ERROR: %d: %s.", oserrnum, oserrtext);
  1001.  
  1002.         srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
  1003.     }
  1004.  
  1005.     // Is this a fatal error for the gateway?
  1006.     //
  1007.     if (severity >= SRV_FATAL_SERVER) {
  1008.         sprintf(log_buffer,
  1009.                 "SERVER: FATAL SERVER ERROR: errornum = %d, \
  1010.                  severity = %d, state = %d: %s.",
  1011.                 errornum, 
  1012.                 severity, 
  1013.                 state, 
  1014.                 errtext);
  1015.  
  1016.         srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
  1017.         return(SRV_EXIT);
  1018.     } 
  1019.     else {
  1020.         //
  1021.         // Did the "srvproc" get a fatal error?
  1022.         //
  1023.         if (severity >= SRV_FATAL_PROCESS) {
  1024.             sprintf(log_buffer,
  1025.             "SERVER: FATAL CONNECT ERROR: errornum = %d, \
  1026.             severity = %d, state = %d: %s.",
  1027.             errornum, 
  1028.             severity, 
  1029.             state, 
  1030.             errtext);
  1031.  
  1032.             srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
  1033.  
  1034.             return(SRV_CANCEL);
  1035.         }
  1036.     }
  1037.  
  1038.     // A non-fatal error or an information message received.
  1039.     // We'll pass it through to the client.
  1040.     //
  1041.     if (srvproc != (SRV_PROC *)NULL && (server != NULL))
  1042.         if (severity < 10) {    // if informational message
  1043.             srv_sendmsg(srvproc, 
  1044.                         SRV_MSG_INFO, 
  1045.                         (DBINT)errornum, 
  1046.                         severity, 
  1047.                         0,
  1048.                         NULL, 
  1049.                         0, 
  1050.                         0, 
  1051.                         errtext, 
  1052.                         SRV_NULLTERM);
  1053.         } 
  1054.         else {            // must be an error message
  1055.         srv_sendmsg(srvproc, 
  1056.                     SRV_MSG_ERROR, 
  1057.                     (DBINT)errornum, 
  1058.                     severity, 
  1059.                     0,
  1060.                     NULL, 
  1061.                     0, 
  1062.                     0, 
  1063.                     errtext, 
  1064.                     SRV_NULLTERM);
  1065.     } 
  1066.     else {
  1067.         sprintf(log_buffer, "GATEWAY ERROR: errornum = %d, severity = %d: %s",
  1068.                 errornum, severity, errtext);
  1069.  
  1070.         srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
  1071.     }
  1072.     return(SRV_CONTINUE);
  1073. }
  1074.  
  1075. // REMOTEMSGS
  1076. //       DBLIB calls this routine when any messages are received
  1077. //       from the remote DBMS. It gets the remote message information and
  1078. //       sends it back to the client as a message.
  1079. //
  1080. // Parameters:
  1081. //       dbproc     - The handler to the remote DBMS process that sent the message.
  1082. //       msgno     - The message number.
  1083. //       msgstate - The message state.
  1084. //       severity - The message severity.
  1085. //       msgtext  - The text of the message.
  1086. //
  1087. //       The following three parameters are available in TDS4.0 and above:
  1088. //
  1089. //       srvname  - The name of the server that sent the message.
  1090. //       procname - The procedure name, if any, of the remote DBMS command.
  1091. //       line     - The remote DBMS command buffer line to which the msg applies.
  1092. //
  1093. // Returns:
  1094. //       0
  1095. //
  1096.  
  1097. int remotemsgs(dbproc, msgno, msgstate, severity, msgtext)
  1098. DBPROCESS FAR *dbproc;
  1099. DBINT msgno;
  1100. DBSMALLINT msgstate;
  1101. DBSMALLINT severity;
  1102. char FAR *msgtext;
  1103. {
  1104.     SRV_PROC FAR *srvproc;
  1105.  
  1106.     // If a remote DBMS error was received during the remote
  1107.     // open, the dbproc is NULL and a message is sent back on the
  1108.     // most recent srvproc.
  1109.     //
  1110.     if (dbproc == (DBPROCESS *)NULL) {
  1111.         srvproc = Newsrvproc;
  1112.     } 
  1113.     else {
  1114.         if ((srvproc = (SRV_PROC *)dbgetuserdata(dbproc)) == NULL) {
  1115.             //
  1116.             // An error was received after the dbproc was assigned, but
  1117.             // before we were able to associate our srvproc.
  1118.             //
  1119.             srvproc = Newsrvproc;
  1120.         }
  1121.     }
  1122.     if (severity < 10) {    // if informational message
  1123.  
  1124.         srv_sendmsg(srvproc, 
  1125.                     SRV_MSG_INFO, 
  1126.                     msgno, 
  1127.                     (DBTINYINT)severity,
  1128.                     (DBTINYINT)msgstate, 
  1129.                     NULL, 
  1130.                     0, 
  1131.                     0, 
  1132.                     msgtext, 
  1133.                     SRV_NULLTERM);
  1134.     } 
  1135.     else {                // must be an error message
  1136.  
  1137.         srv_sendmsg(srvproc, 
  1138.                     SRV_MSG_ERROR, 
  1139.                     msgno, 
  1140.                     (DBTINYINT)severity,
  1141.                     (DBTINYINT)msgstate, 
  1142.                     NULL, 
  1143.                     0, 
  1144.                     0, 
  1145.                     msgtext, 
  1146.                     SRV_NULLTERM);
  1147.     }
  1148.     return(0);
  1149. }
  1150.  
  1151. // REMOTEERR
  1152. //       This is the handler for error messages from the remote DBMS, in this
  1153. //       case DBLIB.  It gets called whenever a DBLIB error occurs.  It takes
  1154. //       the error message and passes it back to the client.
  1155. //
  1156. // Parameters:
  1157. //       dbproc - The process handle for the remote DBMS.
  1158. //       severity - The severity of the error.
  1159. //       dberr - The DBLIB error number.
  1160. //       oserr - The operating system error, if any.
  1161. //       dberrstr - The text of the DBLIB error.
  1162. //       oserrstr - The text of operating system error, if any.
  1163. //
  1164. // Returns:
  1165. //       INT_EXIT to exit the program.
  1166. //       INT_CANCEL to cause a FAIL return from the DBLIB routine that got
  1167. //       the error.
  1168. //
  1169. int remoteerr(dbproc, severity, dberr, oserr, dberrstr, oserrstr)
  1170. DBPROCESS FAR *dbproc;
  1171. int severity;
  1172. int dberr;
  1173. int oserr;
  1174. char FAR *dberrstr;
  1175. char FAR *oserrstr;
  1176. {
  1177.     SRV_PROC *srvproc = (SRV_PROC *)NULL;
  1178.  
  1179.     // If the DBLIB process is dead or we get a DBLIB error 10007 
  1180.     // ("General SQL Server Error:...")    then simply ignore it.    The error 
  1181.     // message has already been sent to the client.
  1182.     //
  1183.     if (DBDEAD(dbproc) || dberr == 10007) {
  1184.         return(INT_CANCEL);
  1185.     }
  1186.  
  1187.     //
  1188.     // A remote DBMS error may have been issued during the remote
  1189.     // open. In this case, the dbproc will be NULL and a message
  1190.     // will be sent on the most recent srvproc.
  1191.     //
  1192.     if (dbproc == (DBPROCESS *)NULL) {
  1193.         srvproc = Newsrvproc;
  1194.     } 
  1195.     else {
  1196.         if ((srvproc = (SRV_PROC *)dbgetuserdata(dbproc)) == NULL) {
  1197.             // An error was issued after the dbproc was assigned but before
  1198.             // we were able to associate our srvproc.
  1199.             //
  1200.             srvproc = Newsrvproc;
  1201.         }
  1202.     }
  1203.  
  1204.     //
  1205.     // Send error message to client.
  1206.     //
  1207.     srv_sendmsg(srvproc, 
  1208.                 SRV_MSG_ERROR, 
  1209.                 (DBINT)REMOTE_MSG,
  1210.                 (DBTINYINT)severity, 
  1211.                 (DBTINYINT)0, 
  1212.                 NULL, 
  1213.                 0, 
  1214.                 0, 
  1215.                 dberrstr,
  1216.                 SRV_NULLTERM);
  1217.  
  1218.     if (oserr != DBNOERR) {
  1219.         srv_sendmsg(srvproc, 
  1220.                     SRV_MSG_ERROR, 
  1221.                     (DBINT)REMOTE_MSG,
  1222.                     (DBTINYINT)severity, 
  1223.                     (DBTINYINT)0, 
  1224.                     NULL, 
  1225.                     0, 
  1226.                     0, 
  1227.                     oserrstr,
  1228.                     SRV_NULLTERM);
  1229.     }
  1230.     return(INT_CANCEL);
  1231. }
  1232.  
  1233. // ATTN_HANDLER
  1234. //       This is an event handler that will be called when the gateway receives
  1235. //       an attention from a client.
  1236. //
  1237. // Parameters:
  1238. //       srvproc - Pointer to the client connection structure
  1239. //                   receiving the attention.
  1240. //
  1241. // Returns:
  1242. //       SRV_CONTINUE
  1243. //
  1244. int attn_handler(srvproc)
  1245. SRV_PROC FAR *srvproc;
  1246. {
  1247.     DBPROCESS FAR *rmtproc;    // our DBPROCESS pointer
  1248.  
  1249.     rmtproc = ((REMOTE_DBMS *)srv_getuserdata(srvproc))->dbproc;
  1250.     dbcancel(rmtproc);
  1251.     return(SRV_CONTINUE);
  1252. }
  1253.  
  1254.  
  1255. #pragma check_stack()    // set stack checking to its default setting
  1256.