/*****************************************************************************/ #ifdef COMMENTS_WITH_COMMENTS /* ws_echo.c A demonstrator for CGIplus Web Socket scripts. The JavaScript-driven client-side of this demonstrator is in WASD_ROOT:[SRC.WEBSOCKET]WS_ECHO.HTML An example of using, and test-bench for, the string descriptor versions of the wsLIB (WebSocket library) API. COPYRIGHT --------- Copyright (C) 2010-2012 Mark G.Daniel This program comes with ABSOLUTELY NO WARRANTY. This is free software, and you are welcome to redistribute it under the conditions of the GNU GENERAL PUBLIC LICENSE, version 3, or any later version. http://www.gnu.org/licenses/gpl.txt VERSION HISTORY --------------- 21-JUL-2012 MGD v1.0.1, WsLibDestroy() deprecated 07-AUG-2010 MGD v1.0.0, initial development */ #endif /* COMMENTS_WITH_COMMENTS */ /*****************************************************************************/ #define SOFTWAREVN "1.0.1" #define SOFTWARENM "WS_ECHO" #ifdef __ALPHA # define SOFTWAREID SOFTWARENM " AXP-" SOFTWAREVN #endif #ifdef __ia64 # define SOFTWAREID SOFTWARENM " IA64-" SOFTWAREVN #endif #ifdef __VAX # define SOFTWAREID SOFTWARENM " VAX-" SOFTWAREVN #endif #include #include #include #include #include #include #include #include #include #include #include "wslib.h" /* from libmsg.h */ #define LIB$_INVSTRDES 1409572 #define VMSok(x) ((x) & STS$M_SUCCESS) #define VMSnok(x) !(((x) & STS$M_SUCCESS)) #define Debug 0 #define DEFAULT_UNUSED_TIMEOUT 60 #define FI_LI "WS_ECHO", __LINE__ #define EXIT_FI_LI(status) { printf ("[%s:%d]", FI_LI); exit(status); } int ConnectedCount, UnusedTimeout = DEFAULT_UNUSED_TIMEOUT, UsageCount; struct EchoClient { int CharCount, CloseFrame, LineCount, PerKeyStroke; unsigned int ConnectTime, UnusedTime; char InputBuffer [256], RemoteHost [128], ServerName [128], StatusBuffer [256]; struct WsLibStruct *WsLibPtr; struct dsc$descriptor_s InputDsc, StatusDsc; }; /* function prototypes */ void AddClient (); void EchoThis (struct WsLibStruct*); void HeartBeat (); int HtmlEscapeDsc (struct dsc$descriptor_s*, int); void RemoveClient (struct WsLibStruct *wsptr); /*****************************************************************************/ /* */ main () { /*********/ /* begin */ /*********/ /* don't want the C-RTL fiddling with the carriage control */ stdout = freopen ("SYS$OUTPUT", "w", stdout, "ctx=bin"); WsLibInit (); /* no clients is two minutes in seconds */ WsLibSetLifeSecs (2*60); while (WsLibIsCgiPlus()) { WsLibCgiVar (""); sys$setast (0); UsageCount++; AddClient (); WsLibCgiPlusEof (); sys$setast (1); } exit (SS$_NORMAL); } /*****************************************************************************/ /* Allocate a client structure and add it to the head of the list. */ void AddClient () { static $DESCRIPTOR (GreetingFaoDsc, "!AZHello !AZ;  !AZ at !AZ"); int len, status; unsigned int CurrentTime; char *sptr, *zptr; char Greeting [256]; struct EchoClient *clptr; /*********/ /* begin */ /*********/ CurrentTime = WsLibTime (); clptr = calloc (1, sizeof(struct EchoClient)); if (!clptr) EXIT_FI_LI (vaxc$errno); strcpy (clptr->ServerName, WsLibCgiVar("SERVER_NAME")); strcpy (clptr->RemoteHost, WsLibCgiVar("REMOTE_HOST")); clptr->ConnectTime = CurrentTime; clptr->UnusedTime = CurrentTime + UnusedTimeout; clptr->InputDsc.dsc$b_class = DSC$K_CLASS_S; clptr->InputDsc.dsc$b_dtype = DSC$K_DTYPE_T; clptr->StatusDsc.dsc$b_class = DSC$K_CLASS_S; clptr->StatusDsc.dsc$b_dtype = DSC$K_DTYPE_T; clptr->StatusDsc.dsc$a_pointer = clptr->StatusBuffer; /* create a WebSocket library structure for the client */ if (!(clptr->WsLibPtr = WsLibCreate (clptr, RemoveClient))) { /* failed, commonly on some WebSocket protocol issue */ free (clptr); return; } /* no user interaction is whatever's defined in seconds */ WsLibSetIdleSecs (clptr->WsLibPtr, DEFAULT_UNUSED_TIMEOUT); /* open the IPC to the WebSocket (mailboxes) */ status = WsLibOpen (clptr->WsLibPtr); if (VMSnok(status)) EXIT_FI_LI(status); clptr->StatusDsc.dsc$w_length = sizeof(clptr->StatusBuffer); sys$fao (&GreetingFaoDsc, &clptr->StatusDsc.dsc$w_length, &clptr->StatusDsc, "\x07", clptr->RemoteHost, SOFTWAREID, clptr->ServerName); WsLibWriteDsc (clptr->WsLibPtr, &clptr->StatusDsc, WSLIB_ASYNCH); WsLibSetWakeCallback (clptr->WsLibPtr, HeartBeat, 1); /* queue an asynchronous read from the client */ clptr->InputDsc.dsc$a_pointer = clptr->InputBuffer; clptr->InputDsc.dsc$w_length = sizeof(clptr->InputBuffer); WsLibReadDsc (clptr->WsLibPtr, &clptr->InputDsc, &clptr->InputDsc, EchoThis); ConnectedCount++; } /*****************************************************************************/ /* Remove the client structure from the list and free the memory. */ void RemoveClient (struct WsLibStruct *wsptr) { struct EchoClient *clptr; /*********/ /* begin */ /*********/ clptr = WsLibGetUserData (wsptr); free (clptr); if (ConnectedCount) ConnectedCount--; } /*****************************************************************************/ /* Asynchronous read from a WebSocket client has concluded. Echo the string. Deliberately performed using three separate writes just to illustrate the asynchronous and granular nature of WebSockets. */ void EchoThis (struct WsLibStruct *wsptr) { static $DESCRIPTOR (CountFaoDsc, "!UL: "); static $DESCRIPTOR (NewLineDsc, "\n"); int len; char *cptr, *sptr, *zptr; char CountBuffer [32]; $DESCRIPTOR (CountDsc, CountBuffer); struct EchoClient *clptr; /*********/ /* begin */ /*********/ if (VMSnok (WsLibReadStatus(wsptr))) { /* WEBSOCKET_INPUT read error (can be EOF) */ WsLibClose (wsptr, 0, NULL); return; } clptr = WsLibGetUserData(wsptr); clptr->UnusedTime = WsLibTime() + UnusedTimeout; /* CAUTION! with multiple concurrent write status becomes unreliable */ if (clptr->InputDsc.dsc$w_length && clptr->InputDsc.dsc$a_pointer[0] == 0x07) { /*****************/ /* per keystroke */ /*****************/ clptr->InputDsc.dsc$a_pointer++; clptr->InputDsc.dsc$w_length--; /* dif'rent strokes for dif'rent browsers (prob'ly my fault) */ if (clptr->InputDsc.dsc$a_pointer[0] == 0 || clptr->InputDsc.dsc$a_pointer[0] == 10 || clptr->InputDsc.dsc$a_pointer[0] == 13) { if (clptr->CharCount) WsLibWriteDsc (clptr->WsLibPtr, &NewLineDsc, WSLIB_ASYNCH); clptr->PerKeyStroke = clptr->CharCount = 0; } else { if (!clptr->PerKeyStroke) { clptr->PerKeyStroke = 1; sys$fao (&CountFaoDsc, &CountDsc.dsc$w_length, &CountDsc, ++clptr->LineCount); /* deliberately (test-bench) blocking! */ WsLibWriteDsc (clptr->WsLibPtr, &CountDsc, NULL); } clptr->CharCount++; HtmlEscapeDsc (&clptr->InputDsc, sizeof(clptr->InputBuffer)); WsLibWriteDsc (clptr->WsLibPtr, &clptr->InputDsc, WSLIB_ASYNCH); } } else if (clptr->InputDsc.dsc$w_length) { /************/ /* per line */ /************/ if (clptr->PerKeyStroke) { clptr->PerKeyStroke = clptr->CharCount = 0; WsLibWriteDsc (clptr->WsLibPtr, &NewLineDsc, WSLIB_ASYNCH); } sys$fao (&CountFaoDsc, &CountDsc.dsc$w_length, &CountDsc, ++clptr->LineCount); /* deliberately (test-bench) blocking! */ WsLibWriteDsc (clptr->WsLibPtr, &CountDsc, NULL); HtmlEscapeDsc (&clptr->InputDsc, sizeof(clptr->InputBuffer)); WsLibWriteDsc (clptr->WsLibPtr, &clptr->InputDsc, NULL); WsLibWriteDsc (clptr->WsLibPtr, &NewLineDsc, WSLIB_ASYNCH); } /* queue the next asynchronous read from the client */ clptr->InputDsc.dsc$a_pointer = clptr->InputBuffer; clptr->InputDsc.dsc$w_length = sizeof(clptr->InputBuffer); WsLibReadDsc (clptr->WsLibPtr, &clptr->InputDsc, &clptr->InputDsc, EchoThis); } /*****************************************************************************/ /* Called by wsLIB every second to provide status information to each client. */ void HeartBeat (struct WsLibStruct *wsptr) { static $DESCRIPTOR (StatusFaoDsc, "!AZHello !AZ;  Connected: !ULS;  Timeout: !ULS;  \ Connected: !UL;  Usage: !UL"); int len, status; unsigned int CurrentTime; char StatusBuffer [256]; $DESCRIPTOR (StatusDsc, StatusBuffer); struct EchoClient *clptr; /*********/ /* begin */ /*********/ CurrentTime = WsLibTime (); clptr = WsLibGetUserData (wsptr); clptr->StatusDsc.dsc$w_length = sizeof(clptr->StatusBuffer); sys$fao (&StatusFaoDsc, &clptr->StatusDsc.dsc$w_length, &clptr->StatusDsc, "\x07", clptr->RemoteHost, CurrentTime - clptr->ConnectTime, clptr->UnusedTime - CurrentTime, ConnectedCount, UsageCount); WsLibWriteDsc (wsptr, &clptr->StatusDsc, WSLIB_ASYNCH); } /*****************************************************************************/ /* Escape HTML-reserved characters. Does this in-situ! Provided there is enough trailing storage. If not the escaped string gets truncated in various (safe) ways. */ int HtmlEscapeDsc ( struct dsc$descriptor_s *StringDsc, int SizeOfString ) { int extralen; char *cptr, *czptr, *sptr, *zptr; /*********/ /* begin */ /*********/ if (StringDsc->dsc$b_class != DSC$K_CLASS_S && StringDsc->dsc$b_dtype != DSC$K_DTYPE_T) return (LIB$_INVSTRDES); czptr = (cptr = StringDsc->dsc$a_pointer) + StringDsc->dsc$w_length; extralen = 0; while (cptr < czptr) { switch (*cptr) { case '<' : extralen += 3; break; case '>' : extralen += 3; break; case '&' : extralen += 4; break; } cptr++; } if (!extralen) return (SS$_NORMAL); zptr = StringDsc->dsc$a_pointer + SizeOfString - 1; sptr = cptr + extralen; if (sptr < zptr) *(unsigned short*)sptr = '\0\0'; else *zptr = '\0'; while (cptr >= StringDsc->dsc$a_pointer) { switch (*cptr) { case '<' : sptr -= 3; if (sptr' : sptr -= 3; if (sptrdsc$w_length = sptr - StringDsc->dsc$a_pointer; return (SS$_NORMAL); } /*****************************************************************************/