Skip to content

UltraVNC repeater implemented in NodeJS (can be used as library or fully functional CLI tool)

License

Notifications You must be signed in to change notification settings

Tomas2D/uvnc-repeater

Repository files navigation

💻 🔄 uvnc-repeater (UltraVNC Repeater)

UltraVNC Repeater Implementation in Node.js, for use as a library within a larger application or as a standalone CLI tool. This library draws inspiration from the Perl implementation.

Show some love and ⭐️ this project!

Library

Installation

yarn add uvnc-repeater
npm install uvnc-repeater
// CommonJS
const { UltraVNCRepeater } = require("uvnc-repeater");

// ESM / Typescript
import { UltraVNCRepeater } from "uvnc-repeater";

Examples

Minimal working setup

import { UltraVNCRepeater } from "uvnc-repeater";

const repeater = new UltraVNCRepeater();

// The server gateway will be started on port 5500.
// The client gateway will be started on port 5900.
await repeater.start();

// Get active connections (both client and server are defined)
const activeConnections = repeater.getActiveConnections();

// Get pending connections (either client or server will be defined)
const pendingConnections = repeater.getPendingConnections();

// Close repeater (pass true as first parameter to force shutdown)
await repeater.stop();

Custom settings

import { UltraVNCRepeater } from "uvnc-repeater";

const repeater = new UltraVNCRepeater({
  serverPort: 5500,
  clientPort: 5900,

  // Reads and waits for the first N bytes from an incoming connection.
  // In this data chunk, the target ID or 'host:port' should be contained.
  bufferSize: 250,

  // Timeout (seconds) in which we should gather `bufferSize` bytes from the connection socket
  socketFirstDataTimeout: 60 * 5,

  // Prevent direct connecting to the target when 'host:port' is sent instead of ID.
  refuseDirectHookup: false,
  
  // Prevents multiple connections to the same target, instead override the existing connection.
  refuse: false,

  // Does not send "RFB 000.000" to the client when the connection starts.
  noRFB: false,

  // Specify a custom path for logging (default is standard output).
  logFile: "/tmp/repeater.log",

  // Log levels: "info", "debug", "fatal", "warn", "trace", "error", "warn", "silent".
  logLevel: "debug",

  // Timeout in seconds (in case of inactivity) for all connections.`
  socketTimeout: 60 * 5,

  // Keep-Alive interval in seconds
  keepAlive: 30,

  // Keep-Alive maximal retries before connection is closed
  keepAliveRetries: 1,

  // Time in seconds for a graceful shutdown (close() method or on SIGTERM/SIGINT signal).
  // pass Infinity (Number.POSITIVE_INFINITY) to disable such a feature
  killTimeout: 30,
});

await repeater.start();
// ...
// ...
await repeater.stop();

Custom logger

import {
  UltraVNCRepeater,
  createLogger,
  createFileLogger, // for files
} from "uvnc-repeater";

const repeater = new UltraVNCRepeater({
  logger: createLogger({
    /* custom options for a pino logger */
  }),
});
await repeater.start();

Custom logic

If you want to extend the repeater functionality, you can extend the UltraVNCRepeater class and override some of its methods.

import {
  UltraVNCRepeater,
  NewClientConnectionEvent,
  NewServerConnectionEvent,
  CloseConnectionEvent,
} from "uvnc-repeater";
import { Socket } from "node:net";

class MyRepeater extends UltraVNCRepeater {
  constructor(options: Partial<VNCRepeaterOptions>) {
    super(options);
  }

  protected async _onNewClient(event: NewClientConnectionEvent): Promise<void> {
    await super._onNewClient(event);
    // do something
  }

  protected async _onCloseClient(event: CloseConnectionEvent): Promise<void> {
    await super._onCloseClient(event);
    // do something
  }

  protected async _onNewServer(event: NewServerConnectionEvent): Promise<void> {
    return super._onNewServer(event);
    // do something
  }

