home *** CD-ROM | disk | FTP | other *** search
- // This module defines all of the "callback" functions. These modules
- // are defined here due to the fact that they must be mult-threaded and
- // therefore completely reentrant. They must be linked with the proper
- // C Runtime Multi-Threaded Library (LLIBCDLL.LIB).
-
- #include <stdlib.h>
- #include <stdio.h>
- #include <string.h>
- #include <ctype.h>
-
- #define INCL_BASE
- #include <os2.h>
- #include <srv.h>
-
- #ifdef LMPTK
- #define INCL_NETMAILSLOT
- #define INCL_NETERRORS
- #include <lan.h>
- #endif
-
- // Define some user message codes.
- //
- #define SRV_MAXERROR 20000
- #define SP_UNKNOWN SRV_MAXERROR + 1
- #define INVALID_SP_SYNTAX SRV_MAXERROR + 2
- #define BAD_SP_PARAMETER SRV_MAXERROR + 3
- #define BROADCAST SRV_MAXERROR + 4
- #define EXEC SRV_MAXERROR + 5
-
- // Miscellaneous defines used by sp-handling routines.
- //
- #define EXEC_CMD "exec"
- #define MAXNAME 32
- #define MAXLEN 80
- #define MAXPARAMS 4
- #define SP_EXT ";1" // Default extension when Server initiated
- #define VALUE 0x0000
- #define REFERENCE 0x0001
-
- #define BUF_SIZE 2048
- #define XBASE_HDR_SIZE 32
- #define XBASE_MAX_COLUMNS 128
-
- // Event handlers for SP requests.
- //
- RETCODE proclist(); // List SP associated with Server
- RETCODE sp_exec(); // Execute the specified command string
- RETCODE sp_exec_trig(); // Execute the specified command string
- RETCODE diskfree(); // Return the space on a given drive
- RETCODE disklist(); // List available drives and their space
- RETCODE scan_xbase(); // Open and read an xBase file
-
- #ifdef LMPTK
- RETCODE broadcast(); // Broadcast message to specifed mailslot
- #endif
-
- // Global arguments
- //
- DBCHAR FAR *ProcSrvrName; // The server name
-
- // Stored Procedure parameter information structure.
- //
- typedef struct sp_params {
- DBCHAR name[MAXNAME + 1]; // Parameter name
- int type; // Parameter data type
- DBINT length; // Parameter type length
- int status; // Parameter return status
- DBCHAR defaultvalue[MAXLEN * 10]; // Optional default value
- } SP_PARAMS;
-
- // Stored Procedure information structure.
- //
- typedef struct sp_info {
- DBCHAR name[MAXNAME + 1]; // Procedure name
- DBCHAR usage[MAXLEN]; // Usage string
- int numparams; // Number of parameters
- SP_PARAMS params[MAXPARAMS]; // Parameter array
- RETCODE(FAR *handler) (void FAR *, ...);// Pointer to function with
- // variable arguments
- } SP_INFO;
-
- // Array of Stored Procedure handlers.
- //
- SP_INFO Sps[] =
- {
- "sp_exec", // Procedure name
- "usage: sp_exec <[@command =] command string>", // Procedure usage
- 1, // Number of parameters
- { // Parameter definitions
- "command", // Parameter name
- SRVCHAR, // Parameter type
- MAXLEN, // Parameter length (0 if fixed)
- VALUE, // Pass by value
- "dir *.*", // Default parameter value
- },
- sp_exec, // Procedure function pointer
-
- "sp_exec_trig", // Procedure name
- "usage: sp_exec_trig <[@command =] command string>", // Procedure usage
- 1, // Number of parameters
- { // Parameter definitions
- "command", // Parameter name
- SRVCHAR, // Parameter type
- MAXLEN, // Parameter length (0 if fixed)
- VALUE, // Pass by value
- "dir *.*", // Default parameter value
- },
- sp_exec_trig, // Procedure function pointer
-
- "proclist",
- "usage: proclist",
- 0,
- {
- 0
- },
- proclist,
-
- "disklist",
- "usage: disklist",
- 0,
- {
- 0
- },
- disklist,
-
- "diskfree",
- "usage: diskfree <[@drive =] drive letter> [,] <[@space =] free space>",
- 2,
- {
- "drive",
- SRVCHAR,
- 1,
- VALUE, // pass by value
- "c",
-
- "space",
- SRVINT4,
- 8,
- REFERENCE, // pass by reference
- "0",
- },
- diskfree,
-
- "scan_xbase", // rpc name
- "usage: scan_xbase <[@file_name =] xbase file name>", // rpc usage
- 1, // number of parameters
- { // parameter definitions
- "file_name", // parameter name
- SRVCHAR, // parameter type
- MAXLEN, // parameter length (0 if fixed)
- VALUE, // pass by value
- "\\sql\\opends\\samples\\procsrv\\build\\sales.dbf", // default parameter value
- },
- scan_xbase, // rpc function pointer
-
- #ifdef LMPTK
- "broadcast",
- "usage: broadcast <[@mailslot =] mailslot> [,] <[@message =] message>",
- 2,
- {
- "mailslot",
- SRVCHAR,
- MAXLEN,
- VALUE, // pass by value
- "\\\\*\\mailslot\\broadcast",
-
- "message",
- SRVCHAR,
- MAXLEN,
- VALUE, // pass by value
- "This is a default broadcast msg.\n"
- },
- broadcast,
- #endif
- };
-
-
- #define Rpcnumber sizeof(Sps) / sizeof(SP_INFO)
-
- // Prototype dblib API's
- //
- RETCODE attn_handler(SRV_PROC FAR *srvproc);
-
- RETCODE chk_err(SRV_SERVER FAR *server,
- SRV_PROC FAR *srvproc,
- int srverror,
- BYTE severity,
- BYTE state,
- int oserrnum,
- DBCHAR FAR *errtext,
- int errtextlen,
- DBCHAR FAR *oserrtext,
- int oserrtextlen);
-
- RETCODE init_remote(SRV_PROC FAR *srvproc);
-
- RETCODE init_server(SRV_SERVER FAR *server);
-
- RETCODE sp_execute(SRV_PROC FAR *srvproc);
-
- RETCODE exit_remote(SRV_PROC FAR *srvproc);
-
- RETCODE lang_execute(SRV_PROC FAR *srvproc);
-
- void set_remote_server_name(char FAR *name);
-
- // Miscellaneous prototypes for sp language events
- //
- RETCODE lang_parser(SRV_PROC FAR *srvproc,
- SP_INFO **sp,
- DBCHAR paramvalues[MAXPARAMS][MAXLEN]);
-
- DBCHAR *scan_next(DBCHAR *string, DBCHAR *word);
-
- //
- // This section defines the "call-back" functions supplied to the procedure server
- // application.
- //
- #pragma check_stack(off) // turn off stack checking
-
- //
- // SET_REMOTE_SERVER_NAME
- // This function sets the name of the procedure server.
- //
- // Parameters:
- // name - Pointer to name of server.
- //
- // Returns:
- // none
- //
- void set_remote_server_name(name)
- char FAR *name;
- {
- ProcSrvrName = name;
- return;
- }
-
- //
- // INIT_SERVER
- // Initialize the server on a SRV_START event.
- // Event handlers for the server are installed.
- //
- // Parameters:
- // server - Pointer to SRV_SERVER structure
- //
- // Returns:
- // SRV_CONTINUE
- //
- RETCODE init_server(server)
- SRV_SERVER FAR *server;
- {
- char log_buffer[256];
-
- // When we get a connection request from a client, we want to
- // call "init_remote()" to make a connection to the remote
- // server.
- //
- srv_handle(server, (DBINT)SRV_CONNECT, init_remote);
-
- // When the client issues a language request, call
- // "lang_execute()" to send the SQL statement to the remote DBMS.
- //
- srv_handle(server, (DBINT)SRV_LANGUAGE, lang_execute);
-
- // When the client issues an RSP, call "sp_execute()"
- // to send the RSP to the remote DBMS (the SQL Server).
- //
- srv_handle(server, (DBINT)SRV_RPC, sp_execute);
-
- // When a disconnect request is issued, call "exit_remote()"
- // to close the connection to the remote DBMS.
- //
- srv_handle(server, (DBINT)SRV_DISCONNECT, exit_remote);
-
- // Log Server information to log file
- //
- sprintf(log_buffer, "Server pipe name = %s",
- srv_sfield(server, SRV_SERVERNAME, (int *)NULL));
-
- srv_log(server, FALSE, log_buffer, SRV_NULLTERM);
- printf("%s\n", log_buffer);
-
- sprintf(log_buffer, "Client connections allowed = %s",
- srv_sfield(server, SRV_CONNECTIONS, (int *)NULL));
-
- srv_log(server, FALSE, log_buffer, SRV_NULLTERM);
- printf("%s\n", log_buffer);
-
- return(SRV_CONTINUE);
- }
-
- // INIT_REMOTE
- // Event handler for a SRV_CONNECT event.
- // A connection is made to the procedure server.
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- //
- // Returns:
- // SRV_CONTINUE
- //
- // Side Effects:
- // If the connection to the remote dbms cannot be made, then issue
- // a SRV_DISCONNECT request.
- //
- //
- RETCODE init_remote(srvproc)
- SRV_PROC FAR *srvproc;
- {
- char *string;
- int len;
-
- // Set server name
- //
- srvproc->serverlen = (BYTE)strlen(ProcSrvrName);
- srvproc->servername = srv_alloc((DBINT)srvproc->serverlen);
- strcpy(srvproc->servername, ProcSrvrName);
-
- // Display info on console
- //
- string = srv_pfield(srvproc, SRV_CPID, &len);
- string[len] = '\0';
- printf("\nClient process ID: %s\n", string);
-
- string = srv_pfield(srvproc, SRV_USER, &len);
- string[len] = '\0';
- printf("User name: %s\n", string);
-
- string = srv_pfield(srvproc, SRV_APPLNAME, &len);
- string[len] = '\0';
- if (len > 0)
- printf("Application program name: %s\n", string);
-
- string = srv_pfield(srvproc, SRV_RMTSERVER, &len);
- string[len] = '\0';
- if (len > 0)
- printf("Remote Server: %s\n", string);
-
- return(SRV_CONTINUE);
- }
-
- //
- // LANG_EXECUTE
- // Execute a client language request on the procedure server.
- //
- // Parameters:
- // srvproc - process handle to the current client connection.
- //
- // Returns:
- // SRV_CONTINUE
- //
-
- RETCODE lang_execute(srvproc)
- SRV_PROC FAR *srvproc;
- {
- int i;
- DBCHAR paramvalues[MAXPARAMS][MAXLEN];
- BYTE convertvalues[MAXPARAMS][MAXLEN];
- SP_INFO *sp = NULL;
-
- // Initialize parameter storage
- //
- for (i = 0; i < MAXPARAMS; i++) {
- memset(paramvalues[i], 0, MAXLEN);
- memset(convertvalues[i], 0, MAXLEN);
- }
- if (lang_parser(srvproc, &sp, paramvalues) == SUCCEED) {
- for (i = 0; i < sp->numparams; i++) {
- if (sp->params[i].status == REFERENCE) {
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- INVALID_SP_SYNTAX,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Procedure contains a return parameter.\
- Unable to execute as a language event.",
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(SRV_CONTINUE);
-
- }
- if (strlen(paramvalues[i]) == 0 &&
- strlen(sp->params[i].defaultvalue) == 0) {
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- INVALID_SP_SYNTAX,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- sp->usage,
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(SRV_CONTINUE);
- }
- if (strlen(paramvalues[i]) == 0 &&
- strlen(sp->params[i].defaultvalue) != 0)
- strcpy(paramvalues[i], sp->params[i].defaultvalue);
-
- // convert parameters from character string to parmeter type
- //
- srv_convert(srvproc,
- SRVCHAR,
- paramvalues[i],
- -1,
- sp->params[i].type,
- &convertvalues[i],
- sp->params[i].length);
- }
-
- // Execute the procedure
- //
- (sp->handler)(srvproc,
- &convertvalues[0],
- &convertvalues[1],
- &convertvalues[2],
- &convertvalues[3]);
- }
- return(SRV_CONTINUE);
- }
-
- //
- // LANG_PARSER
- // A procedure server specific language event parser.
- //
- // Parameters:
- // srvproc - process handle to the current client connection.
- // sp - Pointer to the stored procedure structure
- // paramvalues - An array of the values of the parameters.
- //
- // Returns:
- // SUCCEED
- //
- RETCODE lang_parser(srvproc, sp, paramvalues)
- SRV_PROC FAR *srvproc;
- SP_INFO **sp;
- DBCHAR paramvalues[MAXPARAMS][MAXLEN];
- {
- DBCHAR FAR *query; // pointer to language buffer
- int i;
- int numparams;
- DBCHAR msg[MAXLEN * 5];
- DBINT msgnum;
- DBCHAR spname[MAXLEN];
- BOOL paramsyntax = FALSE;
- DBCHAR paramname[MAXLEN];
- DBCHAR equalstring[2];
- DBCHAR *paramvalue = NULL;
-
- query = srv_langptr(srvproc);
-
- query = scan_next(query,
- spname);
- if (strlen(spname) == 0) {
- srv_senddone(srvproc, SRV_DONE_FINAL, 0, 0);
- return(FAIL);
- }
- if (strnicmp(spname, EXEC_CMD, (sizeof(EXEC_CMD) - 1)) == 0) {
- // stored procedure name
- //
- query = scan_next(query,
- spname);
- if (strlen(spname) == 0)
- goto syntax_error;
- }
-
- // Check for existence
- //
- for (i = 0; i < Rpcnumber; i++)
- if (strcmp(Sps[i].name, spname) == 0) {
- *sp = &Sps[i];
- break;
- }
- if (*sp == NULL) {
- sprintf(msg, "Procedure \'%s \' not found.", spname);
- msgnum = SP_UNKNOWN;
- goto error;
- }
-
- // Parameters
- //
- numparams = 0;
- while (*query != '\0') {
- if (++numparams > (*sp)->numparams) {
- sprintf(msg, (*sp)->usage);
- msgnum = INVALID_SP_SYNTAX;
- goto error;
- }
- if (!paramsyntax && *query == '@')
- paramsyntax = TRUE; // parameter name mode
-
- if (!paramsyntax)
- if (paramvalue == NULL)
- paramvalue = paramvalues[0];
- else
- paramvalue += MAXLEN;
-
- if (paramsyntax) {
- if (*query != '@') {
- sprintf(msg, "Once the form '@name = value' has been used, \
- all subsequent parameters must be passed in \
- the form '@name = value'."
- );
- msgnum = INVALID_SP_SYNTAX;
- goto error;
- }
- else
- query++;
-
- query = scan_next(query,
- paramname);
- if (strlen(paramname) == 0)
- goto syntax_error;
-
- // Get parameter index
- //
- paramvalue = NULL;
- for (i = 0; i < (*sp)->numparams; i++)
- if (strcmp((*sp)->params[i].name, paramname) == 0) {
- paramvalue = paramvalues[i];
- break;
- }
- if (paramvalue == NULL) {
- sprintf(msg,
- "Procedure '%s' does not recognize parameter name: %s",
- spname,
- paramname);
- msgnum = BAD_SP_PARAMETER;
- goto error;
- }
-
- // Already assigned value
- //
- if (strlen(paramvalue) > 0)
- goto syntax_error;
-
- // Check for '='
- //
- query = scan_next(query,
- equalstring);
- if (*equalstring != '=')
- goto syntax_error;
-
- }
- query = scan_next(query,
- paramvalue);
- if (strlen(paramvalue) == 0)
- goto syntax_error;
-
- if (*query == ',') {
- query++;
- while (*query == ' ' || *query == '\t')
- query++;
- }
- }
- return(SUCCEED);
-
- syntax_error:
- sprintf(msg, "Incorrect syntax found near '%s'.", query);
- msgnum = INVALID_SP_SYNTAX;
-
- error:
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- msgnum,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- msg,
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(FAIL);
- }
-
- //
- // SCAN_NEXT
- // Reads the next token in a string, ignoring whitespace.
- //
- // Parameters:
- // string - The language event string
- // word - The next token the string
- //
- // Returns:
- // The string incremented passed the token.
- //
- DBCHAR *scan_next(string, word)
- DBCHAR *string;
- DBCHAR *word;
- {
- DBCHAR *p;
-
- word[0] = '\0';
- if (*string == '\"' || *string == '\'') {
- // check for unclosed quote
- //
- p = strchr(string + 1,
- *string);
- if (p == NULL)
- return(string);
- strncpy(word, string + 1, p - (string + 1));
- word[p - (string + 1)] = '\0';
- string += 2;
- }
- else {
- // clear proceeding white space
- //
- while (*string == ' ' || *string == '\t' || *string == '\n')
- string++;
- sscanf(string, "%s", word);
-
- // ignore comments
- //
- while (strncmp(word, "/*", 2) == 0) {
- string = strstr(string, "*/");
- if (string != NULL) {
- string += 2;
- word[0] = '\0';
- while (*string == ' ' || *string == '\t' || *string == '\n')
- string++;
- sscanf(string, "%s", word);
- }
- else
- return(string);
- }
- }
- if (strlen(word) > 0)
- string += strlen(word);
-
- // clear trailing white space
- //
- while (*string == ' ' || *string == '\t' || *string == '\n')
- string++;
-
- return(string);
- }
-
- //
- // SP_EXECUTE
- // Execute a client stored procedure.
- //
- // Scans the list of defined stored procedures, checks the parameters and
- // executes the procedure. If results are returned it is the responsiblity
- // of the underlying proedure.
- //
- // Parameters:
- // srvproc - The process handle to use to send results to the client.
- //
- // Returns:
- // SRV_CONTINUE
- //
- RETCODE sp_execute(srvproc)
- SRV_PROC FAR *srvproc;
- {
- int i;
- int x;
- int y;
- int len;
- int numparams;
- SP_INFO *sp = NULL;
- DBCHAR msg[MAXLEN];
- DBINT msgnum;
- DBCHAR paramvalues[MAXPARAMS][MAXLEN];
- BOOL paramnamemode = FALSE;
- DBCHAR *paramname;
- DBINT paramtype;
- DBCHAR *value;
- DBINT type;
- DBINT status;
- DBCHAR server_spname[MAXNAME + 1];
-
- // Initialize parameter storage
- //
- for (i = 0; i < MAXPARAMS; i++) {
- memset(paramvalues[i], 0, MAXLEN);
- }
- for (i = 0; i < Rpcnumber; i++) {
- // Create name generated by calling server
- //
- if (strcmp(Sps[i].name, srv_rpcname(srvproc, (int *)NULL)) == 0 ||
- strcmp(srv_rpcname(srvproc, (int *)NULL),
- strcat(strcpy(server_spname, Sps[i].name),
- SP_EXT)) == 0) {
- sp = &Sps[i];
- break;
- }
- }
- if (sp == NULL) {
- sprintf(msg, "Procedure \'%s \' not found.",
- srv_rpcname(srvproc, (int *)NULL));
- msgnum = SP_UNKNOWN;
- goto error;
- }
- numparams = srv_rpcparams(srvproc);
- if (srv_paramname(srvproc, 1, &len) && len > 0)
- paramnamemode = TRUE;
- for (y = 1; y <= numparams; y++) {
- // Find parameter number
- //
- if (paramnamemode) {
- paramname = srv_paramname(srvproc, y, &len);
- if (strlen(paramname) == 0)
- goto parameter_error;
-
- if (*paramname == '@')
- paramname++;
- else
- goto parameter_error;
- value = NULL;
- for (x = 0; x < sp->numparams; x++) {
- if (strcmp(sp->params[x].name, paramname) == 0) {
- value = paramvalues[x];
- type = sp->params[x].type;
- status = sp->params[x].status;
- break;
- }
- }
- if (value == NULL)
- goto parameter_error;
- }
- else {
- value = paramvalues[y - 1];
- type = sp->params[y - 1].type;
- status = sp->params[y - 1].status;
- }
-
- // Check parameters for correct type
- //
- paramtype = srv_paramtype(srvproc, y);
- switch (paramtype) {
- case SRVVARCHAR: // Type sent by Servers instead of SRVCHAR
- paramtype = SRVCHAR;
- break;
- case SRVINTN: // Type sent by Servers instead of SRVINT
- paramtype = SRVINT4;
- break;
- default:
- break;
- }
-
- if (type != paramtype) {
- if (paramnamemode)
- sprintf(msg, "Parameter \'%s \' is incorrect type.", paramname);
- else
- sprintf(msg, "Parameter \'%d \' is incorrect type.", y);
- msgnum = BAD_SP_PARAMETER;
- goto error;
- }
-
- // Check parameters for correct status
- //
- if ((DBINT)srv_paramstatus(srvproc, y) != status) {
- if (paramnamemode)
- sprintf(msg, "Parameter \'%s \' has incorrect status.",
- paramname);
- else
- sprintf(msg, "Parameter \'%d \' had incorrect status.", y);
- msgnum = BAD_SP_PARAMETER;
- goto error;
- }
-
- // Move SP parameters to local variables
- //
- srv_bmove(srv_paramdata(srvproc, y), value, srv_paramlen(srvproc, y));
- value[srv_paramlen(srvproc, y)] = '\0';
-
- }
-
- // If unspecified, use default value
- //
- for (i = 0; i < sp->numparams; i++) {
- if (strlen(paramvalues[i]) == 0 &&
- strlen(sp->params[i].defaultvalue) == 0) {
- strcpy(msg, sp->usage);
- msgnum = INVALID_SP_SYNTAX;
- goto error;
- }
- if (strlen(paramvalues[i]) == 0 &&
- strlen(sp->params[i].defaultvalue) != 0)
- strcpy(paramvalues[i], sp->params[i].defaultvalue);
- }
-
- // Execute procedure
- //
- (*sp->handler)(srvproc,
- paramvalues[0],
- paramvalues[1],
- paramvalues[2],
- paramvalues[3]);
- return(SRV_CONTINUE);
-
- parameter_error:
- sprintf(msg, "Procedure '%s' does not recognize parameter name: %s",
- sp->name, paramname);
- msgnum = BAD_SP_PARAMETER;
-
- error:
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- msgnum,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- msg,
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(SRV_CONTINUE);
- }
-
- // EXIT_REMOTE
- // Handler for SRV_DISCONNECT events.
- //
- // The code to disconnect from the procedure server.
- //
- // Parameters:
- // srvproc - the handle to the client connection
- //
- // Returns:
- // SRV_DISCONNECT
- ///
- RETCODE exit_remote(srvproc)
- SRV_PROC FAR *srvproc;
- {
- char *string;
- int len;
-
- // Display info on console
- //
- string = srv_pfield(srvproc, SRV_CPID, &len);
- string[len] = '\0';
- printf("\nClient connection closed, process ID: %s\n", string);
-
- return(SRV_CONTINUE);
- }
-
- // CHK_ERR
- // Print out errors.
- //
- // Parameters:
- // server - pointer to procedure server server structure.
- // srvproc - pointer to client connection structure
- // errornum - error number.
- // severity - error severity.
- // state - error state.
- // oserrnum - operating system error number, if any.
- // errtext - the text of the error message.
- // errtextlen - length of the errtext message
- // oserrtext - the text of the operating system error message.
- // oserrtextlen - length of the errtext message
- //
- // Returns:
- // SRV_CONTINUE, SRV_CANCEL, or SRV_EXIT_PROGRAM
- //
-
- RETCODE chk_err(server, srvproc, errornum, severity, state, oserrnum, errtext,
- errtextlen, oserrtext, oserrtextlen)
- SRV_SERVER FAR *server;
- SRV_PROC FAR *srvproc;
- int errornum;
- BYTE severity;
- BYTE state;
- int oserrnum;
- DBCHAR FAR *errtext;
- int errtextlen;
- DBCHAR FAR *oserrtext;
- int oserrtextlen;
- {
- char log_buffer[256];
-
- // Operating system error?
- //
- if (oserrnum != SRV_ENO_OS_ERR) {
- sprintf(log_buffer, "SERVER OS ERROR: %d: %s.", oserrnum, oserrtext);
-
- srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
- }
-
- // Is this a fatal error for the server?
- //
- if (severity >= SRV_FATAL_SERVER) {
- sprintf(log_buffer,
- "SERVER: FATAL SERVER ERROR: errornum = %d, "
- "severity = %d, state = %d: %s.",
- errornum,
- severity,
- state,
- errtext);
-
- srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
- return(SRV_EXIT);
- }
- else {
- //
- // Did the "srvproc" get a fatal error?
- //
- if (severity >= SRV_FATAL_PROCESS) {
- sprintf(log_buffer,
- "SERVER: FATAL CONNECT ERROR: errornum = %d, "
- "severity = %d, state = %d: %s.",
- errornum,
- severity,
- state,
- errtext);
-
- srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
-
- return(SRV_CANCEL);
- }
- }
-
- // A non-fatal error or an information message received.
- // We'll pass it through to the client.
- //
- if (srvproc != (SRV_PROC *)NULL && (server != NULL))
- if (severity < 10) { // if informational message
- srv_sendmsg(srvproc,
- SRV_MSG_INFO,
- (DBINT)errornum,
- severity,
- 0,
- NULL,
- 0,
- 0,
- errtext,
- SRV_NULLTERM);
- }
- else { // must be an error message
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- (DBINT)errornum,
- severity,
- 0,
- NULL,
- 0,
- 0,
- errtext,
- SRV_NULLTERM);
- }
- else {
- sprintf(log_buffer, "GATEWAY ERROR: errornum = %d, severity = %d: %s",
- errornum, severity, errtext);
-
- srv_log(server, TRUE, log_buffer, SRV_NULLTERM);
- }
- return(SRV_CONTINUE);
- }
-
- // The following are the supported store procedure functions
- //
-
- //
- // PROCLIST
- // Returns the usage for all defined stored procedures
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- //
- // Returns:
- // SUCCEED
- //
- // Side Effects:
- // Returns a result set to client
- //
- RETCODE proclist(srvproc)
- SRV_PROC FAR *srvproc;
- {
- DBCHAR colname1[MAXNAME + 1];
- DBCHAR colname2[MAXNAME + 1];
- int i;
-
- sprintf(colname1, "spname");
- srv_describe(srvproc, 1, colname1, SRV_NULLTERM, SRVCHAR, MAXNAME + 1,
- SRVCHAR, 0, NULL);
-
- sprintf(colname2, "spusage");
- srv_describe(srvproc, 2, colname2, SRV_NULLTERM, SRVCHAR, MAXLEN, SRVCHAR,
- 0, NULL);
-
- // Return each SP handler as a row
- //
- for (i = 0; i < Rpcnumber; i++) {
- srv_setcoldata(srvproc, 1, Sps[i].name);
- srv_setcollen(srvproc, 1, strlen(Sps[i].name));
-
- srv_setcoldata(srvproc, 2, Sps[i].usage);
- srv_setcollen(srvproc, 2, strlen(Sps[i].usage));
-
- srv_sendrow(srvproc);
- }
- srv_senddone(srvproc, (SRV_DONE_COUNT | SRV_DONE_FINAL), 0, i);
-
- return(SUCCEED);
- }
-
- //
- // SP_EXEC
- // Execute a given command string and returns any output as rows of
- // text.
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- // command - the command string to execute
- //
- // Returns:
- // SUCCEED or FAIL
- //
- // Side Effects:
- // Returns messages and/or a result set to client
- //
- RETCODE sp_exec(srvproc, command)
- SRV_PROC FAR *srvproc;
- DBCHAR *command;
- {
- DBCHAR buf[MAXLEN * 5];
- FILE *cmdpipe;
- int length;
- DBINT rows;
- int status;
-
- // Execute string
- //
- if ((cmdpipe = _popen(command, "r")) == NULL) {
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Error executing command",
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(FAIL);
- }
-
- // Command successful
- //
- for (rows = 0; !feof(cmdpipe); ) {
- if (fgets(buf, sizeof(buf), cmdpipe) != NULL) {
- if (rows == 0) {
- // Describe result row:
- // it will be one column of size sizeof(buf)
- // We do this in the retrieval loop to ensure that the
- // row description will occur only if there are rows to be
- // outputted
- srv_describe(srvproc, 1, command, SRV_NULLTERM, SRVCHAR,
- sizeof(buf), SRVCHAR, sizeof(buf), buf);
- }
- length = strlen(buf) - 1;
- // get rid of new line character
- //
- buf[length] = '\0';
- srv_setcollen(srvproc, 1, length);
- srv_sendrow(srvproc);
- rows++;
- }
- }
- status = _pclose(cmdpipe);
- if (rows == 0) {
- if (status == 0) {
- srv_sendstatus(srvproc, 0);
- srv_sendmsg(srvproc,
- SRV_MSG_INFO,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Command executed successfully",
- SRV_NULLTERM);
- }
- else {
- sprintf(buf, "Error with command %s: OS error = %d", command,
- status);
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- buf,
- SRV_NULLTERM);
- }
- }
- srv_senddone(srvproc, SRV_DONE_FINAL | SRV_DONE_COUNT, 0, rows);
- return(SUCCEED);
- }
-
- //
- // SP_EXEC_TRIG
- // Execute a given command string without returning any output.
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- // command - the command string to execute
- //
- // Returns:
- // SUCCEED or FAIL
- //
- // Side Effects:
- // Returns success, error messages to client
- //
- RETCODE sp_exec_trig(srvproc, command)
- SRV_PROC FAR *srvproc;
- DBCHAR *command;
- {
- int status;
- DBCHAR buf[MAXLEN * 5];
-
- // Execute string
- //
- if ((status = system(command)) != 0) {
- sprintf(buf, "Error with command '%s': OS error = %d", command,
- status);
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- buf,
- SRV_NULLTERM);
-
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(FAIL);
- }
- srv_senddone(srvproc, SRV_DONE_FINAL, 0, 0);
- return(SUCCEED);
- }
-
-
- //
- // DISKLIST
- // Returns a row for each defined drive containing its name and the
- // amount of disk space available.
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- //
- // Returns:
- // SUCCEED
- //
- // Side Effects:
- // Returns a result set to client
- //
- RETCODE disklist(srvproc)
- SRV_PROC FAR *srvproc;
- {
- DBCHAR colname1[MAXNAME + 1];
- DBCHAR colname2[MAXNAME + 1];
- DBCHAR drivename;
- int drivenum;
- unsigned short disk;
- unsigned long drivenums;
- FSALLOCATE data;
- long space_remaining;
- int i = 0;
-
- sprintf(colname1, "drive");
- srv_describe(srvproc, 1, colname1, SRV_NULLTERM, SRVCHAR, 1, SRVCHAR, 1,
- (BYTE *)&drivename);
-
- sprintf(colname2, "bytes free");
- srv_describe(srvproc, 2, colname2, SRV_NULLTERM, SRVINT4, 4, SRVINT4, 4,
- (BYTE *)&space_remaining);
-
- DosQCurDisk(&disk, &drivenums);
-
- drivenums >>= 2; //Ignore drives A and B
- for (drivename = 'C', drivenum = 3; drivename <= 'Z';
- drivename++, drivenum++) {
- if (drivenums & 1) {
- i++;
- DosQFSInfo(drivenum, 1, (char far *)&data, sizeof(data));
- space_remaining = (long)data.cSectorUnit * (long)data.cUnitAvail;
- space_remaining *= (long)data.cbSector;
- srv_sendrow(srvproc);
- }
- drivenums >>= 1;
- }
- srv_senddone(srvproc, (SRV_DONE_COUNT | SRV_DONE_FINAL), 0, i);
- return(SUCCEED);
- }
-
- //
- // DISKFREE
- // Returns the amount of space available on a given drive. The value
- // is placed into the defined return parameter of the stored procedure.
- //
- // NOTE: This routine can not be called via a language event.
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- // drive - the drive letter to check
- //
- // Returns:
- // SUCCEED
- //
- // Side Effects:
- // Returns messages and/or a result set to client. Returns a value in the
- // defined return parameter.
- //
- RETCODE diskfree(srvproc, drive)
- SRV_PROC FAR *srvproc;
- DBCHAR *drive;
- {
- DBCHAR colname1[MAXNAME + 1];
- int drivenum;
- unsigned short disk;
- unsigned long drivenums;
- FSALLOCATE data;
- long space_remaining = -1;
- int i = 0;
-
- drive = strupr(drive);
-
- sprintf(colname1, "drive");
- srv_describe(srvproc, 1, colname1, SRV_NULLTERM, SRVCHAR, 1, SRVCHAR, 1,
- (BYTE *)drive);
- srv_sendrow(srvproc);
- srv_senddone(srvproc, (SRV_DONE_COUNT | SRV_DONE_MORE), 0, 1);
-
- DosQCurDisk(&disk, &drivenums);
-
- drivenum = drive[0] - 'A' + 1;
-
- drivenums >>= drivenum - 1; //Ignore drives A and B
- if (drivenums & 0x01) {
- DosQFSInfo(drivenum, 1, (char far *)&data, sizeof(data));
- space_remaining = (long)data.cSectorUnit * (long)data.cUnitAvail;
- space_remaining *= (long)data.cbSector;
- }
-
- // Process return parameter
- //
- if (srv_paramstatus(srvproc, 2) & 0x0001)
- srv_paramset(srvproc, 2, (BYTE *)&space_remaining, 4);
-
- srv_senddone(srvproc, SRV_DONE_FINAL, 0, 0);
- return(SUCCEED);
- }
-
-
- //
- // SCAN_XBASE
- // Reads an xBase file and sends it to the client as if it were a SQL
- // Server query result set (the equivalent of a 'SELECT * FROM
- // tablename' SQL statement).
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- // szFileName - dbase file path name
- //
- // Returns:
- // SUCCEED or FAIL
- //
- // Side Effects:
- // Returns messages and/or a result set to client
- //
- RETCODE scan_xbase(srvproc, filename)
- SRV_PROC FAR *srvproc;
- char *filename;
- {
- FILE *xbasefile;
- size_t count;
- char buffer[BUF_SIZE];
- int numrecords;
- int headerlength;
- int recordlength;
- int lengthlist[XBASE_MAX_COLUMNS];
- long i;
- int j;
- int position;
- int numcolumns;
-
- // now read the database header info
- //
- if ((xbasefile = fopen(filename, "r")) == NULL){
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Error reading xBase file",
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(FAIL);
- }
- count = fread(buffer, XBASE_HDR_SIZE, 1, xbasefile);
-
- if (count == 0) {
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Error reading xBase file",
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- fclose (xbasefile);
- return(FAIL);
- }
-
- numrecords = *((unsigned int *) &buffer[4]);
- headerlength = *((unsigned int *) &buffer[8]);
- recordlength = *((unsigned int *) &buffer[10]);
- numcolumns = (headerlength - 32 -1) / 32;
-
- // now get the column header information
- //
- for (j = 0; j < numcolumns; j++) {
- count = fread(buffer, XBASE_HDR_SIZE, 1, xbasefile);
- if ( count == 0 ) {
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Error reading xBase file",
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- fclose (xbasefile);
- return(FAIL);
- }
-
- // we need to NULL terminate the column name (if it is a
- // full 11 characters long)
- //
- buffer[11] = '\0';
-
- // now find our the column length for this data buffer
- //
- lengthlist[j] = (int) buffer[16];
-
- // now 'describe' this column
- //
- srv_describe(srvproc,
- j + 1, // column number
- buffer, // pointer to column name
- SRV_NULLTERM, // column name is NULL terminated
- SRVCHAR, // datatype is char (xBase numbers are ASCII
- lengthlist[j], // column length
- SRVCHAR, // destination datatype is also char
- lengthlist[j], // destination column length
- NULL); // pointer to where the data will be
-
- }
- // now read the one byte 'column header seperator'
- //
- count = fread(buffer, 1, 1, xbasefile);
- if (count == 0) {
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Error reading xBase file",
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- fclose (xbasefile);
- return(FAIL);
- }
-
- for (i = 0; i < numrecords; i++) {
- count = fread(buffer, recordlength, 1, xbasefile);
- if (count == 0) {
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- EXEC,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Error reading xBase file",
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- fclose (xbasefile);
- return(FAIL);
- }
-
- // check to see if this is a deleted row
- //
- if (buffer[0] == '*')
- break;
-
- // Now set the length and data pointers for each column
- for (j = 0, position = 1; j < numcolumns; j++) {
- srv_setcollen(srvproc, j+1, lengthlist[j]);
- srv_setcoldata(srvproc, j+1, &buffer[position]);
- position += lengthlist[j];
- }
-
- // send the row to the client.
- //
- srv_sendrow(srvproc);
- }
- srv_senddone(srvproc, SRV_DONE_COUNT | SRV_DONE_FINAL, 0, i);
- fclose (xbasefile);
- return(SUCCEED);
- }
-
-
-
- #ifdef LMPTK
- //
- // BROADCAST
- // Send the given message to the specified mailslot.
- // This is a Lan Manager specific stored procedure.
- //
- // Parameters:
- // srvproc - the handle to the client connection that got the SRV_CONNECT.
- // mailslot - the mailslot to broadcast to
- // msg - the text message
- //
- // Returns:
- // SUCCEED or FAIL
- //
- // Side Effects:
- // Returns a message to client
- //
- RETCODE broadcast(srvproc, mailslot, msg)
- SRV_PROC FAR *srvproc;
- DBCHAR *mailslot;
- DBCHAR *msg;
- {
- unsigned retcode;
- DBCHAR errmsg[MAXLEN];
-
- // Send message as second class mail
- //
- retcode = DosWriteMailslot(mailslot, msg, strlen(msg), 0, 2, 0);
- if (retcode) {
- sprintf(errmsg, "Unable to broadcast message. Error: %u", retcode);
- srv_sendstatus(srvproc, 1);
- srv_sendmsg(srvproc,
- SRV_MSG_ERROR,
- BROADCAST,
- SRV_INFO,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- errmsg,
- SRV_NULLTERM);
- srv_senddone(srvproc, (SRV_DONE_ERROR | SRV_DONE_FINAL), 0, 0);
- return(FAIL);
- }
-
- // Broadcast successful
- //
- srv_sendstatus(srvproc, 0);
- srv_sendmsg(srvproc,
- SRV_MSG_INFO,
- BROADCAST,
- (DBTINYINT)0,
- (DBTINYINT)0,
- NULL,
- 0,
- 0,
- "Broadcast complete.",
- SRV_NULLTERM);
- srv_senddone(srvproc, SRV_DONE_FINAL, 0, 0);
- return(SUCCEED);
- }
- #endif
-
-
- #pragma check_stack() // set stack checking to its default setting
-