From a51da2742a4c03fac570a99494722afa51507a62 Mon Sep 17 00:00:00 2001 From: Jeremy Allison Date: Thu, 17 Sep 2015 15:54:40 -0700 Subject: [PATCH 1/4] s3: smbclient: Move cmd_setmode out of clitar.c and back into client.c setmode attribute is a valid smbclient command even if libarchive isn't on the system and tarmode isn't compiled in. Signed-off-by: Jeremy Allison Reviewed-by: Ralph Boehme (cherry picked from commit a47012d5429044c9a3616718bac21360f281aa81) --- source3/client/client.c | 118 ++++++++++++++++++++++++++++++++++++++ source3/client/client_proto.h | 6 ++ source3/client/clitar.c | 129 ------------------------------------------ 3 files changed, 124 insertions(+), 129 deletions(-) diff --git a/source3/client/client.c b/source3/client/client.c index f49f3ff..034b48a 100644 --- a/source3/client/client.c +++ b/source3/client/client.c @@ -4656,6 +4656,124 @@ static int cmd_show_connect( void ) return 0; } +/** + * set_remote_attr - set DOS attributes of a remote file + * @filename: path to the file name + * @new_attr: attribute bit mask to use + * @mode: one of ATTR_SET or ATTR_UNSET + * + * Update the file attributes with the one provided. + */ +int set_remote_attr(const char *filename, uint16_t new_attr, int mode) +{ + extern struct cli_state *cli; + uint16_t old_attr; + NTSTATUS status; + + status = cli_getatr(cli, filename, &old_attr, NULL, NULL); + if (!NT_STATUS_IS_OK(status)) { + d_printf("cli_getatr failed: %s\n", nt_errstr(status)); + return 1; + } + + if (mode == ATTR_SET) { + new_attr |= old_attr; + } else { + new_attr = old_attr & ~new_attr; + } + + status = cli_setatr(cli, filename, new_attr, 0); + if (!NT_STATUS_IS_OK(status)) { + d_printf("cli_setatr failed: %s\n", nt_errstr(status)); + return 1; + } + + return 0; +} + +/** + * cmd_setmode - interactive command to set DOS attributes + * + * Read a filename and mode from the client command line and update + * the file DOS attributes. + */ +int cmd_setmode(void) +{ + const extern char *cmd_ptr; + char *buf; + char *fname = NULL; + uint16_t attr[2] = {0}; + int mode = ATTR_SET; + int err = 0; + bool ok; + TALLOC_CTX *ctx = talloc_new(NULL); + if (ctx == NULL) { + return 1; + } + + ok = next_token_talloc(ctx, &cmd_ptr, &buf, NULL); + if (!ok) { + d_printf("setmode <[+|-]rsha>\n"); + err = 1; + goto out; + } + + fname = talloc_asprintf(ctx, + "%s%s", + client_get_cur_dir(), + buf); + if (fname == NULL) { + err = 1; + goto out; + } + + while (next_token_talloc(ctx, &cmd_ptr, &buf, NULL)) { + const char *s = buf; + + while (*s) { + switch (*s++) { + case '+': + mode = ATTR_SET; + break; + case '-': + mode = ATTR_UNSET; + break; + case 'r': + attr[mode] |= FILE_ATTRIBUTE_READONLY; + break; + case 'h': + attr[mode] |= FILE_ATTRIBUTE_HIDDEN; + break; + case 's': + attr[mode] |= FILE_ATTRIBUTE_SYSTEM; + break; + case 'a': + attr[mode] |= FILE_ATTRIBUTE_ARCHIVE; + break; + default: + d_printf("setmode \n"); + err = 1; + goto out; + } + } + } + + if (attr[ATTR_SET] == 0 && attr[ATTR_UNSET] == 0) { + d_printf("setmode <[+|-]rsha>\n"); + err = 1; + goto out; + } + + DEBUG(2, ("perm set %d %d\n", attr[ATTR_SET], attr[ATTR_UNSET])); + + /* ignore return value: server might not store DOS attributes */ + set_remote_attr(fname, attr[ATTR_SET], ATTR_SET); + set_remote_attr(fname, attr[ATTR_UNSET], ATTR_UNSET); +out: + talloc_free(ctx); + return err; +} + /**************************************************************************** iosize command ***************************************************************************/ diff --git a/source3/client/client_proto.h b/source3/client/client_proto.h index 86f1d18..d3d4036 100644 --- a/source3/client/client_proto.h +++ b/source3/client/client_proto.h @@ -26,6 +26,11 @@ struct cli_state; struct file_info; +enum { + ATTR_UNSET, + ATTR_SET, +}; + /* The following definitions come from client/client.c */ const char *client_get_cur_dir(void); @@ -36,6 +41,7 @@ NTSTATUS do_list(const char *mask, const char *dir), bool rec, bool dirs); +int set_remote_attr(const char *filename, uint16_t new_attr, int mode); int cmd_iosize(void); /* The following definitions come from client/dnsbrowse.c */ diff --git a/source3/client/clitar.c b/source3/client/clitar.c index 5561845..7eb3fa0 100644 --- a/source3/client/clitar.c +++ b/source3/client/clitar.c @@ -121,11 +121,6 @@ enum tar_selection { TAR_EXCLUDE, /* X flag */ }; -enum { - ATTR_UNSET, - ATTR_SET, -}; - struct tar { TALLOC_CTX *talloc_ctx; @@ -216,7 +211,6 @@ static int make_remote_path(const char *full_path); static int max_token (const char *str); static NTSTATUS is_subpath(const char *sub, const char *full, bool *_subpath_match); -static int set_remote_attr(const char *filename, uint16_t new_attr, int mode); /** * tar_get_ctx - retrieve global tar context handle @@ -383,88 +377,6 @@ out: return err; } -/** - * cmd_setmode - interactive command to set DOS attributes - * - * Read a filename and mode from the client command line and update - * the file DOS attributes. - */ -int cmd_setmode(void) -{ - const extern char *cmd_ptr; - char *buf; - char *fname = NULL; - uint16_t attr[2] = {0}; - int mode = ATTR_SET; - int err = 0; - bool ok; - TALLOC_CTX *ctx = talloc_new(NULL); - if (ctx == NULL) { - return 1; - } - - ok = next_token_talloc(ctx, &cmd_ptr, &buf, NULL); - if (!ok) { - DBG(0, ("setmode <[+|-]rsha>\n")); - err = 1; - goto out; - } - - fname = talloc_asprintf(ctx, - "%s%s", - client_get_cur_dir(), - buf); - if (fname == NULL) { - err = 1; - goto out; - } - - while (next_token_talloc(ctx, &cmd_ptr, &buf, NULL)) { - const char *s = buf; - - while (*s) { - switch (*s++) { - case '+': - mode = ATTR_SET; - break; - case '-': - mode = ATTR_UNSET; - break; - case 'r': - attr[mode] |= FILE_ATTRIBUTE_READONLY; - break; - case 'h': - attr[mode] |= FILE_ATTRIBUTE_HIDDEN; - break; - case 's': - attr[mode] |= FILE_ATTRIBUTE_SYSTEM; - break; - case 'a': - attr[mode] |= FILE_ATTRIBUTE_ARCHIVE; - break; - default: - DBG(0, ("setmode \n")); - err = 1; - goto out; - } - } - } - - if (attr[ATTR_SET] == 0 && attr[ATTR_UNSET] == 0) { - DBG(0, ("setmode <[+|-]rsha>\n")); - err = 1; - goto out; - } - - DBG(2, ("perm set %d %d\n", attr[ATTR_SET], attr[ATTR_UNSET])); - - /* ignore return value: server might not store DOS attributes */ - set_remote_attr(fname, attr[ATTR_SET], ATTR_SET); - set_remote_attr(fname, attr[ATTR_UNSET], ATTR_UNSET); -out: - talloc_free(ctx); - return err; -} /** * tar_parse_args - parse and set tar command line arguments @@ -1631,41 +1543,6 @@ out: return status; } -/** - * set_remote_attr - set DOS attributes of a remote file - * @filename: path to the file name - * @new_attr: attribute bit mask to use - * @mode: one of ATTR_SET or ATTR_UNSET - * - * Update the file attributes with the one provided. - */ -static int set_remote_attr(const char *filename, uint16_t new_attr, int mode) -{ - extern struct cli_state *cli; - uint16_t old_attr; - NTSTATUS status; - - status = cli_getatr(cli, filename, &old_attr, NULL, NULL); - if (!NT_STATUS_IS_OK(status)) { - DBG(0, ("cli_getatr failed: %s\n", nt_errstr(status))); - return 1; - } - - if (mode == ATTR_SET) { - new_attr |= old_attr; - } else { - new_attr = old_attr & ~new_attr; - } - - status = cli_setatr(cli, filename, new_attr, 0); - if (!NT_STATUS_IS_OK(status)) { - DBG(1, ("cli_setatr failed: %s\n", nt_errstr(status))); - return 1; - } - - return 0; -} - /** * make_remote_path - recursively make remote dirs @@ -1932,12 +1809,6 @@ int cmd_tarmode(void) return 1; } -int cmd_setmode(void) -{ - NOT_IMPLEMENTED; - return 1; -} - int cmd_tar(void) { NOT_IMPLEMENTED; -- 2.6.0.rc0.131.gf624c3d From 28b19299dbeee8a2b3c29413c9c9d7f76e2b0c80 Mon Sep 17 00:00:00 2001 From: Jeremy Allison Date: Wed, 16 Sep 2015 12:42:46 -0700 Subject: [PATCH 2/4] s3: smbd: Remove unused parameter from build_stream_path(). Signed-off-by: Jeremy Allison Reviewed-by: Ralph Boehme (cherry picked from commit 94e7e707783036b57babc73d320d2a3d8c0648d6) --- source3/smbd/filename.c | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/source3/smbd/filename.c b/source3/smbd/filename.c index 6899d2a..e2c3fe4 100644 --- a/source3/smbd/filename.c +++ b/source3/smbd/filename.c @@ -32,7 +32,6 @@ static NTSTATUS build_stream_path(TALLOC_CTX *mem_ctx, connection_struct *conn, - const char *orig_path, struct smb_filename *smb_fname); /**************************************************************************** @@ -981,7 +980,7 @@ NTSTATUS unix_convert(TALLOC_CTX *ctx, smb_fname->stream_name = stream; /* Check path now that the base_name has been converted. */ - status = build_stream_path(ctx, conn, orig_path, smb_fname); + status = build_stream_path(ctx, conn, smb_fname); if (!NT_STATUS_IS_OK(status)) { goto fail; } @@ -1233,7 +1232,6 @@ int get_real_filename(connection_struct *conn, const char *path, static NTSTATUS build_stream_path(TALLOC_CTX *mem_ctx, connection_struct *conn, - const char *orig_path, struct smb_filename *smb_fname) { NTSTATUS status; -- 2.6.0.rc0.131.gf624c3d From 3aa2b2860235a681fe1b0689fc7c488f56f8041b Mon Sep 17 00:00:00 2001 From: Jeremy Allison Date: Wed, 16 Sep 2015 12:03:34 -0700 Subject: [PATCH 3/4] s3: smbd: Fix opening/creating :stream files on the root share directory. BUG: https://bugzilla.samba.org/show_bug.cgi?id=11522 Signed-off-by: Jeremy Allison Reviewed-by: Ralph Boehme (cherry picked from commit 74fd4f93efe92516fc507edf71a588660782879e) --- source3/smbd/filename.c | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/source3/smbd/filename.c b/source3/smbd/filename.c index e2c3fe4..3ed60e2 100644 --- a/source3/smbd/filename.c +++ b/source3/smbd/filename.c @@ -370,6 +370,29 @@ NTSTATUS unix_convert(TALLOC_CTX *ctx, */ *stream = '\0'; stream = tmp; + + if (smb_fname->base_name[0] == '\0') { + /* + * orig_name was just a stream name. + * This is a stream on the root of + * the share. Replace base_name with + * a "." + */ + smb_fname->base_name = + talloc_strdup(smb_fname, "."); + if (smb_fname->base_name == NULL) { + status = NT_STATUS_NO_MEMORY; + goto err; + } + if (SMB_VFS_STAT(conn, smb_fname) != 0) { + status = map_nt_error_from_unix(errno); + goto err; + } + DEBUG(5, ("conversion finished %s -> %s\n", + orig_path, + smb_fname->base_name)); + goto done; + } } } -- 2.6.0.rc0.131.gf624c3d From 856bab4ede0887bacdcae82c575cb897f0ae7bad Mon Sep 17 00:00:00 2001 From: Jeremy Allison Date: Wed, 16 Sep 2015 16:12:15 -0700 Subject: [PATCH 4/4] s3: tests: smbclient test to ensure we can create and see a :foobar stream on the top level directory in a share. MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Regression test for: BUG: https://bugzilla.samba.org/show_bug.cgi?id=11522 Remember to remove the ARCHIVE attribute from the toplevel share when done (can only be done over SMB2+). Signed-off-by: Jeremy Allison Reviewed-by: Ralph Boehme Autobuild-User(master): Ralph Böhme Autobuild-Date(master): Fri Sep 18 11:00:44 CEST 2015 on sn-devel-104 (cherry picked from commit 6ce3643e45bac6660ae69123738c4b39d7bc1864) --- source3/script/tests/test_smbclient_s3.sh | 36 +++++++++++++++++++++++++++++++ 1 file changed, 36 insertions(+) diff --git a/source3/script/tests/test_smbclient_s3.sh b/source3/script/tests/test_smbclient_s3.sh index 69c7452..c073b43 100755 --- a/source3/script/tests/test_smbclient_s3.sh +++ b/source3/script/tests/test_smbclient_s3.sh @@ -971,6 +971,38 @@ EOF fi } +# Test creating a stream on the root of the share directory filname - :foobar +test_toplevel_stream() +{ + tmpfile=$PREFIX/smbclient_interactive_prompt_commands + cat > $tmpfile <