[lxc-devel] [lxd/master] lxd/migration: Bi-directional rsync negotiation

stgraber on Github lxc-bot at linuxcontainers.org
Sun Nov 25 05:39:57 UTC 2018


A non-text attachment was scrubbed...
Name: not available
Type: text/x-mailbox
Size: 370 bytes
Desc: not available
URL: <http://lists.linuxcontainers.org/pipermail/lxc-devel/attachments/20181125/b5a831e6/attachment.bin>
-------------- next part --------------
From aca0151ed1b54cfb819344cf5509e5eed36520a2 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?St=C3=A9phane=20Graber?= <stgraber at ubuntu.com>
Date: Sun, 25 Nov 2018 00:38:42 -0500
Subject: [PATCH] lxd/migration: Bi-directional rsync negotiation
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Closes #5270

Signed-off-by: Stéphane Graber <stgraber at ubuntu.com>
---
 lxd/migrate.go                 |   5 ++
 lxd/migrate_container.go       |  66 ++++++++++++----
 lxd/migrate_storage_volumes.go |  54 +++++++++----
 lxd/migration/migrate.pb.go    | 137 ++++++++++++++++++---------------
 lxd/migration/migrate.proto    |   1 +
 lxd/rsync.go                   |  23 +++---
 lxd/storage.go                 |   4 +-
 lxd/storage_btrfs.go           |   8 +-
 lxd/storage_ceph.go            |   4 +-
 lxd/storage_ceph_migration.go  |   2 +-
 lxd/storage_dir.go             |   8 +-
 lxd/storage_lvm.go             |   8 +-
 lxd/storage_migration.go       |  21 ++---
 lxd/storage_mock.go            |   4 +-
 lxd/storage_zfs.go             |   6 +-
 15 files changed, 216 insertions(+), 135 deletions(-)

diff --git a/lxd/migrate.go b/lxd/migrate.go
index 86ef2d484c..070c0fd044 100644
--- a/lxd/migrate.go
+++ b/lxd/migrate.go
@@ -273,6 +273,11 @@ type MigrationSinkArgs struct {
 	Refresh bool
 }
 
