What is tunnel-ssh?
The tunnel-ssh npm package allows you to create SSH tunnels in Node.js applications. This can be useful for securely connecting to remote servers, databases, or other services that require SSH tunneling.
What are tunnel-ssh's main functionalities?
Basic SSH Tunnel
This code sample demonstrates how to set up a basic SSH tunnel using the tunnel-ssh package. The configuration object includes details such as the SSH username, host, port, destination host and port, local host and port, and the private key for authentication.
const tunnel = require('tunnel-ssh');
const config = {
username: 'user',
host: 'remote.server.com',
port: 22,
dstHost: '127.0.0.1',
dstPort: 27017,
localHost: '127.0.0.1',
localPort: 27017,
privateKey: require('fs').readFileSync('/path/to/private/key')
};
tunnel(config, function (error, server) {
if (error) {
console.error('SSH connection error: ', error);
} else {
console.log('SSH tunnel established');
}
});
SSH Tunnel with Forwarding
This code sample shows how to set up an SSH tunnel with port forwarding. The 'keepAlive' option is set to true to maintain the connection. This is useful for applications that need to keep a persistent connection to a remote database or service.
const tunnel = require('tunnel-ssh');
const config = {
username: 'user',
host: 'remote.server.com',
port: 22,
dstHost: '127.0.0.1',
dstPort: 3306,
localHost: '127.0.0.1',
localPort: 3306,
privateKey: require('fs').readFileSync('/path/to/private/key'),
keepAlive: true
};
tunnel(config, function (error, server) {
if (error) {
console.error('SSH connection error: ', error);
} else {
console.log('SSH tunnel with forwarding established');
}
});
Dynamic Port Forwarding
This code sample demonstrates how to set up a dynamic SSH tunnel, which can be used as a SOCKS proxy. The 'dynamic' option is set to true, allowing the tunnel to dynamically forward traffic to different destinations.
const tunnel = require('tunnel-ssh');
const config = {
username: 'user',
host: 'remote.server.com',
port: 22,
dstHost: '0.0.0.0',
dstPort: 0,
localHost: '127.0.0.1',
localPort: 1080,
privateKey: require('fs').readFileSync('/path/to/private/key'),
dynamic: true
};
tunnel(config, function (error, server) {
if (error) {
console.error('SSH connection error: ', error);
} else {
console.log('Dynamic SSH tunnel established');
}
});
Other packages similar to tunnel-ssh
ssh2
The ssh2 package is a more general-purpose SSH client for Node.js. It provides a wide range of SSH functionalities, including tunneling, executing commands, and file transfers. Compared to tunnel-ssh, ssh2 offers more flexibility and control over SSH connections but requires more setup and configuration.
node-ssh
The node-ssh package is a high-level SSH client for Node.js that simplifies the process of connecting to SSH servers, executing commands, and transferring files. It is easier to use than ssh2 but offers less control over the SSH connection. It is a good alternative to tunnel-ssh for users who need basic SSH functionalities without the complexity.
ssh2-sftp-client
The ssh2-sftp-client package is built on top of ssh2 and provides a simplified API for SFTP (SSH File Transfer Protocol) operations. While it focuses on file transfers, it can also be used to create SSH tunnels. It is a good choice for users who primarily need SFTP capabilities with occasional SSH tunneling.
Tunnel-SSH
One to connect them all !
Tunnel-ssh is based on the fantastic ssh2 library by Brian White.
Trouble ? Please study the ssh2 configuration.
Changelog 4.1.0 / 2016-08-09
- style: Change codestyle to xo.
- Refactor: Improved configuration error handling.
- refactor: clean up examples
- style: Remove jscs in favor of eslint.
- doc: Improve examples
- Merge pull request #42 from kinsi55/master
- Merge pull request #43 from pedroventura/master
- doc: Update Readme.md
- Fix wrong comparison causing localPort to stay "null" if not explicitly defined.
- Doc: Update Readme.md
related projects
Integration
By default tunnel-ssh will close the tunnel after a client disconnects, so your cli tools should work in the same way, they do if you connect directly.
If you need the tunnel to stay open, use the "keepAlive:true" option within
the configuration.
var config = {
...
keepAlive:true
};
var tnl = tunnel(config, function(error, tnl){
yourClient.connect();
yourClient.disconnect();
setTimeout(function(){
tnl.close();
},2000);
});
setTimeout(function(){
tnl.close();
},2000);
Understanding the configuration
-
A local server listening for connections to forward via ssh
Description: This is where you bind your interface.
Properties:
** localHost (default is '127.0.0.1')
** localPort (default is dstPort)
-
The ssh configuration
Description: The host you want to use as ssh-tunnel server.
Properties:
** host
** port (22)
** username
** ...
-
The destination host configuration (based on the ssh host)
Imagine you just connected to The host you want to connect to. (via host:port)
now that server connects requires a target to tunnel to.
Properties:
** dstHost (localhost)
** dstPort
Config example
var config = {
username:'root',
host:sshServer,
port:22,
dstHost:destinationServer,
dstPort:27017,
localHost:'127.0.0.1',
localPort: 27000
};
var tunnel = require('tunnel-ssh');
tunnel(config, function (error, server) {
});
Sugar configuration
In many cases host 1. and 2. are the same, for example if you want to connect to a database
where the port from that database is bound to a local interface (127.0.0.1:27017)
but you are able to connect via ssh (port 22 by default).
You can skip the "dstHost" or the "host" configuration if they are the same.
You can also skip the local configuration if you want to connect to localhost and
the same port as "dstPort".
var config = {
username:'root',
dstHost:destinationServer,
dstPort:27017
};
var tunnel = require('tunnel-ssh');
tunnel(config, function (error, server) {
});
More configuration options
tunnel-ssh pipes the configuration direct into the ssh2 library so every config option
provided by ssh2 still works.
Common examples are:
var config = {
agent : process.env.SSH_AUTH_SOCK,
privateKey:require('fs').readFileSync('/here/is/my/key'),
password:'secret'
}
####catch errors:
var tunnel = require('tunnel-ssh');
var server = tunnel({host: '172.16.0.8', dstPort: 3306}, function (error, server) {
if(error){
}
});
server.on('error', function(err){
console.error('Something bad happened:', err);
});