Unlike Mysql, I found it quite challenging trying to sync MongoDB files -
They can not be piped back, since they don't send the data to stdout
(If I understand correctly).
So, I'm trying to find another way, that doesn't involve two ssh calls.
What needs to be done is this:
- Log into the ssh server
- Export all MongoDB files
- Compress them to gzip
- Send them back to the local machine
- Extract and import
The key thing here, though, is leaving no trace behind -
I don't want the compressed files stay in the remote machine,
which would usually require me for another ssh login.
So something along the lines of "move files into archive" is the ideal solution,
if that could be later piped back to the local machine seamlessly.
I realize MongoDB has a method to connect to a server credentials using mongodump, but the port is closed atm, so I need the SSH method. Any other ideas would be welcome, BTW.
Edit - 11.06.14
Since this questions seems to be somewhat popular, I'd care to share a script that evolved from this questions' answers, and other resources during the last year (credit is where credit is due).
The script basically manages a sync from/to a remote server, for either type of db possible (probably. postgres, mysql and mongo for the time being).
It does have a few assumptions, like the root user having no password for the db, but that can be changed according to need.
The script can be found here: https://github.com/iwfmp/zsh/blob/master/scripts/db/db-sync
You can accomplish this with SSH Tunneling, setting up your remote MongoDB instance to run on one of your local ports. By default, MongoDB runs on 27017, so in the example below, I've chosen to map my remote MongoDB instance to my local 27018 port.
If on your trying to copy a database from SERVER1 to LOCALHOST, you could run this command on your LOCALHOST:
ssh -L27018:localhost:27017 SERVER1
(Obviously replace SERVER1 with your actual server or ssh alias)
This opens an SSH connection to SERVER1, but also maps the port 27018 on LOCALHOST to the remote port 27017 on SERVER1. Don't close that SSH connection, and now try to connect to MongoDB on your localhost machine with port 27018, like so:
mongo --port 27018
You'll notice this is now the data on SERVER1, except you're accessing it from your local machine.
Just running MongoDB normally:
mongo
(ormongo --port 27107
)Will be your local machine.
Now, since you technically have (on your LOCALHOST, where you ran the SSH tunnel):
You can just use the
db.copyDatabase()
function inside MongoDB (LOCALHOST) to copy over data.FROM LOCALHOST ON PORT 27017 (Executing on live will DROP YOUR DATA)
You should be able to wrap this all up into a shell script that can execute all of these commands for you. I have one myself, but it actually has a few extra steps that would probably make it a bit more confusing :)
Doing this, and using MongoDB's native db.copyDatabase() function will prevent you from having to dump/zip/restore. Of course, if you still want to go that route, it wouldn't be too hard to run
mongodump
, export the data, tar/gzip it, then usescp TARGETSERVER:/path/to/file /local/path/to/file
to pull it down and run amongorestore
on it.Just seems like more work!
Edit - Here's a SH and JS file that go together to make a shell script you can run this with. Run these on your LOCALHOST, don't run them on live or it'll do the db.dropDatabase on live. Put these two files in the same folder, and replace YOURSERVERNAME in
pull-db.sh
with the domain/ip/ssh alias, and then inpull-db.js
change DBNAMEHERE to whatever your database name is.I normally create a folder called
scripts
in my projects, and using Textmate, I just have to hit⌘+R
while havingpull-db.sh
open to edit in order to execute it.pull-db.sh
pull-db.js
I added some extra code to the shell script to echo out what it's doing (sorta). The sleep timers in the script are just to give the SSH connections time to get connected before the next line is run. Basically, here's what happens:
You should now have all of the data from your remote database in your localhost.
To complete Jesta great answer, if you want to do the reverse (copy from local database to distant database), you have to bind the port the other way, with the -R command instead of the -L command :
and now, being logged to the REMOTE server, you can copy the db from the local database :
I don't really like to have one database connect to another - IMHO it breaks the separation of environments and makes it complicated to automate such scripts.
My solution is to use the mongo dump/restore which use "dump directories" and to use tar for streaming the files. A trivial implementation (for copying from one remote to another remote) might look like this:
Notes:
mongodump
andmongorestore
has very verbose output, butmongodump
's will both mess with the tar streaming and also apparentlymongodump
will actually refuse to work if you run without a pseudo-terminal and without output redirection.mongodump
has an option to dump to stdout, but I couldn't figure out what kind of format it uses and I didn't understand how to getmongorestore
to load that.