From: Junio C Hamano <gitster@pobox.com>
To: Jens Lehmann <Jens.Lehmann@web.de>
Cc: Git Mailing List <git@vger.kernel.org>,
Heiko Voigt <hvoigt@hvoigt.net>,
Jonathan Nieder <jrnieder@gmail.com>, Jeff King <peff@peff.net>
Subject: Re: [PATCH 02/14] submodules: Add the lib-submodule-update.sh test library
Date: Mon, 16 Jun 2014 15:49:23 -0700 [thread overview]
Message-ID: <xmqqwqcgo4gc.fsf@gitster.dls.corp.google.com> (raw)
In-Reply-To: <539DD09A.7010200@web.de> (Jens Lehmann's message of "Sun, 15 Jun 2014 18:58:02 +0200")
Jens Lehmann <Jens.Lehmann@web.de> writes:
> Add this test library to simplify covering all combinations of submodule
> update scenarios without having to add those to a test of each work tree
> manipulating command over and over again.
>
> The functions test_submodule_switch() and test_submodule_forced_switch()
> are intended to be called from a test script with a single argument. This
> argument is either a work tree manipulating command (including any command
> line options) or a function (when more than a single git command is needed
> to switch work trees from the current HEAD to another commit). This
> command (or function) is passed a target branch as argument. The two new
> functions check that each submodule transition is handled as expected,
> which currently means that submodule work trees are not affected until
> "git submodule update" is called. The "forced" variant is for commands
> using their '-f' or '--hard' option and expects them to overwrite local
> modifications as a result. Each of these two functions contains 14
> tests_expect_* calls.
>
> Calling one of these test functions the first time creates a repository
> named "submodule_update_repo". At first it contains two files, then a
> single submodule is added in another commit followed by commits covering
> all relevant submodule modifications. This repository is newly cloned into
> the "submodule_update" for each test_expect_* to avoid interference
> between different parts of the test functions (some to-be-tested commands
> also manipulate refs along with the work tree, e.g. "git reset").
>
> Follow-up commits will then call these two test functions for all work
> tree manipulating commands (with a combination of all their options
> relevant to what they do with the work tree) making sure they work as
> expected. Later this test library will be extended to cover merges
> resulting in conflicts too. Also it is intended to be easily extendable
> for the recursive update functionality, where even more combinations of
> submodule modifications have to be tested for.
>
> This version documents two bugs in current Git with expected failures:
>
> *) When a submodule is replaced with a tracked file of the same name the
> submodule work tree including any local modifications (and even the
> whole history if it uses a .git directory instead of a gitfile!) is
> silently removed.
>
> *) Forced work tree updates happily manipulate files in the directory of a
> submodule that has just been removed in the superproject (but is of
> course still present in the work tree due to the way submodules are
> currently handled). This becomes dangerous when files in the submodule
> directory are overwritten by files from the new superproject commit, as
> any modifications to the submodule files will be lost) and is expected
> to also destroy history in the - admittedly unlikely case - the new
> commit adds a file named ".git" to the submodule directory.
>
> Signed-off-by: Jens Lehmann <Jens.Lehmann@web.de>
> ---
> t/lib-submodule-update.sh | 630 ++++++++++++++++++++++++++++++++++++++++++++++
> 1 file changed, 630 insertions(+)
> create mode 100755 t/lib-submodule-update.sh
>
> diff --git a/t/lib-submodule-update.sh b/t/lib-submodule-update.sh
> new file mode 100755
> index 0000000..c6c842a
> --- /dev/null
> +++ b/t/lib-submodule-update.sh
> @@ -0,0 +1,630 @@
> +# Create a submodule layout used for all tests below.
> +#
> +# The following use cases are covered:
> +# - New submodule (no_submodule => add_sub1)
> +# - Removed submodule (add_sub1 => remove_sub1)
> +# - Updated submodule (add_sub1 => modify_sub1)
> +# - Submodule updated to invalid commit (add_sub1 => invalid_sub1)
> +# - Submodule updated from invalid commit (invalid_sub1 => valid_sub1)
> +# - Submodule replaced by tracked files in directory (add_sub1 =>
> +# replace_sub1_with_directory)
> +# - Directory containing tracked files replaced by submodule
> +# (replace_sub1_with_directory => replace_directory_with_sub1)
> +# - Submodule replaced by tracked file with the same name (add_sub1 =>
> +# replace_sub1_with_file)
> +# - Tracked file replaced by submodule (replace_sub1_with_file =>
> +# replace_file_with_sub1)
> +#
> +# --O-----O
> +# / ^ replace_directory_with_sub1
> +# / replace_sub1_with_directory
> +# /----O
> +# / ^
> +# / modify_sub1
> +# O------O-------O
> +# ^ ^\ ^
> +# | | \ remove_sub1
> +# | | -----O-----O
> +# | | \ ^ replace_file_with_sub1
> +# | | \ replace_sub1_with_file
> +# | add_sub1 --O-----O
> +# no_submodule ^ valid_sub1
> +# invalid_sub1
> +#
> +create_lib_submodule_repo () {
> + git init submodule_update_repo &&
> + (
> + cd submodule_update_repo &&
> + echo "expect" >>.gitignore &&
> + echo "actual" >>.gitignore &&
> + echo "x" >file1 &&
> + echo "y" >file2 &&
> + git add .gitignore file1 file2 &&
> + git commit -m "Base" &&
> + git branch "no_submodule" &&
> +
> + git checkout -b "add_sub1" &&
> + git submodule add ./. sub1 &&
This is not technically wrong per-se, but having the project's
history itself as its own submodule *is* something nobody sane would
do in the real life. Do we really have to do it this unusual way?
> + git config -f .gitmodules submodule.sub1.ignore all &&
> + git config submodule.sub1.ignore all &&
> + git add .gitmodules &&
> + git commit -m "Add sub1" &&
> + git checkout -b remove_sub1 &&
> + git revert HEAD &&
> +
> + git checkout -b "modify_sub1" "add_sub1" &&
> + git submodule update &&
> + (
> + cd sub1 &&
> + git fetch &&
> + git checkout -b "modifications" &&
> + echo "z" >file2 &&
> + echo "x" >file3 &&
> + git add file2 file3 &&
> + git commit -m "modified file2 and added file3" &&
> + git push origin modifications
> + ) &&
> + git add sub1 &&
> + git commit -m "Modify sub1" &&
> +
> + git checkout -b "replace_sub1_with_directory" "add_sub1" &&
> + git submodule update &&
> + (
> + cd sub1 &&
> + git checkout modifications
> + ) &&
> + git rm --cached sub1 &&
> + rm sub1/.git* &&
> + git config -f .gitmodules --remove-section "submodule.sub1" &&
> + git add .gitmodules sub1/* &&
> + git commit -m "Replace sub1 with directory" &&
> + git checkout -b replace_directory_with_sub1 &&
> + git revert HEAD &&
> +
> + git checkout -b "replace_sub1_with_file" "add_sub1" &&
> + git rm sub1 &&
> + echo "content" >sub1 &&
> + git add sub1 &&
> + git commit -m "Replace sub1 with file" &&
> + git checkout -b replace_file_with_sub1 &&
> + git revert HEAD &&
> +
> + git checkout -b "invalid_sub1" "add_sub1" &&
> + git update-index --cacheinfo 160000 0123456789012345678901234567890123456789 sub1 &&
> + git commit -m "Invalid sub1 commit" &&
> + git checkout -b valid_sub1 &&
> + git revert HEAD &&
> + git checkout master
> + )
> +}
> +
> +# Helper function to replace gitfile with .git directory
> +replace_gitfile_with_git_dir () {
> + (
> + cd "$1" &&
> + git_dir="$(git rev-parse --git-dir)" &&
> + rm -f .git &&
> + cp -a "$git_dir" .git &&
We avoid "cp -a" for portability, don't we?
> + GIT_WORK_TREE=. git config --unset core.worktree
Hmph. What does GIT_WORK_TREE=. alone without GIT_DIR=<somewhere>
do? It's not like it is a workaround for "git config" that complains
when you do not have a working tree, right? Puzzled...
> + )
> +}
> +
> +# Test that the .git directory in the submodule is unchanged (except for the
> +# core.worktree setting)
> +test_git_directory_is_unchanged () {
> + (
> + cd "$1" &&
> + git config core.worktree "../../../$1"
> + ) &&
> + git diff -r ".git/modules/$1" "$1/.git" &&
I'd prefer to see "--no-index" spelled out, if that is what is going
on.
> + (
> + cd "$1" &&
> + GIT_WORK_TREE=. git config --unset core.worktree
> + )
> +}
> +
> +# Helper function to be executed at the start of every test below, it sets up
> +# the submodule repo if it doesn't exist and configures the most problematic
> +# settings for diff.ignoreSubmodules.
> +prolog () {
> + (test -d submodule_update_repo || create_lib_submodule_repo) &&
> + test_config_global diff.ignoreSubmodules all &&
> + test_config diff.ignoreSubmodules all
> +}
> +
> +# Helper function to bring work tree back into the state given by the
> +# commit. This includes trying to populate sub1 accordingly if it exists and
> +# should be updated to an existing commit.
> +reset_work_tree_to () {
> + rm -rf submodule_update &&
> + git clone submodule_update_repo submodule_update &&
> + (
> + cd submodule_update &&
> + rm -rf sub1 &&
> + git checkout -f "$1" &&
> + git status -u -s >actual &&
> + test_must_be_empty actual &&
> + sha1=$(git ls-tree HEAD "sub1" 2>/dev/null | grep 160000 | tr '\t' ' ' | cut -d ' ' -f3) &&
Why discard the standard error stream?
grep|tr|cut looks somewhat stupid. Can't we do that with a single
sed?
sha1=$(git ls-tree HEAD sub1 | sed -ne "s/^160000 commit \($_x40\) .*/\1/p")
or better yet, perhaps
sha1=$(git rev-parse HEAD:sub1)
> +# Test that the given submodule at path "$1" contains the content according
> +# to the submodule commit recorded in the superproject's commit "$2"
> +test_submodule_content () {
> + if test $# != 2
> + then
> + echo "test_submodule_content needs two arguments"
> + return 1
> + fi &&
> + submodule="$1" &&
> + commit="$2" &&
> + test -d "$submodule"/ &&
> + if ! test -f "$submodule"/.git && ! test -d "$submodule"/.git
I wonder if we can get away with a single "test -e" (we do not
expect us to be creating device nodes or fifos there, do we?).
> + then
> + echo "Submodule $submodule is not populated"
> + return 1
> + fi &&
> + sha1=$(git ls-tree "$commit" "$submodule" 2>/dev/null | tr '\t' ' ' | cut -d ' ' -f3) &&
Likewise.
next prev parent reply other threads:[~2014-06-16 22:49 UTC|newest]
Thread overview: 65+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-06-15 16:56 [PATCH 00/14] Add submodule test harness Jens Lehmann
2014-06-15 16:57 ` [PATCH 01/14] test-lib: add test_dir_is_empty() Jens Lehmann
2014-06-16 22:05 ` Junio C Hamano
2014-06-17 16:47 ` Jens Lehmann
2014-06-19 20:12 ` [PATCH v2 " Jens Lehmann
2014-06-15 16:58 ` [PATCH 02/14] submodules: Add the lib-submodule-update.sh test library Jens Lehmann
2014-06-16 22:49 ` Junio C Hamano [this message]
2014-06-17 17:33 ` Jens Lehmann
2014-06-17 18:44 ` Junio C Hamano
2014-06-17 20:46 ` Jens Lehmann
2014-06-17 21:05 ` Junio C Hamano
2014-06-19 20:12 ` [PATCH v2 " Jens Lehmann
2014-06-20 17:31 ` Junio C Hamano
2014-07-01 21:24 ` [PATCH v3 " Jens Lehmann
2014-06-15 16:58 ` [PATCH 03/14] checkout: call the new submodule update test framework Jens Lehmann
2014-06-15 16:59 ` [PATCH 04/14] apply: add t4137 for submodule updates Jens Lehmann
2014-06-15 16:59 ` [PATCH 05/14] read-tree: add t1013 " Jens Lehmann
2014-06-15 17:00 ` [PATCH 06/14] reset: add t7112 " Jens Lehmann
2014-06-15 17:01 ` [PATCH 07/14] bisect: add t6041 " Jens Lehmann
2014-06-19 20:12 ` [PATCH v2 " Jens Lehmann
2014-06-15 17:01 ` [PATCH 08/14] merge: add t7613 " Jens Lehmann
2014-06-15 17:02 ` [PATCH 09/14] rebase: add t3426 " Jens Lehmann
2014-06-16 9:57 ` Eric Sunshine
2014-06-17 17:41 ` Jens Lehmann
2014-06-19 20:12 ` [PATCH v2 " Jens Lehmann
2014-06-15 17:02 ` [PATCH 10/14] pull: add t5572 " Jens Lehmann
2014-06-15 17:03 ` [PATCH 11/14] cherry-pick: add t3512 " Jens Lehmann
2014-06-15 17:03 ` [PATCH 12/14] am: add t4255 " Jens Lehmann
2014-06-15 17:04 ` [PATCH 13/14] stash: add t3906 " Jens Lehmann
2014-06-19 20:12 ` [PATCH v2 " Jens Lehmann
2014-06-15 17:04 ` [PATCH 14/14] revert: add t3513 " Jens Lehmann
2014-06-19 20:12 ` [PATCH v2 " Jens Lehmann
2014-07-02 14:54 ` [PATCH 00/14] Add submodule test harness Torsten Bögershausen
2014-07-02 19:57 ` Jens Lehmann
2014-07-03 5:56 ` Torsten Bögershausen
2014-07-03 21:14 ` Jens Lehmann
2014-07-07 17:05 ` Junio C Hamano
2014-07-07 19:40 ` Torsten Bögershausen
2014-07-08 19:34 ` Jens Lehmann
2014-07-08 20:25 ` Ramsay Jones
2014-07-08 21:03 ` Ramsay Jones
2014-07-09 6:39 ` No fchmod() under msygit - Was: " Torsten Bögershausen
2014-07-09 20:00 ` Eric Wong
2014-07-14 11:31 ` Erik Faye-Lund
2014-07-14 13:55 ` Nico Williams
2014-07-14 14:02 ` Nico Williams
2014-07-14 19:30 ` Karsten Blees
2014-07-14 21:18 ` Junio C Hamano
2014-07-09 6:14 ` Torsten Bögershausen
2014-07-09 15:20 ` Junio C Hamano
2014-07-09 18:19 ` Jens Lehmann
2014-07-09 19:31 ` Junio C Hamano
2014-07-10 20:52 ` Junio C Hamano
2014-07-12 18:23 ` Jens Lehmann
2014-07-14 1:01 ` Junio C Hamano
2014-07-14 18:22 ` Jens Lehmann
2014-07-14 21:18 ` Junio C Hamano
2014-07-09 17:21 ` Johannes Sixt
2014-07-09 19:22 ` Junio C Hamano
2014-07-09 19:56 ` Eric Wong
2014-07-09 21:57 ` Junio C Hamano
2014-07-10 6:22 ` No fchmd. was: " Torsten Bögershausen
2014-07-10 19:49 ` Junio C Hamano
2014-07-10 20:55 ` Torsten Bögershausen
2014-07-10 21:43 ` Junio C Hamano
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
List information: http://vger.kernel.org/majordomo-info.html
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=xmqqwqcgo4gc.fsf@gitster.dls.corp.google.com \
--to=gitster@pobox.com \
--cc=Jens.Lehmann@web.de \
--cc=git@vger.kernel.org \
--cc=hvoigt@hvoigt.net \
--cc=jrnieder@gmail.com \
--cc=peff@peff.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
Code repositories for project(s) associated with this public inbox
https://80x24.org/mirrors/git.git
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).