I’ve been using docker a lot, and on occasion I need to transfer images between two machines that are on a local network. If a particular image is large, I might not want to download it twice from two machines, so I download it on one machine and transfer it to the other over the local network.
Now, I could stand up a local docker registry and use that, but it’s a bit of work. Instead, I’ve found that the quickest and easiest solution is to combine the docker ‘save’ and ‘load’ commands with a bit of netcat magic, and it’s pretty fast and easy. (Update: you can do it easily using SSH too, see the end of the post). Check it out.
First, on the destination machine (make sure your firewall allows traffic to the specified port, in this case 1234):
nc -v -l 1234 | docker load
Next, on the source machine, transfer the image (virtuald/etcd:0.4.6) to the destination IPÂ (192.168.0.42):
docker save virtuald/etcd:0.4.6 | nc -v 192.168.0.42 1234
And that’s it!
The sad thing is that docker save/load doesn’t show a status message when saving/loading, so it might look like it’s not doing anything. However, using the -v flag for netcat shows when the connection is successfully opened/closed, so that’s something.
Security warning:Â Obviously, running netcat like this is a *huge* security hole while its up and listening, as anyone who can connect to the port can upload arbitrary images into your docker registry. This is mitigated a bit since netcat will immediately disconnect after the first client disconnects, but still risky on an untrusted network. Only use this on trusted networks!
Note:Â due to this bug, you’ll want to be using docker 1.2+, otherwise you may get unexpected results.
Update! As Joshua Barratt points out, since this method generalizes to any transport that allows piping via stdin/stdout, you can also do the transfer via SSH too, which is certainly more secure. Use the -C option to enable compression for faster transfers (thanks Andreas Steffan).
docker save virtuald/etcd:0.4.6 | ssh -C 192.168.0.42 ‘docker load’
Update II:Â As a number of people have pointed out, you can use PV to show a status message:
docker save virtuald/etcd:0.4.6 | pv | ssh -C 192.168.0.42 'docker load'
I haven’t tested this, but can’t you do the same thing over ssh?
docker save virtuald/etcd:0.4.6 | ssh 192.168.0.42 ‘docker load’
Excellent point, thanks! Clearly, this can generalize to any transport that allows using stdin/stdout. I’ve updated the post.
I haven’t tried it my self, but I think you can get the transfer status using the “pv” command, something like:
docker save virtuald/etcd:0.4.6 | pv |ssh 192.168.0.42 ‘docker load’
“pv” works fine for me and its very helpful because you see there is actually something happening. If you use ssh, you will most likely want to use it with “-C” to use compression. Reduced the total time by 66% for me.