diff --git a/cmd/zfs/zfs_main.c b/cmd/zfs/zfs_main.c index 85a37c925cf0..bfc9e1c62a4a 100644 --- a/cmd/zfs/zfs_main.c +++ b/cmd/zfs/zfs_main.c @@ -247,9 +247,9 @@ get_usage(zfs_help_t idx) case HELP_PROMOTE: return (gettext("\tpromote \n")); case HELP_RECEIVE: - return (gettext("\treceive [-vnFu] \n" - "\treceive [-vnFu] [-d | -e] \n")); + "\treceive [-vnFus] [-d | -e] \n")); case HELP_RENAME: return (gettext("\trename [-f] " "\n" @@ -3853,7 +3853,7 @@ zfs_do_send(int argc, char **argv) } /* - * zfs receive [-vnFu] [-d | -e] + * zfs receive [-vnFus] [-d | -e] * * Restore a backup stream from stdin. */ @@ -3864,8 +3864,11 @@ zfs_do_receive(int argc, char **argv) recvflags_t flags = { 0 }; /* check options */ - while ((c = getopt(argc, argv, ":denuvF")) != -1) { + while ((c = getopt(argc, argv, ":denuvFs")) != -1) { switch (c) { + case 's': + flags.skip = B_TRUE; + break; case 'd': flags.isprefix = B_TRUE; break; diff --git a/include/libzfs.h b/include/libzfs.h index 08a493579c24..126ccb3b73f9 100644 --- a/include/libzfs.h +++ b/include/libzfs.h @@ -673,6 +673,9 @@ typedef struct recvflags { /* do not mount file systems as they are extracted (private) */ boolean_t nomount; + + /* skip existing snapshots */ + boolean_t skip; } recvflags_t; extern int zfs_receive(libzfs_handle_t *, const char *, recvflags_t *, diff --git a/lib/libzfs/libzfs_sendrecv.c b/lib/libzfs/libzfs_sendrecv.c index ebd8a7be2ce4..067338d8e2ce 100644 --- a/lib/libzfs/libzfs_sendrecv.c +++ b/lib/libzfs/libzfs_sendrecv.c @@ -2825,7 +2825,7 @@ zfs_receive_one(libzfs_handle_t *hdl, int infd, const char *tosnap, * away (and have therefore specified -F and removed any * snapshots). */ - if (stream_wantsnewfs) { + if (stream_wantsnewfs && !flags->skip) { if (!flags->force) { zcmd_free_nvlists(&zc); zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, @@ -2913,6 +2913,17 @@ zfs_receive_one(libzfs_handle_t *hdl, int infd, const char *tosnap, zc.zc_begin_record = drr_noswap->drr_u.drr_begin; zc.zc_cookie = infd; zc.zc_guid = flags->force; + + if (zfs_dataset_exists(hdl, zc.zc_value, ZFS_TYPE_SNAPSHOT) && + flags->skip) { + (void) printf("skipping snapshot %s - %s already exists\n", + drrb->drr_toname, zc.zc_value); + (void) fflush(stdout); + + zcmd_free_nvlists(&zc); + return (recv_skip(hdl, infd, flags->byteswap)); + } + if (flags->verbose) { (void) printf("%s %s stream of %s into %s\n", flags->dryrun ? "would receive" : "receiving", diff --git a/man/man8/zfs.8 b/man/man8/zfs.8 index 6a6439fce0b5..8d3be29611a8 100644 --- a/man/man8/zfs.8 +++ b/man/man8/zfs.8 @@ -184,12 +184,12 @@ zfs \- configures ZFS file systems .LP .nf -\fBzfs\fR \fBreceive | recv\fR [\fB-vnFu\fR] \fIfilesystem\fR|\fIvolume\fR|\fIsnapshot\fR +\fBzfs\fR \fBreceive | recv\fR [\fB-vnFus\fR] \fIfilesystem\fR|\fIvolume\fR|\fIsnapshot\fR .fi .LP .nf -\fBzfs\fR \fBreceive | recv\fR [\fB-vnFu\fR] [\fB-d\fR|\fB-e\fR] \fIfilesystem\fR +\fBzfs\fR \fBreceive | recv\fR [\fB-vnFus\fR] [\fB-d\fR|\fB-e\fR] \fIfilesystem\fR .fi .LP @@ -3001,11 +3001,11 @@ or the origin's origin, etc. .ne 2 .mk .na -\fB\fBzfs receive\fR [\fB-vnFu\fR] \fIfilesystem\fR|\fIvolume\fR|\fIsnapshot\fR\fR +\fB\fBzfs receive\fR [\fB-vnFus\fR] \fIfilesystem\fR|\fIvolume\fR|\fIsnapshot\fR\fR .ad .br .na -\fB\fBzfs receive\fR [\fB-vnFu\fR] [\fB-d\fR|\fB-e\fR] \fIfilesystem\fR\fR +\fB\fBzfs receive\fR [\fB-vnFus\fR] [\fB-d\fR|\fB-e\fR] \fIfilesystem\fR\fR .ad .sp .6 .RS 4n @@ -3101,6 +3101,17 @@ then the receiving system must have that feature enabled as well. See \fBzpool-features\fR(5) for details on ZFS feature flags and the \fBembedded_data\fR feature. .RE + +.sp +.ne 2 +.mk +.na +\fB\fB-s\fR\fR +.ad +.sp .6 +.RS 4n +Skip existing snapshots in a stream. +.RE .RE .sp