+type MigrationSourceArgs struct {
+	// transport specific fields
+	RsyncArgs []string
+}
+
 func (c *migrationSink) connectWithSecret(secret string) (*websocket.Conn, error) {
 	query := url.Values{"secret": []string{secret}}
 
diff --git a/lxd/migrate_container.go b/lxd/migrate_container.go
index bbc22fabcd..ec4f279c3e 100644
--- a/lxd/migrate_container.go
+++ b/lxd/migrate_container.go
@@ -220,6 +220,7 @@ type preDumpLoopArgs struct {
 	preDumpDir    string
 	dumpDir       string
 	final         bool
+	rsyncArgs     []string
 }
 
 // The function preDumpLoop is the main logic behind the pre-copy migration.
@@ -250,7 +251,7 @@ func (s *migrationSourceWs) preDumpLoop(args *preDumpLoopArgs) (bool, error) {
 	// Send the pre-dump.
 	ctName, _, _ := containerGetParentAndSnapshotName(s.container.Name())
 	state := s.container.DaemonState()
-	err = RsyncSend(ctName, shared.AddSlash(args.checkpointDir), s.criuConn, nil, args.bwlimit, state.OS.ExecPath)
+	err = RsyncSend(ctName, shared.AddSlash(args.checkpointDir), s.criuConn, nil, args.rsyncArgs, args.bwlimit, state.OS.ExecPath)
 	if err != nil {
 		return final, err
 	}
@@ -355,14 +356,12 @@ func (s *migrationSourceWs) Do(migrateOp *operation) error {
 		}
 	}
 
-	driver, fsErr := s.container.Storage().MigrationSource(s.container, s.containerOnly)
-
 	snapshots := []*migration.Snapshot{}
 	snapshotNames := []string{}
 	// Only send snapshots when requested.
 	if !s.containerOnly {
-		if fsErr == nil {
-			fullSnaps := driver.Snapshots()
+		fullSnaps, err := s.container.Snapshots()
+		if err == nil {
 			for _, snap := range fullSnaps {
 				snapshots = append(snapshots, snapshotToProtobuf(snap))
 				snapshotNames = append(snapshotNames, shared.ExtractSnapshotName(snap.Name()))
@@ -388,9 +387,10 @@ func (s *migrationSourceWs) Do(migrateOp *operation) error {
 		Snapshots:     snapshots,
 		Predump:       proto.Bool(use_pre_dumps),
 		RsyncFeatures: &migration.RsyncFeatures{
-			Xattrs:   &rsyncHasFeature,
-			Delete:   &rsyncHasFeature,
-			Compress: &rsyncHasFeature,
+			Xattrs:        &rsyncHasFeature,
+			Delete:        &rsyncHasFeature,
+			Compress:      &rsyncHasFeature,
+			Bidirectional: &rsyncHasFeature,
 		},
 	}
 
@@ -400,26 +400,52 @@ func (s *migrationSourceWs) Do(migrateOp *operation) error {
 		return err
 	}
 
-	if fsErr != nil {
-		s.sendControl(fsErr)
-		return fsErr
-	}
-
 	err = s.recv(&header)
 	if err != nil {
 		s.sendControl(err)
 		return err
 	}
 
+	// Handle rsync options
+	rsyncArgs := []string{}
+	rsyncFeatures := header.GetRsyncFeatures()
+	if !rsyncFeatures.GetBidirectional() {
+		// If no bi-directional support, assume LXD 3.7 level
+		// NOTE: Do NOT extend this list of arguments
+		rsyncArgs = append(rsyncArgs, "--xattrs")
+		rsyncArgs = append(rsyncArgs, "--delete")
+		rsyncArgs = append(rsyncArgs, "--compress")
+		rsyncArgs = append(rsyncArgs, "--compress-level=2")
+	} else {
+		if rsyncFeatures.GetXattrs() {
+			rsyncArgs = append(rsyncArgs, "--xattrs")
+		}
+		if rsyncFeatures.GetDelete() {
+			rsyncArgs = append(rsyncArgs, "--delete")
+		}
+		if rsyncFeatures.GetCompress() {
+			rsyncArgs = append(rsyncArgs, "--compress")
+			rsyncArgs = append(rsyncArgs, "--compress-level=2")
+		}
+	}
+	sourceArgs := MigrationSourceArgs{rsyncArgs}
+
+	// Initialize storage driver
+	driver, fsErr := s.container.Storage().MigrationSource(s.container, s.containerOnly, sourceArgs)
+	if fsErr != nil {
+		s.sendControl(fsErr)
+		return fsErr
+	}
+
 	bwlimit := ""
 	if header.GetRefresh() || *header.Fs != myType {
 		myType = migration.MigrationFSType_RSYNC
 		header.Fs = &myType
 
 		if header.GetRefresh() {
-			driver, _ = rsyncRefreshSource(s.container, s.containerOnly, header.GetSnapshotNames())
+			driver, _ = rsyncRefreshSource(s.container, s.containerOnly, header.GetSnapshotNames(), sourceArgs)
 		} else {
-			driver, _ = rsyncMigrationSource(s.container, s.containerOnly)
+			driver, _ = rsyncMigrationSource(s.container, s.containerOnly, sourceArgs)
 		}
 
 		// Check if this storage pool has a rate limit set for rsync.
@@ -561,6 +587,7 @@ func (s *migrationSourceWs) Do(migrateOp *operation) error {
 						preDumpDir:    preDumpDir,
 						dumpDir:       dumpDir,
 						final:         final,
+						rsyncArgs:     rsyncArgs,
 					}
 					final, err = s.preDumpLoop(&loop_args)
 					if err != nil {
@@ -631,7 +658,7 @@ func (s *migrationSourceWs) Do(migrateOp *operation) error {
 		 */
 		ctName, _, _ := containerGetParentAndSnapshotName(s.container.Name())
 		state := s.container.DaemonState()
-		err = RsyncSend(ctName, shared.AddSlash(checkpointDir), s.criuConn, nil, bwlimit, state.OS.ExecPath)
+		err = RsyncSend(ctName, shared.AddSlash(checkpointDir), s.criuConn, nil, rsyncArgs, bwlimit, state.OS.ExecPath)
 		if err != nil {
 			return abort(err)
 		}
@@ -818,12 +845,19 @@ func (c *migrationSink) Do(migrateOp *operation) error {
 	}
 
 	myType := c.src.container.Storage().MigrationType()
+	rsyncHasFeature := true
 	resp := migration.MigrationHeader{
 		Fs:            &myType,
 		Criu:          criuType,
 		Snapshots:     header.Snapshots,
 		SnapshotNames: header.SnapshotNames,
 		Refresh:       &c.refresh,
+		RsyncFeatures: &migration.RsyncFeatures{
+			Xattrs:        &rsyncHasFeature,
+			Delete:        &rsyncHasFeature,
+			Compress:      &rsyncHasFeature,
+			Bidirectional: &rsyncHasFeature,
+		},
 	}
 
 	if c.refresh {
diff --git a/lxd/migrate_storage_volumes.go b/lxd/migrate_storage_volumes.go
index eb10bafc33..9b30157aea 100644
--- a/lxd/migrate_storage_volumes.go
+++ b/lxd/migrate_storage_volumes.go
@@ -51,9 +51,10 @@ func (s *migrationSourceWs) DoStorage(migrateOp *operation) error {
 	header := migration.MigrationHeader{
 		Fs: &myType,
 		RsyncFeatures: &migration.RsyncFeatures{
-			Xattrs:   &rsyncHasFeature,
-			Delete:   &rsyncHasFeature,
-			Compress: &rsyncHasFeature,
+			Xattrs:        &rsyncHasFeature,
+			Delete:        &rsyncHasFeature,
+			Compress:      &rsyncHasFeature,
+			Bidirectional: &rsyncHasFeature,
 		},
 	}
 
@@ -64,13 +65,6 @@ func (s *migrationSourceWs) DoStorage(migrateOp *operation) error {
 		return err
 	}
 
-	driver, fsErr := s.storage.StorageMigrationSource()
-	if fsErr != nil {
-		logger.Errorf("Failed to initialize new storage volume migration driver")
-		s.sendControl(fsErr)
-		return fsErr
-	}
-
 	err = s.recv(&header)
 	if err != nil {
 		logger.Errorf("Failed to receive storage volume migration header")
@@ -78,12 +72,43 @@ func (s *migrationSourceWs) DoStorage(migrateOp *operation) error {
 		return err
 	}
 
+	// Handle rsync options
+	rsyncArgs := []string{}
+	rsyncFeatures := header.GetRsyncFeatures()
+	if !rsyncFeatures.GetBidirectional() {
+		// If no bi-directional support, assume LXD 3.7 level
+		// NOTE: Do NOT extend this list of arguments
+		rsyncArgs = append(rsyncArgs, "--xattrs")
+		rsyncArgs = append(rsyncArgs, "--delete")
+		rsyncArgs = append(rsyncArgs, "--compress")
+		rsyncArgs = append(rsyncArgs, "--compress-level=2")
+	} else {
+		if rsyncFeatures.GetXattrs() {
+			rsyncArgs = append(rsyncArgs, "--xattrs")
+		}
+		if rsyncFeatures.GetDelete() {
+			rsyncArgs = append(rsyncArgs, "--delete")
+		}
+		if rsyncFeatures.GetCompress() {
+			rsyncArgs = append(rsyncArgs, "--compress")
+			rsyncArgs = append(rsyncArgs, "--compress-level=2")
+		}
+	}
+	sourceArgs := MigrationSourceArgs{rsyncArgs}
+
+	driver, fsErr := s.storage.StorageMigrationSource(sourceArgs)
+	if fsErr != nil {
+		logger.Errorf("Failed to initialize new storage volume migration driver")
+		s.sendControl(fsErr)
+		return fsErr
+	}
+
 	bwlimit := ""
 	if *header.Fs != myType {
 		myType = migration.MigrationFSType_RSYNC
 		header.Fs = &myType
 
-		driver, _ = rsyncStorageMigrationSource()
+		driver, _ = rsyncStorageMigrationSource(sourceArgs)
 
 		// Check if this storage pool has a rate limit set for rsync.
 		poolwritable := s.storage.GetStoragePoolWritable()
@@ -223,9 +248,10 @@ func (c *migrationSink) DoStorage(migrateOp *operation) error {
 	resp := migration.MigrationHeader{
 		Fs: &myType,
 		RsyncFeatures: &migration.RsyncFeatures{
-			Xattrs:   &rsyncHasFeature,
-			Delete:   &rsyncHasFeature,
-			Compress: &rsyncHasFeature,
+			Xattrs:        &rsyncHasFeature,
+			Delete:        &rsyncHasFeature,
+			Compress:      &rsyncHasFeature,
+			Bidirectional: &rsyncHasFeature,
 		},
 	}
 
diff --git a/lxd/migration/migrate.pb.go b/lxd/migration/migrate.pb.go
index f31f9b7140..9ba7f19cfc 100644
--- a/lxd/migration/migrate.pb.go
+++ b/lxd/migration/migrate.pb.go
@@ -298,6 +298,7 @@ type RsyncFeatures struct {
 	Xattrs           *bool  `protobuf:"varint,1,opt,name=xattrs" json:"xattrs,omitempty"`
 	Delete           *bool  `protobuf:"varint,2,opt,name=delete" json:"delete,omitempty"`
 	Compress         *bool  `protobuf:"varint,3,opt,name=compress" json:"compress,omitempty"`
+	Bidirectional    *bool  `protobuf:"varint,4,opt,name=bidirectional" json:"bidirectional,omitempty"`
 	XXX_unrecognized []byte `json:"-"`
 }
 
@@ -327,6 +328,13 @@ func (m *RsyncFeatures) GetCompress() bool {
 	return false
 }
 
+func (m *RsyncFeatures) GetBidirectional() bool {
+	if m != nil && m.Bidirectional != nil {
+		return *m.Bidirectional
+	}
+	return false
+}
+
 type MigrationHeader struct {
 	Fs               *MigrationFSType `protobuf:"varint,1,req,name=fs,enum=migration.MigrationFSType" json:"fs,omitempty"`
 	Criu             *CRIUType        `protobuf:"varint,2,opt,name=criu,enum=migration.CRIUType" json:"criu,omitempty"`
@@ -634,68 +642,69 @@ func init() {
 func init() { proto.RegisterFile("lxd/migration/migrate.proto", fileDescriptor0) }
 
 var fileDescriptor0 = []byte{
-	// 1005 bytes of a gzipped FileDescriptorProto
-	0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x84, 0x55, 0xcd, 0x6e, 0xdb, 0x46,
-	0x10, 0xae, 0x24, 0xca, 0x16, 0x47, 0x92, 0xa3, 0x6c, 0x82, 0x82, 0x48, 0xfa, 0xa3, 0x32, 0x29,
-	0xaa, 0xfa, 0x10, 0xa7, 0x0a, 0x0a, 0xb4, 0x97, 0x02, 0xb5, 0x5c, 0x37, 0x01, 0x12, 0xd7, 0x58,
-	0xd9, 0x28, 0xda, 0x1e, 0x88, 0x2d, 0x39, 0x94, 0x17, 0xe6, 0x1f, 0x76, 0x29, 0xdb, 0xf2, 0xa5,
-	0x4f, 0xd3, 0xe7, 0xe9, 0xa9, 0xe7, 0xbe, 0x4a, 0xb1, 0xb3, 0x24, 0x4d, 0x39, 0x05, 0x7a, 0xdb,
-	0xf9, 0xe6, 0xe3, 0xcc, 0xce, 0x7c, 0x33, 0x4b, 0x78, 0x9a, 0xdc, 0x44, 0x07, 0xa9, 0x5c, 0x29,
-	0x51, 0xca, 0x3c, 0xab, 0x4e, 0xf8, 0xa2, 0x50, 0x79, 0x99, 0x33, 0xb7, 0x71, 0xf8, 0x7f, 0x80,
-	0xfb, 0xe6, 0xe8, 0x9d, 0x28, 0xce, 0x36, 0x05, 0xb2, 0xc7, 0xd0, 0x97, 0x7a, 0x2d, 0x23, 0xaf,
-	0x33, 0xed, 0xce, 0x06, 0xdc, 0x1a, 0x16, 0x5d, 0xc9, 0xc8, 0xeb, 0xd6, 0xe8, 0x4a, 0x46, 0xec,
-	0x43, 0xd8, 0xb9, 0xc8, 0x75, 0x29, 0x23, 0xaf, 0x37, 0xed, 0xce, 0xfa, 0xbc, 0xb2, 0x18, 0x03,
-	0x27, 0xd3, 0x32, 0xf2, 0x1c, 0x42, 0xe9, 0xcc, 0x9e, 0xc0, 0x20, 0x15, 0x85, 0x12, 0xd9, 0x0a,
-	0xbd, 0x3e, 0xe1, 0x8d, 0xed, 0xbf, 0x84, 0x9d, 0x45, 0x9e, 0xc5, 0x72, 0xc5, 0x26, 0xd0, 0xbb,
-	0xc4, 0x0d, 0xe5, 0x76, 0xb9, 0x39, 0x9a, 0xcc, 0x57, 0x22, 0x59, 0x23, 0x65, 0x76, 0xb9, 0x35,
-	0xfc, 0x1f, 0x61, 0xe7, 0x08, 0xaf, 0x64, 0x88, 0x94, 0x4b, 0xa4, 0x58, 0x7d, 0x42, 0x67, 0xf6,
-	0x25, 0xec, 0x84, 0x14, 0xcf, 0xeb, 0x4e, 0x7b, 0xb3, 0xe1, 0xfc, 0xe1, 0x8b, 0xa6, 0xd8, 0x17,
-	0x36, 0x11, 0xaf, 0x08, 0xfe, 0x5f, 0x5d, 0x18, 0x2c, 0x33, 0x51, 0xe8, 0x8b, 0xbc, 0xfc, 0xcf,
-	0x58, 0xaf, 0x60, 0x98, 0xe4, 0xa1, 0x48, 0x16, 0xff, 0x13, 0xb0, 0xcd, 0x32, 0xc5, 0x16, 0x2a,
-	0x8f, 0x65, 0x82, 0xda, 0xeb, 0x4d, 0x7b, 0x33, 0x97, 0x37, 0x36, 0xfb, 0x08, 0x5c, 0x2c, 0x2e,
-	0x30, 0x45, 0x25, 0x12, 0xea, 0xd0, 0x80, 0xdf, 0x01, 0xec, 0x6b, 0x18, 0x51, 0x20, 0x5b, 0x9d,
-	0xf6, 0xfa, 0xef, 0xe5, 0xb3, 0x1e, 0xbe, 0x45, 0x63, 0x3e, 0x8c, 0x84, 0x0a, 0x2f, 0x64, 0x89,
-	0x61, 0xb9, 0x56, 0xe8, 0xed, 0x50, 0x87, 0xb7, 0x30, 0x73, 0x29, 0x5d, 0x8a, 0x12, 0xe3, 0x75,
-	0xe2, 0xed, 0x52, 0xde, 0xc6, 0x66, 0xcf, 0x60, 0x1c, 0x2a, 0xa4, 0x04, 0x41, 0x24, 0x4a, 0xf4,
-	0x06, 0xd3, 0xce, 0xac, 0xc7, 0x47, 0x35, 0x78, 0x24, 0x4a, 0x64, 0xcf, 0x61, 0x2f, 0x11, 0xba,
-	0x0c, 0xd6, 0x1a, 0x23, 0xcb, 0x72, 0x2d, 0xcb, 0xa0, 0xe7, 0x1a, 0x23, 0xc3, 0xf2, 0x7f, 0x83,
-	0xb1, 0xd2, 0x9b, 0x2c, 0x3c, 0x46, 0x61, 0xd2, 0x6a, 0x33, 0x25, 0x37, 0xa2, 0x2c, 0x95, 0xf6,
-	0x3a, 0xd3, 0xce, 0x6c, 0xc0, 0x2b, 0xcb, 0xe0, 0x11, 0x26, 0x58, 0x1a, 0x69, 0x09, 0xb7, 0x96,
-	0xb9, 0x67, 0x98, 0xa7, 0x85, 0x42, 0x6d, 0x9a, 0x67, 0x3c, 0x8d, 0xed, 0xff, 0xd3, 0x85, 0x07,
-	0xef, 0xea, 0x56, 0xbc, 0x46, 0x11, 0xa1, 0x62, 0xfb, 0xd0, 0x8d, 0x35, 0x69, 0xb6, 0x37, 0x7f,
-	0xd2, 0x6a, 0x54, 0xc3, 0x3b, 0x5e, 0x9a, 0xc9, 0xe6, 0xdd, 0x58, 0xb3, 0x2f, 0xc0, 0x09, 0x95,
-	0x5c, 0x53, 0xc6, 0xbd, 0xf9, 0xa3, 0xb6, 0x8c, 0xfc, 0xcd, 0x39, 0xd1, 0x88, 0xc0, 0xf6, 0xa1,
-	0x2f, 0xa3, 0x54, 0x14, 0x24, 0xdf, 0x70, 0xfe, 0xb8, 0xc5, 0x6c, 0x76, 0x85, 0x5b, 0x0a, 0x7b,
-	0x0e, 0x63, 0x5d, 0x8d, 0xd0, 0x89, 0x48, 0x51, 0x7b, 0x0e, 0x49, 0xbe, 0x0d, 0xb2, 0xaf, 0xc0,
-	0xad, 0x81, 0x5a, 0xd6, 0x76, 0xfe, 0x7a, 0x08, 0xf9, 0x1d, 0x8b, 0x79, 0xb0, 0x5b, 0x28, 0x8c,
-	0xd6, 0x69, 0xe1, 0xed, 0x52, 0x23, 0x6a, 0x93, 0x7d, 0x77, 0xaf, 0xc9, 0xa4, 0xd7, 0x70, 0xee,
-	0xb5, 0x02, 0x6e, 0xf9, 0xf9, 0x3d, 0x4d, 0x3c, 0xd8, 0x55, 0x18, 0x2b, 0xd4, 0x17, 0xa4, 0xe1,
-	0x80, 0xd7, 0xa6, 0x7f, 0x0c, 0x93, 0xa6, 0x71, 0x8b, 0x3c, 0x2b, 0x55, 0x9e, 0x18, 0xb6, 0x5e,
-	0x87, 0xa1, 0x11, 0xc4, 0xbe, 0x0a, 0xb5, 0x69, 0x3c, 0x29, 0x6a, 0x2d, 0x56, 0x56, 0x44, 0x97,
-	0xd7, 0xa6, 0xff, 0x0a, 0xc6, 0x4d, 0x9c, 0xe5, 0x26, 0x0b, 0xcd, 0x88, 0xc6, 0x32, 0x13, 0xc9,
-	0xa9, 0xc2, 0x23, 0x53, 0x91, 0x8d, 0xb4, 0x85, 0xf9, 0x7f, 0xf6, 0x60, 0x62, 0xea, 0x0b, 0xcc,
-	0x60, 0xea, 0x00, 0xb3, 0x52, 0x6d, 0xcc, 0x6c, 0xc6, 0x0a, 0xf1, 0x56, 0x66, 0xab, 0xa0, 0x94,
-	0xd5, 0x7a, 0x8e, 0xf9, 0xa8, 0x06, 0xcf, 0x64, 0x8a, 0xec, 0x53, 0x18, 0xc6, 0x2a, 0xbf, 0xc5,
-	0xcc, 0x52, 0xba, 0x44, 0x01, 0x0b, 0x11, 0xe1, 0x33, 0x18, 0xa5, 0x98, 0x52, 0x70, 0x62, 0xf4,
-	0x88, 0x31, 0xac, 0x30, 0xa2, 0x3c, 0x83, 0x71, 0x8a, 0xe9, 0xb5, 0x92, 0x25, 0x5a, 0x8e, 0x63,
-	0x13, 0xd5, 0x60, 0x4d, 0x2a, 0xc4, 0x0a, 0x75, 0xa0, 0x43, 0x91, 0x65, 0x18, 0xd1, 0x63, 0xe6,
-	0xf0, 0x11, 0x81, 0x4b, 0x8b, 0xb1, 0x97, 0xf0, 0xb8, 0x22, 0x5d, 0xca, 0xa2, 0xc0, 0x28, 0x28,
-	0x84, 0xc2, 0xac, 0xa4, 0xb5, 0x74, 0x38, 0xb3, 0x5c, 0xeb, 0x3a, 0x25, 0xcf, 0x5d, 0x58, 0x93,
-	0xa9, 0xc4, 0x8c, 0x36, 0xb4, 0x0e, 0xfb, 0xb3, 0xc5, 0x0c, 0x49, 0xaa, 0x54, 0x14, 0x81, 0x42,
-	0x9d, 0x27, 0x57, 0x76, 0x4b, 0xc7, 0x7c, 0x44, 0x20, 0xb7, 0x18, 0xfb, 0x18, 0xc0, 0x46, 0x4a,
-	0xc4, 0xed, 0xc6, 0x73, 0x29, 0x8c, 0x4b, 0xc8, 0x5b, 0x71, 0xbb, 0xa9, 0xdd, 0x41, 0x21, 0x0b,
-	0xd4, 0x1e, 0x4c, 0x3b, 0xb5, 0xfb, 0xd4, 0x00, 0x66, 0xc7, 0x1b, 0x77, 0xf0, 0xfb, 0x3a, 0xd6,
-	0xde, 0x90, 0x28, 0xa3, 0x9a, 0x72, 0xb8, 0x8e, 0xb5, 0xff, 0x77, 0x07, 0x1e, 0x29, 0xd4, 0x65,
-	0xae, 0x70, 0x4b, 0xaa, 0xcf, 0xed, 0xd7, 0x3a, 0x30, 0x0b, 0x2b, 0x14, 0xda, 0xbf, 0x88, 0xc3,
-	0x6d, 0x6d, 0x8b, 0x0a, 0x64, 0xfb, 0xf0, 0x70, 0xbb, 0x3d, 0x61, 0x7e, 0x4d, 0x92, 0x39, 0xfc,
-	0x41, 0xbb, 0x37, 0x8b, 0xfc, 0xda, 0xe8, 0x16, 0xe7, 0xea, 0xb2, 0x11, 0xbf, 0xd2, 0xad, 0xc2,
-	0x6a, 0x69, 0xeb, 0xcb, 0xb4, 0x64, 0x1b, 0x56, 0x18, 0x51, 0x9a, 0x8b, 0x55, 0xa0, 0x91, 0xad,
-	0xd3, 0x5c, 0x8c, 0x57, 0xa0, 0x7f, 0x03, 0xc3, 0x76, 0x39, 0x07, 0xe0, 0x44, 0x76, 0x54, 0xcd,
-	0x72, 0x3d, 0x6d, 0x2d, 0xd7, 0xfd, 0x21, 0xe5, 0x44, 0x64, 0xdf, 0x98, 0xb5, 0xa2, 0x58, 0xb4,
-	0x0e, 0xc3, 0xf9, 0x27, 0xed, 0x85, 0x7c, 0xbf, 0x61, 0xbc, 0xa6, 0xef, 0x7f, 0xdb, 0x7a, 0xd7,
-	0xec, 0x7b, 0xc5, 0x5c, 0xe8, 0xf3, 0xe5, 0x2f, 0x27, 0x8b, 0xc9, 0x07, 0xe6, 0x78, 0x78, 0xc6,
-	0x8f, 0x97, 0x93, 0x0e, 0xdb, 0x85, 0xde, 0xaf, 0xc7, 0xcb, 0x49, 0xd7, 0x1c, 0xf8, 0xe1, 0xd1,
-	0xa4, 0xb7, 0x7f, 0x00, 0x83, 0xfa, 0xf1, 0x62, 0x7b, 0x00, 0xe6, 0x1c, 0xb4, 0x3e, 0x3c, 0x7d,
-	0xfd, 0xfd, 0xf9, 0xdb, 0x49, 0x87, 0x0d, 0xc0, 0x39, 0xf9, 0xe9, 0xe4, 0x87, 0x49, 0xf7, 0xdf,
-	0x00, 0x00, 0x00, 0xff, 0xff, 0xd0, 0x14, 0x86, 0xd7, 0x18, 0x08, 0x00, 0x00,
+	// 1022 bytes of a gzipped FileDescriptorProto
+	0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x84, 0x55, 0x4d, 0x6f, 0xdb, 0x46,
+	0x13, 0x7e, 0x25, 0x51, 0xb6, 0x38, 0x92, 0x1c, 0x65, 0x13, 0xbc, 0x20, 0x92, 0x7e, 0xa8, 0x4c,
+	0x8a, 0xaa, 0x3e, 0xc4, 0xa9, 0x82, 0x02, 0xed, 0xa5, 0x40, 0x2d, 0xd7, 0x4d, 0x80, 0xc4, 0x35,
+	0x56, 0x36, 0x8a, 0xf6, 0x42, 0x6c, 0xc8, 0xa1, 0xbc, 0x30, 0xbf, 0xb0, 0x4b, 0xd9, 0x96, 0x2f,
+	0x45, 0x7f, 0x4c, 0x7f, 0x4f, 0x4f, 0x3d, 0xf7, 0xaf, 0x14, 0x3b, 0x4b, 0xd2, 0x94, 0x53, 0xa0,
+	0xb7, 0x9d, 0x67, 0x1e, 0xce, 0xcc, 0xce, 0x33, 0xb3, 0x84, 0xa7, 0xc9, 0x4d, 0x74, 0x90, 0xca,
+	0x95, 0x12, 0xa5, 0xcc, 0xb3, 0xea, 0x84, 0x2f, 0x0a, 0x95, 0x97, 0x39, 0x73, 0x1b, 0x87, 0xff,
+	0x1b, 0xb8, 0x6f, 0x8e, 0xde, 0x89, 0xe2, 0x6c, 0x53, 0x20, 0x7b, 0x0c, 0x7d, 0xa9, 0xd7, 0x32,
+	0xf2, 0x3a, 0xd3, 0xee, 0x6c, 0xc0, 0xad, 0x61, 0xd1, 0x95, 0x8c, 0xbc, 0x6e, 0x8d, 0xae, 0x64,
+	0xc4, 0xfe, 0x0f, 0x3b, 0x17, 0xb9, 0x2e, 0x65, 0xe4, 0xf5, 0xa6, 0xdd, 0x59, 0x9f, 0x57, 0x16,
+	0x63, 0xe0, 0x64, 0x5a, 0x46, 0x9e, 0x43, 0x28, 0x9d, 0xd9, 0x13, 0x18, 0xa4, 0xa2, 0x50, 0x22,
+	0x5b, 0xa1, 0xd7, 0x27, 0xbc, 0xb1, 0xfd, 0x97, 0xb0, 0xb3, 0xc8, 0xb3, 0x58, 0xae, 0xd8, 0x04,
+	0x7a, 0x97, 0xb8, 0xa1, 0xdc, 0x2e, 0x37, 0x47, 0x93, 0xf9, 0x4a, 0x24, 0x6b, 0xa4, 0xcc, 0x2e,
+	0xb7, 0x86, 0xff, 0x23, 0xec, 0x1c, 0xe1, 0x95, 0x0c, 0x91, 0x72, 0x89, 0x14, 0xab, 0x4f, 0xe8,
+	0xcc, 0xbe, 0x84, 0x9d, 0x90, 0xe2, 0x79, 0xdd, 0x69, 0x6f, 0x36, 0x9c, 0x3f, 0x7c, 0xd1, 0x5c,
+	0xf6, 0x85, 0x4d, 0xc4, 0x2b, 0x82, 0xff, 0x67, 0x17, 0x06, 0xcb, 0x4c, 0x14, 0xfa, 0x22, 0x2f,
+	0xff, 0x35, 0xd6, 0x2b, 0x18, 0x26, 0x79, 0x28, 0x92, 0xc5, 0x7f, 0x04, 0x6c, 0xb3, 0xcc, 0x65,
+	0x0b, 0x95, 0xc7, 0x32, 0x41, 0xed, 0xf5, 0xa6, 0xbd, 0x99, 0xcb, 0x1b, 0x9b, 0x7d, 0x04, 0x2e,
+	0x16, 0x17, 0x98, 0xa2, 0x12, 0x09, 0x75, 0x68, 0xc0, 0xef, 0x00, 0xf6, 0x35, 0x8c, 0x28, 0x90,
+	0xbd, 0x9d, 0xf6, 0xfa, 0x1f, 0xe4, 0xb3, 0x1e, 0xbe, 0x45, 0x63, 0x3e, 0x8c, 0x84, 0x0a, 0x2f,
+	0x64, 0x89, 0x61, 0xb9, 0x56, 0xe8, 0xed, 0x50, 0x87, 0xb7, 0x30, 0x53, 0x94, 0x2e, 0x45, 0x89,
+	0xf1, 0x3a, 0xf1, 0x76, 0x29, 0x6f, 0x63, 0xb3, 0x67, 0x30, 0x0e, 0x15, 0x52, 0x82, 0x20, 0x12,
+	0x25, 0x7a, 0x83, 0x69, 0x67, 0xd6, 0xe3, 0xa3, 0x1a, 0x3c, 0x12, 0x25, 0xb2, 0xe7, 0xb0, 0x97,
+	0x08, 0x5d, 0x06, 0x6b, 0x8d, 0x91, 0x65, 0xb9, 0x96, 0x65, 0xd0, 0x73, 0x8d, 0x91, 0x61, 0xf9,
+	0xbf, 0x77, 0x60, 0xac, 0xf4, 0x26, 0x0b, 0x8f, 0x51, 0x98, 0xbc, 0xda, 0x8c, 0xc9, 0x8d, 0x28,
+	0x4b, 0xa5, 0xbd, 0xce, 0xb4, 0x33, 0x1b, 0xf0, 0xca, 0x32, 0x78, 0x84, 0x09, 0x96, 0x46, 0x5b,
+	0xc2, 0xad, 0x65, 0x0a, 0x0d, 0xf3, 0xb4, 0x50, 0xa8, 0x4d, 0xf7, 0x8c, 0xa7, 0xb1, 0xd9, 0x73,
+	0x18, 0xbf, 0x97, 0x91, 0x54, 0x18, 0x9a, 0xb2, 0xa8, 0x83, 0x86, 0xb0, 0x0d, 0xfa, 0x7f, 0x77,
+	0xe1, 0xc1, 0xbb, 0xba, 0x63, 0xaf, 0x51, 0x44, 0xa8, 0xd8, 0x3e, 0x74, 0x63, 0x4d, 0xd2, 0xee,
+	0xcd, 0x9f, 0xb4, 0xfa, 0xd9, 0xf0, 0x8e, 0x97, 0x66, 0x01, 0x78, 0x37, 0xd6, 0xec, 0x0b, 0x70,
+	0x42, 0x25, 0xd7, 0x54, 0xd7, 0xde, 0xfc, 0x51, 0x5b, 0x6d, 0xfe, 0xe6, 0x9c, 0x68, 0x44, 0x60,
+	0xfb, 0xd0, 0x97, 0x51, 0x2a, 0x0a, 0x52, 0x79, 0x38, 0x7f, 0xdc, 0x62, 0x36, 0x2b, 0xc5, 0x2d,
+	0xc5, 0x94, 0xae, 0xab, 0x49, 0x3b, 0x11, 0x29, 0x6a, 0xcf, 0xa1, 0xc9, 0xd8, 0x06, 0xd9, 0x57,
+	0xe0, 0xd6, 0x40, 0xad, 0x7e, 0x3b, 0x7f, 0x3d, 0xab, 0xfc, 0x8e, 0xc5, 0x3c, 0xd8, 0x2d, 0x14,
+	0x46, 0xeb, 0xb4, 0xf0, 0x76, 0xa9, 0x1b, 0xb5, 0xc9, 0xbe, 0xbb, 0x27, 0x05, 0xc9, 0x3a, 0x9c,
+	0x7b, 0xad, 0x80, 0x5b, 0x7e, 0x7e, 0x4f, 0x39, 0x0f, 0x76, 0x15, 0xc6, 0x0a, 0xf5, 0x05, 0x49,
+	0x3d, 0xe0, 0xb5, 0xe9, 0x1f, 0xc3, 0xa4, 0x69, 0xdc, 0x22, 0xcf, 0x4a, 0x95, 0x27, 0x86, 0xad,
+	0xd7, 0x61, 0x68, 0x64, 0xb3, 0x8f, 0x47, 0x6d, 0x1a, 0x4f, 0x8a, 0x5a, 0x8b, 0x95, 0x95, 0xda,
+	0xe5, 0xb5, 0xe9, 0xbf, 0x82, 0x71, 0x13, 0x67, 0xb9, 0xc9, 0x42, 0x33, 0xc9, 0xb1, 0xcc, 0x44,
+	0x72, 0xaa, 0xf0, 0xc8, 0xdc, 0xc8, 0x46, 0xda, 0xc2, 0xfc, 0x3f, 0x7a, 0x30, 0x31, 0xf7, 0x0b,
+	0xcc, 0xfc, 0xea, 0x00, 0xb3, 0x52, 0x6d, 0xcc, 0x08, 0xc7, 0x0a, 0xf1, 0x56, 0x66, 0xab, 0xa0,
+	0x94, 0xd5, 0x16, 0x8f, 0xf9, 0xa8, 0x06, 0xcf, 0x64, 0x8a, 0xec, 0x53, 0x18, 0xc6, 0x2a, 0xbf,
+	0xc5, 0xcc, 0x52, 0xba, 0x44, 0x01, 0x0b, 0x11, 0xe1, 0x33, 0x18, 0xa5, 0x98, 0x52, 0x70, 0x62,
+	0xf4, 0x88, 0x31, 0xac, 0x30, 0xa2, 0x3c, 0x83, 0x71, 0x8a, 0xe9, 0xb5, 0x92, 0x25, 0x5a, 0x8e,
+	0x63, 0x13, 0xd5, 0x60, 0x4d, 0x2a, 0xc4, 0x0a, 0x75, 0xa0, 0x43, 0x91, 0x65, 0x18, 0xd1, 0x9b,
+	0xe7, 0xf0, 0x11, 0x81, 0x4b, 0x8b, 0xb1, 0x97, 0xf0, 0xb8, 0x22, 0x5d, 0xca, 0xa2, 0xc0, 0x28,
+	0x28, 0x84, 0xc2, 0xac, 0xa4, 0xed, 0x75, 0x38, 0xb3, 0x5c, 0xeb, 0x3a, 0x25, 0xcf, 0x5d, 0x58,
+	0x93, 0xa9, 0xc4, 0x8c, 0x16, 0xb9, 0x0e, 0xfb, 0xb3, 0xc5, 0x0c, 0x49, 0xaa, 0x54, 0x14, 0x81,
+	0x42, 0x9d, 0x27, 0x57, 0x76, 0x99, 0xc7, 0x7c, 0x44, 0x20, 0xb7, 0x18, 0xfb, 0x18, 0xc0, 0x46,
+	0x4a, 0xc4, 0xed, 0xc6, 0x73, 0x29, 0x8c, 0x4b, 0xc8, 0x5b, 0x71, 0xbb, 0xa9, 0xdd, 0x41, 0x21,
+	0x0b, 0xd4, 0x1e, 0x4c, 0x3b, 0xb5, 0xfb, 0xd4, 0x00, 0xe6, 0x29, 0x68, 0xdc, 0xc1, 0xfb, 0x75,
+	0xac, 0xbd, 0x21, 0x51, 0x46, 0x35, 0xe5, 0x70, 0x1d, 0x6b, 0xff, 0xaf, 0x0e, 0x3c, 0x52, 0xa8,
+	0xcb, 0x5c, 0xe1, 0x96, 0x54, 0x9f, 0xdb, 0xaf, 0x75, 0x60, 0xd6, 0x5a, 0x28, 0xb4, 0x3f, 0x1b,
+	0x87, 0xdb, 0xbb, 0x2d, 0x2a, 0x90, 0xed, 0xc3, 0xc3, 0xed, 0xf6, 0x84, 0xf9, 0x35, 0x49, 0xe6,
+	0xf0, 0x07, 0xed, 0xde, 0x2c, 0xf2, 0x6b, 0xa3, 0x5b, 0x9c, 0xab, 0xcb, 0x46, 0xfc, 0x4a, 0xb7,
+	0x0a, 0xab, 0xa5, 0xad, 0x8b, 0x69, 0xc9, 0x36, 0xac, 0x30, 0xa2, 0x34, 0x85, 0x55, 0xa0, 0x91,
+	0xad, 0xd3, 0x14, 0xc6, 0x2b, 0xd0, 0xbf, 0x81, 0x61, 0xfb, 0x3a, 0x07, 0xe0, 0x44, 0x76, 0x54,
+	0xcd, 0x72, 0x3d, 0x6d, 0x2d, 0xd7, 0xfd, 0x21, 0xe5, 0x44, 0x64, 0xdf, 0x98, 0xb5, 0xa2, 0x58,
+	0xb4, 0x0e, 0xc3, 0xf9, 0x27, 0xed, 0x85, 0xfc, 0xb0, 0x61, 0xbc, 0xa6, 0xef, 0x7f, 0xdb, 0x7a,
+	0xd7, 0xec, 0x7b, 0xc5, 0x5c, 0xe8, 0xf3, 0xe5, 0x2f, 0x27, 0x8b, 0xc9, 0xff, 0xcc, 0xf1, 0xf0,
+	0x8c, 0x1f, 0x2f, 0x27, 0x1d, 0xb6, 0x0b, 0xbd, 0x5f, 0x8f, 0x97, 0x93, 0xae, 0x39, 0xf0, 0xc3,
+	0xa3, 0x49, 0x6f, 0xff, 0x00, 0x06, 0xf5, 0xe3, 0xc5, 0xf6, 0x00, 0xcc, 0x39, 0x68, 0x7d, 0x78,
+	0xfa, 0xfa, 0xfb, 0xf3, 0xb7, 0x93, 0x0e, 0x1b, 0x80, 0x73, 0xf2, 0xd3, 0xc9, 0x0f, 0x93, 0xee,
+	0x3f, 0x01, 0x00, 0x00, 0xff, 0xff, 0xe0, 0xe0, 0x8b, 0x09, 0x3f, 0x08, 0x00, 0x00,
 }
diff --git a/lxd/migration/migrate.proto b/lxd/migration/migrate.proto
index 009aa8fcba..938e73af0d 100644
--- a/lxd/migration/migrate.proto
+++ b/lxd/migration/migrate.proto
@@ -50,6 +50,7 @@ message rsyncFeatures {
 	optional bool		xattrs = 1;
 	optional bool		delete = 2;
 	optional bool		compress = 3;
+	optional bool		bidirectional = 4;
 }
 
 message MigrationHeader {
diff --git a/lxd/rsync.go b/lxd/rsync.go
index cb8874ac12..e6bf3543ad 100644
--- a/lxd/rsync.go
+++ b/lxd/rsync.go
@@ -62,7 +62,7 @@ func rsyncLocalCopy(source string, dest string, bwlimit string) (string, error)
 	return msg, nil
 }
 
-func rsyncSendSetup(name string, path string, bwlimit string, execPath string) (*exec.Cmd, net.Conn, io.ReadCloser, error) {
+func rsyncSendSetup(name string, path string, bwlimit string, execPath string, extraArgs []string) (*exec.Cmd, net.Conn, io.ReadCloser, error) {
 	/*
 	 * The way rsync works, it invokes a subprocess that does the actual
 	 * talking (given to it by a -E argument). Since there isn't an easy
@@ -104,22 +104,27 @@ func rsyncSendSetup(name string, path string, bwlimit string, execPath string) (
 		bwlimit = "0"
 	}
 
-	cmd := exec.Command("rsync",
+	args := []string{
 		"-ar",
 		"--devices",
 		"--numeric-ids",
 		"--partial",
 		"--sparse",
-		"--xattrs",
-		"--delete",
-		"--compress",
-		"--compress-level=2",
+	}
+
+	if extraArgs != nil && len(extraArgs) > 0 {
+		args = append(args, extraArgs...)
+	}
+
+	args = append(args, []string{
 		path,
 		"localhost:/tmp/foo",
 		"-e",
 		rsyncCmd,
 		"--bwlimit",
-		bwlimit)
+		bwlimit}...)
+
+	cmd := exec.Command("rsync", args...)
 
 	stderr, err := cmd.StderrPipe()
 	if err != nil {
@@ -143,8 +148,8 @@ func rsyncSendSetup(name string, path string, bwlimit string, execPath string) (
 
 // RsyncSend sets up the sending half of an rsync, to recursively send the
 // directory pointed to by path over the websocket.
-func RsyncSend(name string, path string, conn *websocket.Conn, readWrapper func(io.ReadCloser) io.ReadCloser, bwlimit string, execPath string) error {
-	cmd, dataSocket, stderr, err := rsyncSendSetup(name, path, bwlimit, execPath)
+func RsyncSend(name string, path string, conn *websocket.Conn, readWrapper func(io.ReadCloser) io.ReadCloser, extraArgs []string, bwlimit string, execPath string) error {
+	cmd, dataSocket, stderr, err := rsyncSendSetup(name, path, bwlimit, execPath, extraArgs)
 	if err != nil {
 		return err
 	}
diff --git a/lxd/storage.go b/lxd/storage.go
index 5b81367b71..b394968d8d 100644
--- a/lxd/storage.go
+++ b/lxd/storage.go
@@ -232,7 +232,7 @@ type storage interface {
 	// We leave sending containers which are snapshots of other containers
 	// already present on the target instance as an exercise for the
 	// enterprising developer.
-	MigrationSource(c container, containerOnly bool) (MigrationStorageSourceDriver, error)
+	MigrationSource(c container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error)
 	MigrationSink(
 		live bool,
 		c container,
@@ -243,7 +243,7 @@ type storage interface {
 		containerOnly bool,
 		args MigrationSinkArgs) error
 
-	StorageMigrationSource() (MigrationStorageSourceDriver, error)
+	StorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error)
 	StorageMigrationSink(conn *websocket.Conn, op *operation, storage storage, args MigrationSinkArgs) error
 }
 
diff --git a/lxd/storage_btrfs.go b/lxd/storage_btrfs.go
index bd710bf4fc..3325fa0ea7 100644
--- a/lxd/storage_btrfs.go
+++ b/lxd/storage_btrfs.go
@@ -2625,9 +2625,9 @@ func (s *storageBtrfs) PreservesInodes() bool {
 	return true
 }
 
-func (s *storageBtrfs) MigrationSource(c container, containerOnly bool) (MigrationStorageSourceDriver, error) {
+func (s *storageBtrfs) MigrationSource(c container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
 	if s.s.OS.RunningInUserNS {
-		return rsyncMigrationSource(c, containerOnly)
+		return rsyncMigrationSource(c, containerOnly, args)
 	}
 
 	/* List all the snapshots in order of reverse creation. The idea here
@@ -3038,8 +3038,8 @@ func (s *btrfsMigrationSourceDriver) SendStorageVolume(conn *websocket.Conn, op
 	return fmt.Errorf(msg)
 }
 
-func (s *storageBtrfs) StorageMigrationSource() (MigrationStorageSourceDriver, error) {
-	return rsyncStorageMigrationSource()
+func (s *storageBtrfs) StorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncStorageMigrationSource(args)
 }
 
 func (s *storageBtrfs) StorageMigrationSink(conn *websocket.Conn, op *operation, storage storage, args MigrationSinkArgs) error {
diff --git a/lxd/storage_ceph.go b/lxd/storage_ceph.go
index 2e79ba5e11..2382757dab 100644
--- a/lxd/storage_ceph.go
+++ b/lxd/storage_ceph.go
@@ -2734,8 +2734,8 @@ func (s *rbdMigrationSourceDriver) SendStorageVolume(conn *websocket.Conn, op *o
 	return fmt.Errorf(msg)
 }
 
-func (s *storageCeph) StorageMigrationSource() (MigrationStorageSourceDriver, error) {
-	return rsyncStorageMigrationSource()
+func (s *storageCeph) StorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncStorageMigrationSource(args)
 }
 
 func (s *storageCeph) StorageMigrationSink(conn *websocket.Conn, op *operation, storage storage, args MigrationSinkArgs) error {
diff --git a/lxd/storage_ceph_migration.go b/lxd/storage_ceph_migration.go
index 08c539e89b..6abd4d76af 100644
--- a/lxd/storage_ceph_migration.go
+++ b/lxd/storage_ceph_migration.go
@@ -159,7 +159,7 @@ func (s *storageCeph) PreservesInodes() bool {
 	return false
 }
 
-func (s *storageCeph) MigrationSource(c container, containerOnly bool) (MigrationStorageSourceDriver, error) {
+func (s *storageCeph) MigrationSource(c container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
 	// If the container is a snapshot, let's just send that. We don't need
 	// to send anything else, because that's all the user asked for.
 	if c.IsSnapshot() {
diff --git a/lxd/storage_dir.go b/lxd/storage_dir.go
index 66cc9a7803..20059b4993 100644
--- a/lxd/storage_dir.go
+++ b/lxd/storage_dir.go
@@ -1272,8 +1272,8 @@ func (s *storageDir) PreservesInodes() bool {
 	return false
 }
 
-func (s *storageDir) MigrationSource(container container, containerOnly bool) (MigrationStorageSourceDriver, error) {
-	return rsyncMigrationSource(container, containerOnly)
+func (s *storageDir) MigrationSource(container container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncMigrationSource(container, containerOnly, args)
 }
 
 func (s *storageDir) MigrationSink(live bool, container container, snapshots []*migration.Snapshot, conn *websocket.Conn, srcIdmap *idmap.IdmapSet, op *operation, containerOnly bool, args MigrationSinkArgs) error {
@@ -1355,8 +1355,8 @@ func (s *storageDir) StoragePoolVolumeCopy(source *api.StorageVolumeSource) erro
 	return nil
 }
 
-func (s *storageDir) StorageMigrationSource() (MigrationStorageSourceDriver, error) {
-	return rsyncStorageMigrationSource()
+func (s *storageDir) StorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncStorageMigrationSource(args)
 }
 
 func (s *storageDir) StorageMigrationSink(conn *websocket.Conn, op *operation, storage storage, args MigrationSinkArgs) error {
diff --git a/lxd/storage_lvm.go b/lxd/storage_lvm.go
index 46281ca6f3..a2b5df8d82 100644
--- a/lxd/storage_lvm.go
+++ b/lxd/storage_lvm.go
@@ -2069,8 +2069,8 @@ func (s *storageLvm) PreservesInodes() bool {
 	return false
 }
 
-func (s *storageLvm) MigrationSource(container container, containerOnly bool) (MigrationStorageSourceDriver, error) {
-	return rsyncMigrationSource(container, containerOnly)
+func (s *storageLvm) MigrationSource(container container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncMigrationSource(container, containerOnly, args)
 }
 
 func (s *storageLvm) MigrationSink(live bool, container container, snapshots []*migration.Snapshot, conn *websocket.Conn, srcIdmap *idmap.IdmapSet, op *operation, containerOnly bool, args MigrationSinkArgs) error {
@@ -2278,8 +2278,8 @@ func (s *storageLvm) StoragePoolVolumeCopy(source *api.StorageVolumeSource) erro
 	return nil
 }
 
-func (s *storageLvm) StorageMigrationSource() (MigrationStorageSourceDriver, error) {
-	return rsyncStorageMigrationSource()
+func (s *storageLvm) StorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncStorageMigrationSource(args)
 }
 
 func (s *storageLvm) StorageMigrationSink(conn *websocket.Conn, op *operation, storage storage, args MigrationSinkArgs) error {
diff --git a/lxd/storage_migration.go b/lxd/storage_migration.go
index cd3c7b03ad..c7e7c8e972 100644
--- a/lxd/storage_migration.go
+++ b/lxd/storage_migration.go
@@ -43,6 +43,7 @@ type MigrationStorageSourceDriver interface {
 type rsyncStorageSourceDriver struct {
 	container container
 	snapshots []container
+	rsyncArgs []string
 }
 
 func (s rsyncStorageSourceDriver) Snapshots() []container {
@@ -66,7 +67,7 @@ func (s rsyncStorageSourceDriver) SendStorageVolume(conn *websocket.Conn, op *op
 	path := getStoragePoolVolumeMountPoint(pool.Name, volume.Name)
 	path = shared.AddSlash(path)
 	logger.Debugf("Starting to send storage volume %s on storage pool %s from %s", volume.Name, pool.Name, path)
-	return RsyncSend(volume.Name, path, conn, wrapper, bwlimit, state.OS.ExecPath)
+	return RsyncSend(volume.Name, path, conn, wrapper, s.rsyncArgs, bwlimit, state.OS.ExecPath)
 }
 
 func (s rsyncStorageSourceDriver) SendWhileRunning(conn *websocket.Conn, op *operation, bwlimit string, containerOnly bool) error {
@@ -85,7 +86,7 @@ func (s rsyncStorageSourceDriver) SendWhileRunning(conn *websocket.Conn, op *ope
 			path := send.Path()
 			wrapper := StorageProgressReader(op, "fs_progress", send.Name())
 			state := s.container.DaemonState()
-			err = RsyncSend(projectPrefix(s.container.Project(), ctName), shared.AddSlash(path), conn, wrapper, bwlimit, state.OS.ExecPath)
+			err = RsyncSend(projectPrefix(s.container.Project(), ctName), shared.AddSlash(path), conn, wrapper, s.rsyncArgs, bwlimit, state.OS.ExecPath)
 			if err != nil {
 				return err
 			}
@@ -94,25 +95,25 @@ func (s rsyncStorageSourceDriver) SendWhileRunning(conn *websocket.Conn, op *ope
 
 	wrapper := StorageProgressReader(op, "fs_progress", s.container.Name())
 	state := s.container.DaemonState()
-	return RsyncSend(projectPrefix(s.container.Project(), ctName), shared.AddSlash(s.container.Path()), conn, wrapper, bwlimit, state.OS.ExecPath)
+	return RsyncSend(projectPrefix(s.container.Project(), ctName), shared.AddSlash(s.container.Path()), conn, wrapper, s.rsyncArgs, bwlimit, state.OS.ExecPath)
 }
 
 func (s rsyncStorageSourceDriver) SendAfterCheckpoint(conn *websocket.Conn, bwlimit string) error {
 	ctName, _, _ := containerGetParentAndSnapshotName(s.container.Name())
 	// resync anything that changed between our first send and the checkpoint
 	state := s.container.DaemonState()
-	return RsyncSend(projectPrefix(s.container.Project(), ctName), shared.AddSlash(s.container.Path()), conn, nil, bwlimit, state.OS.ExecPath)
+	return RsyncSend(projectPrefix(s.container.Project(), ctName), shared.AddSlash(s.container.Path()), conn, nil, s.rsyncArgs, bwlimit, state.OS.ExecPath)
 }
 
 func (s rsyncStorageSourceDriver) Cleanup() {
 	// noop
 }
 
-func rsyncStorageMigrationSource() (MigrationStorageSourceDriver, error) {
-	return rsyncStorageSourceDriver{nil, nil}, nil
+func rsyncStorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncStorageSourceDriver{nil, nil, args.RsyncArgs}, nil
 }
 
-func rsyncRefreshSource(c container, containerOnly bool, refreshSnapshots []string) (MigrationStorageSourceDriver, error) {
+func rsyncRefreshSource(c container, containerOnly bool, refreshSnapshots []string, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
 	var snapshots = []container{}
 	if !containerOnly {
 		allSnapshots, err := c.Snapshots()
@@ -130,10 +131,10 @@ func rsyncRefreshSource(c container, containerOnly bool, refreshSnapshots []stri
 		}
 	}
 
-	return rsyncStorageSourceDriver{c, snapshots}, nil
+	return rsyncStorageSourceDriver{c, snapshots, args.RsyncArgs}, nil
 }
 
-func rsyncMigrationSource(c container, containerOnly bool) (MigrationStorageSourceDriver, error) {
+func rsyncMigrationSource(c container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
 	var err error
 	var snapshots = []container{}
 	if !containerOnly {
@@ -143,7 +144,7 @@ func rsyncMigrationSource(c container, containerOnly bool) (MigrationStorageSour
 		}
 	}
 
-	return rsyncStorageSourceDriver{c, snapshots}, nil
+	return rsyncStorageSourceDriver{c, snapshots, args.RsyncArgs}, nil
 }
 
 func snapshotProtobufToContainerArgs(project string, containerName string, snap *migration.Snapshot) db.ContainerArgs {
diff --git a/lxd/storage_mock.go b/lxd/storage_mock.go
index ed81e1a841..d7757612f4 100644
--- a/lxd/storage_mock.go
+++ b/lxd/storage_mock.go
@@ -226,7 +226,7 @@ func (s *storageMock) PreservesInodes() bool {
 	return false
 }
 
-func (s *storageMock) MigrationSource(container container, containerOnly bool) (MigrationStorageSourceDriver, error) {
+func (s *storageMock) MigrationSource(container container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
 	return nil, fmt.Errorf("not implemented")
 }
 
@@ -246,7 +246,7 @@ func (s *storageMock) StoragePoolVolumeCopy(source *api.StorageVolumeSource) err
 	return nil
 }
 
-func (s *storageMock) StorageMigrationSource() (MigrationStorageSourceDriver, error) {
+func (s *storageMock) StorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
 	return nil, nil
 }
 
diff --git a/lxd/storage_zfs.go b/lxd/storage_zfs.go
index d10935b5ed..9f89244317 100644
--- a/lxd/storage_zfs.go
+++ b/lxd/storage_zfs.go
@@ -2660,7 +2660,7 @@ func (s *storageZfs) PreservesInodes() bool {
 	return true
 }
 
-func (s *storageZfs) MigrationSource(ct container, containerOnly bool) (MigrationStorageSourceDriver, error) {
+func (s *storageZfs) MigrationSource(ct container, containerOnly bool, args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
 	/* If the container is a snapshot, let's just send that; we don't need
 	* to send anything else, because that's all the user asked for.
 	 */
@@ -3243,8 +3243,8 @@ func (s *zfsMigrationSourceDriver) SendStorageVolume(conn *websocket.Conn, op *o
 	return fmt.Errorf(msg)
 }
 
-func (s *storageZfs) StorageMigrationSource() (MigrationStorageSourceDriver, error) {
-	return rsyncStorageMigrationSource()
+func (s *storageZfs) StorageMigrationSource(args MigrationSourceArgs) (MigrationStorageSourceDriver, error) {
+	return rsyncStorageMigrationSource(args)
 }
 
 func (s *storageZfs) StorageMigrationSink(conn *websocket.Conn, op *operation, storage storage, args MigrationSinkArgs) error {


More information about the lxc-devel mailing list