All I'm trying to do here is replicate one file system. So I create it, touch a file so I know it's made it.
zfs create -o rpool/t1
touch /rpool/t1/1
OK, snapshot it and send it.
zfs snapshot rpool/t1@t1s1
zfs send rpool/t1@t1s1 | zfs recv rpool/t2
Create another file, and create a snapshot at both source and destination.
touch /rpool/t1/2
zfs snapshot rpool/t1@t1s2
zfs snapshot rpool/t2@t2s2
And now send an incremental stream from the original.
zfs send -i rpool/t1@t1s1 rpool/t1@t1s2 | zfs recv -F rpool/t2
That works, the whole point of the -F flag is to discard any subsequent changes. (You'll usually need this if the file system is mounted at the receiver, because even access time updates count as updates that will need to be discarded.) It will roll back rpool/t2 to the original @t1s1 snapshot, discarding the local @t2s2 snapshot, then update the rpool/t2 file system to the @t1s2 snapshot.
So far so good.
Now a minor variation.
I create it, touch a file so I know it's made it.
zfs create rpool/t1
touch /rpool/t1/1
OK, snapshot it and send it.
zfs snapshot rpool/t1@s1
zfs send rpool/t1@s1 | zfs recv rpool/t2
Create another file, and create a snapshot at both source and destination.
touch /rpool/t1/2
zfs snapshot rpool/t1@s2
zfs snapshot rpool/t2@s2
And now send the incremental stream just like last time.
zfs send -i rpool/t1@s1 rpool/t1@s2 | zfs recv -F rpool/t2
Kaboom. This fails, reporting:
cannot restore to rpool/t2@s2: destination already exists
What? The problem is hinted at in the zfs man page, where the description of -F says:
If receiving an incremental replication stream (for example,
one generated by zfs send -R [-i|-I]), destroy snapshots and
file systems that do not exist on the sending side.
The problem, then, is that zfs won't destroy the @s2 snapshot that exists at the receiver, because a snapshot of the same name exists in the source. It's not the same snapshot, of course, but it has the same name. This prevents the rollback, and the receive fails.
Snapshot name collisions are pretty common. We have an automatic snapshot regime, so pretty much every file system we have has a daily snapshot that embeds the date, and being automatic, they all have the same name.
What this means in practice is that if you have snapshots created on the receiving side, you'll have to explicitly roll the file system back to the snapshot you sent to previously, to avoid hitting name collisions.
I think this behaviour is wrong, although I'm not quite confident enough to call it a bug. The point is that on the receiving side, any snapshots created after the one that was sent are irrelevant - it shouldn't matter what their names are, and I'm not at all sure why zfs even bothers checking the names of snapshots that ought to be deleted.