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