Skip to content

C# replacement of the Apache Guacamole server-side Java servlet.

License

Notifications You must be signed in to change notification settings

m-expunged/guacamole-sharp

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

guacamole-sharp

Docker Image Version (latest by date)

⚠️ Deprecated previous version of guacamole-sharp docker image is now available under the "1.0.0" tag and "1.0.0" branch of the repository.

guacamole-sharp is a C# replacement of the Apache Guacamole server-side Java servlet. It is intended for customizable integration of Apache Guacamole into existing frontend projects with their own user and connection management.

Like the official Java guacamole-client guacamole-common, guacamole-sharp makes use of the guacamole-protocol and is intended to be used with guacamole-common-js (frontend client) and guacd (server-side proxy) by Apache. Check out the Apache Guacamole Docs for detailed information. Definitely skim over the developer section of the docs at least once so you know what you are dealing with!

Inspired by Vadim Pronin's guacamole-lite.

Usage

While not strictly required, the intended use of guacamole-sharp is with Docker.

Quick rundown

  • Prepare SSH/RDP/VNC on the target machine
  • Run guacd docker container
  • Run guacamole-sharp docker container
  • Generate connection token through guacamole-sharp endpoint
  • Send connection token to guacamole-sharp via guacamole-common-js client
  • Display connection in your frontend

guacamole-sharp diagram

Docker

guacamole-sharp requires two containers to work:

  • guacamole/guacd
docker run --name guacd
-p 4822:4822
-d guacamole/guacd
docker run --name guacamolesharp
--link guacd:guacd
-e LOG_LEVEL=Information
-e GuacamoleSharp:Password=YourTokenEncryptionPasswordHere
-e Guacd:Hostname=guacd
-e Guacd:Port=4822
-p 80:80
-d manuelexpunged/guacamolesharp:2.1.0

With the configuration above, the guacamole-sharp container will listen for API/WebSocket connections on port 80.

Logging level can be controlled with the "LOG_LEVEL" environment variable. The default level is "Information", anything below that should not be used in production since sensitive information is exposed in the logs.

Accepted log levels:

//
// Summary:
//     Specifies the meaning and relative importance of a log event.
public enum LogEventLevel
{
    //
    // Summary:
    //     Anything and everything you might want to know about a running block of code.
    Verbose,
    //
    // Summary:
    //     Internal system events that aren't necessarily observable from the outside.
    Debug,
    //
    // Summary:
    //     The lifeblood of operational intelligence - things happen.
    Information,
    //
    // Summary:
    //     Service is degraded or endangered.
    Warning,
    //
    // Summary:
    //     Functionality is unavailable, invariants are broken or data is lost.
    Error,
    //
    // Summary:
    //     If you have a pager, it goes off when one of these occurs.
    Fatal
}

A docker compose example is included in the repository.

The connection token

guacamole-sharp transports the arguments for remote connections via an encrypted connection token. This is done to provide a bare minimum of protection against rogue connections without to much hassle for users. Ideally you should build additional authentication/authorization layers on top of the token.

The token is passed to the guacamole-common-js client instance as a parameter of the connect method.

let tunnel = new Guacamole.WebSocketTunnel("ws://localhost:80/connect");
let client = new Guacamole.Client(tunnel);

client.connect(
  "token=hD12AB5Js4WD0Cse6mtgw_8msAieiSi1-vHajL2vAZgAo24yPufxuLKefeZxEYyWXhcbW21iv53Pv18gTXTnXp1i7wClkQ2tDutnIHqrHRo"
);

To generate such a token, guacamole-sharp exposes an endpoint.

'HOSTNAME:PORT/token/TOKEN_ENCRYPTION_PASSWORD'

Curl example:

curl -X 'POST' \
 'http://localhost:5072/token/YourTokenEncryptionPasswordHere' \
 -H 'accept: text/plain' \
 -H 'Content-Type: application/json' \
 -d '{
"arguments": {
"hostname": "127.0.0.1",
"port": "22"
},
"type": "ssh"
}'

Connection object example:

{
  "arguments": {
    "hostname": "127.0.0.1",
    "port": "3389",
    "domain": "mycooldomain",
    "username": "user",
    "password": "password",
    "security": "any",
    "ignore-cert": "true"
  },
  "type": "rdp"
}

The request requires two things:

  • The token encryption password as query parameter
  • A connection object inside the request body

The token encryption password is defined while creating the Docker container through the environment variables (alternatively inside the appsettings.json if you are building your own image).

The connection object contains all parameters guacd needs to create a connection.

