]> git.proxmox.com Git - mirror_zfs-debian.git/blame - tests/zfs-tests/tests/functional/cli_root/zfs_receive/receive-o-x_props_override.ksh
New upstream version 0.7.6
[mirror_zfs-debian.git] / tests / zfs-tests / tests / functional / cli_root / zfs_receive / receive-o-x_props_override.ksh
CommitLineData
cae5b340
AX
1#!/bin/ksh -p
2#
3# CDDL HEADER START
4#
5# This file and its contents are supplied under the terms of the
6# Common Development and Distribution License ("CDDL"), version 1.0.
7# You may only use this file in accordance with the terms of version
8# 1.0 of the CDDL.
9#
10# A full copy of the text of the CDDL should have accompanied this
11# source. A copy of the CDDL is also available via the Internet at
12# http://www.illumos.org/license/CDDL.
13#
14# CDDL HEADER END
15#
16
17#
18# Copyright 2017, loli10K <ezomori.nozomu@gmail.com>. All rights reserved.
19#
20
21. $STF_SUITE/include/libtest.shlib
22. $STF_SUITE/tests/functional/cli_root/zfs_set/zfs_set_common.kshlib
23
24#
25# DESCRIPTION:
26# Verify ZFS property override (-o) and exclude (-x) options work when
27# receiving a send stream
28#
29# STRATEGY:
30# 1. Create a filesystem with children.
31# 2. Snapshot the filesystems.
32# 3. Create various send streams (full, incremental, replication) and verify
33# we can both override and exclude native and user properties.
34#
35
36verify_runnable "both"
37
38function cleanup
39{
40 log_must rm -f $streamfile_full
41 log_must rm -f $streamfile_incr
42 log_must rm -f $streamfile_repl
43 log_must rm -f $streamfile_trun
44 log_must zfs destroy -r -f $orig
45 log_must zfs destroy -r -f $dest
46}
47
48#
49# Verify property $2 is set from source $4 on dataset $1 and has value $3.
50#
51# $1 checked dataset
52# $2 user property
53# $3 property value
54# $4 source
55#
56function check_prop_source
57{
58 typeset dataset="$1"
59 typeset prop="$2"
60 typeset value="$3"
61 typeset source="$4"
62 typeset chk_value=$(get_prop "$prop" "$dataset")
63 typeset chk_source=$(get_source "$prop" "$dataset")
64
65 if [[ "$chk_value" != "$value" || "$chk_source" != "$4" ]]
66 then
67 return 1
68 else
69 return 0
70 fi
71}
72
73#
74# Verify target dataset $1 inherit property $2 from dataset $3.
75#
76# $1 checked dataset
77# $2 property
78# $3 inherited dataset
79#
80function check_prop_inherit
81{
82 typeset checked_dtst="$1"
83 typeset prop="$2"
84 typeset inherited_dtst="$3"
85 typeset inherited_value=$(get_prop "$prop" "$inherited_dtst")
86 typeset value=$(get_prop "$prop" "$checked_dtst")
87 typeset source=$(get_source "$prop" "$checked_dtst")
88
89 if [[ "$value" != "$inherited_value" || \
90 "$source" != "inherited from $inherited_dtst" ]]
91 then
92 return 1
93 else
94 return 0
95 fi
96}
97
98#
99# Verify property $2 received value on dataset $1 has value $3
100#
101# $1 checked dataset
102# $2 property name
103# $3 checked value
104#
105function check_prop_received
106{
107 typeset dataset="$1"
108 typeset prop="$2"
109 typeset value="$3"
110
111 received=$(zfs get -H -o received "$prop" "$dataset")
112 if (($? != 0)); then
113 log_fail "Unable to get $prop received value for dataset " \
114 "$dataset"
115 fi
116 if [[ "$received" == "$value" ]]
117 then
118 return 0
119 else
120 return 1
121 fi
122}
123
124#
125# Verify user property $2 is not set on dataset $1
126#
127# $1 checked dataset
128# $2 property name
129#
130function check_prop_missing
131{
132 typeset dataset="$1"
133 typeset prop="$2"
134
135 value=$(zfs get -H -o value "$prop" "$dataset")
136 if (($? != 0)); then
137 log_fail "Unable to get $prop value for dataset $dataset"
138 fi
139 if [[ "-" == "$value" ]]
140 then
141 return 0
142 else
143 return 1
144 fi
145}
146
147log_assert "ZFS receive property override and exclude options work as expected."
148log_onexit cleanup
149
150orig=$TESTPOOL/$TESTFS1
151origsub=$orig/sub
152dest=$TESTPOOL/$TESTFS2
153destsub=$dest/sub
154typeset userprop=$(valid_user_property 8)
155typeset userval=$(user_property_value 8)
156typeset streamfile_full=$TESTDIR/streamfile_full.$$
157typeset streamfile_incr=$TESTDIR/streamfile_incr.$$
158typeset streamfile_repl=$TESTDIR/streamfile_repl.$$
159typeset streamfile_trun=$TESTDIR/streamfile_trun.$$
160
161#
162# 3.1 Verify we can't specify the same property in multiple -o or -x options
163# or an invalid value was specified.
164#
165# Create a full send stream
166log_must zfs create $orig
167log_must zfs snapshot $orig@snap1
168log_must eval "zfs send $orig@snap1 > $streamfile_full"
169# Verify we reject invalid options
170log_mustnot eval "zfs recv $dest -o atime < $streamfile_full"
171log_mustnot eval "zfs recv $dest -x atime=off < $streamfile_full"
172log_mustnot eval "zfs recv $dest -o atime=off -x atime < $streamfile_full"
173log_mustnot eval "zfs recv $dest -o atime=off -o atime=on < $streamfile_full"
174log_mustnot eval "zfs recv $dest -x atime -x atime < $streamfile_full"
175log_mustnot eval "zfs recv $dest -o version=1 < $streamfile_full"
176log_mustnot eval "zfs recv $dest -x version < $streamfile_full"
177log_mustnot eval "zfs recv $dest -x normalization < $streamfile_full"
178# Verify we also reject invalid ZVOL options
179log_must zfs create -V 32K -s $orig/zvol
180log_must eval "zfs send $orig@snap1 > $streamfile_full"
181log_mustnot eval "zfs recv $dest -x volsize < $streamfile_full"
182log_mustnot eval "zfs recv $dest -o volsize=32K < $streamfile_full"
183# Cleanup
184block_device_wait
185log_must_busy zfs destroy -r -f $orig
186
187#
188# 3.2 Verify -o property=value works on streams without properties.
189#
190# Create a full send stream
191log_must zfs create $orig
192log_must zfs snapshot $orig@snap1
193log_must eval "zfs send $orig@snap1 > $streamfile_full"
194# Receive the full stream, override some properties
195log_must eval "zfs recv -o compression=on -o '$userprop:dest'='$userval' "\
196 "$dest < $streamfile_full"
197log_must eval "check_prop_source $dest compression on local"
198log_must eval "check_prop_source $dest '$userprop:dest' '$userval' local"
199# Cleanup
200log_must zfs destroy -r -f $orig
201log_must zfs destroy -r -f $dest
202
203#
204# 3.3 Verify -o property=value and -x work on both native and user properties
205# for an incremental replication send stream.
206#
207# Create a dataset tree and receive it
208log_must zfs create $orig
209log_must zfs create $origsub
210log_must zfs snapshot -r $orig@snap1
211log_must eval "zfs send -R $orig@snap1 > $streamfile_repl"
212log_must eval "zfs recv $dest < $streamfile_repl"
213# Fill the datasets with properties and create an incremental replication stream
214log_must zfs snapshot -r $orig@snap2
215log_must eval "zfs set copies=2 $orig"
216log_must eval "zfs set '$userprop:orig'='$userval' $orig"
217log_must eval "zfs set '$userprop:orig'='$userval' $origsub"
218log_must eval "zfs set '$userprop:snap'='$userval' $orig@snap1"
219log_must eval "zfs set '$userprop:snap'='$userval' $origsub@snap2"
220log_must eval "zfs send -R -I $orig@snap1 $orig@snap2 > $streamfile_incr"
221# Sets various combination of override and exclude options
222log_must eval "zfs recv -F -o atime=off -o '$userprop:dest2'='$userval' "\
223 "-o quota=123456789 -x compression -x '$userprop:orig' " \
224 "-x '$userprop:snap2' $dest < $streamfile_incr"
225# Verify we can correctly override and exclude properties
226log_must eval "check_prop_source $dest copies 2 received"
227log_must eval "check_prop_source $dest atime off local"
228log_must eval "check_prop_source $dest '$userprop:dest2' '$userval' local"
229log_must eval "check_prop_source $dest quota 123456789 local"
230log_must eval "check_prop_inherit $destsub copies $dest"
231log_must eval "check_prop_inherit $destsub atime $dest"
232log_must eval "check_prop_inherit $destsub '$userprop:dest2' $dest"
233log_must eval "check_prop_source $destsub quota 0 default"
234log_must eval "check_prop_source $destsub compression off default"
235log_must eval "check_prop_missing $dest '$userprop:orig'"
236log_must eval "check_prop_missing $destsub '$userprop:orig'"
237log_must eval "check_prop_source " \
238 "$dest@snap1 '$userprop:snap' '$userval' received"
239log_must eval "check_prop_source " \
240 "$destsub@snap2 '$userprop:snap' '$userval' received"
241log_must eval "check_prop_missing $dest@snap2 '$userprop:snap2'"
242log_must eval "check_prop_missing $destsub@snap2 '$userprop:snap2'"
243# Cleanup
244log_must zfs destroy -r -f $orig
245log_must zfs destroy -r -f $dest
246
247#
248# 3.4 Verify '-x property' does not remove existing local properties and a
249# modified sent property is received and updated to the new value but can
250# still be excluded.
251#
252# Create a dataset tree
253log_must zfs create $orig
254log_must zfs create $origsub
255log_must zfs snapshot -r $orig@snap1
256log_must eval "zfs set copies=2 $orig"
257log_must eval "zfs set '$userprop:orig'='oldval' $orig"
258log_must eval "zfs set '$userprop:orig'='oldsubval' $origsub"
259log_must eval "zfs send -R $orig@snap1 > $streamfile_repl"
260log_must eval "zfs receive $dest < $streamfile_repl"
261log_must eval "check_prop_source $dest copies 2 received"
262log_must eval "check_prop_inherit $destsub copies $dest"
263log_must eval "check_prop_source $dest '$userprop:orig' 'oldval' received"
264log_must eval "check_prop_source $destsub '$userprop:orig' 'oldsubval' received"
265# Set new custom properties on both source and destination
266log_must eval "zfs set copies=3 $orig"
267log_must eval "zfs set '$userprop:orig'='newval' $orig"
268log_must eval "zfs set '$userprop:orig'='newsubval' $origsub"
269log_must eval "zfs set compression=gzip $dest"
270log_must eval "zfs set '$userprop:dest'='localval' $dest"
271# Receive the new stream, verify we preserve locally set properties
272log_must zfs snapshot -r $orig@snap2
273log_must eval "zfs send -R -I $orig@snap1 $orig@snap2 > $streamfile_incr"
274log_must eval "zfs recv -F -x copies -x compression -x '$userprop:orig' " \
275 "-x '$userprop:dest' $dest < $streamfile_incr"
276log_must eval "check_prop_source $dest '$userprop:dest' 'localval' local"
277log_must eval "check_prop_received $dest '$userprop:orig' 'newval'"
278log_must eval "check_prop_received $destsub '$userprop:orig' 'newsubval'"
279log_must eval "check_prop_missing $dest '$userprop:orig'"
280log_must eval "check_prop_missing $destsub '$userprop:orig'"
281log_must eval "check_prop_source $dest copies 1 default"
282log_must eval "check_prop_received $dest copies 3"
283log_must eval "check_prop_source $destsub copies 1 default"
284log_must eval "check_prop_received $destsub copies '-'"
285log_must eval "check_prop_source $dest compression gzip local"
286log_must eval "check_prop_inherit $destsub compression $dest"
287# Cleanup
288log_must zfs destroy -r -f $orig
289log_must zfs destroy -r -f $dest
290
291#
292# 3.5 Verify we can exclude non-inheritable properties from a send stream
293#
294# Create a dataset tree and replication stream
295log_must zfs create $orig
296log_must zfs create $origsub
297log_must zfs snapshot -r $orig@snap1
298log_must eval "zfs set quota=123456789 $orig"
299log_must eval "zfs send -R $orig@snap1 > $streamfile_repl"
300# Receive the stream excluding non-inheritable properties
301log_must eval "zfs recv -F -x quota $dest < $streamfile_repl"
302log_must eval "check_prop_source $dest quota 0 default"
303log_must eval "check_prop_source $destsub quota 0 default"
304# Set some non-inheritable properties on the destination, verify we keep them
305log_must eval "zfs set quota=123456789 $dest"
306log_must eval "zfs set canmount=off $destsub"
307log_must zfs snapshot -r $orig@snap2
308log_must eval "zfs send -R -I $orig@snap1 $orig@snap2 > $streamfile_incr"
309log_must eval "zfs recv -F -x quota -x canmount $dest < $streamfile_incr"
310log_must eval "check_prop_source $dest quota 123456789 local"
311log_must eval "check_prop_source $destsub quota 0 default"
312log_must eval "check_prop_source $destsub canmount off local"
313# Cleanup
314log_must zfs destroy -r -f $orig
315log_must zfs destroy -r -f $dest
316
317#
318# 3.6 Verify we correctly restore existing properties on a failed receive
319#
320# Receive a "clean" dataset tree
321log_must zfs create $orig
322log_must zfs create $origsub
323log_must zfs snapshot -r $orig@snap1
324log_must eval "zfs send -R $orig@snap1 > $streamfile_repl"
325log_must eval "zfs receive $dest < $streamfile_repl"
326# Set custom properties on the destination
327log_must eval "zfs set atime=off $dest"
328log_must eval "zfs set quota=123456789 $dest"
329log_must eval "zfs set '$userprop:orig'='$userval' $dest"
330log_must eval "zfs set '$userprop:origsub'='$userval' $destsub"
331# Create a truncated incremental replication stream
332mntpnt=$(get_prop mountpoint $orig)
333log_must eval "dd if=/dev/urandom of=$mntpnt/file bs=1024k count=10"
334log_must zfs snapshot -r $orig@snap2
335log_must eval "zfs send -R -I $orig@snap1 $orig@snap2 > $streamfile_incr"
336log_must eval "dd if=$streamfile_incr of=$streamfile_trun bs=1024k count=9"
337# Receive the truncated stream, verify original properties are kept
338log_mustnot eval "zfs recv -F -o copies=3 -o quota=987654321 "\
339 "-o '$userprop:new'='badval' $dest < $streamfile_trun"
340log_must eval "check_prop_source $dest copies 1 default"
341log_must eval "check_prop_source $destsub copies 1 default"
342log_must eval "check_prop_source $dest atime off local"
343log_must eval "check_prop_inherit $destsub atime $dest"
344log_must eval "check_prop_source $dest quota 123456789 local"
345log_must eval "check_prop_source $destsub quota 0 default"
346log_must eval "check_prop_source $dest '$userprop:orig' '$userval' local"
347log_must eval "check_prop_inherit $destsub '$userprop:orig' $dest"
348log_must eval "check_prop_source $destsub '$userprop:origsub' '$userval' local"
349log_must eval "check_prop_missing $dest '$userprop:new'"
350# Cleanup
351log_must zfs destroy -r -f $orig
352log_must zfs destroy -r -f $dest
353
354#
355# 3.7 Verify we can't receive a send stream overriding or excluding properties
356# invalid for the dataset type unless the stream it's recursive, in which
357# case only the appropriate properties are set on the destination.
358#
359log_must zfs create -V 128K -s $orig
360log_must zfs snapshot $orig@snap1
361log_must eval "zfs send $orig@snap1 > $streamfile_full"
362log_mustnot eval "zfs receive -x atime $dest < $streamfile_full"
363log_mustnot eval "zfs receive -o atime=off $dest < $streamfile_full"
364log_must zfs destroy -r -f $orig
365log_must zfs create $orig
366log_must zfs create -V 128K -s $origsub
367log_must zfs snapshot -r $orig@snap1
368log_must eval "zfs send -R $orig@snap1 > $streamfile_repl"
369log_must eval "zfs receive -o atime=off $dest < $streamfile_repl"
370log_must eval "check_prop_source $dest type filesystem -"
371log_must eval "check_prop_source $dest atime off local"
372log_must eval "check_prop_source $destsub type volume -"
373log_must eval "check_prop_source $destsub atime - -"
047218e2
AX
374# Cleanup
375log_must zfs destroy -r -f $orig
376log_must zfs destroy -r -f $dest
377
378#
379# 3.8 Verify 'zfs recv -x|-o' works correctly when used in conjunction with -d
380# and -e options.
381#
382log_must zfs create -p $orig/1/2/3/4
383log_must eval "zfs set copies=2 $orig"
384log_must eval "zfs set atime=on $orig"
385log_must eval "zfs set '$userprop:orig'='oldval' $orig"
386log_must zfs snapshot -r $orig@snap1
387log_must eval "zfs send -R $orig/1/2@snap1 > $streamfile_repl"
388# Verify 'zfs recv -e'
389log_must zfs create $dest
390log_must eval "zfs receive -e -o copies=3 -x atime "\
391 "-o '$userprop:orig'='newval' $dest < $streamfile_repl"
392log_must datasetexists $dest/2/3/4
393log_must eval "check_prop_source $dest/2 copies 3 local"
394log_must eval "check_prop_inherit $dest/2/3/4 copies $dest/2"
395log_must eval "check_prop_source $dest/2/3/4 atime on default"
396log_must eval "check_prop_source $dest/2 '$userprop:orig' 'newval' local"
397log_must eval "check_prop_inherit $dest/2/3/4 '$userprop:orig' $dest/2"
398log_must zfs destroy -r -f $dest
399# Verify 'zfs recv -d'
400log_must zfs create $dest
401typeset fs="$(echo $orig | awk -F'/' '{print $NF}')"
402log_must eval "zfs receive -d -o copies=3 -x atime "\
403 "-o '$userprop:orig'='newval' $dest < $streamfile_repl"
404log_must datasetexists $dest/$fs/1/2/3/4
405log_must eval "check_prop_source $dest/$fs/1/2 copies 3 local"
406log_must eval "check_prop_inherit $dest/$fs/1/2/3/4 copies $dest/$fs/1/2"
407log_must eval "check_prop_source $dest/$fs/1/2/3/4 atime on default"
408log_must eval "check_prop_source $dest/$fs/1/2 '$userprop:orig' 'newval' local"
409log_must eval "check_prop_inherit $dest/$fs/1/2/3/4 '$userprop:orig' $dest/$fs/1/2"
cae5b340
AX
410# We don't need to cleanup here
411
412log_pass "ZFS receive property override and exclude options passed."