npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

mg_web_js

v1.1.2

Published

A JavaScript server for mg_web.

Downloads

26

Readme

mg_web_js

A JavaScript server for mg_web.

Chris Munt [email protected]
10 June 2024, MGateway Ltd http://www.mgateway.com

  • Current Release: Version: 1.1; Revision 2.
  • Release Notes can be found at the end of this document.

Contents

Overview

mg_web_js is a JavaScript server for mg_web. This server completely replaces the functionality that would otherwise be provided by the DB Server based superserver mgsi.

mg_web provides a high-performance minimalistic interface between three popular web servers ( Microsoft IIS, Apache and Nginx ) and M-like DB Servers ( YottaDB, InterSystems IRIS and Cache ). It is compliant with HTTP version 1.1 and 2.0 and WebSockets are supported. mg_web can connect to a local DB Server via its high-performance API or to local or remote DB Servers via the network.

Prerequisites

  • mg_web. The mg_web package can be downloaded from here.

  • A supported web server. Currently mg_web supports Microsoft IIS, Apache and Nginx.

  • A database. InterSystems Cache/IRIS or YottaDB (or similar M DB Server):

    • https://www.intersystems.com/
    • https://yottadb.com/

mg_web configuration

This section will describe how to configure mg_web to communicate with the JavaScript server (mg_web_js) as opposed to the DB Server based mgsi superserver.

For JavaScript, the server type should be defined as Node.js in the mg_web configuration file (mgweb.conf). For example:

 <server NodeJS>
   type Node.JS
   host 127.0.0.1
   tcp_port 7777
 </server>

Define a mapping to your JavaScript application

In the package there is an example application module named application.mjs. This can be defined in the mg_web configuration as follows:

 <location /mgweb/js>
   function ./application.mjs
   servers NodeJS
 </location>

Web requests with a path of /mgweb/js will be routed to application.mjs for processing. The handler function in application.mjs takes the following form and will be described in more detail in a later section.

 let handler = function(web_server, cgi, content, sys) {
   // Process HTTP web request and return response
   return result;
 }
 export {handler};

Define a mapping to a JavaScript WebSocket application

In the package there is an example WebSocket module named websocket.mjs. This can be defined in the mg_web configuration as follows:

 <location /mgweb/js>
   function ./application.mjs
   websocket websocket.mgw ./websocket.mjs
   servers NodeJS
 </location>

This WebSocket application can be invoked from the client using:

 ws = new WebSocket(((window.location.protocol == "https:") ? "wss:" : "ws:") + "//" + window.location.host + "/mgweb/js/websocket.mgw");

The handler function in websocket.mjs takes the following form and will be described in more detail in a later section.

 let handler = function(websocket_server, cgi, content, sys) {
   // Server-side of the websocket
   return "";
 }
 export {handler};

Starting the server

An example, showing how to start the server to listen on a particular TCP port is given in the /example/server.mjs module. Using this module, the server is started as follows:

 node server.mjs

Contents of server.mjs:

 import {mg_web} from 'mg_web_js';

 mg_web({
   port: 7777
 });

This will start the server listening on TCP port 7777. Modify this file if you wish to use an alternative TCP port.

Application code

The handler function in the application takes the following form:

 let handler = function(web_server, cgi, content, system) {
   // Process HTTP web request and return response
   return result;
 }
 export {handler};

Where:

  • web_server: Methods and properties related to the hosting web server.
  • cgi: List of CGI Environment Variables.
  • content: The request payload (if any).
  • system: Read-only system array reserved for mg_web_js use.

There are two methods for returning a response to the client via mg_web and the hosting web server. You can generate a complete response (including a HTTP response header) as a single string. Alternatively, responses involving large volumes of data can be streamed to the client.

Returning a complete response as a string

 let handler = function(web_server, cgi, content, system) {
   //
   // Create HTTP response headers
   let result = "HTTP/1.1 200 OK\r\n";
   result = result + "Content-type: text/html\r\n";
   result = result + "\r\n";
   //
   // Add the HTML content
   result = result + "<html>\r\n";
   result = result + "<head><title>\r\n";
   result = result + "Hello World\r\n";
   result = result + "</title></head>\r\n";
   result = result + "<h1>Hello World</h1>\r\n";
   return result;
 }

