Node.js SSH Tunneling to MongoDB using Mongoose

17,838

Solution 1

dstHost needs to be a hostname/IP, not a MongoDB connection string. In this particular case, you can actually omit dstHost, localHost, and localPort in this particular case because they already default to the values you're providing.

Solution 2

The final working config for future reference. Thanks to mscdex -- I simply needed to provide the correct dstPort and include it in my Mongo URI string (the 27017). So simple. Hope this helps.

var config = {
    username:'myusername',
    host:'my.ip.address',
    agent : process.env.SSH_AUTH_SOCK,
    privateKey:require('fs').readFileSync('/Users/myusername/.ssh/id_rsa'),
    port:22,
    dstPort:27017,
    password:'mypassword'
};

var server = tunnel(config, function (error, server) {
    if(error){
        console.log("SSH connection error: " + error);
    }
    mongoose.connect('mongodb://localhost:27017/mydbname');

    var db = mongoose.connection;
    db.on('error', console.error.bind(console, 'DB connection error:'));
    db.once('open', function() {
        // we're connected!
        console.log("DB connection successful");
    });
});

Solution 3

Or if you don't want to change your code, provided that you have your ssh public key on the tunnel server, you can create a tunnel via ssh on the terminal:

ssh -fNL <local_port>:<mongodb_server_hostname_or_ip>:<mongodb_server_port> <tunnel_server_user>@<tunnel_server_hostname_or_ip>

Example with made-up IPs connecting to a fake AWS EC2 AMI Linux

ssh -fNL 27000:101.202.10.20:27000 [email protected]

Now this mongoose.connect('mongodb://localhost:27000/mydbname'); works like a charm. ;)

Share:
17,838
jeremytripp
Author by

jeremytripp

Updated on June 08, 2022

Comments

  • jeremytripp
    jeremytripp almost 2 years

    I have two Mongo DBs. One for my dev environment, one for production as seen here from my Robomongo setup:

    Two Mongo DBs

    The production db is SSH tunneled to my Digital Ocean virtual server as seen here (specifics have been changed obviously). This setup works fine when connecting to/from my production website:

    SSH Tunnel Settings

    I am now working on a different/related project and need to connect my local machine to my production DB, so I assumed I'd need to use something like tunnel-ssh to get it done. I've followed this answer as an example, but I'm either getting Error: (SSH) Channel open failure: open failed OR it's connecting to my Dev db (when I use 27017 as my dstHost/dstPort/localPort). I must be thinking about this wrong or I'm being dumb with my configuration. I am admittedly a total Mongo/Mongoose novice, so both are equally possible. Any advice?

    var fs = require("fs");
    var mongoose = require('mongoose');
    var tunnel = require('tunnel-ssh');
    
    //===== db connection =====
    
    var config = {
        username:'my.username',
        host:'my.ip.address',
        agent : process.env.SSH_AUTH_SOCK,
        privateKey:require('fs').readFileSync('/Users/myusername/.ssh/id_rsa'),
        port:22,
        dstHost:'mongodb://localhost:27000/mydbname',
        dstPort:27000,
        localHost:'127.0.0.1',
        password:'mypassword',
        localPort: 27000
    };
    
    var server = tunnel(config, function (error, server) {
        if(error){
            console.log("SSH connection error: " + error);
        }
        mongoose.connect('mongodb://localhost:27000/mydbname');
    
        var db = mongoose.connection;
        db.on('error', console.error.bind(console, 'DB connection error:'));
        db.once('open', function() {
            // we're connected!
            console.log("DB connection successful");
            // console.log(server);
        });
    });
    

    Thanks in advance!

  • jeremytripp
    jeremytripp over 7 years
    Well, progress at least. Commenting out those three, now I get Error: (SSH) Channel open failure: Connection refused. Mean anything to you? Off I go to the Google machine...
  • mscdex
    mscdex over 7 years
    You're not using the same port as you show in your screenshot. So try changing it accordingly to match that instead. You'll also need to change the port in your connection string as well.
  • jeremytripp
    jeremytripp over 7 years
    That's the ticket. Painfully obvious and fixed in 5 seconds after reading your comment this morning. Thanks so much!
  • Yoga
    Yoga over 6 years
    Hi I have tried your solution it is pretty much helpful but i had to add passphrase to make it work. Thank you very much :)
  • Emanoel Carlos
    Emanoel Carlos almost 4 years
    Awesome. Thanks.
  • Titou
    Titou about 3 years
    Best answer IMO !
  • MartianMartian
    MartianMartian almost 2 years
    what is agent please?