]> git.ipfire.org Git - thirdparty/git.git/commitdiff
t7527: create test for fsmonitor--daemon
authorJeff Hostetler <jeffhost@microsoft.com>
Fri, 25 Mar 2022 18:03:04 +0000 (18:03 +0000)
committerJunio C Hamano <gitster@pobox.com>
Fri, 25 Mar 2022 23:04:17 +0000 (16:04 -0700)
Signed-off-by: Jeff Hostetler <jeffhost@microsoft.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
t/t7527-builtin-fsmonitor.sh [new file with mode: 0755]

diff --git a/t/t7527-builtin-fsmonitor.sh b/t/t7527-builtin-fsmonitor.sh
new file mode 100755 (executable)
index 0000000..062e01c
--- /dev/null
@@ -0,0 +1,494 @@
+#!/bin/sh
+
+test_description='built-in file system watcher'
+
+. ./test-lib.sh
+
+if ! test_have_prereq FSMONITOR_DAEMON
+then
+       skip_all="fsmonitor--daemon is not supported on this platform"
+       test_done
+fi
+
+stop_daemon_delete_repo () {
+       r=$1 &&
+       test_might_fail git -C $r fsmonitor--daemon stop &&
+       rm -rf $1
+}
+
+start_daemon () {
+       r= tf= t2= tk= &&
+
+       while test "$#" -ne 0
+       do
+               case "$1" in
+               -C)
+                       r="-C ${2?}"
+                       shift
+                       ;;
+               --tf)
+                       tf="${2?}"
+                       shift
+                       ;;
+               --t2)
+                       t2="${2?}"
+                       shift
+                       ;;
+               --tk)
+                       tk="${2?}"
+                       shift
+                       ;;
+               -*)
+                       BUG "error: unknown option: '$1'"
+                       ;;
+               *)
+                       BUG "error: unbound argument: '$1'"
+                       ;;
+               esac
+               shift
+       done &&
+
+       (
+               if test -n "$tf"
+               then
+                       GIT_TRACE_FSMONITOR="$tf"
+                       export GIT_TRACE_FSMONITOR
+               fi &&
+
+               if test -n "$t2"
+               then
+                       GIT_TRACE2_PERF="$t2"
+                       export GIT_TRACE2_PERF
+               fi &&
+
+               if test -n "$tk"
+               then
+                       GIT_TEST_FSMONITOR_TOKEN="$tk"
+                       export GIT_TEST_FSMONITOR_TOKEN
+               fi &&
+
+               git $r fsmonitor--daemon start &&
+               git $r fsmonitor--daemon status
+       )
+}
+
+# Is a Trace2 data event present with the given catetory and key?
+# We do not care what the value is.
+#
+have_t2_data_event () {
+       c=$1 &&
+       k=$2 &&
+
+       grep -e '"event":"data".*"category":"'"$c"'".*"key":"'"$k"'"'
+}
+
+test_expect_success 'explicit daemon start and stop' '
+       test_when_finished "stop_daemon_delete_repo test_explicit" &&
+
+       git init test_explicit &&
+       start_daemon -C test_explicit &&
+
+       git -C test_explicit fsmonitor--daemon stop &&
+       test_must_fail git -C test_explicit fsmonitor--daemon status
+'
+
+test_expect_success 'implicit daemon start' '
+       test_when_finished "stop_daemon_delete_repo test_implicit" &&
+
+       git init test_implicit &&
+       test_must_fail git -C test_implicit fsmonitor--daemon status &&
+
+       # query will implicitly start the daemon.
+       #
+       # for test-script simplicity, we send a V1 timestamp rather than
+       # a V2 token.  either way, the daemon response to any query contains
+       # a new V2 token.  (the daemon may complain that we sent a V1 request,
+       # but this test case is only concerned with whether the daemon was
+       # implicitly started.)
+
+       GIT_TRACE2_EVENT="$PWD/.git/trace" \
+               test-tool -C test_implicit fsmonitor-client query --token 0 >actual &&
+       nul_to_q <actual >actual.filtered &&
+       grep "builtin:" actual.filtered &&
+
+       # confirm that a daemon was started in the background.
+       #
+       # since the mechanism for starting the background daemon is platform
+       # dependent, just confirm that the foreground command received a
+       # response from the daemon.
+
+       have_t2_data_event fsm_client query/response-length <.git/trace &&
+
+       git -C test_implicit fsmonitor--daemon status &&
+       git -C test_implicit fsmonitor--daemon stop &&
+       test_must_fail git -C test_implicit fsmonitor--daemon status
+'
+
+test_expect_success 'implicit daemon stop (delete .git)' '
+       test_when_finished "stop_daemon_delete_repo test_implicit_1" &&
+
+       git init test_implicit_1 &&
+
+       start_daemon -C test_implicit_1 &&
+
+       # deleting the .git directory will implicitly stop the daemon.
+       rm -rf test_implicit_1/.git &&
+
+       # [1] Create an empty .git directory so that the following Git
+       #     command will stay relative to the `-C` directory.
+       #
+       #     Without this, the Git command will override the requested
+       #     -C argument and crawl out to the containing Git source tree.
+       #     This would make the test result dependent upon whether we
+       #     were using fsmonitor on our development worktree.
+       #
+       sleep 1 &&
+       mkdir test_implicit_1/.git &&
+
+       test_must_fail git -C test_implicit_1 fsmonitor--daemon status
+'
+
+test_expect_success 'implicit daemon stop (rename .git)' '
+       test_when_finished "stop_daemon_delete_repo test_implicit_2" &&
+
+       git init test_implicit_2 &&
+
+       start_daemon -C test_implicit_2 &&
+
+       # renaming the .git directory will implicitly stop the daemon.
+       mv test_implicit_2/.git test_implicit_2/.xxx &&
+
+       # See [1] above.
+       #
+       sleep 1 &&
+       mkdir test_implicit_2/.git &&
+
+       test_must_fail git -C test_implicit_2 fsmonitor--daemon status
+'
+
+test_expect_success 'cannot start multiple daemons' '
+       test_when_finished "stop_daemon_delete_repo test_multiple" &&
+
+       git init test_multiple &&
+
+       start_daemon -C test_multiple &&
+
+       test_must_fail git -C test_multiple fsmonitor--daemon start 2>actual &&
+       grep "fsmonitor--daemon is already running" actual &&
+
+       git -C test_multiple fsmonitor--daemon stop &&
+       test_must_fail git -C test_multiple fsmonitor--daemon status
+'
+
+# These tests use the main repo in the trash directory
+
+test_expect_success 'setup' '
+       >tracked &&
+       >modified &&
+       >delete &&
+       >rename &&
+       mkdir dir1 &&
+       >dir1/tracked &&
+       >dir1/modified &&
+       >dir1/delete &&
+       >dir1/rename &&
+       mkdir dir2 &&
+       >dir2/tracked &&
+       >dir2/modified &&
+       >dir2/delete &&
+       >dir2/rename &&
+       mkdir dirtorename &&
+       >dirtorename/a &&
+       >dirtorename/b &&
+
+       cat >.gitignore <<-\EOF &&
+       .gitignore
+       expect*
+       actual*
+       EOF
+
+       git -c core.fsmonitor=false add . &&
+       test_tick &&
+       git -c core.fsmonitor=false commit -m initial &&
+
+       git config core.fsmonitor true
+'
+
+# The test already explicitly stopped (or tried to stop) the daemon.
+# This is here in case something else fails first.
+#
+redundant_stop_daemon () {
+       test_might_fail git fsmonitor--daemon stop
+}
+
+test_expect_success 'update-index implicitly starts daemon' '
+       test_when_finished redundant_stop_daemon &&
+
+       test_must_fail git fsmonitor--daemon status &&
+
+       GIT_TRACE2_EVENT="$PWD/.git/trace_implicit_1" \
+               git update-index --fsmonitor &&
+
+       git fsmonitor--daemon status &&
+       test_might_fail git fsmonitor--daemon stop &&
+
+       # Confirm that the trace2 log contains a record of the
+       # daemon starting.
+       test_subcommand git fsmonitor--daemon start <.git/trace_implicit_1
+'
+
+test_expect_success 'status implicitly starts daemon' '
+       test_when_finished redundant_stop_daemon &&
+
+       test_must_fail git fsmonitor--daemon status &&
+
+       GIT_TRACE2_EVENT="$PWD/.git/trace_implicit_2" \
+               git status >actual &&
+
+       git fsmonitor--daemon status &&
+       test_might_fail git fsmonitor--daemon stop &&
+
+       # Confirm that the trace2 log contains a record of the
+       # daemon starting.
+       test_subcommand git fsmonitor--daemon start <.git/trace_implicit_2
+'
+
+edit_files () {
+       echo 1 >modified &&
+       echo 2 >dir1/modified &&
+       echo 3 >dir2/modified &&
+       >dir1/untracked
+}
+
+delete_files () {
+       rm -f delete &&
+       rm -f dir1/delete &&
+       rm -f dir2/delete
+}
+
+create_files () {
+       echo 1 >new &&
+       echo 2 >dir1/new &&
+       echo 3 >dir2/new
+}
+
+rename_files () {
+       mv rename renamed &&
+       mv dir1/rename dir1/renamed &&
+       mv dir2/rename dir2/renamed
+}
+
+file_to_directory () {
+       rm -f delete &&
+       mkdir delete &&
+       echo 1 >delete/new
+}
+
+directory_to_file () {
+       rm -rf dir1 &&
+       echo 1 >dir1
+}
+
+# The next few test cases confirm that our fsmonitor daemon sees each type
+# of OS filesystem notification that we care about.  At this layer we just
+# ensure we are getting the OS notifications and do not try to confirm what
+# is reported by `git status`.
+#
+# We run a simple query after modifying the filesystem just to introduce
+# a bit of a delay so that the trace logging from the daemon has time to
+# get flushed to disk.
+#
+# We `reset` and `clean` at the bottom of each test (and before stopping the
+# daemon) because these commands might implicitly restart the daemon.
+
+clean_up_repo_and_stop_daemon () {
+       git reset --hard HEAD &&
+       git clean -fd &&
+       test_might_fail git fsmonitor--daemon stop &&
+       rm -f .git/trace
+}
+
+test_expect_success 'edit some files' '
+       test_when_finished clean_up_repo_and_stop_daemon &&
+
+       start_daemon --tf "$PWD/.git/trace" &&
+
+       edit_files &&
+
+       test-tool fsmonitor-client query --token 0 &&
+
+       grep "^event: dir1/modified$"  .git/trace &&
+       grep "^event: dir2/modified$"  .git/trace &&
+       grep "^event: modified$"       .git/trace &&
+       grep "^event: dir1/untracked$" .git/trace
+'
+
+test_expect_success 'create some files' '
+       test_when_finished clean_up_repo_and_stop_daemon &&
+
+       start_daemon --tf "$PWD/.git/trace" &&
+
+       create_files &&
+
+       test-tool fsmonitor-client query --token 0 &&
+
+       grep "^event: dir1/new$" .git/trace &&
+       grep "^event: dir2/new$" .git/trace &&
+       grep "^event: new$"      .git/trace
+'
+
+test_expect_success 'delete some files' '
+       test_when_finished clean_up_repo_and_stop_daemon &&
+
+       start_daemon --tf "$PWD/.git/trace" &&
+
+       delete_files &&
+
+       test-tool fsmonitor-client query --token 0 &&
+
+       grep "^event: dir1/delete$" .git/trace &&
+       grep "^event: dir2/delete$" .git/trace &&
+       grep "^event: delete$"      .git/trace
+'
+
+test_expect_success 'rename some files' '
+       test_when_finished clean_up_repo_and_stop_daemon &&
+
+       start_daemon --tf "$PWD/.git/trace" &&
+
+       rename_files &&
+
+       test-tool fsmonitor-client query --token 0 &&
+
+       grep "^event: dir1/rename$"  .git/trace &&
+       grep "^event: dir2/rename$"  .git/trace &&
+       grep "^event: rename$"       .git/trace &&
+       grep "^event: dir1/renamed$" .git/trace &&
+       grep "^event: dir2/renamed$" .git/trace &&
+       grep "^event: renamed$"      .git/trace
+'
+
+test_expect_success 'rename directory' '
+       test_when_finished clean_up_repo_and_stop_daemon &&
+
+       start_daemon --tf "$PWD/.git/trace" &&
+
+       mv dirtorename dirrenamed &&
+
+       test-tool fsmonitor-client query --token 0 &&
+
+       grep "^event: dirtorename/*$" .git/trace &&
+       grep "^event: dirrenamed/*$"  .git/trace
+'
+
+test_expect_success 'file changes to directory' '
+       test_when_finished clean_up_repo_and_stop_daemon &&
+
+       start_daemon --tf "$PWD/.git/trace" &&
+
+       file_to_directory &&
+
+       test-tool fsmonitor-client query --token 0 &&
+
+       grep "^event: delete$"     .git/trace &&
+       grep "^event: delete/new$" .git/trace
+'
+
+test_expect_success 'directory changes to a file' '
+       test_when_finished clean_up_repo_and_stop_daemon &&
+
+       start_daemon --tf "$PWD/.git/trace" &&
+
+       directory_to_file &&
+
+       test-tool fsmonitor-client query --token 0 &&
+
+       grep "^event: dir1$" .git/trace
+'
+
+# The next few test cases exercise the token-resync code.  When filesystem
+# drops events (because of filesystem velocity or because the daemon isn't
+# polling fast enough), we need to discard the cached data (relative to the
+# current token) and start collecting events under a new token.
+#
+# the 'test-tool fsmonitor-client flush' command can be used to send a
+# "flush" message to a running daemon and ask it to do a flush/resync.
+
+test_expect_success 'flush cached data' '
+       test_when_finished "stop_daemon_delete_repo test_flush" &&
+
+       git init test_flush &&
+
+       start_daemon -C test_flush --tf "$PWD/.git/trace_daemon" --tk true &&
+
+       # The daemon should have an initial token with no events in _0 and
+       # then a few (probably platform-specific number of) events in _1.
+       # These should both have the same <token_id>.
+
+       test-tool -C test_flush fsmonitor-client query --token "builtin:test_00000001:0" >actual_0 &&
+       nul_to_q <actual_0 >actual_q0 &&
+
+       >test_flush/file_1 &&
+       >test_flush/file_2 &&
+
+       test-tool -C test_flush fsmonitor-client query --token "builtin:test_00000001:0" >actual_1 &&
+       nul_to_q <actual_1 >actual_q1 &&
+
+       grep "file_1" actual_q1 &&
+
+       # Force a flush.  This will change the <token_id>, reset the <seq_nr>, and
+       # flush the file data.  Then create some events and ensure that the file
+       # again appears in the cache.  It should have the new <token_id>.
+
+       test-tool -C test_flush fsmonitor-client flush >flush_0 &&
+       nul_to_q <flush_0 >flush_q0 &&
+       grep "^builtin:test_00000002:0Q/Q$" flush_q0 &&
+
+       test-tool -C test_flush fsmonitor-client query --token "builtin:test_00000002:0" >actual_2 &&
+       nul_to_q <actual_2 >actual_q2 &&
+
+       grep "^builtin:test_00000002:0Q$" actual_q2 &&
+
+       >test_flush/file_3 &&
+
+       test-tool -C test_flush fsmonitor-client query --token "builtin:test_00000002:0" >actual_3 &&
+       nul_to_q <actual_3 >actual_q3 &&
+
+       grep "file_3" actual_q3
+'
+
+# The next few test cases create repos where the .git directory is NOT
+# inside the one of the working directory.  That is, where .git is a file
+# that points to a directory elsewhere.  This happens for submodules and
+# non-primary worktrees.
+
+test_expect_success 'setup worktree base' '
+       git init wt-base &&
+       echo 1 >wt-base/file1 &&
+       git -C wt-base add file1 &&
+       git -C wt-base commit -m "c1"
+'
+
+test_expect_success 'worktree with .git file' '
+       git -C wt-base worktree add ../wt-secondary &&
+
+       start_daemon -C wt-secondary \
+               --tf "$PWD/trace_wt_secondary" \
+               --t2 "$PWD/trace2_wt_secondary" &&
+
+       git -C wt-secondary fsmonitor--daemon stop &&
+       test_must_fail git -C wt-secondary fsmonitor--daemon status
+'
+
+# NEEDSWORK: Repeat one of the "edit" tests on wt-secondary and
+# confirm that we get the same events and behavior -- that is, that
+# fsmonitor--daemon correctly watches BOTH the working directory and
+# the external GITDIR directory and behaves the same as when ".git"
+# is a directory inside the working directory.
+
+test_expect_success 'cleanup worktrees' '
+       stop_daemon_delete_repo wt-secondary &&
+       stop_daemon_delete_repo wt-base
+'
+
+test_done