Streaming a response

Response data can be incrementally streamed back to the client using the write method:

 web_server.write(<data>);

Where:

  • data: Data string to be sent.

Example:

 let handler = function(web_server, cgi, content, system) {
   //
   // Create HTTP response headers
   let result = "HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\nConnection: close\r\n\r\n";
   //
   // Return CGI environment variables
   //
   cgi.forEach((value, name) => {
     result = result + "CGI variable " + name + " : " + value + "\r\n";
   });
   //
   // Stream back what we have so far
   //
   web_server.write(result);
   //
   // Stream back the request data
   //
   web_server.write("Request payload: " + content.toString());
   //
   // return with a final message
   //
   result = "\r\n*** End of Response ***\r\n";
   return result;
 }

Server-sent event code

When the client requests a SSE channel, the web_server sse property will be set to true. The SSE channel should be initialised by calling the initsse method.

 web_server.initsse(system, <options>);

Where:

  • options: Reserved for future use.

Data is dispatched to the client using the write method.

 web_server.write(<data>);

Where:

  • data: Data string to be sent.

The SSE server can close the channel using the close method.

 web_server.close();

Example:

let handler = function(web_server, cgi, content, system) {
  if (web_server.sse === true) {
    //
    // Initialise SSE server
    //
    let result = web_server.initsse(sys, "");
    let n = 0;
    let d = 0;
    var intervalId = setInterval(function () {
      n ++;
      d = new Date();
      //
      // Write some data to the client
      //
      web_server.write('data: SSE data line ' + n + ' Date: ' + d.toLocaleString() + '\r\n\r\n');
      //
      // Close channel after 10 data items have been dispatched
      //
      if (n > 9) {
        clearInterval(intervalId);
        web_server.close();
      }
    }, 5000);
    return result;
  }

The above simple example sends a line of data to the client every 5 seconds and closes the channel after the tenth line has been sent.

WebSocket code

The handler function in the WebSocket application takes the following form:

 let handler = function(websocket_server, cgi, content, sys) {
   // Server-side of the websocket
   return "";
 }
 export {handler};
  • websocket_server: Methods and properties related to the WebSocket opened by the hosting web server.
  • cgi: List of CGI Environment Variables.
  • content: The request payload (if any).
  • system: Read-only system array reserved for mg_web_js use.

The first task is to create a WebSocket client class containing a read method for accepting incoming data from the client.

   class websocket_client {
     read(websocket_server, data) {
       // process the data received from the client in 'data'
     }
   }

The WebSocket channel should be initialised by calling the initialisation method.

websocket_server.init(system, <binary>, <options>);

Where:

  • binary: Set to 0 for textual data; 1 for binary data.
  • options: Reserved for future use.

Now create an instance of the WebSocket client class and register it in the websocket_server.

  websocket_server.client = new websocket_client();

Having completed these initial tasks, the WebSocket server is ready for use. Data is dispatched to the client using the write method.

 websocket_server.write(<data>);

Where:

  • data: Data string (or array of binary data) to be sent.

The WebSocket server can close the channel using the close method.

 websocket_server.close();

Example:

 let handler = function (websocket_server, cgi, content, sys) {

   // Define a class to represent client data
   class websocket_client {

     // Mandatory read method to accept client data 
     read(websocket_server, data) {
       let d = new Date();
       // Acknowledge data from client
       websocket_server.conn.write('Data received from Client: ' + data + ' at: ' + d.toLocaleString());
     }
   }

   // Initialise server side
   websocket_server.init(sys, 0, "");

   // Create instance of client class for this WebSocket connection
   websocket_server.client = new ws_client();

   // Send inital message to client
   websocket_server.write('Hello from Server');

   return "";
 }

This simple example echoes back data received from the client with the date and time appended.

License

Copyright (c) 2019-2024 MGateway Ltd, Surrey UK.
All rights reserved.

http://www.mgateway.com
Email: [email protected]

Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at

  http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.

Release Notes

v1.0.1 (4 April 2024)

  • Initial Release

v1.0.1a (5 April 2024)

  • Add some configuration notes to this README file.

v1.1.2 (10 June 2024)

  • Introduce support for Server-Sent Events (SSE).