2 * Third attempt at a shadow copy module
4 * Copyright (C) Andrew Tridgell 2007 (portions taken from shadow_copy2)
5 * Copyright (C) Ed Plese 2009
6 * Copyright (C) Volker Lendecke 2011
7 * Copyright (C) Christian Ambach 2011
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
26 This is a 3rd implemetation of a shadow copy module for exposing
27 snapshots to windows clients as shadow copies. This version has the
30 1) you don't need to populate your shares with symlinks to the
31 snapshots. This can be very important when you have thousands of
32 shares, or use [homes]
34 2) the inode number of the files is altered so it is different
35 from the original. This allows the 'restore' button to work
36 without a sharing violation
38 3) shadow copy results can be sorted before being sent to the
39 client. This is beneficial for filesystems that don't read
40 directories alphabetically (the default unix).
42 4) vanity naming for snapshots. Snapshots can be named in any
43 format compatible with str[fp]time conversions.
45 5) time stamps in snapshot names can be represented in localtime
50 shadow:snapdir = <directory where snapshots are kept>
52 This is the directory containing the @GMT-* snapshot directories. If it is an absolute
53 path it is used as-is. If it is a relative path, then it is taken relative to the mount
54 point of the filesystem that the root of this share is on
56 shadow:basedir = <base directory that snapshots are from>
58 This is an optional parameter that specifies the directory that
59 the snapshots are relative to. It defaults to the filesystem
62 shadow:fixinodes = yes/no
64 If you enable shadow:fixinodes then this module will modify the
65 apparent inode number of files in the snapshot directories using
66 a hash of the files path. This is needed for snapshot systems
67 where the snapshots have the same device:inode number as the
68 original files (such as happens with GPFS snapshots). If you
69 don't set this option then the 'restore' button in the shadow
70 copy UI will fail with a sharing violation.
72 shadow:sort = asc/desc, or not specified for unsorted (default)
74 This is an optional parameter that specifies that the shadow
75 copy directories should be sorted before sending them to the
76 client. This can be beneficial as unix filesystems are usually
77 not listed alphabetically sorted. If enabled, you typically
78 want to specify descending order.
80 shadow:format = <format specification for snapshot names>
82 This is an optional parameter that specifies the format
83 specification for the naming of snapshots. The format must
84 be compatible with the conversion specifications recognized
85 by str[fp]time. The default value is "@GMT-%Y.%m.%d-%H.%M.%S".
87 shadow:sscanf = yes/no (default is no)
89 The time is the unsigned long integer (%lu) in the format string
90 rather than a time strptime() can parse. The result must be a unix time_t
93 shadow:localtime = yes/no (default is no)
95 This is an optional parameter that indicates whether the
96 snapshot names are in UTC/GMT or the local time.
99 The following command would generate a correctly formatted directory name
100 for use with the default parameters:
101 date -u +@GMT-%Y.%m.%d-%H.%M.%S
104 #include "includes.h"
105 #include "system/filesys.h"
106 #include "include/ntioctl.h"
107 #include <ccan/hash/hash.h>
108 #include "util_tdb.h"
110 static bool shadow_copy2_find_slashes(TALLOC_CTX
*mem_ctx
, const char *str
,
112 unsigned *pnum_offsets
)
114 unsigned num_offsets
;
121 while ((p
= strchr(p
, '/')) != NULL
) {
126 offsets
= talloc_array(mem_ctx
, size_t, num_offsets
);
127 if (offsets
== NULL
) {
133 while ((p
= strchr(p
, '/')) != NULL
) {
134 offsets
[num_offsets
] = p
-str
;
140 *pnum_offsets
= num_offsets
;
144 static char *shadow_copy2_insert_string(TALLOC_CTX
*mem_ctx
,
145 struct vfs_handle_struct
*handle
,
150 fstring snaptime_string
;
153 fmt
= lp_parm_const_string(SNUM(handle
->conn
), "shadow",
154 "format", GMT_FORMAT
);
156 if (lp_parm_bool(SNUM(handle
->conn
), "shadow", "sscanf", false)) {
157 snaptime_len
= snprintf(snaptime_string
, sizeof(snaptime_string
), fmt
,
158 (unsigned long)snapshot
);
159 if (snaptime_len
<= 0) {
160 DEBUG(10, ("snprintf failed\n"));
164 if (lp_parm_bool(SNUM(handle
->conn
), "shadow", "localtime", false)) {
165 if (localtime_r(&snapshot
, &snap_tm
) == 0) {
166 DEBUG(10, ("gmtime_r failed\n"));
170 if (gmtime_r(&snapshot
, &snap_tm
) == 0) {
171 DEBUG(10, ("gmtime_r failed\n"));
175 snaptime_len
= strftime(snaptime_string
, sizeof(snaptime_string
), fmt
,
177 if (snaptime_len
== 0) {
178 DEBUG(10, ("strftime failed\n"));
182 return talloc_asprintf(mem_ctx
, "/%s/%s",
183 lp_parm_const_string(
184 SNUM(handle
->conn
), "shadow", "snapdir",
190 * Strip a snapshot component from an filename as
191 * handed in via the smb layer.
192 * Returns the parsed timestamp and the stripped filename.
194 static bool shadow_copy2_strip_snapshot(TALLOC_CTX
*mem_ctx
,
195 struct vfs_handle_struct
*handle
,
205 size_t rest_len
, dst_len
;
207 p
= strstr_m(name
, "@GMT-");
211 if ((p
> name
) && (p
[-1] != '/')) {
214 q
= strptime(p
, GMT_FORMAT
, &tm
);
219 timestamp
= timegm(&tm
);
220 if (timestamp
== (time_t)-1) {
223 if ((p
== name
) && (q
[0] == '\0')) {
224 if (pstripped
!= NULL
) {
225 stripped
= talloc_strdup(mem_ctx
, "");
226 if (stripped
== NULL
) {
229 *pstripped
= stripped
;
231 *ptimestamp
= timestamp
;
239 rest_len
= strlen(q
);
240 dst_len
= (p
-name
) + rest_len
;
242 if (lp_parm_bool(SNUM(handle
->conn
), "shadow", "snapdirseverywhere",
246 insert
= shadow_copy2_insert_string(talloc_tos(), handle
,
248 if (insert
== NULL
) {
253 have_insert
= (strstr(name
, insert
+1) != NULL
);
260 if (pstripped
!= NULL
) {
261 stripped
= talloc_array(mem_ctx
, char, dst_len
+1);
262 if (stripped
== NULL
) {
267 memcpy(stripped
, name
, p
-name
);
270 memcpy(stripped
+ (p
-name
), q
, rest_len
);
272 stripped
[dst_len
] = '\0';
273 *pstripped
= stripped
;
275 *ptimestamp
= timestamp
;
282 static char *shadow_copy2_find_mount_point(TALLOC_CTX
*mem_ctx
,
283 vfs_handle_struct
*handle
)
285 char *path
= talloc_strdup(mem_ctx
, handle
->conn
->connectpath
);
290 if (stat(path
, &st
) != 0) {
297 while ((p
= strrchr(path
, '/')) && p
> path
) {
299 if (stat(path
, &st
) != 0) {
303 if (st
.st_dev
!= dev
) {
312 static char *shadow_copy2_convert(TALLOC_CTX
*mem_ctx
,
313 struct vfs_handle_struct
*handle
,
314 const char *name
, time_t timestamp
)
316 struct smb_filename converted_fname
;
318 size_t *slashes
= NULL
;
319 unsigned num_slashes
;
323 char *converted
= NULL
;
328 path
= talloc_asprintf(mem_ctx
, "%s/%s", handle
->conn
->connectpath
,
334 pathlen
= talloc_get_size(path
)-1;
336 DEBUG(10, ("converting %s\n", path
));
338 if (!shadow_copy2_find_slashes(talloc_tos(), path
,
339 &slashes
, &num_slashes
)) {
342 insert
= shadow_copy2_insert_string(talloc_tos(), handle
, timestamp
);
343 if (insert
== NULL
) {
346 insertlen
= talloc_get_size(insert
)-1;
347 converted
= talloc_array(mem_ctx
, char, pathlen
+ insertlen
+ 1);
348 if (converted
== NULL
) {
352 if (path
[pathlen
-1] != '/') {
354 * Append a fake slash to find the snapshot root
357 tmp
= talloc_realloc(talloc_tos(), slashes
,
358 size_t, num_slashes
+1);
363 slashes
[num_slashes
] = pathlen
;
369 if (!lp_parm_bool(SNUM(handle
->conn
), "shadow", "crossmountpoints",
373 mount_point
= shadow_copy2_find_mount_point(talloc_tos(),
375 if (mount_point
== NULL
) {
378 min_offset
= strlen(mount_point
);
379 TALLOC_FREE(mount_point
);
382 memcpy(converted
, path
, pathlen
+1);
383 converted
[pathlen
+insertlen
] = '\0';
385 ZERO_STRUCT(converted_fname
);
386 converted_fname
.base_name
= converted
;
388 for (i
= num_slashes
-1; i
>=0; i
--) {
394 if (offset
< min_offset
) {
399 memcpy(converted
+offset
, insert
, insertlen
);
402 memcpy(converted
+offset
, path
+ slashes
[i
],
403 pathlen
- slashes
[i
]);
405 ret
= SMB_VFS_NEXT_LSTAT(handle
, &converted_fname
);
407 DEBUG(10, ("Trying %s: %d (%s)\n", converted
,
408 ret
, ret
== 0 ? "ok" : strerror(errno
)));
413 if (errno
== ENOTDIR
) {
415 * This is a valid condition: We appended the
416 * .snaphots/@GMT.. to a file name. Just try
417 * with the upper levels.
421 if (errno
!= ENOENT
) {
422 /* Other problem than "not found" */
431 DEBUG(10, ("Found %s\n", converted
));
439 TALLOC_FREE(converted
);
441 TALLOC_FREE(slashes
);
448 modify a sbuf return to ensure that inodes in the shadow directory
449 are different from those in the main directory
451 static void convert_sbuf(vfs_handle_struct
*handle
, const char *fname
,
452 SMB_STRUCT_STAT
*sbuf
)
454 if (lp_parm_bool(SNUM(handle
->conn
), "shadow", "fixinodes", False
)) {
455 /* some snapshot systems, like GPFS, return the name
456 device:inode for the snapshot files as the current
457 files. That breaks the 'restore' button in the shadow copy
458 GUI, as the client gets a sharing violation.
460 This is a crude way of allowing both files to be
461 open at once. It has a slight chance of inode
462 number collision, but I can't see a better approach
463 without significant VFS changes
467 shash
= hash(fname
, strlen(fname
), 0) & 0xFF000000;
471 sbuf
->st_ex_ino
^= shash
;
475 static DIR *shadow_copy2_opendir(vfs_handle_struct
*handle
,
486 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
487 ×tamp
, &stripped
)) {
490 if (timestamp
== 0) {
491 return SMB_VFS_NEXT_OPENDIR(handle
, fname
, mask
, attr
);
493 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
494 TALLOC_FREE(stripped
);
498 ret
= SMB_VFS_NEXT_OPENDIR(handle
, conv
, mask
, attr
);
505 static int shadow_copy2_rename(vfs_handle_struct
*handle
,
506 const struct smb_filename
*smb_fname_src
,
507 const struct smb_filename
*smb_fname_dst
)
509 time_t timestamp_src
, timestamp_dst
;
511 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
512 smb_fname_src
->base_name
,
513 ×tamp_src
, NULL
)) {
516 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
517 smb_fname_dst
->base_name
,
518 ×tamp_dst
, NULL
)) {
521 if (timestamp_src
!= 0) {
525 if (timestamp_dst
!= 0) {
529 return SMB_VFS_NEXT_RENAME(handle
, smb_fname_src
, smb_fname_dst
);
532 static int shadow_copy2_symlink(vfs_handle_struct
*handle
,
533 const char *oldname
, const char *newname
)
535 time_t timestamp_old
, timestamp_new
;
537 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, oldname
,
538 ×tamp_old
, NULL
)) {
541 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, newname
,
542 ×tamp_new
, NULL
)) {
545 if ((timestamp_old
!= 0) || (timestamp_new
!= 0)) {
549 return SMB_VFS_NEXT_SYMLINK(handle
, oldname
, newname
);
552 static int shadow_copy2_link(vfs_handle_struct
*handle
,
553 const char *oldname
, const char *newname
)
555 time_t timestamp_old
, timestamp_new
;
557 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, oldname
,
558 ×tamp_old
, NULL
)) {
561 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, newname
,
562 ×tamp_new
, NULL
)) {
565 if ((timestamp_old
!= 0) || (timestamp_new
!= 0)) {
569 return SMB_VFS_NEXT_LINK(handle
, oldname
, newname
);
572 static int shadow_copy2_stat(vfs_handle_struct
*handle
,
573 struct smb_filename
*smb_fname
)
576 char *stripped
, *tmp
;
577 int ret
, saved_errno
;
579 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
580 smb_fname
->base_name
,
581 ×tamp
, &stripped
)) {
584 if (timestamp
== 0) {
585 return SMB_VFS_NEXT_STAT(handle
, smb_fname
);
588 tmp
= smb_fname
->base_name
;
589 smb_fname
->base_name
= shadow_copy2_convert(
590 talloc_tos(), handle
, stripped
, timestamp
);
591 TALLOC_FREE(stripped
);
593 if (smb_fname
->base_name
== NULL
) {
594 smb_fname
->base_name
= tmp
;
598 ret
= SMB_VFS_NEXT_STAT(handle
, smb_fname
);
601 TALLOC_FREE(smb_fname
->base_name
);
602 smb_fname
->base_name
= tmp
;
605 convert_sbuf(handle
, smb_fname
->base_name
, &smb_fname
->st
);
611 static int shadow_copy2_lstat(vfs_handle_struct
*handle
,
612 struct smb_filename
*smb_fname
)
615 char *stripped
, *tmp
;
616 int ret
, saved_errno
;
618 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
619 smb_fname
->base_name
,
620 ×tamp
, &stripped
)) {
623 if (timestamp
== 0) {
624 return SMB_VFS_NEXT_LSTAT(handle
, smb_fname
);
627 tmp
= smb_fname
->base_name
;
628 smb_fname
->base_name
= shadow_copy2_convert(
629 talloc_tos(), handle
, stripped
, timestamp
);
630 TALLOC_FREE(stripped
);
632 if (smb_fname
->base_name
== NULL
) {
633 smb_fname
->base_name
= tmp
;
637 ret
= SMB_VFS_NEXT_LSTAT(handle
, smb_fname
);
640 TALLOC_FREE(smb_fname
->base_name
);
641 smb_fname
->base_name
= tmp
;
644 convert_sbuf(handle
, smb_fname
->base_name
, &smb_fname
->st
);
650 static int shadow_copy2_fstat(vfs_handle_struct
*handle
, files_struct
*fsp
,
651 SMB_STRUCT_STAT
*sbuf
)
656 ret
= SMB_VFS_NEXT_FSTAT(handle
, fsp
, sbuf
);
660 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
661 fsp
->fsp_name
->base_name
,
665 if (timestamp
!= 0) {
666 convert_sbuf(handle
, fsp
->fsp_name
->base_name
, sbuf
);
671 static int shadow_copy2_open(vfs_handle_struct
*handle
,
672 struct smb_filename
*smb_fname
, files_struct
*fsp
,
673 int flags
, mode_t mode
)
676 char *stripped
, *tmp
;
677 int ret
, saved_errno
;
679 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
680 smb_fname
->base_name
,
681 ×tamp
, &stripped
)) {
684 if (timestamp
== 0) {
685 return SMB_VFS_NEXT_OPEN(handle
, smb_fname
, fsp
, flags
, mode
);
688 tmp
= smb_fname
->base_name
;
689 smb_fname
->base_name
= shadow_copy2_convert(
690 talloc_tos(), handle
, stripped
, timestamp
);
691 TALLOC_FREE(stripped
);
693 if (smb_fname
->base_name
== NULL
) {
694 smb_fname
->base_name
= tmp
;
698 ret
= SMB_VFS_NEXT_OPEN(handle
, smb_fname
, fsp
, flags
, mode
);
701 TALLOC_FREE(smb_fname
->base_name
);
702 smb_fname
->base_name
= tmp
;
708 static int shadow_copy2_unlink(vfs_handle_struct
*handle
,
709 const struct smb_filename
*smb_fname
)
713 int ret
, saved_errno
;
714 struct smb_filename
*conv
;
717 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
718 smb_fname
->base_name
,
719 ×tamp
, &stripped
)) {
722 if (timestamp
== 0) {
723 return SMB_VFS_NEXT_UNLINK(handle
, smb_fname
);
725 status
= copy_smb_filename(talloc_tos(), smb_fname
, &conv
);
726 if (!NT_STATUS_IS_OK(status
)) {
730 conv
->base_name
= shadow_copy2_convert(
731 conv
, handle
, stripped
, timestamp
);
732 TALLOC_FREE(stripped
);
733 if (conv
->base_name
== NULL
) {
736 ret
= SMB_VFS_NEXT_UNLINK(handle
, conv
);
743 static int shadow_copy2_chmod(vfs_handle_struct
*handle
, const char *fname
,
748 int ret
, saved_errno
;
751 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
752 ×tamp
, &stripped
)) {
755 if (timestamp
== 0) {
756 return SMB_VFS_NEXT_CHMOD(handle
, fname
, mode
);
758 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
759 TALLOC_FREE(stripped
);
763 ret
= SMB_VFS_NEXT_CHMOD(handle
, conv
, mode
);
770 static int shadow_copy2_chown(vfs_handle_struct
*handle
, const char *fname
,
771 uid_t uid
, gid_t gid
)
775 int ret
, saved_errno
;
778 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
779 ×tamp
, &stripped
)) {
782 if (timestamp
== 0) {
783 return SMB_VFS_NEXT_CHOWN(handle
, fname
, uid
, gid
);
785 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
786 TALLOC_FREE(stripped
);
790 ret
= SMB_VFS_NEXT_CHOWN(handle
, conv
, uid
, gid
);
797 static int shadow_copy2_chdir(vfs_handle_struct
*handle
,
802 int ret
, saved_errno
;
805 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
806 ×tamp
, &stripped
)) {
809 if (timestamp
== 0) {
810 return SMB_VFS_NEXT_CHDIR(handle
, fname
);
812 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
813 TALLOC_FREE(stripped
);
817 ret
= SMB_VFS_NEXT_CHDIR(handle
, conv
);
824 static int shadow_copy2_ntimes(vfs_handle_struct
*handle
,
825 const struct smb_filename
*smb_fname
,
826 struct smb_file_time
*ft
)
830 int ret
, saved_errno
;
831 struct smb_filename
*conv
;
834 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
835 smb_fname
->base_name
,
836 ×tamp
, &stripped
)) {
839 if (timestamp
== 0) {
840 return SMB_VFS_NEXT_NTIMES(handle
, smb_fname
, ft
);
842 status
= copy_smb_filename(talloc_tos(), smb_fname
, &conv
);
843 if (!NT_STATUS_IS_OK(status
)) {
847 conv
->base_name
= shadow_copy2_convert(
848 conv
, handle
, stripped
, timestamp
);
849 TALLOC_FREE(stripped
);
850 if (conv
->base_name
== NULL
) {
853 ret
= SMB_VFS_NEXT_NTIMES(handle
, conv
, ft
);
860 static int shadow_copy2_readlink(vfs_handle_struct
*handle
,
861 const char *fname
, char *buf
, size_t bufsiz
)
865 int ret
, saved_errno
;
868 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
869 ×tamp
, &stripped
)) {
872 if (timestamp
== 0) {
873 return SMB_VFS_NEXT_READLINK(handle
, fname
, buf
, bufsiz
);
875 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
876 TALLOC_FREE(stripped
);
880 ret
= SMB_VFS_NEXT_READLINK(handle
, conv
, buf
, bufsiz
);
887 static int shadow_copy2_mknod(vfs_handle_struct
*handle
,
888 const char *fname
, mode_t mode
, SMB_DEV_T dev
)
892 int ret
, saved_errno
;
895 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
896 ×tamp
, &stripped
)) {
899 if (timestamp
== 0) {
900 return SMB_VFS_NEXT_MKNOD(handle
, fname
, mode
, dev
);
902 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
903 TALLOC_FREE(stripped
);
907 ret
= SMB_VFS_NEXT_MKNOD(handle
, conv
, mode
, dev
);
914 static char *shadow_copy2_realpath(vfs_handle_struct
*handle
,
918 char *stripped
= NULL
;
921 char *inserted
= NULL
;
922 char *inserted_to
, *inserted_end
;
925 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
926 ×tamp
, &stripped
)) {
929 if (timestamp
== 0) {
930 return SMB_VFS_NEXT_REALPATH(handle
, fname
);
933 tmp
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
938 result
= SMB_VFS_NEXT_REALPATH(handle
, tmp
);
939 if (result
== NULL
) {
944 * Take away what we've inserted. This removes the @GMT-thingy
945 * completely, but will give a path under the share root.
947 inserted
= shadow_copy2_insert_string(talloc_tos(), handle
, timestamp
);
948 if (inserted
== NULL
) {
951 inserted_to
= strstr_m(result
, inserted
);
952 if (inserted_to
== NULL
) {
953 DEBUG(2, ("SMB_VFS_NEXT_REALPATH removed %s\n", inserted
));
956 inserted_end
= inserted_to
+ talloc_get_size(inserted
) - 1;
957 memmove(inserted_to
, inserted_end
, strlen(inserted_end
)+1);
961 TALLOC_FREE(inserted
);
963 TALLOC_FREE(stripped
);
969 * Check whether a given directory contains a
970 * snapshot directory as direct subdirectory.
971 * If yes, return the path of the snapshot-subdir,
972 * otherwise return NULL.
974 static char *have_snapdir(struct vfs_handle_struct
*handle
,
977 struct smb_filename smb_fname
;
980 ZERO_STRUCT(smb_fname
);
981 smb_fname
.base_name
= talloc_asprintf(
982 talloc_tos(), "%s/%s", path
,
983 lp_parm_const_string(SNUM(handle
->conn
), "shadow", "snapdir",
985 if (smb_fname
.base_name
== NULL
) {
989 ret
= SMB_VFS_NEXT_STAT(handle
, &smb_fname
);
990 if ((ret
== 0) && (S_ISDIR(smb_fname
.st
.st_ex_mode
))) {
991 return smb_fname
.base_name
;
993 TALLOC_FREE(smb_fname
.base_name
);
997 static char *shadow_copy2_find_snapdir(TALLOC_CTX
*mem_ctx
,
998 struct vfs_handle_struct
*handle
,
999 struct smb_filename
*smb_fname
)
1004 path
= talloc_asprintf(mem_ctx
, "%s/%s",
1005 handle
->conn
->connectpath
,
1006 smb_fname
->base_name
);
1011 snapdir
= have_snapdir(handle
, path
);
1012 if (snapdir
!= NULL
) {
1017 while ((p
= strrchr(path
, '/')) && (p
> path
)) {
1021 snapdir
= have_snapdir(handle
, path
);
1022 if (snapdir
!= NULL
) {
1031 static bool shadow_copy2_snapshot_to_gmt(vfs_handle_struct
*handle
,
1033 char *gmt
, size_t gmt_len
)
1035 struct tm timestamp
;
1037 unsigned long int timestamp_long
;
1040 fmt
= lp_parm_const_string(SNUM(handle
->conn
), "shadow",
1041 "format", GMT_FORMAT
);
1043 ZERO_STRUCT(timestamp
);
1044 if (lp_parm_bool(SNUM(handle
->conn
), "shadow", "sscanf", false)) {
1045 if (sscanf(name
, fmt
, ×tamp_long
) != 1) {
1046 DEBUG(10, ("shadow_copy2_snapshot_to_gmt: "
1047 "no sscanf match %s: %s\n",
1051 timestamp_t
= timestamp_long
;
1052 gmtime_r(×tamp_t
, ×tamp
);
1054 if (strptime(name
, fmt
, ×tamp
) == NULL
) {
1055 DEBUG(10, ("shadow_copy2_snapshot_to_gmt: "
1056 "no match %s: %s\n",
1060 DEBUG(10, ("shadow_copy2_snapshot_to_gmt: match %s: %s\n",
1063 if (lp_parm_bool(SNUM(handle
->conn
), "shadow", "localtime", false)) {
1064 timestamp
.tm_isdst
= -1;
1065 timestamp_t
= mktime(×tamp
);
1066 gmtime_r(×tamp_t
, ×tamp
);
1070 strftime(gmt
, gmt_len
, GMT_FORMAT
, ×tamp
);
1074 static int shadow_copy2_label_cmp_asc(const void *x
, const void *y
)
1076 return strncmp((const char *)x
, (const char *)y
, sizeof(SHADOW_COPY_LABEL
));
1079 static int shadow_copy2_label_cmp_desc(const void *x
, const void *y
)
1081 return -strncmp((const char *)x
, (const char *)y
, sizeof(SHADOW_COPY_LABEL
));
1085 sort the shadow copy data in ascending or descending order
1087 static void shadow_copy2_sort_data(vfs_handle_struct
*handle
,
1088 struct shadow_copy_data
*shadow_copy2_data
)
1090 int (*cmpfunc
)(const void *, const void *);
1093 sort
= lp_parm_const_string(SNUM(handle
->conn
), "shadow",
1099 if (strcmp(sort
, "asc") == 0) {
1100 cmpfunc
= shadow_copy2_label_cmp_asc
;
1101 } else if (strcmp(sort
, "desc") == 0) {
1102 cmpfunc
= shadow_copy2_label_cmp_desc
;
1107 if (shadow_copy2_data
&& shadow_copy2_data
->num_volumes
> 0 &&
1108 shadow_copy2_data
->labels
)
1110 TYPESAFE_QSORT(shadow_copy2_data
->labels
,
1111 shadow_copy2_data
->num_volumes
,
1116 static int shadow_copy2_get_shadow_copy_data(
1117 vfs_handle_struct
*handle
, files_struct
*fsp
,
1118 struct shadow_copy_data
*shadow_copy2_data
,
1122 const char *snapdir
;
1124 TALLOC_CTX
*tmp_ctx
= talloc_stackframe();
1126 snapdir
= shadow_copy2_find_snapdir(tmp_ctx
, handle
, fsp
->fsp_name
);
1127 if (snapdir
== NULL
) {
1128 DEBUG(0,("shadow:snapdir not found for %s in get_shadow_copy_data\n",
1129 handle
->conn
->connectpath
));
1131 talloc_free(tmp_ctx
);
1135 p
= SMB_VFS_NEXT_OPENDIR(handle
, snapdir
, NULL
, 0);
1138 DEBUG(2,("shadow_copy2: SMB_VFS_NEXT_OPENDIR() failed for '%s'"
1139 " - %s\n", snapdir
, strerror(errno
)));
1140 talloc_free(tmp_ctx
);
1145 shadow_copy2_data
->num_volumes
= 0;
1146 shadow_copy2_data
->labels
= NULL
;
1148 while ((d
= SMB_VFS_NEXT_READDIR(handle
, p
, NULL
))) {
1149 char snapshot
[GMT_NAME_LEN
+1];
1150 SHADOW_COPY_LABEL
*tlabels
;
1153 * ignore names not of the right form in the snapshot
1156 if (!shadow_copy2_snapshot_to_gmt(
1158 snapshot
, sizeof(snapshot
))) {
1160 DEBUG(6, ("shadow_copy2_get_shadow_copy_data: "
1161 "ignoring %s\n", d
->d_name
));
1164 DEBUG(6,("shadow_copy2_get_shadow_copy_data: %s -> %s\n",
1165 d
->d_name
, snapshot
));
1168 /* the caller doesn't want the labels */
1169 shadow_copy2_data
->num_volumes
++;
1173 tlabels
= talloc_realloc(shadow_copy2_data
,
1174 shadow_copy2_data
->labels
,
1176 shadow_copy2_data
->num_volumes
+1);
1177 if (tlabels
== NULL
) {
1178 DEBUG(0,("shadow_copy2: out of memory\n"));
1179 SMB_VFS_NEXT_CLOSEDIR(handle
, p
);
1180 talloc_free(tmp_ctx
);
1184 strlcpy(tlabels
[shadow_copy2_data
->num_volumes
], snapshot
,
1187 shadow_copy2_data
->num_volumes
++;
1188 shadow_copy2_data
->labels
= tlabels
;
1191 SMB_VFS_NEXT_CLOSEDIR(handle
,p
);
1193 shadow_copy2_sort_data(handle
, shadow_copy2_data
);
1195 talloc_free(tmp_ctx
);
1199 static NTSTATUS
shadow_copy2_fget_nt_acl(vfs_handle_struct
*handle
,
1200 struct files_struct
*fsp
,
1201 uint32 security_info
,
1202 TALLOC_CTX
*mem_ctx
,
1203 struct security_descriptor
**ppdesc
)
1210 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
,
1211 fsp
->fsp_name
->base_name
,
1212 ×tamp
, &stripped
)) {
1213 return map_nt_error_from_unix(errno
);
1215 if (timestamp
== 0) {
1216 return SMB_VFS_NEXT_FGET_NT_ACL(handle
, fsp
, security_info
,
1220 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1221 TALLOC_FREE(stripped
);
1223 return map_nt_error_from_unix(errno
);
1225 status
= SMB_VFS_NEXT_GET_NT_ACL(handle
, conv
, security_info
,
1231 static NTSTATUS
shadow_copy2_get_nt_acl(vfs_handle_struct
*handle
,
1233 uint32 security_info
,
1234 TALLOC_CTX
*mem_ctx
,
1235 struct security_descriptor
**ppdesc
)
1242 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1243 ×tamp
, &stripped
)) {
1244 return map_nt_error_from_unix(errno
);
1246 if (timestamp
== 0) {
1247 return SMB_VFS_NEXT_GET_NT_ACL(handle
, fname
, security_info
,
1250 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1251 TALLOC_FREE(stripped
);
1253 return map_nt_error_from_unix(errno
);
1255 status
= SMB_VFS_NEXT_GET_NT_ACL(handle
, conv
, security_info
,
1261 static int shadow_copy2_mkdir(vfs_handle_struct
*handle
,
1262 const char *fname
, mode_t mode
)
1266 int ret
, saved_errno
;
1269 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1270 ×tamp
, &stripped
)) {
1273 if (timestamp
== 0) {
1274 return SMB_VFS_NEXT_MKDIR(handle
, fname
, mode
);
1276 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1277 TALLOC_FREE(stripped
);
1281 ret
= SMB_VFS_NEXT_MKDIR(handle
, conv
, mode
);
1282 saved_errno
= errno
;
1284 errno
= saved_errno
;
1288 static int shadow_copy2_rmdir(vfs_handle_struct
*handle
, const char *fname
)
1292 int ret
, saved_errno
;
1295 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1296 ×tamp
, &stripped
)) {
1299 if (timestamp
== 0) {
1300 return SMB_VFS_NEXT_RMDIR(handle
, fname
);
1302 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1303 TALLOC_FREE(stripped
);
1307 ret
= SMB_VFS_NEXT_RMDIR(handle
, conv
);
1308 saved_errno
= errno
;
1310 errno
= saved_errno
;
1314 static int shadow_copy2_chflags(vfs_handle_struct
*handle
, const char *fname
,
1319 int ret
, saved_errno
;
1322 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1323 ×tamp
, &stripped
)) {
1326 if (timestamp
== 0) {
1327 return SMB_VFS_NEXT_CHFLAGS(handle
, fname
, flags
);
1329 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1330 TALLOC_FREE(stripped
);
1334 ret
= SMB_VFS_NEXT_CHFLAGS(handle
, conv
, flags
);
1335 saved_errno
= errno
;
1337 errno
= saved_errno
;
1341 static ssize_t
shadow_copy2_getxattr(vfs_handle_struct
*handle
,
1342 const char *fname
, const char *aname
,
1343 void *value
, size_t size
)
1351 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1352 ×tamp
, &stripped
)) {
1355 if (timestamp
== 0) {
1356 return SMB_VFS_NEXT_GETXATTR(handle
, fname
, aname
, value
,
1359 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1360 TALLOC_FREE(stripped
);
1364 ret
= SMB_VFS_NEXT_GETXATTR(handle
, conv
, aname
, value
, size
);
1365 saved_errno
= errno
;
1367 errno
= saved_errno
;
1371 static ssize_t
shadow_copy2_listxattr(struct vfs_handle_struct
*handle
,
1373 char *list
, size_t size
)
1381 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1382 ×tamp
, &stripped
)) {
1385 if (timestamp
== 0) {
1386 return SMB_VFS_NEXT_LISTXATTR(handle
, fname
, list
, size
);
1388 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1389 TALLOC_FREE(stripped
);
1393 ret
= SMB_VFS_NEXT_LISTXATTR(handle
, conv
, list
, size
);
1394 saved_errno
= errno
;
1396 errno
= saved_errno
;
1400 static int shadow_copy2_removexattr(vfs_handle_struct
*handle
,
1401 const char *fname
, const char *aname
)
1405 int ret
, saved_errno
;
1408 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1409 ×tamp
, &stripped
)) {
1412 if (timestamp
== 0) {
1413 return SMB_VFS_NEXT_REMOVEXATTR(handle
, fname
, aname
);
1415 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1416 TALLOC_FREE(stripped
);
1420 ret
= SMB_VFS_NEXT_REMOVEXATTR(handle
, conv
, aname
);
1421 saved_errno
= errno
;
1423 errno
= saved_errno
;
1427 static int shadow_copy2_setxattr(struct vfs_handle_struct
*handle
,
1429 const char *aname
, const void *value
,
1430 size_t size
, int flags
)
1438 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1439 ×tamp
, &stripped
)) {
1442 if (timestamp
== 0) {
1443 return SMB_VFS_NEXT_SETXATTR(handle
, fname
, aname
, value
, size
,
1446 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1447 TALLOC_FREE(stripped
);
1451 ret
= SMB_VFS_NEXT_SETXATTR(handle
, conv
, aname
, value
, size
, flags
);
1452 saved_errno
= errno
;
1454 errno
= saved_errno
;
1458 static int shadow_copy2_chmod_acl(vfs_handle_struct
*handle
,
1459 const char *fname
, mode_t mode
)
1467 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, fname
,
1468 ×tamp
, &stripped
)) {
1471 if (timestamp
== 0) {
1472 return SMB_VFS_NEXT_CHMOD_ACL(handle
, fname
, mode
);
1474 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1475 TALLOC_FREE(stripped
);
1479 ret
= SMB_VFS_NEXT_CHMOD_ACL(handle
, conv
, mode
);
1480 saved_errno
= errno
;
1482 errno
= saved_errno
;
1486 static int shadow_copy2_get_real_filename(struct vfs_handle_struct
*handle
,
1489 TALLOC_CTX
*mem_ctx
,
1498 if (!shadow_copy2_strip_snapshot(talloc_tos(), handle
, path
,
1499 ×tamp
, &stripped
)) {
1502 if (timestamp
== 0) {
1503 return SMB_VFS_NEXT_GET_REAL_FILENAME(handle
, path
, name
,
1504 mem_ctx
, found_name
);
1506 if (stripped
[0] == '\0') {
1507 *found_name
= talloc_strdup(mem_ctx
, name
);
1508 if (*found_name
== NULL
) {
1514 conv
= shadow_copy2_convert(talloc_tos(), handle
, stripped
, timestamp
);
1515 TALLOC_FREE(stripped
);
1519 ret
= SMB_VFS_NEXT_GET_REAL_FILENAME(handle
, conv
, name
,
1520 mem_ctx
, found_name
);
1521 saved_errno
= errno
;
1523 errno
= saved_errno
;
1528 static struct vfs_fn_pointers vfs_shadow_copy2_fns
= {
1529 .opendir_fn
= shadow_copy2_opendir
,
1530 .rename_fn
= shadow_copy2_rename
,
1531 .link_fn
= shadow_copy2_link
,
1532 .symlink_fn
= shadow_copy2_symlink
,
1533 .stat_fn
= shadow_copy2_stat
,
1534 .lstat_fn
= shadow_copy2_lstat
,
1535 .fstat_fn
= shadow_copy2_fstat
,
1536 .open_fn
= shadow_copy2_open
,
1537 .unlink_fn
= shadow_copy2_unlink
,
1538 .chmod_fn
= shadow_copy2_chmod
,
1539 .chown_fn
= shadow_copy2_chown
,
1540 .chdir_fn
= shadow_copy2_chdir
,
1541 .ntimes_fn
= shadow_copy2_ntimes
,
1542 .readlink_fn
= shadow_copy2_readlink
,
1543 .mknod_fn
= shadow_copy2_mknod
,
1544 .realpath_fn
= shadow_copy2_realpath
,
1545 .get_nt_acl_fn
= shadow_copy2_get_nt_acl
,
1546 .fget_nt_acl_fn
= shadow_copy2_fget_nt_acl
,
1547 .get_shadow_copy_data_fn
= shadow_copy2_get_shadow_copy_data
,
1548 .mkdir_fn
= shadow_copy2_mkdir
,
1549 .rmdir_fn
= shadow_copy2_rmdir
,
1550 .getxattr_fn
= shadow_copy2_getxattr
,
1551 .listxattr_fn
= shadow_copy2_listxattr
,
1552 .removexattr_fn
= shadow_copy2_removexattr
,
1553 .setxattr_fn
= shadow_copy2_setxattr
,
1554 .chmod_acl_fn
= shadow_copy2_chmod_acl
,
1555 .chflags_fn
= shadow_copy2_chflags
,
1556 .get_real_filename_fn
= shadow_copy2_get_real_filename
,
1559 NTSTATUS
vfs_shadow_copy2_init(void);
1560 NTSTATUS
vfs_shadow_copy2_init(void)
1562 return smb_register_vfs(SMB_VFS_INTERFACE_VERSION
,
1563 "shadow_copy2", &vfs_shadow_copy2_fns
);