2011-06-20 09:52:21 +02:00
/*
2013-12-04 15:50:26 +01:00
* shadow_copy2 : a shadow copy module ( second implementation )
2008-01-16 12:21:38 +03:00
*
2011-06-20 09:52:21 +02:00
* Copyright ( C ) Andrew Tridgell 2007 ( portions taken from shadow_copy2 )
* Copyright ( C ) Ed Plese 2009
* Copyright ( C ) Volker Lendecke 2011
* Copyright ( C ) Christian Ambach 2011
2013-12-04 13:40:14 +01:00
* Copyright ( C ) Michael Adam 2013
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
* Copyright ( C ) Rajesh Joseph 2016
2008-01-16 12:21:38 +03:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
2011-06-20 09:52:21 +02:00
*
2008-01-16 12:21:38 +03:00
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2011-06-20 09:52:21 +02:00
*
2008-01-16 12:21:38 +03:00
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/*
2013-12-04 15:55:19 +01:00
* This is a second implemetation of a shadow copy module for exposing
* file system snapshots to windows clients as shadow copies .
*
* See the manual page for documentation .
2008-01-16 12:21:38 +03:00
*/
2011-06-20 09:52:21 +02:00
# include "includes.h"
2015-10-23 14:54:31 -07:00
# include "smbd/smbd.h"
2011-06-20 09:52:21 +02:00
# include "system/filesys.h"
# include "include/ntioctl.h"
# include "util_tdb.h"
2017-01-20 12:00:08 -08:00
# include "lib/util_path.h"
2018-11-23 14:08:15 +01:00
# include "libcli/security/security.h"
# include "lib/util/tevent_unix.h"
2008-01-16 12:21:38 +03:00
2013-05-24 01:35:44 +02:00
struct shadow_copy2_config {
char * gmt_format ;
bool use_sscanf ;
bool use_localtime ;
char * snapdir ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
char * delimiter ;
2013-05-24 01:35:44 +02:00
bool snapdirseverywhere ;
bool crossmountpoints ;
bool fixinodes ;
char * sort_order ;
bool snapdir_absolute ;
2013-05-30 17:26:44 +02:00
char * mount_point ;
2015-11-03 09:15:12 +02:00
char * rel_connectpath ; /* share root, relative to a snapshot root */
2013-10-04 00:07:15 +02:00
char * snapshot_basepath ; /* the absolute version of snapdir */
2013-05-24 01:35:44 +02:00
} ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
/* Data-structure to hold the list of snap entries */
struct shadow_copy2_snapentry {
char * snapname ;
char * time_fmt ;
struct shadow_copy2_snapentry * next ;
struct shadow_copy2_snapentry * prev ;
} ;
struct shadow_copy2_snaplist_info {
struct shadow_copy2_snapentry * snaplist ; /* snapshot list */
regex_t * regex ; /* Regex to filter snaps */
time_t fetch_time ; /* snaplist update time */
} ;
2016-07-11 11:27:37 +00:00
/*
* shadow_copy2 private structure . This structure will be
* used to keep module specific information
*/
struct shadow_copy2_private {
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
struct shadow_copy2_config * config ;
struct shadow_copy2_snaplist_info * snaps ;
2017-01-20 11:50:49 -08:00
char * shadow_cwd ; /* Absolute $cwd path. */
/* Absolute connectpath - can vary depending on $cwd. */
char * shadow_connectpath ;
2017-06-30 11:32:59 -07:00
/* talloc'ed realpath return. */
struct smb_filename * shadow_realpath ;
2016-07-11 11:27:37 +00:00
} ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
static int shadow_copy2_get_shadow_copy_data (
vfs_handle_struct * handle , files_struct * fsp ,
struct shadow_copy_data * shadow_copy2_data ,
bool labels ) ;
/**
* This function will create a new snapshot list entry and
* return to the caller . This entry will also be added to
* the global snapshot list .
*
* @ param [ in ] priv shadow_copy2 specific data structure
* @ return Newly created snapshot entry or NULL on failure
*/
static struct shadow_copy2_snapentry * shadow_copy2_create_snapentry (
struct shadow_copy2_private * priv )
{
struct shadow_copy2_snapentry * tmpentry = NULL ;
tmpentry = talloc_zero ( priv - > snaps , struct shadow_copy2_snapentry ) ;
if ( tmpentry = = NULL ) {
DBG_ERR ( " talloc_zero() failed \n " ) ;
errno = ENOMEM ;
return NULL ;
}
DLIST_ADD ( priv - > snaps - > snaplist , tmpentry ) ;
return tmpentry ;
}
/**
* This function will delete the entire snaplist and reset
* priv - > snaps - > snaplist to NULL .
*
* @ param [ in ] priv shadow_copye specific data structure
*/
static void shadow_copy2_delete_snaplist ( struct shadow_copy2_private * priv )
{
struct shadow_copy2_snapentry * tmp = NULL ;
while ( ( tmp = priv - > snaps - > snaplist ) ! = NULL ) {
DLIST_REMOVE ( priv - > snaps - > snaplist , tmp ) ;
talloc_free ( tmp ) ;
}
}
/**
* Given a timestamp this function searches the global snapshot list
* and returns the complete snapshot directory name saved in the entry .
*
* @ param [ in ] priv shadow_copy2 specific structure
* @ param [ in ] timestamp timestamp corresponding to one of the snapshot
* @ param [ out ] snap_str buffer to copy the actual snapshot name
* @ param [ in ] len length of snap_str buffer
*
* @ return Length of actual snapshot name , and - 1 on failure
*/
static ssize_t shadow_copy2_saved_snapname ( struct shadow_copy2_private * priv ,
struct tm * timestamp ,
char * snap_str , size_t len )
{
ssize_t snaptime_len = - 1 ;
struct shadow_copy2_snapentry * entry = NULL ;
snaptime_len = strftime ( snap_str , len , GMT_FORMAT , timestamp ) ;
if ( snaptime_len = = 0 ) {
DBG_ERR ( " strftime failed \n " ) ;
return - 1 ;
}
snaptime_len = - 1 ;
for ( entry = priv - > snaps - > snaplist ; entry ; entry = entry - > next ) {
if ( strcmp ( entry - > time_fmt , snap_str ) = = 0 ) {
snaptime_len = snprintf ( snap_str , len , " %s " ,
entry - > snapname ) ;
return snaptime_len ;
}
}
snap_str [ 0 ] = 0 ;
return snaptime_len ;
}
/**
* This function will check if snaplist is updated or not . If snaplist
* is empty then it will create a new list . Each time snaplist is updated
* the time is recorded . If the snapshot time is greater than the snaplist
* update time then chances are we are working on an older list . Then discard
* the old list and fetch a new snaplist .
*
* @ param [ in ] handle VFS handle struct
* @ param [ in ] snap_time time of snapshot
*
* @ return true if the list is updated else false
*/
static bool shadow_copy2_update_snaplist ( struct vfs_handle_struct * handle ,
time_t snap_time )
{
int ret = - 1 ;
bool snaplist_updated = false ;
struct files_struct fsp = { 0 } ;
struct smb_filename smb_fname = { 0 } ;
double seconds = 0.0 ;
struct shadow_copy2_private * priv = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
return false ) ;
seconds = difftime ( snap_time , priv - > snaps - > fetch_time ) ;
/*
* Fetch the snapshot list if either the snaplist is empty or the
* required snapshot time is greater than the last fetched snaplist
* time .
*/
if ( seconds > 0 | | ( priv - > snaps - > snaplist = = NULL ) ) {
2016-09-25 03:30:19 +02:00
smb_fname . base_name = discard_const_p ( char , " . " ) ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
fsp . fsp_name = & smb_fname ;
ret = shadow_copy2_get_shadow_copy_data ( handle , & fsp ,
NULL , false ) ;
if ( ret = = 0 ) {
snaplist_updated = true ;
} else {
DBG_ERR ( " Failed to get shadow copy data \n " ) ;
}
}
return snaplist_updated ;
}
2011-06-20 09:52:21 +02:00
static bool shadow_copy2_find_slashes ( TALLOC_CTX * mem_ctx , const char * str ,
size_t * * poffsets ,
unsigned * pnum_offsets )
2008-01-16 12:21:38 +03:00
{
2011-06-20 09:52:21 +02:00
unsigned num_offsets ;
size_t * offsets ;
2009-09-16 03:22:56 +02:00
const char * p ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
num_offsets = 0 ;
2010-01-21 08:30:01 +01:00
2011-06-20 09:52:21 +02:00
p = str ;
while ( ( p = strchr ( p , ' / ' ) ) ! = NULL ) {
num_offsets + = 1 ;
p + = 1 ;
}
2010-01-21 08:30:01 +01:00
2011-06-20 09:52:21 +02:00
offsets = talloc_array ( mem_ctx , size_t , num_offsets ) ;
if ( offsets = = NULL ) {
return false ;
2010-01-21 08:30:01 +01:00
}
2011-06-20 09:52:21 +02:00
p = str ;
num_offsets = 0 ;
while ( ( p = strchr ( p , ' / ' ) ) ! = NULL ) {
offsets [ num_offsets ] = p - str ;
num_offsets + = 1 ;
p + = 1 ;
2009-12-06 21:03:06 -06:00
}
2010-01-21 08:30:01 +01:00
2011-06-20 09:52:21 +02:00
* poffsets = offsets ;
* pnum_offsets = num_offsets ;
return true ;
2010-01-21 08:30:01 +01:00
}
2013-05-24 17:20:42 +02:00
/**
2013-12-12 12:08:58 +01:00
* Given a timestamp , build the posix level GMT - tag string
2013-05-30 23:51:02 +02:00
* based on the configurable format .
2013-05-24 17:20:42 +02:00
*/
2016-07-11 12:28:46 +00:00
static ssize_t shadow_copy2_posix_gmt_string ( struct vfs_handle_struct * handle ,
2013-05-30 23:51:02 +02:00
time_t snapshot ,
char * snaptime_string ,
size_t len )
2009-09-16 03:22:56 +02:00
{
2011-06-20 09:52:21 +02:00
struct tm snap_tm ;
2016-07-11 12:28:46 +00:00
ssize_t snaptime_len ;
2013-05-24 01:35:44 +02:00
struct shadow_copy2_config * config ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2009-09-16 03:22:56 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-30 23:51:02 +02:00
return 0 ) ;
2012-07-02 19:31:58 +10:00
2016-07-11 11:27:37 +00:00
config = priv - > config ;
2013-05-24 01:35:44 +02:00
if ( config - > use_sscanf ) {
snaptime_len = snprintf ( snaptime_string ,
2013-05-30 23:51:02 +02:00
len ,
2013-05-24 01:35:44 +02:00
config - > gmt_format ,
( unsigned long ) snapshot ) ;
2012-07-02 22:31:49 +10:00
if ( snaptime_len < = 0 ) {
2012-07-02 19:31:58 +10:00
DEBUG ( 10 , ( " snprintf failed \n " ) ) ;
2016-07-11 12:28:46 +00:00
return - 1 ;
2012-07-02 19:31:58 +10:00
}
} else {
2013-05-24 01:35:44 +02:00
if ( config - > use_localtime ) {
2012-07-02 22:31:49 +10:00
if ( localtime_r ( & snapshot , & snap_tm ) = = 0 ) {
DEBUG ( 10 , ( " gmtime_r failed \n " ) ) ;
2013-05-30 23:51:02 +02:00
return - 1 ;
2012-07-02 22:31:49 +10:00
}
} else {
if ( gmtime_r ( & snapshot , & snap_tm ) = = 0 ) {
DEBUG ( 10 , ( " gmtime_r failed \n " ) ) ;
2013-05-30 23:51:02 +02:00
return - 1 ;
2012-07-02 22:31:49 +10:00
}
}
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
if ( priv - > snaps - > regex ! = NULL ) {
snaptime_len = shadow_copy2_saved_snapname ( priv ,
& snap_tm , snaptime_string , len ) ;
if ( snaptime_len > = 0 )
return snaptime_len ;
/*
* If we fail to find the snapshot name , chances are
* that we have not updated our snaplist . Make sure the
* snaplist is updated .
*/
if ( ! shadow_copy2_update_snaplist ( handle , snapshot ) ) {
DBG_DEBUG ( " shadow_copy2_update_snaplist "
" failed \n " ) ;
return - 1 ;
}
return shadow_copy2_saved_snapname ( priv ,
& snap_tm , snaptime_string , len ) ;
}
2013-05-24 01:35:44 +02:00
snaptime_len = strftime ( snaptime_string ,
2013-05-30 23:51:02 +02:00
len ,
2013-05-24 01:35:44 +02:00
config - > gmt_format ,
& snap_tm ) ;
2012-07-02 22:31:49 +10:00
if ( snaptime_len = = 0 ) {
2012-07-02 19:31:58 +10:00
DEBUG ( 10 , ( " strftime failed \n " ) ) ;
2016-07-11 12:28:46 +00:00
return - 1 ;
2012-07-02 19:31:58 +10:00
}
2009-09-16 03:22:56 +02:00
}
2013-05-29 17:12:21 +02:00
2013-05-30 23:51:02 +02:00
return snaptime_len ;
}
/**
2013-12-12 12:08:58 +01:00
* Given a timestamp , build the string to insert into a path
2013-05-30 23:51:02 +02:00
* as a path component for creating the local path to the
* snapshot at the given timestamp of the input path .
*
* In the case of a parallel snapdir ( specified with an
2018-11-28 11:10:17 +13:00
* absolute path ) , this is the initial portion of the
2013-05-30 23:51:02 +02:00
* local path of any snapshot file . The complete path is
* obtained by appending the portion of the file ' s path
* below the share root ' s mountpoint .
*/
static char * shadow_copy2_insert_string ( TALLOC_CTX * mem_ctx ,
struct vfs_handle_struct * handle ,
time_t snapshot )
{
fstring snaptime_string ;
2016-07-11 12:28:46 +00:00
ssize_t snaptime_len = 0 ;
2013-05-30 23:51:02 +02:00
char * result = NULL ;
struct shadow_copy2_config * config ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2013-05-30 23:51:02 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-30 23:51:02 +02:00
return NULL ) ;
2016-07-11 11:27:37 +00:00
config = priv - > config ;
2013-05-30 23:51:02 +02:00
snaptime_len = shadow_copy2_posix_gmt_string ( handle ,
snapshot ,
snaptime_string ,
sizeof ( snaptime_string ) ) ;
if ( snaptime_len < = 0 ) {
return NULL ;
}
2013-05-29 17:12:21 +02:00
if ( config - > snapdir_absolute ) {
result = talloc_asprintf ( mem_ctx , " %s/%s " ,
config - > snapdir , snaptime_string ) ;
} else {
result = talloc_asprintf ( mem_ctx , " /%s/%s " ,
config - > snapdir , snaptime_string ) ;
}
if ( result = = NULL ) {
DEBUG ( 1 , ( __location__ " talloc_asprintf failed \n " ) ) ;
}
return result ;
2009-09-16 03:22:56 +02:00
}
2013-05-31 00:18:52 +02:00
/**
* Build the posix snapshot path for the connection
* at the given timestamp , i . e . the absolute posix path
* that contains the snapshot for this file system .
*
* This only applies to classical case , i . e . not
* to the " snapdirseverywhere " mode .
*/
static char * shadow_copy2_snapshot_path ( TALLOC_CTX * mem_ctx ,
struct vfs_handle_struct * handle ,
time_t snapshot )
{
fstring snaptime_string ;
2016-07-11 12:28:46 +00:00
ssize_t snaptime_len = 0 ;
2013-05-31 00:18:52 +02:00
char * result = NULL ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2013-05-31 00:18:52 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-31 00:18:52 +02:00
return NULL ) ;
snaptime_len = shadow_copy2_posix_gmt_string ( handle ,
snapshot ,
snaptime_string ,
sizeof ( snaptime_string ) ) ;
if ( snaptime_len < = 0 ) {
return NULL ;
}
result = talloc_asprintf ( mem_ctx , " %s/%s " ,
2016-07-11 11:27:37 +00:00
priv - > config - > snapshot_basepath , snaptime_string ) ;
2013-05-31 00:18:52 +02:00
if ( result = = NULL ) {
DEBUG ( 1 , ( __location__ " talloc_asprintf failed \n " ) ) ;
}
return result ;
}
2017-01-20 12:00:08 -08:00
static char * make_path_absolute ( TALLOC_CTX * mem_ctx ,
struct shadow_copy2_private * priv ,
const char * name )
{
char * newpath = NULL ;
char * abs_path = NULL ;
if ( name [ 0 ] ! = ' / ' ) {
newpath = talloc_asprintf ( mem_ctx ,
" %s/%s " ,
priv - > shadow_cwd ,
name ) ;
if ( newpath = = NULL ) {
return NULL ;
}
name = newpath ;
}
abs_path = canonicalize_absolute_path ( mem_ctx , name ) ;
TALLOC_FREE ( newpath ) ;
return abs_path ;
}
/* Return a $cwd-relative path. */
static bool make_relative_path ( const char * cwd , char * abs_path )
{
size_t cwd_len = strlen ( cwd ) ;
size_t abs_len = strlen ( abs_path ) ;
if ( abs_len < cwd_len ) {
return false ;
}
if ( memcmp ( abs_path , cwd , cwd_len ) ! = 0 ) {
return false ;
}
2017-04-11 12:03:20 +02:00
/* The cwd_len != 1 case is for $cwd == '/' */
if ( cwd_len ! = 1 & &
abs_path [ cwd_len ] ! = ' / ' & &
abs_path [ cwd_len ] ! = ' \0 ' )
{
2017-01-20 12:00:08 -08:00
return false ;
}
if ( abs_path [ cwd_len ] = = ' / ' ) {
cwd_len + + ;
}
memmove ( abs_path , & abs_path [ cwd_len ] , abs_len + 1 - cwd_len ) ;
return true ;
}
2017-01-26 10:35:50 -08:00
static bool shadow_copy2_snapshot_to_gmt ( vfs_handle_struct * handle ,
const char * name ,
char * gmt , size_t gmt_len ) ;
/*
* Check if an incoming filename is already a snapshot converted pathname .
*
* If so , it returns the pathname truncated at the snapshot point which
* will be used as the connectpath .
*/
static int check_for_converted_path ( TALLOC_CTX * mem_ctx ,
struct vfs_handle_struct * handle ,
struct shadow_copy2_private * priv ,
char * abs_path ,
bool * ppath_already_converted ,
char * * pconnectpath )
{
size_t snapdirlen = 0 ;
char * p = strstr_m ( abs_path , priv - > config - > snapdir ) ;
char * q = NULL ;
char * connect_path = NULL ;
char snapshot [ GMT_NAME_LEN + 1 ] ;
* ppath_already_converted = false ;
if ( p = = NULL ) {
/* Must at least contain shadow:snapdir. */
return 0 ;
}
if ( priv - > config - > snapdir [ 0 ] = = ' / ' & &
p ! = abs_path ) {
/* Absolute shadow:snapdir must be at the start. */
return 0 ;
}
snapdirlen = strlen ( priv - > config - > snapdir ) ;
if ( p [ snapdirlen ] ! = ' / ' ) {
/* shadow:snapdir must end as a separate component. */
return 0 ;
}
if ( p > abs_path & & p [ - 1 ] ! = ' / ' ) {
/* shadow:snapdir must start as a separate component. */
return 0 ;
}
p + = snapdirlen ;
p + + ; /* Move past the / */
/*
* Need to return up to the next path
* component after the time .
* This will be used as the connectpath .
*/
q = strchr ( p , ' / ' ) ;
if ( q = = NULL ) {
/*
* No next path component .
* Use entire string .
*/
connect_path = talloc_strdup ( mem_ctx ,
abs_path ) ;
} else {
connect_path = talloc_strndup ( mem_ctx ,
abs_path ,
q - abs_path ) ;
}
if ( connect_path = = NULL ) {
return ENOMEM ;
}
/*
* Point p at the same offset in connect_path as
* it is in abs_path .
*/
p = & connect_path [ p - abs_path ] ;
/*
* Now ensure there is a time string at p .
* The SMB - format @ GMT - token string is returned
* in snapshot .
*/
if ( ! shadow_copy2_snapshot_to_gmt ( handle ,
p ,
snapshot ,
sizeof ( snapshot ) ) ) {
TALLOC_FREE ( connect_path ) ;
return 0 ;
}
if ( pconnectpath ! = NULL ) {
* pconnectpath = connect_path ;
}
* ppath_already_converted = true ;
DBG_DEBUG ( " path |%s| is already converted. "
" connect path = |%s| \n " ,
abs_path ,
connect_path ) ;
return 0 ;
}
2013-05-23 23:32:15 +02:00
/**
2017-01-26 10:49:51 -08:00
* This function does two things .
*
* 1 ) . Checks if an incoming filename is already a
* snapshot converted pathname .
* If so , it returns the pathname truncated
* at the snapshot point which will be used
* as the connectpath , and then does an early return .
*
* 2 ) . Checks if an incoming filename contains an
* SMB - layer @ GMT - style timestamp .
* If so , it strips the timestamp , and returns
* both the timestamp and the stripped path
* ( making it cwd - relative ) .
2013-05-23 23:32:15 +02:00
*/
2017-01-26 10:49:51 -08:00
2020-05-05 10:09:01 +02:00
static bool _shadow_copy2_strip_snapshot_internal ( TALLOC_CTX * mem_ctx ,
2011-06-20 09:52:21 +02:00
struct vfs_handle_struct * handle ,
2020-04-30 17:28:16 +02:00
const struct smb_filename * smb_fname ,
2011-06-20 09:52:21 +02:00
time_t * ptimestamp ,
2017-01-20 11:54:56 -08:00
char * * pstripped ,
2018-11-22 11:02:24 +01:00
char * * psnappath ,
2020-05-05 10:09:01 +02:00
bool * _already_converted ,
const char * function )
2011-06-20 09:52:21 +02:00
{
2017-01-20 11:42:39 -08:00
char * stripped = NULL ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2017-01-26 10:49:51 -08:00
char * abs_path = NULL ;
bool ret = true ;
bool already_converted = false ;
int err = 0 ;
2013-05-24 01:35:44 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return false ) ;
2009-06-16 12:01:13 -07:00
2020-05-05 10:09:01 +02:00
DBG_DEBUG ( " [from %s()] Path '%s' \n " ,
function , smb_fname_str_dbg ( smb_fname ) ) ;
2013-05-29 23:57:30 +02:00
2018-11-22 11:02:24 +01:00
if ( _already_converted ! = NULL ) {
* _already_converted = false ;
}
2020-04-30 15:24:44 +02:00
abs_path = make_path_absolute ( mem_ctx , priv , smb_fname - > base_name ) ;
2017-01-26 10:49:51 -08:00
if ( abs_path = = NULL ) {
ret = false ;
goto out ;
}
2020-04-30 15:24:44 +02:00
DBG_DEBUG ( " abs path '%s' \n " , abs_path ) ;
2017-01-26 10:49:51 -08:00
err = check_for_converted_path ( mem_ctx ,
handle ,
priv ,
abs_path ,
& already_converted ,
psnappath ) ;
if ( err ! = 0 ) {
/* error in conversion. */
ret = false ;
goto out ;
}
if ( already_converted ) {
2018-11-22 11:02:24 +01:00
if ( _already_converted ! = NULL ) {
* _already_converted = true ;
}
2017-01-26 10:49:51 -08:00
goto out ;
}
2020-05-02 12:10:38 +02:00
if ( smb_fname - > twrp = = 0 ) {
goto out ;
}
if ( ptimestamp ! = NULL ) {
* ptimestamp = nt_time_to_unix ( smb_fname - > twrp ) ;
}
2011-06-20 09:52:21 +02:00
if ( pstripped ! = NULL ) {
2020-04-30 15:24:44 +02:00
stripped = talloc_strdup ( mem_ctx , abs_path ) ;
2011-06-20 09:52:21 +02:00
if ( stripped = = NULL ) {
2017-01-26 10:49:51 -08:00
ret = false ;
goto out ;
2011-06-20 09:52:21 +02:00
}
2020-04-30 15:24:44 +02:00
if ( smb_fname - > base_name [ 0 ] ! = ' / ' ) {
ret = make_relative_path ( priv - > shadow_cwd , stripped ) ;
if ( ! ret ) {
DBG_DEBUG ( " Path '%s' "
2017-04-11 11:18:30 +02:00
" doesn't start with cwd '%s' \n " ,
2020-04-30 15:24:44 +02:00
stripped , priv - > shadow_cwd ) ;
2017-01-26 10:49:51 -08:00
ret = false ;
errno = ENOENT ;
goto out ;
}
}
2011-06-20 09:52:21 +02:00
* pstripped = stripped ;
}
2020-05-01 16:36:09 +02:00
2017-01-26 10:49:51 -08:00
ret = true ;
out :
TALLOC_FREE ( abs_path ) ;
return ret ;
2011-06-20 09:52:21 +02:00
}
2008-01-16 12:21:38 +03:00
2020-05-05 10:09:01 +02:00
# define shadow_copy2_strip_snapshot_internal(mem_ctx, handle, orig_name, \
ptimestamp , pstripped , psnappath , _already_converted ) \
_shadow_copy2_strip_snapshot_internal ( ( mem_ctx ) , ( handle ) , ( orig_name ) , \
( ptimestamp ) , ( pstripped ) , ( psnappath ) , ( _already_converted ) , \
__FUNCTION__ )
static bool _shadow_copy2_strip_snapshot ( TALLOC_CTX * mem_ctx ,
struct vfs_handle_struct * handle ,
const struct smb_filename * orig_name ,
time_t * ptimestamp ,
char * * pstripped ,
const char * function )
2017-01-20 11:54:56 -08:00
{
2020-05-05 10:09:01 +02:00
return _shadow_copy2_strip_snapshot_internal ( mem_ctx ,
2017-01-20 11:54:56 -08:00
handle ,
orig_name ,
ptimestamp ,
pstripped ,
2018-11-22 11:02:24 +01:00
NULL ,
2020-05-05 10:09:01 +02:00
NULL ,
function ) ;
2017-01-20 11:54:56 -08:00
}
2020-05-05 10:09:01 +02:00
# define shadow_copy2_strip_snapshot(mem_ctx, handle, orig_name, \
ptimestamp , pstripped ) \
_shadow_copy2_strip_snapshot ( ( mem_ctx ) , ( handle ) , ( orig_name ) , \
( ptimestamp ) , ( pstripped ) , __FUNCTION__ )
static bool _shadow_copy2_strip_snapshot_converted ( TALLOC_CTX * mem_ctx ,
2018-11-22 11:04:54 +01:00
struct vfs_handle_struct * handle ,
2020-04-30 17:28:16 +02:00
const struct smb_filename * orig_name ,
2018-11-22 11:04:54 +01:00
time_t * ptimestamp ,
char * * pstripped ,
2020-05-05 10:09:01 +02:00
bool * is_converted ,
const char * function )
2018-11-22 11:04:54 +01:00
{
2020-05-05 10:09:01 +02:00
return _shadow_copy2_strip_snapshot_internal ( mem_ctx ,
2018-11-22 11:04:54 +01:00
handle ,
orig_name ,
ptimestamp ,
pstripped ,
NULL ,
2020-05-05 10:09:01 +02:00
is_converted ,
function ) ;
2018-11-22 11:04:54 +01:00
}
2020-05-05 10:09:01 +02:00
# define shadow_copy2_strip_snapshot_converted(mem_ctx, handle, orig_name, \
ptimestamp , pstripped , is_converted ) \
_shadow_copy2_strip_snapshot_converted ( ( mem_ctx ) , ( handle ) , ( orig_name ) , \
( ptimestamp ) , ( pstripped ) , ( is_converted ) , __FUNCTION__ )
2011-06-20 09:52:21 +02:00
static char * shadow_copy2_find_mount_point ( TALLOC_CTX * mem_ctx ,
vfs_handle_struct * handle )
2008-01-16 12:21:38 +03:00
{
char * path = talloc_strdup ( mem_ctx , handle - > conn - > connectpath ) ;
dev_t dev ;
struct stat st ;
char * p ;
if ( stat ( path , & st ) ! = 0 ) {
talloc_free ( path ) ;
return NULL ;
}
dev = st . st_dev ;
while ( ( p = strrchr ( path , ' / ' ) ) & & p > path ) {
* p = 0 ;
if ( stat ( path , & st ) ! = 0 ) {
talloc_free ( path ) ;
return NULL ;
}
if ( st . st_dev ! = dev ) {
* p = ' / ' ;
break ;
}
}
2011-06-20 09:52:21 +02:00
return path ;
2008-01-16 12:21:38 +03:00
}
2013-05-29 01:13:57 +02:00
/**
* Convert from a name as handed in via the SMB layer
* and a timestamp into the local path of the snapshot
* of the provided file at the provided time .
2015-11-01 22:28:46 +02:00
* Also return the path in the snapshot corresponding
* to the file ' s share root .
2013-05-29 01:13:57 +02:00
*/
2015-11-01 22:28:46 +02:00
static char * shadow_copy2_do_convert ( TALLOC_CTX * mem_ctx ,
struct vfs_handle_struct * handle ,
const char * name , time_t timestamp ,
size_t * snaproot_len )
2008-01-16 12:21:38 +03:00
{
2011-06-20 09:52:21 +02:00
struct smb_filename converted_fname ;
char * result = NULL ;
size_t * slashes = NULL ;
unsigned num_slashes ;
char * path = NULL ;
size_t pathlen ;
char * insert = NULL ;
char * converted = NULL ;
2015-11-01 22:28:46 +02:00
size_t insertlen , connectlen = 0 ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
int i ;
2011-06-20 09:52:21 +02:00
size_t min_offset ;
2013-05-24 01:35:44 +02:00
struct shadow_copy2_config * config ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2015-11-01 22:28:46 +02:00
size_t in_share_offset = 0 ;
2013-05-24 01:35:44 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return NULL ) ;
2011-06-20 09:52:21 +02:00
2016-07-11 11:27:37 +00:00
config = priv - > config ;
2013-05-23 16:23:03 +02:00
DEBUG ( 10 , ( " converting '%s' \n " , name ) ) ;
if ( ! config - > snapdirseverywhere ) {
int ret ;
char * snapshot_path ;
snapshot_path = shadow_copy2_snapshot_path ( talloc_tos ( ) ,
handle ,
timestamp ) ;
if ( snapshot_path = = NULL ) {
goto fail ;
}
if ( config - > rel_connectpath = = NULL ) {
converted = talloc_asprintf ( mem_ctx , " %s/%s " ,
snapshot_path , name ) ;
} else {
converted = talloc_asprintf ( mem_ctx , " %s/%s/%s " ,
snapshot_path ,
config - > rel_connectpath ,
name ) ;
}
if ( converted = = NULL ) {
goto fail ;
}
ZERO_STRUCT ( converted_fname ) ;
converted_fname . base_name = converted ;
ret = SMB_VFS_NEXT_LSTAT ( handle , & converted_fname ) ;
DEBUG ( 10 , ( " Trying[not snapdirseverywhere] %s: %d (%s) \n " ,
converted ,
ret , ret = = 0 ? " ok " : strerror ( errno ) ) ) ;
if ( ret = = 0 ) {
DEBUG ( 10 , ( " Found %s \n " , converted ) ) ;
result = converted ;
converted = NULL ;
2015-11-01 22:28:46 +02:00
if ( snaproot_len ! = NULL ) {
* snaproot_len = strlen ( snapshot_path ) ;
if ( config - > rel_connectpath ! = NULL ) {
* snaproot_len + =
strlen ( config - > rel_connectpath ) + 1 ;
}
}
2013-05-23 16:23:03 +02:00
goto fail ;
} else {
errno = ENOENT ;
goto fail ;
}
/* never reached ... */
}
2015-11-01 22:28:46 +02:00
connectlen = strlen ( handle - > conn - > connectpath ) ;
2012-10-17 12:08:26 +02:00
if ( name [ 0 ] = = 0 ) {
path = talloc_strdup ( mem_ctx , handle - > conn - > connectpath ) ;
} else {
path = talloc_asprintf (
mem_ctx , " %s/%s " , handle - > conn - > connectpath , name ) ;
}
2011-06-20 09:52:21 +02:00
if ( path = = NULL ) {
errno = ENOMEM ;
goto fail ;
2008-01-16 12:21:38 +03:00
}
2011-06-20 09:52:21 +02:00
pathlen = talloc_get_size ( path ) - 1 ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
if ( ! shadow_copy2_find_slashes ( talloc_tos ( ) , path ,
& slashes , & num_slashes ) ) {
goto fail ;
2008-01-16 12:21:38 +03:00
}
2013-05-29 15:06:22 +02:00
2011-06-20 09:52:21 +02:00
insert = shadow_copy2_insert_string ( talloc_tos ( ) , handle , timestamp ) ;
if ( insert = = NULL ) {
goto fail ;
2008-01-16 12:21:38 +03:00
}
2011-06-20 09:52:21 +02:00
insertlen = talloc_get_size ( insert ) - 1 ;
2013-05-29 15:06:22 +02:00
2013-12-11 09:41:38 +01:00
/*
* Note : We deliberatly don ' t expensively initialize the
* array with talloc_zero here : Putting zero into
* converted [ pathlen + insertlen ] below is sufficient , because
* in the following for loop , the insert string is inserted
* at various slash places . So the memory up to position
* pathlen + insertlen will always be initialized when the
* converted string is used .
*/
2013-12-11 09:34:47 +01:00
converted = talloc_array ( mem_ctx , char , pathlen + insertlen + 1 ) ;
2011-06-20 09:52:21 +02:00
if ( converted = = NULL ) {
goto fail ;
2008-01-16 12:21:38 +03:00
}
2011-06-20 09:52:21 +02:00
if ( path [ pathlen - 1 ] ! = ' / ' ) {
/*
* Append a fake slash to find the snapshot root
*/
size_t * tmp ;
tmp = talloc_realloc ( talloc_tos ( ) , slashes ,
size_t , num_slashes + 1 ) ;
if ( tmp = = NULL ) {
goto fail ;
}
slashes = tmp ;
slashes [ num_slashes ] = pathlen ;
num_slashes + = 1 ;
2010-04-14 10:05:56 +02:00
}
2011-06-20 09:52:21 +02:00
min_offset = 0 ;
2013-05-24 01:35:44 +02:00
if ( ! config - > crossmountpoints ) {
2013-05-31 00:46:01 +02:00
min_offset = strlen ( config - > mount_point ) ;
2009-09-16 03:22:56 +02:00
}
2011-06-20 09:52:21 +02:00
memcpy ( converted , path , pathlen + 1 ) ;
converted [ pathlen + insertlen ] = ' \0 ' ;
2010-01-21 08:30:01 +01:00
2011-06-20 09:52:21 +02:00
ZERO_STRUCT ( converted_fname ) ;
converted_fname . base_name = converted ;
2010-01-21 08:30:01 +01:00
2011-06-20 09:52:21 +02:00
for ( i = num_slashes - 1 ; i > = 0 ; i - - ) {
int ret ;
size_t offset ;
2009-12-06 21:03:06 -06:00
2011-06-20 09:52:21 +02:00
offset = slashes [ i ] ;
2010-01-21 08:30:01 +01:00
2011-06-20 09:52:21 +02:00
if ( offset < min_offset ) {
errno = ENOENT ;
goto fail ;
}
2008-01-16 12:21:38 +03:00
2015-11-01 22:28:46 +02:00
if ( offset > = connectlen ) {
in_share_offset = offset ;
}
2011-06-20 09:52:21 +02:00
memcpy ( converted + offset , insert , insertlen ) ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
offset + = insertlen ;
memcpy ( converted + offset , path + slashes [ i ] ,
pathlen - slashes [ i ] ) ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
ret = SMB_VFS_NEXT_LSTAT ( handle , & converted_fname ) ;
2008-01-16 12:21:38 +03:00
2013-05-31 00:45:16 +02:00
DEBUG ( 10 , ( " Trying[snapdirseverywhere] %s: %d (%s) \n " ,
converted ,
2011-06-20 09:52:21 +02:00
ret , ret = = 0 ? " ok " : strerror ( errno ) ) ) ;
if ( ret = = 0 ) {
/* success */
2015-11-01 22:28:46 +02:00
if ( snaproot_len ! = NULL ) {
* snaproot_len = in_share_offset + insertlen ;
}
2011-06-20 09:52:21 +02:00
break ;
}
if ( errno = = ENOTDIR ) {
/*
* This is a valid condition : We appended the
2019-09-19 09:19:40 +02:00
* . snapshots / @ GMT . . to a file name . Just try
2011-06-20 09:52:21 +02:00
* with the upper levels .
*/
continue ;
}
if ( errno ! = ENOENT ) {
/* Other problem than "not found" */
goto fail ;
}
}
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
if ( i > = 0 ) {
/*
* Found something
*/
DEBUG ( 10 , ( " Found %s \n " , converted ) ) ;
result = converted ;
converted = NULL ;
} else {
errno = ENOENT ;
}
fail :
2017-01-23 10:20:13 -08:00
if ( result = = NULL ) {
saved_errno = errno ;
}
2011-06-20 09:52:21 +02:00
TALLOC_FREE ( converted ) ;
TALLOC_FREE ( insert ) ;
TALLOC_FREE ( slashes ) ;
TALLOC_FREE ( path ) ;
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2011-06-20 09:52:21 +02:00
return result ;
2008-01-16 12:21:38 +03:00
}
2015-11-01 22:28:46 +02:00
/**
* Convert from a name as handed in via the SMB layer
* and a timestamp into the local path of the snapshot
* of the provided file at the provided time .
*/
static char * shadow_copy2_convert ( TALLOC_CTX * mem_ctx ,
struct vfs_handle_struct * handle ,
const char * name , time_t timestamp )
{
return shadow_copy2_do_convert ( mem_ctx , handle , name , timestamp , NULL ) ;
}
2008-01-16 12:21:38 +03:00
/*
modify a sbuf return to ensure that inodes in the shadow directory
are different from those in the main directory
*/
2011-06-20 09:52:21 +02:00
static void convert_sbuf ( vfs_handle_struct * handle , const char * fname ,
SMB_STRUCT_STAT * sbuf )
2008-01-16 12:21:38 +03:00
{
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2013-05-24 01:35:44 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return ) ;
2016-07-11 11:27:37 +00:00
if ( priv - > config - > fixinodes ) {
2020-08-26 08:55:30 +02:00
/* some snapshot systems, like GPFS, return the same
2008-01-16 12:21:38 +03:00
device : inode for the snapshot files as the current
files . That breaks the ' restore ' button in the shadow copy
GUI , as the client gets a sharing violation .
This is a crude way of allowing both files to be
open at once . It has a slight chance of inode
number collision , but I can ' t see a better approach
without significant VFS changes
*/
2015-03-12 15:00:58 +00:00
TDB_DATA key = { . dptr = discard_const_p ( uint8_t , fname ) ,
. dsize = strlen ( fname ) } ;
2011-06-20 09:52:21 +02:00
uint32_t shash ;
2015-03-12 15:00:58 +00:00
shash = tdb_jenkins_hash ( & key ) & 0xFF000000 ;
2008-01-16 12:21:38 +03:00
if ( shash = = 0 ) {
shash = 1 ;
}
2009-05-14 15:34:42 +02:00
sbuf - > st_ex_ino ^ = shash ;
2008-01-16 12:21:38 +03:00
}
}
2019-08-09 15:13:24 -07:00
static int shadow_copy2_renameat ( vfs_handle_struct * handle ,
files_struct * srcfsp ,
const struct smb_filename * smb_fname_src ,
files_struct * dstfsp ,
const struct smb_filename * smb_fname_dst )
{
time_t timestamp_src = 0 ;
time_t timestamp_dst = 0 ;
char * snappath_src = NULL ;
char * snappath_dst = NULL ;
if ( ! shadow_copy2_strip_snapshot_internal ( talloc_tos ( ) , handle ,
2020-04-30 17:28:16 +02:00
smb_fname_src ,
2019-08-09 15:13:24 -07:00
& timestamp_src , NULL , & snappath_src ,
NULL ) ) {
return - 1 ;
}
if ( ! shadow_copy2_strip_snapshot_internal ( talloc_tos ( ) , handle ,
2020-04-30 17:28:16 +02:00
smb_fname_dst ,
2019-08-09 15:13:24 -07:00
& timestamp_dst , NULL , & snappath_dst ,
NULL ) ) {
return - 1 ;
}
if ( timestamp_src ! = 0 ) {
errno = EXDEV ;
return - 1 ;
}
if ( timestamp_dst ! = 0 ) {
errno = EROFS ;
return - 1 ;
}
/*
* Don ' t allow rename on already converted paths .
*/
if ( snappath_src ! = NULL ) {
errno = EXDEV ;
return - 1 ;
}
if ( snappath_dst ! = NULL ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_RENAMEAT ( handle ,
srcfsp ,
smb_fname_src ,
dstfsp ,
smb_fname_dst ) ;
}
2019-08-30 13:53:23 -07:00
static int shadow_copy2_symlinkat ( vfs_handle_struct * handle ,
2020-04-30 19:30:50 +02:00
const struct smb_filename * link_contents ,
2019-08-30 13:53:23 -07:00
struct files_struct * dirfsp ,
const struct smb_filename * new_smb_fname )
{
time_t timestamp_old = 0 ;
time_t timestamp_new = 0 ;
char * snappath_old = NULL ;
char * snappath_new = NULL ;
if ( ! shadow_copy2_strip_snapshot_internal ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
link_contents ,
2019-08-30 13:53:23 -07:00
& timestamp_old ,
NULL ,
& snappath_old ,
NULL ) ) {
return - 1 ;
}
if ( ! shadow_copy2_strip_snapshot_internal ( talloc_tos ( ) ,
handle ,
2021-02-03 13:52:33 -08:00
new_smb_fname ,
2019-08-30 13:53:23 -07:00
& timestamp_new ,
NULL ,
& snappath_new ,
NULL ) ) {
return - 1 ;
}
if ( ( timestamp_old ! = 0 ) | | ( timestamp_new ! = 0 ) ) {
errno = EROFS ;
return - 1 ;
}
/*
* Don ' t allow symlinks on already converted paths .
*/
if ( ( snappath_old ! = NULL ) | | ( snappath_new ! = NULL ) ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_SYMLINKAT ( handle ,
link_contents ,
dirfsp ,
new_smb_fname ) ;
}
2019-08-16 15:59:27 -07:00
static int shadow_copy2_linkat ( vfs_handle_struct * handle ,
files_struct * srcfsp ,
const struct smb_filename * old_smb_fname ,
files_struct * dstfsp ,
const struct smb_filename * new_smb_fname ,
int flags )
{
time_t timestamp_old = 0 ;
time_t timestamp_new = 0 ;
char * snappath_old = NULL ;
char * snappath_new = NULL ;
if ( ! shadow_copy2_strip_snapshot_internal ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
old_smb_fname ,
2019-08-16 15:59:27 -07:00
& timestamp_old ,
NULL ,
& snappath_old ,
NULL ) ) {
return - 1 ;
}
if ( ! shadow_copy2_strip_snapshot_internal ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
new_smb_fname ,
2019-08-16 15:59:27 -07:00
& timestamp_new ,
NULL ,
& snappath_new ,
NULL ) ) {
return - 1 ;
}
if ( ( timestamp_old ! = 0 ) | | ( timestamp_new ! = 0 ) ) {
errno = EROFS ;
return - 1 ;
}
/*
* Don ' t allow links on already converted paths .
*/
if ( ( snappath_old ! = NULL ) | | ( snappath_new ! = NULL ) ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_LINKAT ( handle ,
srcfsp ,
old_smb_fname ,
dstfsp ,
new_smb_fname ,
flags ) ;
}
2008-01-16 12:21:38 +03:00
static int shadow_copy2_stat ( vfs_handle_struct * handle ,
2009-06-22 15:26:56 -07:00
struct smb_filename * smb_fname )
2008-01-16 12:21:38 +03:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
char * stripped = NULL ;
char * tmp ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
int ret ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2011-06-20 09:52:21 +02:00
& timestamp , & stripped ) ) {
return - 1 ;
}
if ( timestamp = = 0 ) {
return SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
2008-01-16 12:21:38 +03:00
}
2011-06-20 09:52:21 +02:00
tmp = smb_fname - > base_name ;
smb_fname - > base_name = shadow_copy2_convert (
talloc_tos ( ) , handle , stripped , timestamp ) ;
TALLOC_FREE ( stripped ) ;
2009-07-02 09:27:44 -07:00
2011-06-20 09:52:21 +02:00
if ( smb_fname - > base_name = = NULL ) {
smb_fname - > base_name = tmp ;
2009-07-02 09:27:44 -07:00
return - 1 ;
}
2011-06-20 09:52:21 +02:00
ret = SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
2017-01-23 10:20:13 -08:00
if ( ret = = - 1 ) {
saved_errno = errno ;
}
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
TALLOC_FREE ( smb_fname - > base_name ) ;
smb_fname - > base_name = tmp ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
if ( ret = = 0 ) {
convert_sbuf ( handle , smb_fname - > base_name , & smb_fname - > st ) ;
}
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2011-06-20 09:52:21 +02:00
return ret ;
2008-01-16 12:21:38 +03:00
}
2011-06-20 09:52:21 +02:00
static int shadow_copy2_lstat ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname )
2008-01-16 12:21:38 +03:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
char * stripped = NULL ;
char * tmp ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
int ret ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2011-06-20 09:52:21 +02:00
& timestamp , & stripped ) ) {
return - 1 ;
}
if ( timestamp = = 0 ) {
return SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
}
2009-07-02 13:39:20 -07:00
2011-06-20 09:52:21 +02:00
tmp = smb_fname - > base_name ;
smb_fname - > base_name = shadow_copy2_convert (
talloc_tos ( ) , handle , stripped , timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( smb_fname - > base_name = = NULL ) {
smb_fname - > base_name = tmp ;
2009-07-02 13:39:20 -07:00
return - 1 ;
}
2011-06-20 09:52:21 +02:00
ret = SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
2017-01-23 10:20:13 -08:00
if ( ret = = - 1 ) {
saved_errno = errno ;
}
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
TALLOC_FREE ( smb_fname - > base_name ) ;
smb_fname - > base_name = tmp ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
if ( ret = = 0 ) {
convert_sbuf ( handle , smb_fname - > base_name , & smb_fname - > st ) ;
}
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2011-06-20 09:52:21 +02:00
return ret ;
2008-01-16 12:21:38 +03:00
}
2011-06-20 09:52:21 +02:00
static int shadow_copy2_fstat ( vfs_handle_struct * handle , files_struct * fsp ,
SMB_STRUCT_STAT * sbuf )
2008-01-16 12:21:38 +03:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
2018-11-21 17:20:30 +01:00
struct smb_filename * orig_smb_fname = NULL ;
struct smb_filename vss_smb_fname ;
struct smb_filename * orig_base_smb_fname = NULL ;
struct smb_filename vss_base_smb_fname ;
char * stripped = NULL ;
int saved_errno = 0 ;
bool ok ;
2011-06-20 09:52:21 +02:00
int ret ;
2009-09-16 03:22:56 +02:00
2018-11-21 17:20:30 +01:00
ok = shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle ,
2020-04-30 17:28:16 +02:00
fsp - > fsp_name ,
2018-11-21 17:20:30 +01:00
& timestamp , & stripped ) ;
if ( ! ok ) {
return - 1 ;
}
if ( timestamp = = 0 ) {
TALLOC_FREE ( stripped ) ;
return SMB_VFS_NEXT_FSTAT ( handle , fsp , sbuf ) ;
}
vss_smb_fname = * fsp - > fsp_name ;
vss_smb_fname . base_name = shadow_copy2_convert ( talloc_tos ( ) ,
handle ,
stripped ,
timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( vss_smb_fname . base_name = = NULL ) {
return - 1 ;
}
orig_smb_fname = fsp - > fsp_name ;
fsp - > fsp_name = & vss_smb_fname ;
if ( fsp - > base_fsp ! = NULL ) {
vss_base_smb_fname = * fsp - > base_fsp - > fsp_name ;
vss_base_smb_fname . base_name = vss_smb_fname . base_name ;
orig_base_smb_fname = fsp - > base_fsp - > fsp_name ;
fsp - > base_fsp - > fsp_name = & vss_base_smb_fname ;
}
2011-06-20 09:52:21 +02:00
ret = SMB_VFS_NEXT_FSTAT ( handle , fsp , sbuf ) ;
2018-11-21 17:20:30 +01:00
fsp - > fsp_name = orig_smb_fname ;
if ( fsp - > base_fsp ! = NULL ) {
fsp - > base_fsp - > fsp_name = orig_base_smb_fname ;
2011-06-20 09:52:21 +02:00
}
2018-11-21 17:20:30 +01:00
if ( ret = = - 1 ) {
saved_errno = errno ;
2009-06-30 14:26:32 +02:00
}
2018-11-21 17:20:30 +01:00
if ( ret = = 0 ) {
2011-06-20 09:52:21 +02:00
convert_sbuf ( handle , fsp - > fsp_name - > base_name , sbuf ) ;
}
2018-11-21 17:20:30 +01:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
return ret ;
2008-01-16 12:21:38 +03:00
}
2020-05-20 23:03:05 +02:00
static int shadow_copy2_openat ( vfs_handle_struct * handle ,
const struct files_struct * dirfsp ,
const struct smb_filename * smb_fname_in ,
struct files_struct * fsp ,
int flags ,
mode_t mode )
{
struct smb_filename * smb_fname = NULL ;
time_t timestamp = 0 ;
char * stripped = NULL ;
bool is_converted = false ;
int saved_errno = 0 ;
int ret ;
bool ok ;
2020-10-16 12:28:39 +02:00
smb_fname = full_path_from_dirfsp_atname ( talloc_tos ( ) ,
dirfsp ,
smb_fname_in ) ;
if ( smb_fname = = NULL ) {
errno = ENOMEM ;
return - 1 ;
}
2020-05-20 23:03:05 +02:00
ok = shadow_copy2_strip_snapshot_converted ( talloc_tos ( ) ,
handle ,
2020-10-16 12:28:39 +02:00
smb_fname ,
2020-05-20 23:03:05 +02:00
& timestamp ,
& stripped ,
& is_converted ) ;
if ( ! ok ) {
return - 1 ;
}
if ( timestamp = = 0 ) {
if ( is_converted ) {
/*
* Just pave over the user requested mode and use
* O_RDONLY . Later attempts by the client to write on
* the handle will fail in the pwrite ( ) syscall with
* EINVAL which we carefully map to EROFS . In sum , this
* matches Windows behaviour .
*/
2020-11-12 20:27:24 +05:30
flags & = ~ ( O_WRONLY | O_RDWR | O_CREAT ) ;
2020-05-20 23:03:05 +02:00
}
return SMB_VFS_NEXT_OPENAT ( handle ,
dirfsp ,
smb_fname_in ,
fsp ,
flags ,
mode ) ;
}
smb_fname - > base_name = shadow_copy2_convert ( smb_fname ,
handle ,
stripped ,
timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( smb_fname - > base_name = = NULL ) {
TALLOC_FREE ( smb_fname ) ;
errno = ENOMEM ;
return - 1 ;
}
/*
* Just pave over the user requested mode and use O_RDONLY . Later
* attempts by the client to write on the handle will fail in the
* pwrite ( ) syscall with EINVAL which we carefully map to EROFS . In sum ,
* this matches Windows behaviour .
*/
2020-11-12 20:27:24 +05:30
flags & = ~ ( O_WRONLY | O_RDWR | O_CREAT ) ;
2020-05-20 23:03:05 +02:00
ret = SMB_VFS_NEXT_OPENAT ( handle ,
dirfsp ,
smb_fname ,
fsp ,
flags ,
mode ) ;
if ( ret = = - 1 ) {
saved_errno = errno ;
}
TALLOC_FREE ( smb_fname ) ;
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
return ret ;
}
2019-09-12 13:19:39 -07:00
static int shadow_copy2_unlinkat ( vfs_handle_struct * handle ,
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
int flags )
{
time_t timestamp = 0 ;
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2019-09-12 13:19:39 -07:00
& timestamp , NULL ) ) {
return - 1 ;
}
if ( timestamp ! = 0 ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
smb_fname ,
flags ) ;
}
2021-04-09 15:54:44 +02:00
static int shadow_copy2_fchmod ( vfs_handle_struct * handle ,
struct files_struct * fsp ,
mode_t mode )
{
time_t timestamp = 0 ;
const struct smb_filename * smb_fname = NULL ;
smb_fname = fsp - > fsp_name ;
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
smb_fname ,
& timestamp ,
NULL ) ) {
return - 1 ;
}
if ( timestamp ! = 0 ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_FCHMOD ( handle , fsp , mode ) ;
}
2017-01-20 12:06:55 -08:00
static void store_cwd_data ( vfs_handle_struct * handle ,
const char * connectpath )
{
struct shadow_copy2_private * priv = NULL ;
2017-06-29 14:32:47 -07:00
struct smb_filename * cwd_fname = NULL ;
2017-01-20 12:06:55 -08:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
return ) ;
TALLOC_FREE ( priv - > shadow_cwd ) ;
2017-06-29 14:32:47 -07:00
cwd_fname = SMB_VFS_NEXT_GETWD ( handle , talloc_tos ( ) ) ;
if ( cwd_fname = = NULL ) {
2017-01-20 12:06:55 -08:00
smb_panic ( " getwd failed \n " ) ;
}
2017-06-29 14:32:47 -07:00
DBG_DEBUG ( " shadow cwd = %s \n " , cwd_fname - > base_name ) ;
priv - > shadow_cwd = talloc_strdup ( priv , cwd_fname - > base_name ) ;
TALLOC_FREE ( cwd_fname ) ;
2017-01-20 12:06:55 -08:00
if ( priv - > shadow_cwd = = NULL ) {
smb_panic ( " talloc failed \n " ) ;
}
TALLOC_FREE ( priv - > shadow_connectpath ) ;
if ( connectpath ) {
2019-11-20 11:08:06 +01:00
DBG_DEBUG ( " shadow connectpath = %s \n " , connectpath ) ;
2017-01-20 12:06:55 -08:00
priv - > shadow_connectpath = talloc_strdup ( priv , connectpath ) ;
if ( priv - > shadow_connectpath = = NULL ) {
smb_panic ( " talloc failed \n " ) ;
}
}
}
2011-06-20 09:52:21 +02:00
static int shadow_copy2_chdir ( vfs_handle_struct * handle ,
2017-06-29 11:29:33 -07:00
const struct smb_filename * smb_fname )
2008-01-16 12:21:38 +03:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
char * stripped = NULL ;
2017-01-20 12:06:55 -08:00
char * snappath = NULL ;
int ret = - 1 ;
int saved_errno = 0 ;
char * conv = NULL ;
size_t rootpath_len = 0 ;
2017-06-29 11:29:33 -07:00
struct smb_filename * conv_smb_fname = NULL ;
2011-06-20 09:52:21 +02:00
2017-06-29 11:29:33 -07:00
if ( ! shadow_copy2_strip_snapshot_internal ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2017-06-29 11:29:33 -07:00
& timestamp ,
& stripped ,
2018-11-22 11:02:24 +01:00
& snappath ,
NULL ) ) {
2011-06-20 09:52:21 +02:00
return - 1 ;
}
2017-01-20 12:06:55 -08:00
if ( stripped ! = NULL ) {
conv = shadow_copy2_do_convert ( talloc_tos ( ) ,
handle ,
stripped ,
timestamp ,
& rootpath_len ) ;
TALLOC_FREE ( stripped ) ;
if ( conv = = NULL ) {
return - 1 ;
}
2017-06-29 11:29:33 -07:00
conv_smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
conv ,
NULL ,
NULL ,
2020-04-30 11:48:32 +02:00
0 ,
2017-06-29 11:29:33 -07:00
smb_fname - > flags ) ;
} else {
conv_smb_fname = cp_smb_filename ( talloc_tos ( ) , smb_fname ) ;
}
if ( conv_smb_fname = = NULL ) {
TALLOC_FREE ( conv ) ;
errno = ENOMEM ;
return - 1 ;
2011-06-20 09:52:21 +02:00
}
2017-01-20 12:06:55 -08:00
2017-06-29 11:29:33 -07:00
ret = SMB_VFS_NEXT_CHDIR ( handle , conv_smb_fname ) ;
2017-01-20 12:06:55 -08:00
if ( ret = = - 1 ) {
saved_errno = errno ;
2011-06-20 09:52:21 +02:00
}
2017-01-20 12:06:55 -08:00
if ( ret = = 0 ) {
if ( conv ! = NULL & & rootpath_len ! = 0 ) {
conv [ rootpath_len ] = ' \0 ' ;
} else if ( snappath ! = 0 ) {
TALLOC_FREE ( conv ) ;
conv = snappath ;
}
store_cwd_data ( handle , conv ) ;
}
TALLOC_FREE ( stripped ) ;
2011-06-20 09:52:21 +02:00
TALLOC_FREE ( conv ) ;
2017-06-29 11:29:33 -07:00
TALLOC_FREE ( conv_smb_fname ) ;
2017-01-20 12:06:55 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2011-06-20 09:52:21 +02:00
return ret ;
2008-01-16 12:21:38 +03:00
}
2021-04-13 13:48:36 +02:00
static int shadow_copy2_fntimes ( vfs_handle_struct * handle ,
files_struct * fsp ,
struct smb_file_time * ft )
{
time_t timestamp = 0 ;
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
fsp - > fsp_name ,
& timestamp ,
NULL ) ) {
return - 1 ;
}
if ( timestamp ! = 0 ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_FNTIMES ( handle , fsp , ft ) ;
}
2019-08-22 14:26:47 -07:00
static int shadow_copy2_readlinkat ( vfs_handle_struct * handle ,
2020-10-13 15:19:30 +02:00
const struct files_struct * dirfsp ,
2019-08-22 14:26:47 -07:00
const struct smb_filename * smb_fname ,
char * buf ,
size_t bufsiz )
{
time_t timestamp = 0 ;
char * stripped = NULL ;
int saved_errno = 0 ;
int ret ;
2021-02-11 11:36:22 -08:00
struct smb_filename * full_fname = NULL ;
2019-08-22 14:26:47 -07:00
struct smb_filename * conv = NULL ;
2021-02-11 11:36:22 -08:00
full_fname = full_path_from_dirfsp_atname ( talloc_tos ( ) ,
dirfsp ,
smb_fname ) ;
if ( full_fname = = NULL ) {
errno = ENOMEM ;
2019-08-22 14:26:47 -07:00
return - 1 ;
}
2021-02-11 11:36:22 -08:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
full_fname ,
& timestamp ,
& stripped ) ) {
TALLOC_FREE ( full_fname ) ;
return - 1 ;
}
2019-08-22 14:26:47 -07:00
if ( timestamp = = 0 ) {
2021-02-11 11:36:22 -08:00
TALLOC_FREE ( full_fname ) ;
TALLOC_FREE ( stripped ) ;
2019-08-22 14:26:47 -07:00
return SMB_VFS_NEXT_READLINKAT ( handle ,
dirfsp ,
smb_fname ,
buf ,
bufsiz ) ;
}
2021-02-11 11:36:22 -08:00
conv = cp_smb_filename ( talloc_tos ( ) , full_fname ) ;
2019-08-22 14:26:47 -07:00
if ( conv = = NULL ) {
2021-02-11 11:36:22 -08:00
TALLOC_FREE ( full_fname ) ;
2019-08-22 14:26:47 -07:00
TALLOC_FREE ( stripped ) ;
errno = ENOMEM ;
return - 1 ;
}
2021-02-11 11:36:22 -08:00
TALLOC_FREE ( full_fname ) ;
2019-08-22 14:26:47 -07:00
conv - > base_name = shadow_copy2_convert (
conv , handle , stripped , timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( conv - > base_name = = NULL ) {
return - 1 ;
}
ret = SMB_VFS_NEXT_READLINKAT ( handle ,
2021-02-11 11:36:22 -08:00
handle - > conn - > cwd_fsp ,
2019-08-22 14:26:47 -07:00
conv ,
buf ,
bufsiz ) ;
if ( ret = = - 1 ) {
saved_errno = errno ;
}
TALLOC_FREE ( conv ) ;
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
return ret ;
}
2019-08-20 16:57:24 -07:00
static int shadow_copy2_mknodat ( vfs_handle_struct * handle ,
files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
mode_t mode ,
SMB_DEV_T dev )
{
time_t timestamp = 0 ;
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle ,
2021-02-03 13:55:12 -08:00
smb_fname ,
2019-08-20 16:57:24 -07:00
& timestamp , NULL ) ) {
return - 1 ;
}
if ( timestamp ! = 0 ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_MKNODAT ( handle ,
dirfsp ,
smb_fname ,
mode ,
dev ) ;
}
2017-06-30 11:32:59 -07:00
static struct smb_filename * shadow_copy2_realpath ( vfs_handle_struct * handle ,
TALLOC_CTX * ctx ,
const struct smb_filename * smb_fname )
2008-01-16 12:21:38 +03:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
2011-06-20 09:52:21 +02:00
char * stripped = NULL ;
2017-06-30 11:32:59 -07:00
struct smb_filename * result_fname = NULL ;
struct smb_filename * conv_fname = NULL ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
2011-06-20 09:52:21 +02:00
2017-06-30 11:32:59 -07:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2017-06-30 11:32:59 -07:00
& timestamp , & stripped ) ) {
2011-06-20 09:52:21 +02:00
goto done ;
}
if ( timestamp = = 0 ) {
2017-06-30 11:32:59 -07:00
return SMB_VFS_NEXT_REALPATH ( handle , ctx , smb_fname ) ;
2011-06-20 09:52:21 +02:00
}
2017-06-30 11:32:59 -07:00
conv_fname = cp_smb_filename ( talloc_tos ( ) , smb_fname ) ;
if ( conv_fname = = NULL ) {
goto done ;
}
conv_fname - > base_name = shadow_copy2_convert (
conv_fname , handle , stripped , timestamp ) ;
if ( conv_fname - > base_name = = NULL ) {
2011-06-20 09:52:21 +02:00
goto done ;
}
2017-06-30 11:32:59 -07:00
result_fname = SMB_VFS_NEXT_REALPATH ( handle , ctx , conv_fname ) ;
2011-06-20 09:52:21 +02:00
done :
2017-06-30 11:32:59 -07:00
if ( result_fname = = NULL ) {
2017-01-23 10:20:13 -08:00
saved_errno = errno ;
}
2017-06-30 11:32:59 -07:00
TALLOC_FREE ( conv_fname ) ;
2011-06-20 09:52:21 +02:00
TALLOC_FREE ( stripped ) ;
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2017-06-30 11:32:59 -07:00
return result_fname ;
2008-01-16 12:21:38 +03:00
}
2013-05-23 23:59:49 +02:00
/**
* Check whether a given directory contains a
* snapshot directory as direct subdirectory .
* If yes , return the path of the snapshot - subdir ,
* otherwise return NULL .
*/
2011-06-20 09:52:21 +02:00
static char * have_snapdir ( struct vfs_handle_struct * handle ,
const char * path )
2008-01-16 12:21:38 +03:00
{
2011-06-20 09:52:21 +02:00
struct smb_filename smb_fname ;
int ret ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2013-05-24 01:35:44 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return NULL ) ;
2011-06-20 09:52:21 +02:00
ZERO_STRUCT ( smb_fname ) ;
2013-05-24 01:35:44 +02:00
smb_fname . base_name = talloc_asprintf ( talloc_tos ( ) , " %s/%s " ,
2016-07-11 11:27:37 +00:00
path , priv - > config - > snapdir ) ;
2011-06-20 09:52:21 +02:00
if ( smb_fname . base_name = = NULL ) {
return NULL ;
}
ret = SMB_VFS_NEXT_STAT ( handle , & smb_fname ) ;
if ( ( ret = = 0 ) & & ( S_ISDIR ( smb_fname . st . st_ex_mode ) ) ) {
return smb_fname . base_name ;
}
TALLOC_FREE ( smb_fname . base_name ) ;
return NULL ;
2008-01-16 12:21:38 +03:00
}
2013-05-24 00:01:14 +02:00
/**
* Find the snapshot directory ( if any ) for the given
* filename ( which is relative to the share ) .
*/
2013-05-28 17:01:20 +02:00
static const char * shadow_copy2_find_snapdir ( TALLOC_CTX * mem_ctx ,
struct vfs_handle_struct * handle ,
struct smb_filename * smb_fname )
2008-01-16 12:21:38 +03:00
{
2011-06-20 09:52:21 +02:00
char * path , * p ;
2013-05-28 17:01:20 +02:00
const char * snapdir ;
2013-05-24 01:35:44 +02:00
struct shadow_copy2_config * config ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2013-05-24 01:35:44 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return NULL ) ;
2011-06-20 09:52:21 +02:00
2016-07-11 11:27:37 +00:00
config = priv - > config ;
2013-05-23 16:21:46 +02:00
/*
* If the non - snapdisrseverywhere mode , we should not search !
*/
if ( ! config - > snapdirseverywhere ) {
return config - > snapshot_basepath ;
}
2011-06-20 09:52:21 +02:00
path = talloc_asprintf ( mem_ctx , " %s/%s " ,
handle - > conn - > connectpath ,
smb_fname - > base_name ) ;
if ( path = = NULL ) {
return NULL ;
}
snapdir = have_snapdir ( handle , path ) ;
if ( snapdir ! = NULL ) {
TALLOC_FREE ( path ) ;
return snapdir ;
}
while ( ( p = strrchr ( path , ' / ' ) ) & & ( p > path ) ) {
p [ 0 ] = ' \0 ' ;
snapdir = have_snapdir ( handle , path ) ;
if ( snapdir ! = NULL ) {
TALLOC_FREE ( path ) ;
return snapdir ;
}
}
TALLOC_FREE ( path ) ;
return NULL ;
2008-01-16 12:21:38 +03:00
}
2012-05-08 09:15:12 +02:00
static bool shadow_copy2_snapshot_to_gmt ( vfs_handle_struct * handle ,
2011-06-20 09:52:21 +02:00
const char * name ,
char * gmt , size_t gmt_len )
2008-01-16 12:21:38 +03:00
{
2011-06-20 09:52:21 +02:00
struct tm timestamp ;
time_t timestamp_t ;
2012-07-02 19:31:58 +10:00
unsigned long int timestamp_long ;
2011-06-20 09:52:21 +02:00
const char * fmt ;
2013-05-24 01:35:44 +02:00
struct shadow_copy2_config * config ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
char * tmpstr = NULL ;
char * tmp = NULL ;
bool converted = false ;
int ret = - 1 ;
2011-06-20 09:52:21 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return NULL ) ;
2016-07-11 11:27:37 +00:00
config = priv - > config ;
2013-05-24 01:35:44 +02:00
fmt = config - > gmt_format ;
2011-06-20 09:52:21 +02:00
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
/*
* If regex is provided , then we will have to parse the
* filename which will contain both the prefix and the time format .
* e . g . < prefix > < delimiter > < time_format >
*/
if ( priv - > snaps - > regex ! = NULL ) {
tmpstr = talloc_strdup ( talloc_tos ( ) , name ) ;
/* point "name" to the time format */
name = strstr ( name , priv - > config - > delimiter ) ;
if ( name = = NULL ) {
goto done ;
}
/* Extract the prefix */
tmp = strstr ( tmpstr , priv - > config - > delimiter ) ;
2017-02-16 17:15:38 +01:00
if ( tmp = = NULL ) {
goto done ;
}
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
* tmp = ' \0 ' ;
/* Parse regex */
ret = regexec ( priv - > snaps - > regex , tmpstr , 0 , NULL , 0 ) ;
if ( ret ) {
DBG_DEBUG ( " shadow_copy2_snapshot_to_gmt: "
" no regex match for %s \n " , tmpstr ) ;
goto done ;
}
}
2011-06-20 09:52:21 +02:00
ZERO_STRUCT ( timestamp ) ;
2013-05-24 01:35:44 +02:00
if ( config - > use_sscanf ) {
2012-07-02 19:31:58 +10:00
if ( sscanf ( name , fmt , & timestamp_long ) ! = 1 ) {
2013-10-04 13:15:34 +02:00
DEBUG ( 10 , ( " shadow_copy2_snapshot_to_gmt: "
" no sscanf match %s: %s \n " ,
2012-07-02 19:31:58 +10:00
fmt , name ) ) ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
goto done ;
2012-07-02 19:31:58 +10:00
}
timestamp_t = timestamp_long ;
2011-06-20 09:52:21 +02:00
gmtime_r ( & timestamp_t , & timestamp ) ;
2012-07-02 19:31:58 +10:00
} else {
if ( strptime ( name , fmt , & timestamp ) = = NULL ) {
2013-10-04 13:15:34 +02:00
DEBUG ( 10 , ( " shadow_copy2_snapshot_to_gmt: "
" no match %s: %s \n " ,
2012-07-02 19:31:58 +10:00
fmt , name ) ) ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
goto done ;
2012-07-02 19:31:58 +10:00
}
2013-10-04 13:15:34 +02:00
DEBUG ( 10 , ( " shadow_copy2_snapshot_to_gmt: match %s: %s \n " ,
fmt , name ) ) ;
2012-07-02 19:31:58 +10:00
2013-05-24 01:35:44 +02:00
if ( config - > use_localtime ) {
2012-07-02 19:31:58 +10:00
timestamp . tm_isdst = - 1 ;
timestamp_t = mktime ( & timestamp ) ;
gmtime_r ( & timestamp_t , & timestamp ) ;
}
2011-06-20 09:52:21 +02:00
}
2012-07-02 19:31:58 +10:00
2011-06-20 09:52:21 +02:00
strftime ( gmt , gmt_len , GMT_FORMAT , & timestamp ) ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
converted = true ;
done :
TALLOC_FREE ( tmpstr ) ;
return converted ;
2008-01-16 12:21:38 +03:00
}
2009-12-06 21:00:00 -06:00
static int shadow_copy2_label_cmp_asc ( const void * x , const void * y )
{
2011-05-05 16:19:49 -07:00
return strncmp ( ( const char * ) x , ( const char * ) y , sizeof ( SHADOW_COPY_LABEL ) ) ;
2009-12-06 21:00:00 -06:00
}
static int shadow_copy2_label_cmp_desc ( const void * x , const void * y )
{
2011-05-05 16:19:49 -07:00
return - strncmp ( ( const char * ) x , ( const char * ) y , sizeof ( SHADOW_COPY_LABEL ) ) ;
2009-12-06 21:00:00 -06:00
}
/*
sort the shadow copy data in ascending or descending order
*/
static void shadow_copy2_sort_data ( vfs_handle_struct * handle ,
2011-05-30 12:06:31 +02:00
struct shadow_copy_data * shadow_copy2_data )
2009-12-06 21:00:00 -06:00
{
int ( * cmpfunc ) ( const void * , const void * ) ;
const char * sort ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2013-05-24 01:35:44 +02:00
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return ) ;
2009-12-06 21:00:00 -06:00
2016-07-11 11:27:37 +00:00
sort = priv - > config - > sort_order ;
2009-12-06 21:00:00 -06:00
if ( sort = = NULL ) {
return ;
}
if ( strcmp ( sort , " asc " ) = = 0 ) {
cmpfunc = shadow_copy2_label_cmp_asc ;
} else if ( strcmp ( sort , " desc " ) = = 0 ) {
cmpfunc = shadow_copy2_label_cmp_desc ;
} else {
return ;
}
if ( shadow_copy2_data & & shadow_copy2_data - > num_volumes > 0 & &
shadow_copy2_data - > labels )
{
2010-02-14 10:01:17 +11:00
TYPESAFE_QSORT ( shadow_copy2_data - > labels ,
shadow_copy2_data - > num_volumes ,
cmpfunc ) ;
2009-12-06 21:00:00 -06:00
}
}
2011-06-20 09:52:21 +02:00
static int shadow_copy2_get_shadow_copy_data (
vfs_handle_struct * handle , files_struct * fsp ,
struct shadow_copy_data * shadow_copy2_data ,
bool labels )
2008-01-16 12:21:38 +03:00
{
2020-03-22 12:21:10 +01:00
DIR * p = NULL ;
2008-01-16 12:21:38 +03:00
const char * snapdir ;
2016-02-26 14:53:12 -08:00
struct smb_filename * snapdir_smb_fname = NULL ;
2020-03-22 12:21:10 +01:00
struct files_struct * dirfsp = NULL ;
2020-05-20 22:48:36 +02:00
struct files_struct * fspcwd = NULL ;
2012-03-28 13:18:14 +11:00
struct dirent * d ;
2011-06-20 09:52:21 +02:00
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
struct shadow_copy2_private * priv = NULL ;
struct shadow_copy2_snapentry * tmpentry = NULL ;
bool get_snaplist = false ;
2020-03-22 12:21:10 +01:00
int open_flags = O_RDONLY ;
2020-09-26 21:46:51 +02:00
int fd ;
2016-07-21 09:58:43 +00:00
int ret = - 1 ;
2020-03-22 12:21:10 +01:00
NTSTATUS status ;
2021-06-07 09:44:23 -07:00
int saved_errno = 0 ;
2008-01-16 12:21:38 +03:00
2011-06-20 09:52:21 +02:00
snapdir = shadow_copy2_find_snapdir ( tmp_ctx , handle , fsp - > fsp_name ) ;
2008-01-16 12:21:38 +03:00
if ( snapdir = = NULL ) {
DEBUG ( 0 , ( " shadow:snapdir not found for %s in get_shadow_copy_data \n " ,
handle - > conn - > connectpath ) ) ;
errno = EINVAL ;
2016-07-21 09:58:43 +00:00
goto done ;
2008-01-16 12:21:38 +03:00
}
2016-07-21 09:58:43 +00:00
2016-02-26 14:53:12 -08:00
snapdir_smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
snapdir ,
NULL ,
2016-03-18 21:19:38 -07:00
NULL ,
2020-04-30 11:48:32 +02:00
0 ,
2016-03-18 21:19:38 -07:00
fsp - > fsp_name - > flags ) ;
2016-02-26 14:53:12 -08:00
if ( snapdir_smb_fname = = NULL ) {
errno = ENOMEM ;
2016-07-21 09:58:43 +00:00
goto done ;
2016-02-26 14:53:12 -08:00
}
2020-05-15 15:14:26 +02:00
status = create_internal_dirfsp ( handle - > conn ,
snapdir_smb_fname ,
& dirfsp ) ;
2020-03-22 12:21:10 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_WARNING ( " create_internal_dir_fsp() failed for '%s' "
" - %s \n " , snapdir , nt_errstr ( status ) ) ;
errno = ENOSYS ;
goto done ;
}
2020-05-20 22:48:36 +02:00
status = vfs_at_fspcwd ( talloc_tos ( ) , handle - > conn , & fspcwd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
errno = ENOMEM ;
goto done ;
}
2020-03-22 12:21:10 +01:00
# ifdef O_DIRECTORY
open_flags | = O_DIRECTORY ;
# endif
2008-01-16 12:21:38 +03:00
2020-09-26 21:46:51 +02:00
fd = SMB_VFS_NEXT_OPENAT ( handle ,
fspcwd ,
snapdir_smb_fname ,
dirfsp ,
open_flags ,
0 ) ;
if ( fd = = - 1 ) {
2020-03-22 12:21:10 +01:00
DBG_WARNING ( " SMB_VFS_NEXT_OPEN failed for '%s' "
" - %s \n " , snapdir , strerror ( errno ) ) ;
errno = ENOSYS ;
goto done ;
}
2020-09-26 21:46:51 +02:00
fsp_set_fd ( dirfsp , fd ) ;
2020-03-22 12:21:10 +01:00
2021-06-07 09:47:26 -07:00
/* Now we have the handle, check access here. */
status = smbd_check_access_rights_fsp ( dirfsp ,
false ,
SEC_DIR_LIST ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_ERR ( " user does not have list permission "
" on snapdir %s \n " ,
fsp_str_dbg ( dirfsp ) ) ;
errno = EACCES ;
goto done ;
}
2020-03-22 12:21:10 +01:00
p = SMB_VFS_NEXT_FDOPENDIR ( handle , dirfsp , NULL , 0 ) ;
2008-01-16 12:21:38 +03:00
if ( ! p ) {
2020-06-20 17:14:40 +05:30
DBG_NOTICE ( " shadow_copy2: SMB_VFS_NEXT_FDOPENDIR() failed for '%s' "
" - %s \n " , snapdir , strerror ( errno ) ) ;
2009-04-28 11:16:19 +02:00
errno = ENOSYS ;
2016-07-21 09:58:43 +00:00
goto done ;
2008-01-16 12:21:38 +03:00
}
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
if ( shadow_copy2_data ! = NULL ) {
shadow_copy2_data - > num_volumes = 0 ;
shadow_copy2_data - > labels = NULL ;
}
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
2016-07-21 09:58:43 +00:00
goto done ) ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
/*
* Normally this function is called twice once with labels = false and
* then with labels = true . When labels is false it will return the
* number of volumes so that the caller can allocate memory for that
* many labels . Therefore to eliminate snaplist both the times it is
* good to check if labels is set or not .
*
* shadow_copy2_data is NULL when we only want to update the list and
* don ' t want any labels .
*/
if ( ( priv - > snaps - > regex ! = NULL ) & & ( labels | | shadow_copy2_data = = NULL ) ) {
get_snaplist = true ;
/* Reset the global snaplist */
shadow_copy2_delete_snaplist ( priv ) ;
/* Set the current time as snaplist update time */
time ( & ( priv - > snaps - > fetch_time ) ) ;
}
2008-01-16 12:21:38 +03:00
2020-11-22 13:57:27 +01:00
while ( ( d = SMB_VFS_NEXT_READDIR ( handle , dirfsp , p , NULL ) ) ) {
2011-06-20 09:52:21 +02:00
char snapshot [ GMT_NAME_LEN + 1 ] ;
2008-01-16 12:21:38 +03:00
SHADOW_COPY_LABEL * tlabels ;
2011-06-20 09:52:21 +02:00
/*
* ignore names not of the right form in the snapshot
* directory
*/
if ( ! shadow_copy2_snapshot_to_gmt (
2012-05-08 09:15:12 +02:00
handle , d - > d_name ,
2011-06-20 09:52:21 +02:00
snapshot , sizeof ( snapshot ) ) ) {
DEBUG ( 6 , ( " shadow_copy2_get_shadow_copy_data: "
" ignoring %s \n " , d - > d_name ) ) ;
2008-01-16 12:21:38 +03:00
continue ;
}
2011-06-20 09:52:21 +02:00
DEBUG ( 6 , ( " shadow_copy2_get_shadow_copy_data: %s -> %s \n " ,
d - > d_name , snapshot ) ) ;
2008-01-16 12:21:38 +03:00
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
if ( get_snaplist ) {
/*
* Create a snap entry for each successful
* pattern match .
*/
tmpentry = shadow_copy2_create_snapentry ( priv ) ;
if ( tmpentry = = NULL ) {
DBG_ERR ( " talloc_zero() failed \n " ) ;
2016-07-21 09:58:43 +00:00
goto done ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
}
tmpentry - > snapname = talloc_strdup ( tmpentry , d - > d_name ) ;
tmpentry - > time_fmt = talloc_strdup ( tmpentry , snapshot ) ;
}
if ( shadow_copy2_data = = NULL ) {
continue ;
}
2008-01-16 12:21:38 +03:00
if ( ! labels ) {
/* the caller doesn't want the labels */
shadow_copy2_data - > num_volumes + + ;
continue ;
}
2011-05-30 12:11:53 +02:00
tlabels = talloc_realloc ( shadow_copy2_data ,
2008-01-16 12:21:38 +03:00
shadow_copy2_data - > labels ,
2011-06-20 09:52:21 +02:00
SHADOW_COPY_LABEL ,
shadow_copy2_data - > num_volumes + 1 ) ;
2008-01-16 12:21:38 +03:00
if ( tlabels = = NULL ) {
DEBUG ( 0 , ( " shadow_copy2: out of memory \n " ) ) ;
2016-07-21 09:58:43 +00:00
goto done ;
2008-01-16 12:21:38 +03:00
}
2010-01-21 08:30:01 +01:00
strlcpy ( tlabels [ shadow_copy2_data - > num_volumes ] , snapshot ,
sizeof ( * tlabels ) ) ;
2008-01-16 12:21:38 +03:00
shadow_copy2_data - > num_volumes + + ;
shadow_copy2_data - > labels = tlabels ;
}
2009-12-06 21:00:00 -06:00
shadow_copy2_sort_data ( handle , shadow_copy2_data ) ;
2016-07-21 09:58:43 +00:00
ret = 0 ;
2009-12-06 21:00:00 -06:00
2016-07-21 09:58:43 +00:00
done :
2021-06-07 09:44:23 -07:00
if ( ret ! = 0 ) {
saved_errno = errno ;
}
2020-05-20 22:48:36 +02:00
TALLOC_FREE ( fspcwd ) ;
2020-03-22 12:21:10 +01:00
if ( p ! = NULL ) {
SMB_VFS_NEXT_CLOSEDIR ( handle , p ) ;
p = NULL ;
2020-10-12 16:55:40 +05:30
if ( dirfsp ! = NULL ) {
/*
* VFS_CLOSEDIR implicitly
* closed the associated fd .
*/
2020-09-26 21:46:51 +02:00
fsp_set_fd ( dirfsp , - 1 ) ;
2020-10-12 16:55:40 +05:30
}
2020-03-22 12:21:10 +01:00
}
if ( dirfsp ! = NULL ) {
fd_close ( dirfsp ) ;
file_free ( NULL , dirfsp ) ;
}
2016-07-21 09:58:43 +00:00
TALLOC_FREE ( tmp_ctx ) ;
2021-06-07 09:44:23 -07:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2016-07-21 09:58:43 +00:00
return ret ;
2008-01-16 12:21:38 +03:00
}
2020-04-13 14:05:03 -07:00
static NTSTATUS shadow_copy2_get_nt_acl_at ( vfs_handle_struct * handle ,
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
uint32_t security_info ,
TALLOC_CTX * mem_ctx ,
struct security_descriptor * * ppdesc )
{
time_t timestamp = 0 ;
char * stripped = NULL ;
NTSTATUS status ;
char * conv ;
struct smb_filename * conv_smb_fname = NULL ;
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
smb_fname ,
& timestamp ,
& stripped ) ) {
return map_nt_error_from_unix ( errno ) ;
}
if ( timestamp = = 0 ) {
return SMB_VFS_NEXT_GET_NT_ACL_AT ( handle ,
dirfsp ,
smb_fname ,
security_info ,
mem_ctx ,
ppdesc ) ;
}
conv = shadow_copy2_convert ( talloc_tos ( ) , handle , stripped , timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( conv = = NULL ) {
return map_nt_error_from_unix ( errno ) ;
}
conv_smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
conv ,
NULL ,
NULL ,
0 ,
smb_fname - > flags ) ;
if ( conv_smb_fname = = NULL ) {
TALLOC_FREE ( conv ) ;
return NT_STATUS_NO_MEMORY ;
}
status = SMB_VFS_NEXT_GET_NT_ACL_AT ( handle ,
dirfsp ,
conv_smb_fname ,
security_info ,
mem_ctx ,
ppdesc ) ;
TALLOC_FREE ( conv ) ;
TALLOC_FREE ( conv_smb_fname ) ;
return status ;
}
2019-09-05 10:32:18 -07:00
static int shadow_copy2_mkdirat ( vfs_handle_struct * handle ,
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
mode_t mode )
{
2020-12-14 15:53:33 +01:00
struct smb_filename * full_fname = NULL ;
2019-09-05 10:32:18 -07:00
time_t timestamp = 0 ;
2020-12-14 15:53:33 +01:00
full_fname = full_path_from_dirfsp_atname ( talloc_tos ( ) ,
dirfsp ,
smb_fname ) ;
if ( full_fname = = NULL ) {
errno = ENOMEM ;
return - 1 ;
}
2019-09-05 10:32:18 -07:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
2020-12-14 15:53:33 +01:00
full_fname ,
2019-09-05 10:32:18 -07:00
& timestamp ,
NULL ) ) {
return - 1 ;
}
2020-12-14 15:53:33 +01:00
TALLOC_FREE ( full_fname ) ;
2019-09-05 10:32:18 -07:00
if ( timestamp ! = 0 ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_MKDIRAT ( handle ,
dirfsp ,
smb_fname ,
mode ) ;
}
2017-05-19 16:15:55 -07:00
static int shadow_copy2_chflags ( vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
2011-06-20 09:52:21 +02:00
unsigned int flags )
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
2011-06-20 09:52:21 +02:00
2017-05-19 16:15:55 -07:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2017-05-19 16:15:55 -07:00
& timestamp ,
2019-08-01 13:35:47 -07:00
NULL ) ) {
2011-06-20 09:52:21 +02:00
return - 1 ;
}
2019-08-01 13:35:47 -07:00
if ( timestamp ! = 0 ) {
errno = EROFS ;
2017-05-19 16:15:55 -07:00
return - 1 ;
}
2019-08-01 13:35:47 -07:00
return SMB_VFS_NEXT_CHFLAGS ( handle , smb_fname , flags ) ;
2011-06-20 09:52:21 +02:00
}
static ssize_t shadow_copy2_getxattr ( vfs_handle_struct * handle ,
2017-05-25 16:42:04 -07:00
const struct smb_filename * smb_fname ,
const char * aname ,
void * value ,
size_t size )
2011-06-20 09:52:21 +02:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
char * stripped = NULL ;
2011-06-20 09:52:21 +02:00
ssize_t ret ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
2011-06-20 09:52:21 +02:00
char * conv ;
2017-05-25 16:42:04 -07:00
struct smb_filename * conv_smb_fname = NULL ;
2011-06-20 09:52:21 +02:00
2017-05-25 16:42:04 -07:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2017-05-25 16:42:04 -07:00
& timestamp ,
& stripped ) ) {
2011-06-20 09:52:21 +02:00
return - 1 ;
}
if ( timestamp = = 0 ) {
2017-05-25 16:42:04 -07:00
return SMB_VFS_NEXT_GETXATTR ( handle , smb_fname , aname , value ,
2011-06-20 09:52:21 +02:00
size ) ;
}
conv = shadow_copy2_convert ( talloc_tos ( ) , handle , stripped , timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( conv = = NULL ) {
return - 1 ;
}
2017-05-25 16:42:04 -07:00
conv_smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
conv ,
NULL ,
NULL ,
2020-04-30 11:48:32 +02:00
0 ,
2017-05-25 16:42:04 -07:00
smb_fname - > flags ) ;
if ( conv_smb_fname = = NULL ) {
TALLOC_FREE ( conv ) ;
return - 1 ;
}
ret = SMB_VFS_NEXT_GETXATTR ( handle , conv_smb_fname , aname , value , size ) ;
2017-01-23 10:20:13 -08:00
if ( ret = = - 1 ) {
saved_errno = errno ;
}
2017-05-25 16:42:04 -07:00
TALLOC_FREE ( conv_smb_fname ) ;
2011-06-20 09:52:21 +02:00
TALLOC_FREE ( conv ) ;
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2011-06-20 09:52:21 +02:00
return ret ;
}
2021-02-19 15:03:01 +00:00
static int shadow_copy2_fsetxattr ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
const char * aname , const void * value ,
size_t size , int flags )
{
time_t timestamp = 0 ;
const struct smb_filename * smb_fname = NULL ;
smb_fname = fsp - > fsp_name ;
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
smb_fname ,
& timestamp ,
NULL ) ) {
return - 1 ;
}
if ( timestamp ! = 0 ) {
errno = EROFS ;
return - 1 ;
}
return SMB_VFS_NEXT_FSETXATTR ( handle , fsp ,
aname , value , size , flags ) ;
}
2020-01-09 13:06:36 -08:00
static NTSTATUS shadow_copy2_create_dfs_pathat ( struct vfs_handle_struct * handle ,
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
const struct referral * reflist ,
size_t referral_count )
{
time_t timestamp = 0 ;
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2020-01-09 13:06:36 -08:00
& timestamp ,
NULL ) ) {
return NT_STATUS_NO_MEMORY ;
}
if ( timestamp ! = 0 ) {
return NT_STATUS_MEDIA_WRITE_PROTECTED ;
}
return SMB_VFS_NEXT_CREATE_DFS_PATHAT ( handle ,
dirfsp ,
smb_fname ,
reflist ,
referral_count ) ;
}
2020-01-28 12:12:09 -08:00
static NTSTATUS shadow_copy2_read_dfs_pathat ( struct vfs_handle_struct * handle ,
TALLOC_CTX * mem_ctx ,
struct files_struct * dirfsp ,
2020-05-29 16:32:12 -07:00
struct smb_filename * smb_fname ,
2020-01-28 12:12:09 -08:00
struct referral * * ppreflist ,
size_t * preferral_count )
{
time_t timestamp = 0 ;
char * stripped = NULL ;
struct smb_filename * conv = NULL ;
NTSTATUS status ;
if ( ! shadow_copy2_strip_snapshot ( mem_ctx ,
handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2020-01-28 12:12:09 -08:00
& timestamp ,
& stripped ) ) {
return NT_STATUS_NO_MEMORY ;
}
if ( timestamp = = 0 ) {
return SMB_VFS_NEXT_READ_DFS_PATHAT ( handle ,
mem_ctx ,
dirfsp ,
smb_fname ,
ppreflist ,
preferral_count ) ;
}
conv = cp_smb_filename ( mem_ctx , smb_fname ) ;
if ( conv = = NULL ) {
TALLOC_FREE ( stripped ) ;
return NT_STATUS_NO_MEMORY ;
}
conv - > base_name = shadow_copy2_convert ( conv ,
handle ,
stripped ,
timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( conv - > base_name = = NULL ) {
TALLOC_FREE ( conv ) ;
return NT_STATUS_NO_MEMORY ;
}
status = SMB_VFS_NEXT_READ_DFS_PATHAT ( handle ,
mem_ctx ,
dirfsp ,
conv ,
ppreflist ,
preferral_count ) ;
2020-05-29 17:52:51 -07:00
if ( NT_STATUS_IS_OK ( status ) ) {
/* Return any stat(2) info. */
smb_fname - > st = conv - > st ;
}
2020-01-28 12:12:09 -08:00
TALLOC_FREE ( conv ) ;
return status ;
}
2011-06-20 09:52:21 +02:00
static int shadow_copy2_get_real_filename ( struct vfs_handle_struct * handle ,
2020-04-30 16:40:28 +02:00
const struct smb_filename * fname ,
2011-06-20 09:52:21 +02:00
const char * name ,
TALLOC_CTX * mem_ctx ,
char * * found_name )
{
2020-05-05 10:13:00 +02:00
struct shadow_copy2_private * priv = NULL ;
struct shadow_copy2_config * config = NULL ;
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
char * stripped = NULL ;
2011-06-20 09:52:21 +02:00
ssize_t ret ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
2011-06-20 09:52:21 +02:00
char * conv ;
2020-04-30 16:40:28 +02:00
struct smb_filename conv_fname ;
2011-06-20 09:52:21 +02:00
2020-05-05 10:13:00 +02:00
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
return - 1 ) ;
config = priv - > config ;
DBG_DEBUG ( " Path=[%s] name=[%s] \n " , smb_fname_str_dbg ( fname ) , name ) ;
2012-10-17 12:11:37 +02:00
2020-04-30 17:28:16 +02:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle , fname ,
2011-06-20 09:52:21 +02:00
& timestamp , & stripped ) ) {
2012-10-17 12:11:37 +02:00
DEBUG ( 10 , ( " shadow_copy2_strip_snapshot failed \n " ) ) ;
2011-06-20 09:52:21 +02:00
return - 1 ;
}
if ( timestamp = = 0 ) {
2012-10-17 12:11:37 +02:00
DEBUG ( 10 , ( " timestamp == 0 \n " ) ) ;
2020-04-30 16:40:28 +02:00
return SMB_VFS_NEXT_GET_REAL_FILENAME ( handle , fname , name ,
2011-06-20 09:52:21 +02:00
mem_ctx , found_name ) ;
}
2020-05-05 10:13:00 +02:00
/*
* Note that stripped may be an empty string " " if path was " . " . As
* shadow_copy2_convert ( ) combines " " with the shadow - copy tree connect
* root fullpath and get_real_filename_full_scan ( ) has an explicit check
* for " " this works .
*/
DBG_DEBUG ( " stripped [%s] \n " , stripped ) ;
2011-06-20 09:52:21 +02:00
conv = shadow_copy2_convert ( talloc_tos ( ) , handle , stripped , timestamp ) ;
if ( conv = = NULL ) {
2020-05-05 10:13:00 +02:00
if ( ! config - > snapdirseverywhere ) {
DBG_DEBUG ( " shadow_copy2_convert [%s] failed \n " , stripped ) ;
return - 1 ;
}
/*
* We ' re called in the path traversal loop in unix_convert ( )
* walking down the directory hierarchy . shadow_copy2_convert ( )
* will fail if the snapshot directory is futher down in the
* hierachy . Set conv to the original stripped path and try to
* look it up in the filesystem with
* SMB_VFS_NEXT_GET_REAL_FILENAME ( ) or
* get_real_filename_full_scan ( ) .
*/
DBG_DEBUG ( " Use stripped [%s] as conv \n " , stripped ) ;
conv = talloc_strdup ( talloc_tos ( ) , stripped ) ;
if ( conv = = NULL ) {
TALLOC_FREE ( stripped ) ;
return - 1 ;
}
2011-06-20 09:52:21 +02:00
}
2020-04-30 16:40:28 +02:00
conv_fname = ( struct smb_filename ) {
. base_name = conv ,
} ;
2012-10-17 12:11:37 +02:00
DEBUG ( 10 , ( " Calling NEXT_GET_REAL_FILE_NAME for conv=[%s], "
" name=[%s] \n " , conv , name ) ) ;
2020-04-30 16:40:28 +02:00
ret = SMB_VFS_NEXT_GET_REAL_FILENAME ( handle , & conv_fname , name ,
2011-06-20 09:52:21 +02:00
mem_ctx , found_name ) ;
2012-10-17 12:11:37 +02:00
DEBUG ( 10 , ( " NEXT_REAL_FILE_NAME returned %d \n " , ( int ) ret ) ) ;
2020-05-05 10:13:00 +02:00
if ( ret = = 0 ) {
return 0 ;
2017-01-23 10:20:13 -08:00
}
2020-05-05 10:13:00 +02:00
if ( errno ! = EOPNOTSUPP ) {
TALLOC_FREE ( conv ) ;
errno = EOPNOTSUPP ;
return - 1 ;
}
ret = get_real_filename_full_scan ( handle - > conn ,
conv ,
name ,
false ,
mem_ctx ,
found_name ) ;
if ( ret ! = 0 ) {
saved_errno = errno ;
DBG_DEBUG ( " Scan [%s] for [%s] failed \n " ,
conv , name ) ;
2017-01-23 10:20:13 -08:00
errno = saved_errno ;
2020-05-05 10:13:00 +02:00
return - 1 ;
2017-01-23 10:20:13 -08:00
}
2020-05-05 10:13:00 +02:00
DBG_DEBUG ( " Scan [%s] for [%s] returned [%s] \n " ,
conv , name , * found_name ) ;
TALLOC_FREE ( conv ) ;
return 0 ;
2011-06-20 09:52:21 +02:00
}
2015-11-02 09:08:53 +02:00
static const char * shadow_copy2_connectpath ( struct vfs_handle_struct * handle ,
2017-06-30 13:37:03 -07:00
const struct smb_filename * smb_fname_in )
2015-11-02 09:08:53 +02:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
2015-11-02 09:08:53 +02:00
char * stripped = NULL ;
char * tmp = NULL ;
2017-06-30 13:37:03 -07:00
const char * fname = smb_fname_in - > base_name ;
2017-06-30 11:32:59 -07:00
struct smb_filename smb_fname = { 0 } ;
struct smb_filename * result_fname = NULL ;
2015-11-02 09:08:53 +02:00
char * result = NULL ;
2016-08-24 14:42:23 +03:00
char * parent_dir = NULL ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
2015-11-02 09:08:53 +02:00
size_t rootpath_len = 0 ;
2017-01-20 12:09:08 -08:00
struct shadow_copy2_private * priv = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , priv , struct shadow_copy2_private ,
return NULL ) ;
2015-11-02 09:08:53 +02:00
DBG_DEBUG ( " Calc connect path for [%s] \n " , fname ) ;
2017-01-20 12:09:08 -08:00
if ( priv - > shadow_connectpath ! = NULL ) {
DBG_DEBUG ( " cached connect path is [%s] \n " ,
priv - > shadow_connectpath ) ;
return priv - > shadow_connectpath ;
}
2020-04-30 17:28:16 +02:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) , handle , smb_fname_in ,
2015-11-02 09:08:53 +02:00
& timestamp , & stripped ) ) {
goto done ;
}
if ( timestamp = = 0 ) {
2017-06-30 13:37:03 -07:00
return SMB_VFS_NEXT_CONNECTPATH ( handle , smb_fname_in ) ;
2015-11-02 09:08:53 +02:00
}
tmp = shadow_copy2_do_convert ( talloc_tos ( ) , handle , stripped , timestamp ,
& rootpath_len ) ;
if ( tmp = = NULL ) {
2016-08-24 14:42:23 +03:00
if ( errno ! = ENOENT ) {
goto done ;
}
/*
* If the converted path does not exist , and converting
* the parent yields something that does exist , then
* this path refers to something that has not been
* created yet , relative to the parent path .
* The snapshot finding is relative to the parent .
* ( usually snapshots are read / only but this is not
* necessarily true ) .
* This code also covers getting a wildcard in the
* last component , because this function is called
* prior to sanitizing the path , and in SMB1 we may
* get wildcards in path names .
*/
if ( ! parent_dirname ( talloc_tos ( ) , stripped , & parent_dir ,
NULL ) ) {
errno = ENOMEM ;
goto done ;
}
tmp = shadow_copy2_do_convert ( talloc_tos ( ) , handle , parent_dir ,
timestamp , & rootpath_len ) ;
if ( tmp = = NULL ) {
goto done ;
}
2015-11-02 09:08:53 +02:00
}
DBG_DEBUG ( " converted path is [%s] root path is [%.*s] \n " , tmp ,
( int ) rootpath_len , tmp ) ;
tmp [ rootpath_len ] = ' \0 ' ;
2017-06-30 11:32:59 -07:00
smb_fname = ( struct smb_filename ) { . base_name = tmp } ;
result_fname = SMB_VFS_NEXT_REALPATH ( handle , priv , & smb_fname ) ;
if ( result_fname = = NULL ) {
2015-11-02 09:08:53 +02:00
goto done ;
}
2017-01-23 10:06:44 -08:00
/*
2017-06-30 11:32:59 -07:00
* SMB_VFS_NEXT_REALPATH returns a talloc ' ed string .
2017-01-23 10:06:44 -08:00
* Don ' t leak memory .
*/
2017-06-30 11:32:59 -07:00
TALLOC_FREE ( priv - > shadow_realpath ) ;
priv - > shadow_realpath = result_fname ;
result = priv - > shadow_realpath - > base_name ;
2017-01-23 10:06:44 -08:00
2015-11-02 09:08:53 +02:00
DBG_DEBUG ( " connect path is [%s] \n " , result ) ;
done :
2017-01-23 10:20:13 -08:00
if ( result = = NULL ) {
saved_errno = errno ;
}
2015-11-02 09:08:53 +02:00
TALLOC_FREE ( tmp ) ;
TALLOC_FREE ( stripped ) ;
2016-08-24 14:42:23 +03:00
TALLOC_FREE ( parent_dir ) ;
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2015-11-02 09:08:53 +02:00
return result ;
}
2021-06-01 11:58:55 -07:00
static NTSTATUS shadow_copy2_parent_pathname ( vfs_handle_struct * handle ,
TALLOC_CTX * ctx ,
const struct smb_filename * smb_fname_in ,
struct smb_filename * * parent_dir_out ,
struct smb_filename * * atname_out )
{
time_t timestamp = 0 ;
char * stripped = NULL ;
char * converted_name = NULL ;
struct smb_filename * smb_fname = NULL ;
struct smb_filename * parent = NULL ;
struct smb_filename * atname = NULL ;
struct shadow_copy2_private * priv = NULL ;
bool ok = false ;
bool is_converted = false ;
NTSTATUS status = NT_STATUS_OK ;
TALLOC_CTX * frame = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle ,
priv ,
struct shadow_copy2_private ,
return NT_STATUS_INTERNAL_ERROR ) ;
frame = talloc_stackframe ( ) ;
smb_fname = cp_smb_filename ( frame , smb_fname_in ) ;
if ( smb_fname = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
/* First, call the default PARENT_PATHNAME. */
status = SMB_VFS_NEXT_PARENT_PATHNAME ( handle ,
frame ,
smb_fname ,
& parent ,
& atname ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
if ( parent - > twrp = = 0 ) {
/*
* Parent is not a snapshot path , return
* the regular result .
*/
status = NT_STATUS_OK ;
goto out ;
}
/* See if we can find a snapshot for the parent. */
ok = shadow_copy2_strip_snapshot_converted ( frame ,
handle ,
parent ,
& timestamp ,
& stripped ,
& is_converted ) ;
if ( ! ok ) {
status = map_nt_error_from_unix ( errno ) ;
goto fail ;
}
if ( is_converted ) {
/*
* Already found snapshot for parent so wipe
* out the twrp .
*/
parent - > twrp = 0 ;
goto out ;
}
converted_name = shadow_copy2_convert ( frame ,
handle ,
stripped ,
timestamp ) ;
if ( converted_name = = NULL ) {
/*
* Can ' t find snapshot for parent so wipe
* out the twrp .
*/
parent - > twrp = 0 ;
}
out :
* parent_dir_out = talloc_move ( ctx , & parent ) ;
if ( atname_out ! = NULL ) {
* atname_out = talloc_move ( * parent_dir_out , & atname ) ;
}
fail :
TALLOC_FREE ( frame ) ;
return status ;
}
2013-06-01 02:14:41 +02:00
static uint64_t shadow_copy2_disk_free ( vfs_handle_struct * handle ,
2017-05-23 10:40:47 -07:00
const struct smb_filename * smb_fname ,
uint64_t * bsize ,
uint64_t * dfree ,
uint64_t * dsize )
2013-06-01 02:14:41 +02:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
char * stripped = NULL ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
2017-05-23 10:40:47 -07:00
char * conv = NULL ;
struct smb_filename * conv_smb_fname = NULL ;
uint64_t ret = ( uint64_t ) - 1 ;
2013-06-01 02:14:41 +02:00
2017-05-23 10:40:47 -07:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2017-05-23 10:40:47 -07:00
& timestamp ,
& stripped ) ) {
return ( uint64_t ) - 1 ;
2013-06-01 02:14:41 +02:00
}
if ( timestamp = = 0 ) {
2017-05-23 10:40:47 -07:00
return SMB_VFS_NEXT_DISK_FREE ( handle , smb_fname ,
2013-06-01 02:14:41 +02:00
bsize , dfree , dsize ) ;
}
conv = shadow_copy2_convert ( talloc_tos ( ) , handle , stripped , timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( conv = = NULL ) {
2017-05-23 10:40:47 -07:00
return ( uint64_t ) - 1 ;
2013-06-01 02:14:41 +02:00
}
2017-05-23 10:40:47 -07:00
conv_smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
conv ,
NULL ,
NULL ,
2020-04-30 11:48:32 +02:00
0 ,
2017-05-23 10:40:47 -07:00
smb_fname - > flags ) ;
if ( conv_smb_fname = = NULL ) {
TALLOC_FREE ( conv ) ;
return ( uint64_t ) - 1 ;
}
ret = SMB_VFS_NEXT_DISK_FREE ( handle , conv_smb_fname ,
bsize , dfree , dsize ) ;
if ( ret = = ( uint64_t ) - 1 ) {
2017-01-23 10:20:13 -08:00
saved_errno = errno ;
}
2013-06-01 02:14:41 +02:00
TALLOC_FREE ( conv ) ;
2017-05-23 10:40:47 -07:00
TALLOC_FREE ( conv_smb_fname ) ;
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2013-06-01 02:14:41 +02:00
return ret ;
}
2017-06-01 11:45:25 -07:00
static int shadow_copy2_get_quota ( vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
enum SMB_QUOTA_TYPE qtype ,
unid_t id ,
SMB_DISK_QUOTA * dq )
2016-01-10 14:13:38 +02:00
{
2017-01-20 11:42:39 -08:00
time_t timestamp = 0 ;
char * stripped = NULL ;
2016-01-10 14:13:38 +02:00
int ret ;
2017-01-23 10:20:13 -08:00
int saved_errno = 0 ;
2016-01-10 14:13:38 +02:00
char * conv ;
2017-06-01 11:45:25 -07:00
struct smb_filename * conv_smb_fname = NULL ;
2016-01-10 14:13:38 +02:00
2017-06-01 11:45:25 -07:00
if ( ! shadow_copy2_strip_snapshot ( talloc_tos ( ) ,
handle ,
2020-04-30 17:28:16 +02:00
smb_fname ,
2017-06-01 11:45:25 -07:00
& timestamp ,
& stripped ) ) {
2016-01-10 14:13:38 +02:00
return - 1 ;
}
if ( timestamp = = 0 ) {
2017-06-01 11:45:25 -07:00
return SMB_VFS_NEXT_GET_QUOTA ( handle , smb_fname , qtype , id , dq ) ;
2016-01-10 14:13:38 +02:00
}
conv = shadow_copy2_convert ( talloc_tos ( ) , handle , stripped , timestamp ) ;
TALLOC_FREE ( stripped ) ;
if ( conv = = NULL ) {
return - 1 ;
}
2017-06-01 11:45:25 -07:00
conv_smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
conv ,
NULL ,
NULL ,
2020-04-30 11:48:32 +02:00
0 ,
2017-06-01 11:45:25 -07:00
smb_fname - > flags ) ;
if ( conv_smb_fname = = NULL ) {
TALLOC_FREE ( conv ) ;
return - 1 ;
}
ret = SMB_VFS_NEXT_GET_QUOTA ( handle , conv_smb_fname , qtype , id , dq ) ;
2016-01-10 14:13:38 +02:00
2017-01-23 10:20:13 -08:00
if ( ret = = - 1 ) {
saved_errno = errno ;
}
2016-01-10 14:13:38 +02:00
TALLOC_FREE ( conv ) ;
2017-06-01 11:45:25 -07:00
TALLOC_FREE ( conv_smb_fname ) ;
2017-01-23 10:20:13 -08:00
if ( saved_errno ! = 0 ) {
errno = saved_errno ;
}
2016-01-10 14:13:38 +02:00
return ret ;
}
2018-11-23 14:08:15 +01:00
static ssize_t shadow_copy2_pwrite ( vfs_handle_struct * handle ,
files_struct * fsp ,
const void * data ,
size_t n ,
off_t offset )
{
ssize_t nwritten ;
nwritten = SMB_VFS_NEXT_PWRITE ( handle , fsp , data , n , offset ) ;
if ( nwritten = = - 1 ) {
2020-04-02 17:28:32 +02:00
if ( errno = = EBADF & & fsp - > fsp_flags . can_write ) {
2018-11-23 14:08:15 +01:00
errno = EROFS ;
}
}
return nwritten ;
}
struct shadow_copy2_pwrite_state {
vfs_handle_struct * handle ;
files_struct * fsp ;
ssize_t ret ;
struct vfs_aio_state vfs_aio_state ;
} ;
static void shadow_copy2_pwrite_done ( struct tevent_req * subreq ) ;
static struct tevent_req * shadow_copy2_pwrite_send (
struct vfs_handle_struct * handle , TALLOC_CTX * mem_ctx ,
struct tevent_context * ev , struct files_struct * fsp ,
const void * data , size_t n , off_t offset )
{
struct tevent_req * req = NULL , * subreq = NULL ;
struct shadow_copy2_pwrite_state * state = NULL ;
req = tevent_req_create ( mem_ctx , & state ,
struct shadow_copy2_pwrite_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > handle = handle ;
state - > fsp = fsp ;
subreq = SMB_VFS_NEXT_PWRITE_SEND ( state ,
ev ,
handle ,
fsp ,
data ,
n ,
offset ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , shadow_copy2_pwrite_done , req ) ;
return req ;
}
static void shadow_copy2_pwrite_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct shadow_copy2_pwrite_state * state = tevent_req_data (
req , struct shadow_copy2_pwrite_state ) ;
state - > ret = SMB_VFS_PWRITE_RECV ( subreq , & state - > vfs_aio_state ) ;
TALLOC_FREE ( subreq ) ;
if ( state - > ret = = - 1 ) {
tevent_req_error ( req , state - > vfs_aio_state . error ) ;
return ;
}
tevent_req_done ( req ) ;
}
static ssize_t shadow_copy2_pwrite_recv ( struct tevent_req * req ,
struct vfs_aio_state * vfs_aio_state )
{
struct shadow_copy2_pwrite_state * state = tevent_req_data (
req , struct shadow_copy2_pwrite_state ) ;
if ( tevent_req_is_unix_error ( req , & vfs_aio_state - > error ) ) {
if ( ( vfs_aio_state - > error = = EBADF ) & &
2020-04-02 17:28:32 +02:00
state - > fsp - > fsp_flags . can_write )
2018-11-23 14:08:15 +01:00
{
vfs_aio_state - > error = EROFS ;
errno = EROFS ;
}
return - 1 ;
}
* vfs_aio_state = state - > vfs_aio_state ;
return state - > ret ;
}
2013-05-24 01:35:44 +02:00
static int shadow_copy2_connect ( struct vfs_handle_struct * handle ,
const char * service , const char * user )
{
struct shadow_copy2_config * config ;
2016-07-11 11:27:37 +00:00
struct shadow_copy2_private * priv ;
2013-05-24 01:35:44 +02:00
int ret ;
const char * snapdir ;
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
const char * snapprefix = NULL ;
const char * delimiter ;
2013-05-24 01:35:44 +02:00
const char * gmt_format ;
const char * sort_order ;
2015-11-03 09:15:12 +02:00
const char * basedir = NULL ;
2015-11-03 10:57:13 +02:00
const char * snapsharepath = NULL ;
2013-05-31 16:36:33 +02:00
const char * mount_point ;
2013-05-24 01:35:44 +02:00
DEBUG ( 10 , ( __location__ " : cnum[%u], connectpath[%s] \n " ,
( unsigned ) handle - > conn - > cnum ,
handle - > conn - > connectpath ) ) ;
ret = SMB_VFS_NEXT_CONNECT ( handle , service , user ) ;
if ( ret < 0 ) {
return ret ;
}
2016-07-11 11:27:37 +00:00
priv = talloc_zero ( handle - > conn , struct shadow_copy2_private ) ;
if ( priv = = NULL ) {
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
DBG_ERR ( " talloc_zero() failed \n " ) ;
errno = ENOMEM ;
return - 1 ;
}
priv - > snaps = talloc_zero ( priv , struct shadow_copy2_snaplist_info ) ;
if ( priv - > snaps = = NULL ) {
DBG_ERR ( " talloc_zero() failed \n " ) ;
2016-07-11 11:27:37 +00:00
errno = ENOMEM ;
return - 1 ;
}
config = talloc_zero ( priv , struct shadow_copy2_config ) ;
2013-05-24 01:35:44 +02:00
if ( config = = NULL ) {
DEBUG ( 0 , ( " talloc_zero() failed \n " ) ) ;
errno = ENOMEM ;
return - 1 ;
}
2016-07-11 11:27:37 +00:00
priv - > config = config ;
2013-05-24 01:35:44 +02:00
gmt_format = lp_parm_const_string ( SNUM ( handle - > conn ) ,
" shadow " , " format " ,
GMT_FORMAT ) ;
config - > gmt_format = talloc_strdup ( config , gmt_format ) ;
if ( config - > gmt_format = = NULL ) {
DEBUG ( 0 , ( " talloc_strdup() failed \n " ) ) ;
errno = ENOMEM ;
return - 1 ;
}
2017-01-26 10:24:52 -08:00
/* config->gmt_format must not contain a path separator. */
if ( strchr ( config - > gmt_format , ' / ' ) ! = NULL ) {
DEBUG ( 0 , ( " shadow:format %s must not contain a / "
" character. Unable to initialize module. \n " ,
config - > gmt_format ) ) ;
errno = EINVAL ;
return - 1 ;
}
2013-05-24 01:35:44 +02:00
config - > use_sscanf = lp_parm_bool ( SNUM ( handle - > conn ) ,
" shadow " , " sscanf " , false ) ;
config - > use_localtime = lp_parm_bool ( SNUM ( handle - > conn ) ,
" shadow " , " localtime " ,
false ) ;
snapdir = lp_parm_const_string ( SNUM ( handle - > conn ) ,
" shadow " , " snapdir " ,
" .snapshots " ) ;
config - > snapdir = talloc_strdup ( config , snapdir ) ;
if ( config - > snapdir = = NULL ) {
DEBUG ( 0 , ( " talloc_strdup() failed \n " ) ) ;
errno = ENOMEM ;
return - 1 ;
}
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
snapprefix = lp_parm_const_string ( SNUM ( handle - > conn ) ,
" shadow " , " snapprefix " ,
NULL ) ;
if ( snapprefix ! = NULL ) {
priv - > snaps - > regex = talloc_zero ( priv - > snaps , regex_t ) ;
if ( priv - > snaps - > regex = = NULL ) {
DBG_ERR ( " talloc_zero() failed \n " ) ;
errno = ENOMEM ;
return - 1 ;
}
/* pre-compute regex rule for matching pattern later */
ret = regcomp ( priv - > snaps - > regex , snapprefix , 0 ) ;
if ( ret ) {
DBG_ERR ( " Failed to create regex object \n " ) ;
return - 1 ;
}
}
delimiter = lp_parm_const_string ( SNUM ( handle - > conn ) ,
" shadow " , " delimiter " ,
" _GMT " ) ;
if ( delimiter ! = NULL ) {
priv - > config - > delimiter = talloc_strdup ( priv - > config , delimiter ) ;
if ( priv - > config - > delimiter = = NULL ) {
DBG_ERR ( " talloc_strdup() failed \n " ) ;
errno = ENOMEM ;
return - 1 ;
}
}
2013-05-24 01:35:44 +02:00
config - > snapdirseverywhere = lp_parm_bool ( SNUM ( handle - > conn ) ,
" shadow " ,
" snapdirseverywhere " ,
false ) ;
config - > crossmountpoints = lp_parm_bool ( SNUM ( handle - > conn ) ,
" shadow " , " crossmountpoints " ,
false ) ;
2015-11-03 10:42:00 +02:00
if ( config - > crossmountpoints & & ! config - > snapdirseverywhere ) {
DBG_WARNING ( " Warning: 'crossmountpoints' depends on "
" 'snapdirseverywhere'. Disabling crossmountpoints. \n " ) ;
}
2013-05-24 01:35:44 +02:00
config - > fixinodes = lp_parm_bool ( SNUM ( handle - > conn ) ,
" shadow " , " fixinodes " ,
false ) ;
sort_order = lp_parm_const_string ( SNUM ( handle - > conn ) ,
" shadow " , " sort " , " desc " ) ;
config - > sort_order = talloc_strdup ( config , sort_order ) ;
if ( config - > sort_order = = NULL ) {
DEBUG ( 0 , ( " talloc_strdup() failed \n " ) ) ;
errno = ENOMEM ;
return - 1 ;
}
2013-05-31 16:36:33 +02:00
mount_point = lp_parm_const_string ( SNUM ( handle - > conn ) ,
" shadow " , " mountpoint " , NULL ) ;
if ( mount_point ! = NULL ) {
if ( mount_point [ 0 ] ! = ' / ' ) {
DEBUG ( 1 , ( __location__ " Warning: 'mountpoint' is "
" relative ('%s'), but it has to be an "
" absolute path. Ignoring provided value. \n " ,
mount_point ) ) ;
mount_point = NULL ;
} else {
char * p ;
p = strstr ( handle - > conn - > connectpath , mount_point ) ;
if ( p ! = handle - > conn - > connectpath ) {
2016-01-13 01:37:24 +01:00
DBG_WARNING ( " Warning: the share root (%s) is "
" not a subdirectory of the "
" specified mountpoint (%s). "
" Ignoring provided value. \n " ,
handle - > conn - > connectpath ,
mount_point ) ;
2013-05-31 16:36:33 +02:00
mount_point = NULL ;
}
}
}
if ( mount_point ! = NULL ) {
config - > mount_point = talloc_strdup ( config , mount_point ) ;
if ( config - > mount_point = = NULL ) {
DEBUG ( 0 , ( __location__ " talloc_strdup() failed \n " ) ) ;
return - 1 ;
}
} else {
config - > mount_point = shadow_copy2_find_mount_point ( config ,
handle ) ;
if ( config - > mount_point = = NULL ) {
2015-06-25 18:43:50 +02:00
DBG_WARNING ( " shadow_copy2_find_mount_point "
" of the share root '%s' failed: %s \n " ,
handle - > conn - > connectpath , strerror ( errno ) ) ;
2013-05-31 16:36:33 +02:00
return - 1 ;
}
2013-05-30 17:26:44 +02:00
}
basedir = lp_parm_const_string ( SNUM ( handle - > conn ) ,
" shadow " , " basedir " , NULL ) ;
if ( basedir ! = NULL ) {
if ( basedir [ 0 ] ! = ' / ' ) {
DEBUG ( 1 , ( __location__ " Warning: 'basedir' is "
" relative ('%s'), but it has to be an "
" absolute path. Disabling basedir. \n " ,
basedir ) ) ;
2015-11-03 09:15:12 +02:00
basedir = NULL ;
2013-05-30 17:26:44 +02:00
} else {
char * p ;
p = strstr ( basedir , config - > mount_point ) ;
if ( p ! = basedir ) {
DEBUG ( 1 , ( " Warning: basedir (%s) is not a "
" subdirectory of the share root's "
" mount point (%s). "
" Disabling basedir \n " ,
basedir , config - > mount_point ) ) ;
2015-11-03 09:15:12 +02:00
basedir = NULL ;
2013-05-30 17:26:44 +02:00
}
}
}
2015-11-03 09:15:12 +02:00
if ( config - > snapdirseverywhere & & basedir ! = NULL ) {
2013-05-30 17:26:44 +02:00
DEBUG ( 1 , ( __location__ " Warning: 'basedir' is incompatible "
" with 'snapdirseverywhere'. Disabling basedir. \n " ) ) ;
2015-11-03 09:15:12 +02:00
basedir = NULL ;
2013-05-30 17:26:44 +02:00
}
2015-11-03 10:57:13 +02:00
snapsharepath = lp_parm_const_string ( SNUM ( handle - > conn ) , " shadow " ,
" snapsharepath " , NULL ) ;
if ( snapsharepath ! = NULL ) {
if ( snapsharepath [ 0 ] = = ' / ' ) {
DBG_WARNING ( " Warning: 'snapsharepath' is "
" absolute ('%s'), but it has to be a "
" relative path. Disabling snapsharepath. \n " ,
snapsharepath ) ;
snapsharepath = NULL ;
}
if ( config - > snapdirseverywhere & & snapsharepath ! = NULL ) {
DBG_WARNING ( " Warning: 'snapsharepath' is incompatible "
" with 'snapdirseverywhere'. Disabling "
" snapsharepath. \n " ) ;
snapsharepath = NULL ;
}
}
if ( basedir ! = NULL & & snapsharepath ! = NULL ) {
DBG_WARNING ( " Warning: 'snapsharepath' is incompatible with "
" 'basedir'. Disabling snapsharepath \n " ) ;
snapsharepath = NULL ;
}
if ( snapsharepath ! = NULL ) {
config - > rel_connectpath = talloc_strdup ( config , snapsharepath ) ;
if ( config - > rel_connectpath = = NULL ) {
DBG_ERR ( " talloc_strdup() failed \n " ) ;
errno = ENOMEM ;
return - 1 ;
}
}
2015-11-03 09:15:12 +02:00
if ( basedir = = NULL ) {
basedir = config - > mount_point ;
2013-05-30 17:26:44 +02:00
}
2015-11-03 10:57:13 +02:00
if ( config - > rel_connectpath = = NULL & &
2017-01-20 11:48:40 -08:00
strlen ( basedir ) < strlen ( handle - > conn - > connectpath ) ) {
2013-10-04 00:04:06 +02:00
config - > rel_connectpath = talloc_strdup ( config ,
2015-11-03 09:15:12 +02:00
handle - > conn - > connectpath + strlen ( basedir ) ) ;
2013-10-04 00:04:06 +02:00
if ( config - > rel_connectpath = = NULL ) {
DEBUG ( 0 , ( " talloc_strdup() failed \n " ) ) ;
errno = ENOMEM ;
return - 1 ;
}
}
2013-05-29 17:10:51 +02:00
if ( config - > snapdir [ 0 ] = = ' / ' ) {
config - > snapdir_absolute = true ;
2013-10-04 00:04:06 +02:00
2013-05-29 17:10:51 +02:00
if ( config - > snapdirseverywhere = = true ) {
DEBUG ( 1 , ( __location__ " Warning: An absolute snapdir "
" is incompatible with 'snapdirseverywhere', "
" setting 'snapdirseverywhere' to false. \n " ) ) ;
config - > snapdirseverywhere = false ;
}
2013-05-30 13:19:50 +02:00
if ( config - > crossmountpoints = = true ) {
DEBUG ( 1 , ( __location__ " Warning: 'crossmountpoints' "
" is not supported with an absolute snapdir. "
" Disabling it. \n " ) ) ;
config - > crossmountpoints = false ;
}
2013-10-04 00:07:15 +02:00
config - > snapshot_basepath = config - > snapdir ;
} else {
config - > snapshot_basepath = talloc_asprintf ( config , " %s/%s " ,
config - > mount_point , config - > snapdir ) ;
if ( config - > snapshot_basepath = = NULL ) {
DEBUG ( 0 , ( " talloc_asprintf() failed \n " ) ) ;
errno = ENOMEM ;
return - 1 ;
}
2013-05-29 17:10:51 +02:00
}
2017-01-20 11:45:54 -08:00
trim_string ( config - > mount_point , NULL , " / " ) ;
trim_string ( config - > rel_connectpath , " / " , " / " ) ;
trim_string ( config - > snapdir , NULL , " / " ) ;
trim_string ( config - > snapshot_basepath , NULL , " / " ) ;
2013-05-31 17:17:27 +02:00
DEBUG ( 10 , ( " shadow_copy2_connect: configuration: \n "
" share root: '%s' \n "
" mountpoint: '%s' \n "
" rel share root: '%s' \n "
" snapdir: '%s' \n "
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
" snapprefix: '%s' \n "
" delimiter: '%s' \n "
2013-05-31 17:17:27 +02:00
" snapshot base path: '%s' \n "
" format: '%s' \n "
" use sscanf: %s \n "
" snapdirs everywhere: %s \n "
" cross mountpoints: %s \n "
" fix inodes: %s \n "
" sort order: %s \n "
" " ,
handle - > conn - > connectpath ,
config - > mount_point ,
config - > rel_connectpath ,
config - > snapdir ,
shadow_copy2: allow configurable prefix for snapshot name
With growing number of snapshots file-systems need some mechanism
to differentiate one set of snapshots from other, e.g. monthly, weekly,
manual, special events, etc. Therefore these file-systems provide
different ways to tag snapshots, e.g. provide a configurable way to
name snapshots, which is not just based on time. With only shadow:format
it is very difficult to filter these snapshots.
As part of this change added two new options, shadow:snapprefix and
shadow:delimiter, in shadow_copy2 config. This option will accept regular
expression (BRE) as input. With this optional parameter, one can specify a
variable prefix component for names of the snapshot directories in the
file-system. If this parameter is set, together with the shadow:format and
shadow:delimiter parameters it determines the possible names of snapshot
directories in the file-system.
e.g.
shadow:snapprefix = [a-z]*[0-9]
When this option is provided then shadow:format option should always
start with <delimiter> string. This delimiter is configurable via a new option,
i.e. shadow:delimiter. Default value for this is "_GMT",
e.g. _GMT-%Y.%m.%d-%H.%M.%S
Signed-off-by: Rajesh Joseph <rjoseph@redhat.com>
Reviewed-by: Uri Simchoni <uri@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-07-13 18:59:18 +00:00
snapprefix ,
config - > delimiter ,
2013-05-31 17:17:27 +02:00
config - > snapshot_basepath ,
config - > gmt_format ,
config - > use_sscanf ? " yes " : " no " ,
config - > snapdirseverywhere ? " yes " : " no " ,
config - > crossmountpoints ? " yes " : " no " ,
config - > fixinodes ? " yes " : " no " ,
config - > sort_order
) ) ;
2016-07-11 11:27:37 +00:00
SMB_VFS_HANDLE_SET_DATA ( handle , priv ,
NULL , struct shadow_copy2_private ,
2013-05-24 01:35:44 +02:00
return - 1 ) ;
return 0 ;
}
2011-06-20 09:52:21 +02:00
static struct vfs_fn_pointers vfs_shadow_copy2_fns = {
2013-05-24 01:35:44 +02:00
. connect_fn = shadow_copy2_connect ,
2013-06-01 02:14:41 +02:00
. disk_free_fn = shadow_copy2_disk_free ,
2016-01-10 14:13:38 +02:00
. get_quota_fn = shadow_copy2_get_quota ,
2020-01-09 13:06:36 -08:00
. create_dfs_pathat_fn = shadow_copy2_create_dfs_pathat ,
2020-01-28 12:12:09 -08:00
. read_dfs_pathat_fn = shadow_copy2_read_dfs_pathat ,
2019-08-09 15:13:24 -07:00
. renameat_fn = shadow_copy2_renameat ,
2019-08-16 15:59:27 -07:00
. linkat_fn = shadow_copy2_linkat ,
2019-08-30 13:53:23 -07:00
. symlinkat_fn = shadow_copy2_symlinkat ,
2011-12-03 20:45:04 -08:00
. stat_fn = shadow_copy2_stat ,
. lstat_fn = shadow_copy2_lstat ,
. fstat_fn = shadow_copy2_fstat ,
2020-05-20 23:03:05 +02:00
. openat_fn = shadow_copy2_openat ,
2019-09-12 13:19:39 -07:00
. unlinkat_fn = shadow_copy2_unlinkat ,
2021-04-09 15:54:44 +02:00
. fchmod_fn = shadow_copy2_fchmod ,
2011-12-03 20:45:04 -08:00
. chdir_fn = shadow_copy2_chdir ,
2021-04-13 13:48:36 +02:00
. fntimes_fn = shadow_copy2_fntimes ,
2019-08-22 14:26:47 -07:00
. readlinkat_fn = shadow_copy2_readlinkat ,
2019-08-20 16:57:24 -07:00
. mknodat_fn = shadow_copy2_mknodat ,
2011-12-03 20:45:04 -08:00
. realpath_fn = shadow_copy2_realpath ,
2020-04-13 14:05:03 -07:00
. get_nt_acl_at_fn = shadow_copy2_get_nt_acl_at ,
2011-12-03 20:45:04 -08:00
. get_shadow_copy_data_fn = shadow_copy2_get_shadow_copy_data ,
2019-09-05 10:32:18 -07:00
. mkdirat_fn = shadow_copy2_mkdirat ,
2011-12-03 20:45:04 -08:00
. getxattr_fn = shadow_copy2_getxattr ,
2018-03-13 08:14:53 +01:00
. getxattrat_send_fn = vfs_not_implemented_getxattrat_send ,
. getxattrat_recv_fn = vfs_not_implemented_getxattrat_recv ,
2021-02-19 15:03:01 +00:00
. fsetxattr_fn = shadow_copy2_fsetxattr ,
2011-12-03 20:45:04 -08:00
. chflags_fn = shadow_copy2_chflags ,
. get_real_filename_fn = shadow_copy2_get_real_filename ,
2018-11-23 14:08:15 +01:00
. pwrite_fn = shadow_copy2_pwrite ,
. pwrite_send_fn = shadow_copy2_pwrite_send ,
. pwrite_recv_fn = shadow_copy2_pwrite_recv ,
2015-11-02 09:08:53 +02:00
. connectpath_fn = shadow_copy2_connectpath ,
2021-06-01 11:58:55 -07:00
. parent_pathname_fn = shadow_copy2_parent_pathname ,
2011-06-20 09:52:21 +02:00
} ;
2017-12-15 15:32:12 -07:00
static_decl_vfs ;
2017-04-20 12:24:43 -07:00
NTSTATUS vfs_shadow_copy2_init ( TALLOC_CTX * ctx )
2011-06-20 09:52:21 +02:00
{
return smb_register_vfs ( SMB_VFS_INTERFACE_VERSION ,
" shadow_copy2 " , & vfs_shadow_copy2_fns ) ;
}