torture/smb2: delayed timestamp update test: single write
authorRalph Boehme <slow@samba.org>
Sat, 14 Mar 2020 15:43:48 +0000 (16:43 +0100)
committerKarolin Seeger <kseeger@samba.org>
Tue, 7 Apr 2020 08:12:36 +0000 (08:12 +0000)
Verify close only updates write-time when a delayed update is actually pending.

This scenario is not covered by basic.delaywrite.

BUG: https://bugzilla.samba.org/show_bug.cgi?id=14320

Signed-off-by: Ralph Boehme <slow@samba.org>
Reviewed-by: Jeremy Allison <jra@samba.org>
(cherry picked from commit 58fa7b4fd7b53d3100459a0c9c7ef4ca7481b58a)

selftest/knownfail.d/samba3.smb2.timestamps [new file with mode: 0644]
source4/torture/smb2/timestamps.c

diff --git a/selftest/knownfail.d/samba3.smb2.timestamps b/selftest/knownfail.d/samba3.smb2.timestamps
new file mode 100644 (file)
index 0000000..dd4aeb5
--- /dev/null
@@ -0,0 +1 @@
+^samba3.smb2.timestamps.delayed-1write\(.*\)$
index 7f1a54c8ad0a5d65cbb96296e6a6ba578bbbeb0a..87137d65414f530b5424cb9e75cc598e0b6dc46a 100644 (file)
@@ -700,6 +700,91 @@ done:
        return ret;
 }
 
+static bool test_delayed_1write(struct torture_context *tctx,
+                               struct smb2_tree *tree)
+{
+       struct smb2_create cr;
+       struct smb2_handle h1 = {{0}};
+       union smb_fileinfo finfo;
+       struct smb2_close c;
+       NTTIME create_time;
+       NTTIME write_time;
+       NTTIME close_time;
+       NTSTATUS status;
+       bool ret = true;
+
+       smb2_deltree(tree, BASEDIR);
+       status = torture_smb2_testdir(tree, BASEDIR, &h1);
+       torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
+                                       "create failed\n");
+       status = smb2_util_close(tree, h1);
+       torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
+                                       "close failed\n");
+
+       torture_comment(tctx, "Open file-handle 1\n");
+
+       cr = (struct smb2_create) {
+               .in.desired_access     = SEC_FLAG_MAXIMUM_ALLOWED,
+               .in.create_disposition = NTCREATEX_DISP_OPEN_IF,
+               .in.share_access       = NTCREATEX_SHARE_ACCESS_MASK,
+               .in.fname              = BASEDIR "\\" FNAME,
+       };
+       status = smb2_create(tree, tctx, &cr);
+       torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
+                                       "create failed\n");
+       h1 = cr.out.file.handle;
+       create_time = cr.out.create_time;
+       sleep(1);
+
+       torture_comment(tctx, "Write to file-handle 1\n");
+
+       status = smb2_util_write(tree, h1, "s", 0, 1);
+       torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
+                                       "write failed\n");
+       sleep(3);
+
+       torture_comment(tctx, "Check writetime has been updated\n");
+
+       finfo = (union smb_fileinfo) {
+               .generic.level = RAW_FILEINFO_SMB2_ALL_INFORMATION,
+               .generic.in.file.handle = h1,
+       };
+       status = smb2_getinfo_file(tree, tree, &finfo);
+       torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
+                                       "getinfo failed\n");
+       write_time = finfo.all_info.out.write_time;
+
+       if (!(write_time > create_time)) {
+               ret = false;
+               torture_fail_goto(tctx, done,
+                                 "Write-time not updated (wrong!)\n");
+       }
+
+       torture_comment(tctx, "Close file-handle 1\n");
+       sleep(1);
+
+       c = (struct smb2_close) {
+               .in.file.handle = h1,
+               .in.flags = SMB2_CLOSE_FLAGS_FULL_INFORMATION,
+       };
+
+       status = smb2_close(tree, &c);
+       torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
+                                       "close failed\n");
+       ZERO_STRUCT(h1);
+       close_time = c.out.write_time;
+
+       torture_assert_nttime_equal(tctx, close_time, write_time,
+                                   "Writetime != close_time (wrong!)\n");
+
+done:
+       if (!smb2_util_handle_empty(h1)) {
+               smb2_util_close(tree, h1);
+       }
+       smb2_deltree(tree, BASEDIR);
+       return ret;
+}
+
 /*
    basic testing of SMB2 timestamps
 */
@@ -722,6 +807,7 @@ struct torture_suite *torture_smb2_timestamps_init(TALLOC_CTX *ctx)
        torture_suite_add_1smb2_test(suite, "delayed-write-vs-seteof", test_delayed_write_vs_seteof);
        torture_suite_add_1smb2_test(suite, "delayed-write-vs-flush", test_delayed_write_vs_flush);
        torture_suite_add_1smb2_test(suite, "delayed-write-vs-setbasic", test_delayed_write_vs_setbasic);
+       torture_suite_add_1smb2_test(suite, "delayed-1write", test_delayed_1write);
 
        suite->description = talloc_strdup(suite, "SMB2 timestamp tests");