Type decides the type of connection (ssh, vnc, rdp, ...) and arguments is used to configure the connection and pass username and password of the machine you want to connect to. The Apache Guacamole documentation contains a full list of arguments and their meaning.

guacamole-common-js

In order to use guacamole-sharp, you will need to use the guacamole-common-js library in your frontend. You can find a detailed explanation in the Apache Guacamole Docs.

A good way to get started is to look at the Angular example that is included in the repository, using the guacamole-common-js npm package. You might need to create a d.ts file if you are using Typescript in your project.

⚠️ Resolution of VNC connections must be handled client side since the VNC protocol does not define any mechanism for the connecting client to indicate the desired screen size.

Also, pay attention to the z-index, height and width of the guacamole-common-js display element and its parent/child elements, specifically the canvas elements. In some cases the lib creates them with weird values (e.g. z-index: -1) and you might need to change them after the element is added to the DOM.

If you want to include your own examples or notice something that could have been done better feel free to open a issue/pull request.

Advanced usage

appsettings.json

In order to simplify the configuration of connections, you can specify default and unencrypted arguments for different connection types inside the appsettings.json and build your own docker image from that.

Default Guacd, WebSocket server and token arguments

By default, guacamole-sharp tries to connect to guacd on port 4822. If you want to specify a different default port you can change them in the appsettings.json. Here you can also specify the token password if Docker isn't an option. Make sure the guacd port here is the same as the port of the actual guacd docker image!

"GuacamoleSharp": {
  "Password": "YourTokenEncryptionPasswordHere"
},
"Guacd": {
  "Hostname": "127.0.0.1",
  "Port": 4822
},

Default connection arguments

Since many connection arguments will be the same across different connections, it might be convenient to define them as default values.

Any valid guacamole-protocol argument can be used. The most common arguments are already preconfigured and will work out of the box. You might want to change 'width' and 'height' to your preferred resolution.

The properties inside 'DefaultArguments' are generally lowercase.

"Client": {
  "DefaultArguments": {
    "rdp": {
      "args": "connect",
      "port": 3389,
      "width": 1024,
      "height": 768,
      "dpi": 96
    },
    "vnc": {
      "args": "connect",
      "port": 5900,
      "width": 1024,
      "height": 768,
      "dpi": 96
    },
    "ssh": {
      "args": "connect",
      "port": 22,
      "width": 1024,
      "height": 768,
      "dpi": 96,
      "server-alive-interval": 10
    },
    "telnet": {
      "args": "connect",
      "port": 23,
      "width": 1024,
      "height": 768,
      "dpi": 96
    }
  }
}

Unencrypted connection arguments

While most arguments for connections should be packed into the encrypted token string, there is also the option to send certain arguments as an unencrypted query parameter.

To allow arguments to be sent unencrypted, add them to the unencrypted connection arguments dictionary for their respective connection type inside the appsettings.json.

All valid guacamole-protocol arguments can used in their unencrypted form, except for the connection type (ssh/rdp/...) which is the only argument that is required to be passed via the connection token.

The properties inside 'UnencryptedArguments' are generally lowercase.

By default, these arguments can be sent unencrypted:

"UnencryptedArguments": {
  "rdp": ["width", "height", "dpi"],
  "vnc": ["width", "height", "dpi"],
  "ssh": ["color-scheme", "font-name", "font-size", "width", "height", "dpi"],
  "telnet": ["color-scheme", "font-name", "font-size", "width", "height", "dpi"]
}

Example unencrypted arguments usage:

let tunnel = new Guacamole.WebSocketTunnel("ws://localhost:80/connect");
let client = new Guacamole.Client(tunnel);

let connectionString =
  "token=hD12AB5Js4WD0Cse6mtgw_8msAieiSi1-vHajL2vAZgAo24yPufxuLKefeZxEYyWXhcbW21iv53Pv18gTXTnXp1i7wClkQ2tDutnIHqrHRo";
connectionString += "&width=1024";
connectionString += "&height=768";

client.connect(connectionString);

Fallback arguments

Some arguments have fallback values and can be left empty in appsettings.json/Dockerfile:

Guacd:Hostname = "localhost"
Guacd:Port = 4822

Arguments overwrite priority

If a guacamole-protocol argument is defined multiple times, it will be overwritten by the argument with the highest priority. The order of priority is as follows (from highest to lowest):

  • Unencrypted values (request query params)
  • Token string values (request)
  • Default values (appsettings.json)
  • Fallback values (default class property values)

Releases

No releases published

Packages

No packages published