7

I'd like to transfer a directory between two servers, but compress the directory on the remote host before transfer, and then uncompress to another host. I'm sure it's possible to pipe everything all the way through and do it in a one liner.

I realise it would be better of course if I could transfer between the hosts directly but that would involve transferring keys and what not, and I love Unix one line power tools. I'm sure people can come up with a few different ways to do this. I'm looking for the shortest syntax and most bandwidth conservative.

To start off I have

ssh -n REMOTEHOST 'tar zcvf - DIRTOCOPY' | localZip.tar.gz 
8
  • Why not replace the local cat with a (cd dest; tar xvzf -)? Commented Aug 16, 2012 at 14:16
  • Can we put another remote ssh before those brackets? ssh -n REMOTEHOST1 'tar zcvf - DIRTOCOPY' | ssh -n REMOTEHOST2 (cd dest; tar xvzf -) Commented Aug 16, 2012 at 14:19
  • why not to use rsync? it can compress on the fly.. Commented Aug 16, 2012 at 14:21
  • 2
    Something like ssh -n host1 'tar cvzf - dir' | ssh -n host2 'cd dest; tar xvzf -'? Commented Aug 16, 2012 at 14:35
  • 4
    @barrymac from man rsync about -z option : Note that this option typically achieves better compression ratios than can be achieved by using a compressing remote shell or a compressing transport because it takes advantage of the implicit information in the matching data blocks that are not explicitly sent over the connection. Commented Aug 16, 2012 at 14:42

3 Answers 3

9

Similar to what jw013 suggested in the comments with separate compression/decompression steps, i.e. combine two ssh commands with a pipe:

compress=gzip
decompress=gunzip

ssh remote1 "cd srcdir; tar cf - dir | $compress" |
ssh remote2 "cd destdir; $decompress | tar xvf -"

Note that the compression/decompression is configurable without depending on the version of tar.

Update

You could also add checksum verification into the pipe:

compress=gzip
decompress=gunzip

ckprg=md5sum
cksum=/tmp/cksum

ssh remote1 "cd srcdir; tar cf - dir | $compress | tee <($ckprg > $cksum)" |
ssh remote2 "cd destdir; tee <($ckprg > $cksum) | $decompress | tar xvf -"

ssh remote1 cat $cksum
ssh remote2 cat $cksum
2
  • nice one decoupling the compression Commented Aug 16, 2012 at 15:21
  • Excellent! checksumming is a worthy addition Commented Aug 17, 2012 at 9:13
2

Your transfer would be faster if you could establish a direct connection between the two hosts. But lacking that, the simplest way is to use cp. First mount the remote filesystems using sshfs

mkdir ~/net ~/net/sourcehost ~/net/destinationhost
sshfs sourcehost: ~/net/sourcehost
sshfs destinationhost: ~/net/destinationhost
cp -Rp ~/net/sourcehost/path/to/source ~/net/destinationhost/path/to/destination

Be sure to activate compression in your ~/.ssh/config:

Host sourcehost
HostName sourcehost.example.com
Compression yes
CompressionLevel 9

Host destinationhost
HostName destinationhost.example.com
Compression yes
CompressionLevel 9
3
  • well that doesn't use compression, and I specified that I actually do want to do it via the local host. It's also a lot more hassle to set up sshfs than a one liner I can pull out of the tool box for a one off situation Commented Aug 17, 2012 at 9:12
  • 1
    @barrymac Compression yes ensures that it uses compression. SSHFS requires one line of permanent setup (mkdir), one line of per-session setup per host (mount), and the actual copying is trivial (cp) and doesn't even require you to remember that the files you're copying are remote. Commented Aug 17, 2012 at 9:25
  • Sorry didn't spot that compression flag, and I was thinking of the hassle of installing it and what not. Would be interesting to benchmark that method. It's not quite the one liner I was thinking of but it's handy alright Commented Aug 17, 2012 at 9:50
1

Your proposed answer:

ssh -n REMOTEHOST 'tar zcvf - DIRTOCOPY' | localZip.tar.gz

did not work for me - the pipe to a file failed.

I did this instead and it worked:

ssh -n REMOTEHOST 'tar zcvf - DIRTOCOPY' | cat - > localZip.tar.gz

Pipe it to 'cat' via standard input and redirect the output to the file.

another solution would be to remove the "| cat -" and just send the SSH output directly to the tarball:

ssh -n REMOTEHOST 'tar zcvf - DIRTOCOPY' > localZip.tar.gz

You must log in to answer this question.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.