  protected async _onCloseServer(event: CloseConnectionEvent): Promise<void> {
    await super._onCloseServer(event);
    // do something
  }
}

const repeater = new MyRepeater({
  level: "debug",
});
await repeater.start();
// ...
// ...
await repeater.stop();

Events

Because the core class UltraVNCRepeater extends the Node.js EventEmitter, one can listen for the following events.

import { Event } from "uvnc-repeater";

// Repeater
Event.BEFORE_REPEATER_START;
Event.AFTER_REPEATER_START;
Event.BEFORE_REPEATER_STOP;
Event.AFTER_REPEATER_STOP;

// Repeater - server
Event.BEFORE_SERVER_GATEWAY_START;
Event.AFTER_SERVER_GATEWAY_START;
Event.BEFORE_SERVER_GATEWAY_STOP;
Event.AFTER_SERVER_GATEWAY_STOP;

// Repeater - client
Event.BEFORE_CLIENT_GATEWAY_START;
Event.AFTER_CLIENT_GATEWAY_START;
Event.BEFORE_CLIENT_GATEWAY_STOP;
Event.AFTER_CLIENT_GATEWAY_STOP;

// Server
Event.BEFORE_SERVER_CLOSE;
Event.AFTER_SERVER_CLOSE;
Event.BEFORE_SERVER_NEW;
Event.SERVER_NEW_ADDED_TO_PENDING;
Event.SERVER_NEW_PREVENT_HOOKUP;
Event.SERVER_NEW_HOOKUP;
Event.SERVER_NEW_OVERRIDE_OLD;

// Client
Event.BEFORE_CLIENT_NEW;
Event.CLIENT_NEW_HOOKUP_DIRECT;
Event.BEFORE_CLIENT_CLOSE;
Event.AFTER_CLIENT_CLOSE;
Event.CLIENT_NEW_ADDED_TO_PENDING;
Event.CLIENT_NEW_DIRECT_INVALID_PORT;
Event.CLIENT_NEW_DIRECT_INVALID_HOST;
Event.CLIENT_NEW_HOOKUP;
Event.CLIENT_NEW_PREVENT_HOOKUP;
Event.CLIENT_NEW_OVERRIDE_OLD;

Example

import { UltraVNCRepeater, Event } from "uvnc-repeater";

const repeater = new UltraVNCRepeater();
repeater.on(Event.CLIENT_NEW_ADDED_TO_PENDING, (event) => {
  console.log(`Client with ID: ${event.id} is waiting for the server.`);
});
await repeater.start();

Manual Connection Closing

import {UltraVNCRepeater, Event} from "uvnc-repeater";
import {createConnection} from "net";

const repeater = new UltraVNCRepeater();
await repeater.start();

// ... after a while
const activeConnections = repeater.getActiveConnections()
for (const [internalId, connection] of activeConnections) {
	if (connection.id === "10000") {
		await repeater.closeConnection(connection)
	}
}

const pendingConnections = repeater.getPendingConnections()
if (pendingConnections.has("10000")) {
  const connection = pendingConnections.get("10000")
  await repeater.closeConnection(connection)
}

⌨️ CLI

If you want to use the repeater as is, you can install it as a CLI tool.

Global Installation

yarn global add uvnc-repeater
npm install -g uvnc-repeater

Commands

Show Help

uvnc-repeater --help

Start Repeater

uvnc-repeater

Start Repeater with Custom Options

uvnc-repeater --serverPort 5555

Or use environmental variables (or both) to set custom options.

REPEATER_SERVER_PORT=5555 uvnc-repeater

To list all possible options, see help (uvnc-repeater --help)

📃 TODO

  • Tests

About

UltraVNC repeater implemented in NodeJS (can be used as library or fully functional CLI tool)

Resources

License

Stars

Watchers

Forks

Sponsor this project

 

Packages

No packages published