| #!/bin/sh |
| |
| test_description='check svn dumpfile importer' |
| |
| . ./test-lib.sh |
| |
| reinit_git () { |
| rm -fr .git && |
| git init |
| } |
| |
| properties () { |
| while test "$#" -ne 0 |
| do |
| property="$1" && |
| value="$2" && |
| printf "%s\n" "K ${#property}" && |
| printf "%s\n" "$property" && |
| printf "%s\n" "V ${#value}" && |
| printf "%s\n" "$value" && |
| shift 2 || |
| return 1 |
| done |
| } |
| |
| text_no_props () { |
| text="$1 |
| " && |
| printf "%s\n" "Prop-content-length: 10" && |
| printf "%s\n" "Text-content-length: ${#text}" && |
| printf "%s\n" "Content-length: $((${#text} + 10))" && |
| printf "%s\n" "" "PROPS-END" && |
| printf "%s\n" "$text" |
| } |
| |
| >empty |
| |
| test_expect_success 'empty dump' ' |
| reinit_git && |
| echo "SVN-fs-dump-format-version: 2" >input && |
| test-svn-fe input >stream && |
| git fast-import <stream |
| ' |
| |
| test_expect_success 'v4 dumps not supported' ' |
| reinit_git && |
| echo "SVN-fs-dump-format-version: 4" >v4.dump && |
| test_must_fail test-svn-fe v4.dump >stream && |
| test_cmp empty stream |
| ' |
| |
| test_expect_failure 'empty revision' ' |
| reinit_git && |
| printf "rev <nobody, nobody@local>: %s\n" "" "" >expect && |
| cat >emptyrev.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 0 |
| Content-length: 0 |
| |
| Revision-number: 2 |
| Prop-content-length: 0 |
| Content-length: 0 |
| |
| EOF |
| test-svn-fe emptyrev.dump >stream && |
| git fast-import <stream && |
| git log -p --format="rev <%an, %ae>: %s" HEAD >actual && |
| test_cmp expect actual |
| ' |
| |
| test_expect_success 'empty properties' ' |
| reinit_git && |
| printf "rev <nobody, nobody@local>: %s\n" "" "" >expect && |
| cat >emptyprop.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Revision-number: 2 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| EOF |
| test-svn-fe emptyprop.dump >stream && |
| git fast-import <stream && |
| git log -p --format="rev <%an, %ae>: %s" HEAD >actual && |
| test_cmp expect actual |
| ' |
| |
| test_expect_success 'author name and commit message' ' |
| reinit_git && |
| echo "<author@example.com, author@example.com@local>" >expect.author && |
| cat >message <<-\EOF && |
| A concise summary of the change |
| |
| A detailed description of the change, why it is needed, what |
| was broken and why applying this is the best course of action. |
| |
| * file.c |
| Details pertaining to an individual file. |
| EOF |
| { |
| properties \ |
| svn:author author@example.com \ |
| svn:log "$(cat message)" && |
| echo PROPS-END |
| } >props && |
| { |
| echo "SVN-fs-dump-format-version: 3" && |
| echo && |
| echo "Revision-number: 1" && |
| echo Prop-content-length: $(wc -c <props) && |
| echo Content-length: $(wc -c <props) && |
| echo && |
| cat props |
| } >log.dump && |
| test-svn-fe log.dump >stream && |
| git fast-import <stream && |
| git log -p --format="%B" HEAD >actual.log && |
| git log --format="<%an, %ae>" >actual.author && |
| test_cmp message actual.log && |
| test_cmp expect.author actual.author |
| ' |
| |
| test_expect_success 'unsupported properties are ignored' ' |
| reinit_git && |
| echo author >expect && |
| cat >extraprop.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 56 |
| Content-length: 56 |
| |
| K 8 |
| nonsense |
| V 1 |
| y |
| K 10 |
| svn:author |
| V 6 |
| author |
| PROPS-END |
| EOF |
| test-svn-fe extraprop.dump >stream && |
| git fast-import <stream && |
| git log -p --format=%an HEAD >actual && |
| test_cmp expect actual |
| ' |
| |
| test_expect_failure 'timestamp and empty file' ' |
| echo author@example.com >expect.author && |
| echo 1999-01-01 >expect.date && |
| echo file >expect.files && |
| reinit_git && |
| { |
| properties \ |
| svn:author author@example.com \ |
| svn:date "1999-01-01T00:01:002.000000Z" \ |
| svn:log "add empty file" && |
| echo PROPS-END |
| } >props && |
| { |
| cat <<-EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| EOF |
| echo Prop-content-length: $(wc -c <props) && |
| echo Content-length: $(wc -c <props) && |
| echo && |
| cat props && |
| cat <<-\EOF |
| |
| Node-path: empty-file |
| Node-kind: file |
| Node-action: add |
| Content-length: 0 |
| |
| EOF |
| } >emptyfile.dump && |
| test-svn-fe emptyfile.dump >stream && |
| git fast-import <stream && |
| git log --format=%an HEAD >actual.author && |
| git log --date=short --format=%ad HEAD >actual.date && |
| git ls-tree -r --name-only HEAD >actual.files && |
| test_cmp expect.author actual.author && |
| test_cmp expect.date actual.date && |
| test_cmp expect.files actual.files && |
| git checkout HEAD empty-file && |
| test_cmp empty file |
| ' |
| |
| test_expect_success 'directory with files' ' |
| reinit_git && |
| printf "%s\n" directory/file1 directory/file2 >expect.files && |
| echo hi >hi && |
| echo hello >hello && |
| { |
| properties \ |
| svn:author author@example.com \ |
| svn:date "1999-02-01T00:01:002.000000Z" \ |
| svn:log "add directory with some files in it" && |
| echo PROPS-END |
| } >props && |
| { |
| cat <<-EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| EOF |
| echo Prop-content-length: $(wc -c <props) && |
| echo Content-length: $(wc -c <props) && |
| echo && |
| cat props && |
| cat <<-\EOF && |
| |
| Node-path: directory |
| Node-kind: dir |
| Node-action: add |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: directory/file1 |
| Node-kind: file |
| Node-action: add |
| EOF |
| text_no_props hello && |
| cat <<-\EOF && |
| Node-path: directory/file2 |
| Node-kind: file |
| Node-action: add |
| EOF |
| text_no_props hi |
| } >directory.dump && |
| test-svn-fe directory.dump >stream && |
| git fast-import <stream && |
| |
| git ls-tree -r --name-only HEAD >actual.files && |
| git checkout HEAD directory && |
| test_cmp expect.files actual.files && |
| test_cmp hello directory/file1 && |
| test_cmp hi directory/file2 |
| ' |
| |
| test_expect_success 'node without action' ' |
| cat >inaction.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: directory |
| Node-kind: dir |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| EOF |
| test_must_fail test-svn-fe inaction.dump |
| ' |
| |
| test_expect_success 'action: add node without text' ' |
| cat >textless.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: textless |
| Node-kind: file |
| Node-action: add |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| EOF |
| test_must_fail test-svn-fe textless.dump |
| ' |
| |
| test_expect_failure 'change file mode but keep old content' ' |
| reinit_git && |
| cat >expect <<-\EOF && |
| OBJID |
| :120000 100644 OBJID OBJID T greeting |
| OBJID |
| :100644 120000 OBJID OBJID T greeting |
| OBJID |
| :000000 100644 OBJID OBJID A greeting |
| EOF |
| echo "link hello" >expect.blob && |
| echo hello >hello && |
| cat >filemode.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: add |
| Prop-content-length: 10 |
| Text-content-length: 11 |
| Content-length: 21 |
| |
| PROPS-END |
| link hello |
| |
| Revision-number: 2 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: change |
| Prop-content-length: 33 |
| Content-length: 33 |
| |
| K 11 |
| svn:special |
| V 1 |
| * |
| PROPS-END |
| |
| Revision-number: 3 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: change |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| EOF |
| test-svn-fe filemode.dump >stream && |
| git fast-import <stream && |
| { |
| git rev-list HEAD | |
| git diff-tree --root --stdin | |
| sed "s/$_x40/OBJID/g" |
| } >actual && |
| git show HEAD:greeting >actual.blob && |
| git show HEAD^:greeting >actual.target && |
| test_cmp expect actual && |
| test_cmp expect.blob actual.blob && |
| test_cmp hello actual.target |
| ' |
| |
| test_expect_success 'NUL in property value' ' |
| reinit_git && |
| echo "commit message" >expect.message && |
| { |
| properties \ |
| unimportant "something with a NUL (Q)" \ |
| svn:log "commit message"&& |
| echo PROPS-END |
| } | |
| q_to_nul >props && |
| { |
| cat <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| EOF |
| echo Prop-content-length: $(wc -c <props) && |
| echo Content-length: $(wc -c <props) && |
| echo && |
| cat props |
| } >nulprop.dump && |
| test-svn-fe nulprop.dump >stream && |
| git fast-import <stream && |
| git diff-tree --always -s --format=%s HEAD >actual.message && |
| test_cmp expect.message actual.message |
| ' |
| |
| test_expect_success 'NUL in log message, file content, and property name' ' |
| # Caveat: svnadmin 1.6.16 (r1073529) truncates at \0 in the |
| # svn:specialQnotreally example. |
| reinit_git && |
| cat >expect <<-\EOF && |
| OBJID |
| :100644 100644 OBJID OBJID M greeting |
| OBJID |
| :000000 100644 OBJID OBJID A greeting |
| EOF |
| printf "\n%s\n" "something with an ASCII NUL (Q)" >expect.message && |
| printf "%s\n" "helQo" >expect.hello1 && |
| printf "%s\n" "link hello" >expect.hello2 && |
| { |
| properties svn:log "something with an ASCII NUL (Q)" && |
| echo PROPS-END |
| } | |
| q_to_nul >props && |
| { |
| q_to_nul <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: add |
| Prop-content-length: 10 |
| Text-content-length: 6 |
| Content-length: 16 |
| |
| PROPS-END |
| helQo |
| |
| Revision-number: 2 |
| EOF |
| echo Prop-content-length: $(wc -c <props) && |
| echo Content-length: $(wc -c <props) && |
| echo && |
| cat props && |
| q_to_nul <<-\EOF |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: change |
| Prop-content-length: 43 |
| Text-content-length: 11 |
| Content-length: 54 |
| |
| K 21 |
| svn:specialQnotreally |
| V 1 |
| * |
| PROPS-END |
| link hello |
| EOF |
| } >8bitclean.dump && |
| test-svn-fe 8bitclean.dump >stream && |
| git fast-import <stream && |
| { |
| git rev-list HEAD | |
| git diff-tree --root --stdin | |
| sed "s/$_x40/OBJID/g" |
| } >actual && |
| { |
| git cat-file commit HEAD | nul_to_q && |
| echo |
| } | |
| sed -ne "/^\$/,\$ p" >actual.message && |
| git cat-file blob HEAD^:greeting | nul_to_q >actual.hello1 && |
| git cat-file blob HEAD:greeting | nul_to_q >actual.hello2 && |
| test_cmp expect actual && |
| test_cmp expect.message actual.message && |
| test_cmp expect.hello1 actual.hello1 && |
| test_cmp expect.hello2 actual.hello2 |
| ' |
| |
| test_expect_success 'change file mode and reiterate content' ' |
| reinit_git && |
| cat >expect <<-\EOF && |
| OBJID |
| :120000 100644 OBJID OBJID T greeting |
| OBJID |
| :100644 120000 OBJID OBJID T greeting |
| OBJID |
| :000000 100644 OBJID OBJID A greeting |
| EOF |
| echo "link hello" >expect.blob && |
| echo hello >hello && |
| cat >filemode.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: add |
| Prop-content-length: 10 |
| Text-content-length: 11 |
| Content-length: 21 |
| |
| PROPS-END |
| link hello |
| |
| Revision-number: 2 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: change |
| Prop-content-length: 33 |
| Text-content-length: 11 |
| Content-length: 44 |
| |
| K 11 |
| svn:special |
| V 1 |
| * |
| PROPS-END |
| link hello |
| |
| Revision-number: 3 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: change |
| Prop-content-length: 10 |
| Text-content-length: 11 |
| Content-length: 21 |
| |
| PROPS-END |
| link hello |
| EOF |
| test-svn-fe filemode.dump >stream && |
| git fast-import <stream && |
| { |
| git rev-list HEAD | |
| git diff-tree --root --stdin | |
| sed "s/$_x40/OBJID/g" |
| } >actual && |
| git show HEAD:greeting >actual.blob && |
| git show HEAD^:greeting >actual.target && |
| test_cmp expect actual && |
| test_cmp expect.blob actual.blob && |
| test_cmp hello actual.target |
| ' |
| |
| test_expect_success 'deltas not supported' ' |
| { |
| # (old) h + (inline) ello + (old) \n |
| printf "SVNQ%b%b%s" "Q\003\006\005\004" "\001Q\0204\001\002" "ello" | |
| q_to_nul |
| } >delta && |
| { |
| properties \ |
| svn:author author@example.com \ |
| svn:date "1999-01-05T00:01:002.000000Z" \ |
| svn:log "add greeting" && |
| echo PROPS-END |
| } >props && |
| { |
| properties \ |
| svn:author author@example.com \ |
| svn:date "1999-01-06T00:01:002.000000Z" \ |
| svn:log "change it" && |
| echo PROPS-END |
| } >props2 && |
| { |
| echo SVN-fs-dump-format-version: 3 && |
| echo && |
| echo Revision-number: 1 && |
| echo Prop-content-length: $(wc -c <props) && |
| echo Content-length: $(wc -c <props) && |
| echo && |
| cat props && |
| cat <<-\EOF && |
| |
| Node-path: hello |
| Node-kind: file |
| Node-action: add |
| Prop-content-length: 10 |
| Text-content-length: 3 |
| Content-length: 13 |
| |
| PROPS-END |
| hi |
| |
| EOF |
| echo Revision-number: 2 && |
| echo Prop-content-length: $(wc -c <props2) && |
| echo Content-length: $(wc -c <props2) && |
| echo && |
| cat props2 && |
| cat <<-\EOF && |
| |
| Node-path: hello |
| Node-kind: file |
| Node-action: change |
| Text-delta: true |
| Prop-content-length: 10 |
| EOF |
| echo Text-content-length: $(wc -c <delta) && |
| echo Content-length: $((10 + $(wc -c <delta))) && |
| echo && |
| echo PROPS-END && |
| cat delta |
| } >delta.dump && |
| test_must_fail test-svn-fe delta.dump |
| ' |
| |
| test_expect_success 'property deltas supported' ' |
| reinit_git && |
| cat >expect <<-\EOF && |
| OBJID |
| :100755 100644 OBJID OBJID M script.sh |
| EOF |
| { |
| properties \ |
| svn:author author@example.com \ |
| svn:date "1999-03-06T00:01:002.000000Z" \ |
| svn:log "make an executable, or chmod -x it" && |
| echo PROPS-END |
| } >revprops && |
| { |
| echo SVN-fs-dump-format-version: 3 && |
| echo && |
| echo Revision-number: 1 && |
| echo Prop-content-length: $(wc -c <revprops) && |
| echo Content-length: $(wc -c <revprops) && |
| echo && |
| cat revprops && |
| echo && |
| cat <<-\EOF && |
| Node-path: script.sh |
| Node-kind: file |
| Node-action: add |
| Text-content-length: 0 |
| Prop-content-length: 39 |
| Content-length: 39 |
| |
| K 14 |
| svn:executable |
| V 4 |
| true |
| PROPS-END |
| |
| EOF |
| echo Revision-number: 2 && |
| echo Prop-content-length: $(wc -c <revprops) && |
| echo Content-length: $(wc -c <revprops) && |
| echo && |
| cat revprops && |
| echo && |
| cat <<-\EOF |
| Node-path: script.sh |
| Node-kind: file |
| Node-action: change |
| Prop-delta: true |
| Prop-content-length: 30 |
| Content-length: 30 |
| |
| D 14 |
| svn:executable |
| PROPS-END |
| EOF |
| } >propdelta.dump && |
| test-svn-fe propdelta.dump >stream && |
| git fast-import <stream && |
| { |
| git rev-list HEAD | |
| git diff-tree --stdin | |
| sed "s/$_x40/OBJID/g" |
| } >actual && |
| test_cmp expect actual |
| ' |
| |
| test_expect_success 'properties on /' ' |
| reinit_git && |
| cat <<-\EOF >expect && |
| OBJID |
| OBJID |
| :000000 100644 OBJID OBJID A greeting |
| EOF |
| sed -e "s/X$//" <<-\EOF >changeroot.dump && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: greeting |
| Node-kind: file |
| Node-action: add |
| Text-content-length: 0 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Revision-number: 2 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: X |
| Node-kind: dir |
| Node-action: change |
| Prop-delta: true |
| Prop-content-length: 43 |
| Content-length: 43 |
| |
| K 10 |
| svn:ignore |
| V 11 |
| build-area |
| |
| PROPS-END |
| EOF |
| test-svn-fe changeroot.dump >stream && |
| git fast-import <stream && |
| { |
| git rev-list HEAD | |
| git diff-tree --root --always --stdin | |
| sed "s/$_x40/OBJID/g" |
| } >actual && |
| test_cmp expect actual |
| ' |
| |
| test_expect_success 'deltas for typechange' ' |
| reinit_git && |
| cat >expect <<-\EOF && |
| OBJID |
| :120000 100644 OBJID OBJID T test-file |
| OBJID |
| :100755 120000 OBJID OBJID T test-file |
| OBJID |
| :000000 100755 OBJID OBJID A test-file |
| EOF |
| cat >deleteprop.dump <<-\EOF && |
| SVN-fs-dump-format-version: 3 |
| |
| Revision-number: 1 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: test-file |
| Node-kind: file |
| Node-action: add |
| Prop-delta: true |
| Prop-content-length: 35 |
| Text-content-length: 17 |
| Content-length: 52 |
| |
| K 14 |
| svn:executable |
| V 0 |
| |
| PROPS-END |
| link testing 123 |
| |
| Revision-number: 2 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: test-file |
| Node-kind: file |
| Node-action: change |
| Prop-delta: true |
| Prop-content-length: 53 |
| Text-content-length: 17 |
| Content-length: 70 |
| |
| K 11 |
| svn:special |
| V 1 |
| * |
| D 14 |
| svn:executable |
| PROPS-END |
| link testing 231 |
| |
| Revision-number: 3 |
| Prop-content-length: 10 |
| Content-length: 10 |
| |
| PROPS-END |
| |
| Node-path: test-file |
| Node-kind: file |
| Node-action: change |
| Prop-delta: true |
| Prop-content-length: 27 |
| Text-content-length: 17 |
| Content-length: 44 |
| |
| D 11 |
| svn:special |
| PROPS-END |
| link testing 321 |
| EOF |
| test-svn-fe deleteprop.dump >stream && |
| git fast-import <stream && |
| { |
| git rev-list HEAD | |
| git diff-tree --root --stdin | |
| sed "s/$_x40/OBJID/g" |
| } >actual && |
| test_cmp expect actual |
| ' |
| |
| |
| test_expect_success 'set up svn repo' ' |
| svnconf=$PWD/svnconf && |
| mkdir -p "$svnconf" && |
| |
| if |
| svnadmin -h >/dev/null 2>&1 && |
| svnadmin create simple-svn && |
| svnadmin load simple-svn <"$TEST_DIRECTORY/t9135/svn.dump" && |
| svn export --config-dir "$svnconf" "file://$PWD/simple-svn" simple-svnco |
| then |
| test_set_prereq SVNREPO |
| fi |
| ' |
| |
| test_expect_success SVNREPO 't9135/svn.dump' ' |
| git init simple-git && |
| test-svn-fe "$TEST_DIRECTORY/t9135/svn.dump" >simple.fe && |
| ( |
| cd simple-git && |
| git fast-import <../simple.fe |
| ) && |
| ( |
| cd simple-svnco && |
| git init && |
| git add . && |
| git fetch ../simple-git master && |
| git diff --exit-code FETCH_HEAD |
| ) |
| ' |
| |
| test_done |