-
Notifications
You must be signed in to change notification settings - Fork 120
/
Copy pathgit-svn-migrate.sh
executable file
·245 lines (223 loc) · 7.79 KB
/
git-svn-migrate.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
#!/bin/bash
# Copyright 2010-2011 John Albin Wilkins and contributors.
# Available under the GPL v2 license. See LICENSE.txt.
script=`basename $0`;
dir=`pwd`/`dirname $0`;
usage=$(cat <<EOF_USAGE
USAGE: $script --url-file=<filename> --authors-file=<filename> [destination folder]
\n
\nFor more info, see: $script --help
EOF_USAGE
);
help=$(cat <<EOF_HELP
NAME
\n\t$script - Migrates Subversion repositories to Git
\n
\nSYNOPSIS
\n\t$script [options] [arguments]
\n
\nDESCRIPTION
\n\tThe $script utility migrates a list of Subversion
\n\trepositories to Git using the specified authors list. The
\n\turl-file and authors-file parameters are required. The
\n\tdestination folder is optional and can be specified as an
\n\targument or as a named parameter.
\n
\n\tThe following options are available:
\n
\n\t-u=<filename>, -u <filename>,
\n\t--url-file=<filename>, --url-file <filename>
\n\t\tSpecify the file containing the Subversion repository list.
\n
\n\t-a=<filename>, -a <filename>,
\n\t--authors-file=[filename], --authors-file [filename]
\n\t\tSpecify the file containing the authors transformation data.
\n
\n\t-d=<folder>, -d <folder>,
\n\t--destination=<folder>, --destination <folder>
\n\t\tThe directory where the new Git repositories should be
\n\t\tsaved. Defaults to the current directory.
\n
\n\t-i=<filename>, -i <filename>,
\n\t--ignore-file=<filename>, --ignore-file <filename>
\n\t\tThe location of a .gitignore file to add to all repositories.
\n
\n\t--quiet
\n\t\tBy default this script is rather verbose since it outputs each revision
\n\t\tnumber as it is processed from Subversion. Since conversion can sometimes
\n\t\ttake hours to complete, this output can be useful. However, this option
\n\t\twill surpress that output.
\n
\n\t--no-metadata
\n\t\tBy default, all converted log messages will include a line starting with
\n\t\t"git-svn-id:" which makes it easy to track down old references to
\n\t\tSubversion revision numbers in existing documentation, bug reports and
\n\t\tarchives. Use this option to get rid of that data. See git svn --help for
\n\t\ta fuller discussion on this option.
\n
\n\t--shared[=(false|true|umask|group|all|world|everybody|0xxx)]
\n\t\tSpecify that the generated git repositories are to be shared amongst
\n\t\tseveral users. See git init --help for more info about this option.
\n
\n\tAny additional options are assumed to be git-svn options and will be passed
\n\talong to that utility directly. Some useful git-svn options are:
\n\t\t--trunk --tags --branches --no-minimize-url
\n\tSee git svn --help for more info about its options.
\n
\nBASIC EXAMPLES
\n\t# Use the long parameter names
\n\t$script --url-file=my-repository-list.txt --authors-file=authors-file.txt --destination=/var/git
\n
\n\t# Use short parameter names
\n\t$script -u my-repository-list.txt -a authors-file.txt /var/git
\n
\nSEE ALSO
\n\tfetch-svn-authors.sh
\n\tsvn-lookup-author.sh
EOF_HELP
);
# Set defaults for any optional parameters or arguments.
destination='.';
gitinit_params='';
gitsvn_params='';
# Process parameters.
until [[ -z "$1" ]]; do
option=$1;
# Strip off leading '--' or '-'.
if [[ ${option:0:1} == '-' ]]; then
flag_delimiter='-';
if [[ ${option:0:2} == '--' ]]; then
tmp=${option:2};
flag_delimiter='--';
else
tmp=${option:1};
fi
else
# Any argument given is assumed to be the destination folder.
tmp="destination=$option";
fi
parameter=${tmp%%=*}; # Extract option's name.
value=${tmp##*=}; # Extract option's value.
# If a value is expected, but not specified inside the parameter, grab the next param.
if [[ $value == $tmp ]]; then
if [[ ${2:0:1} == '-' ]]; then
# The next parameter is a new option, so unset the value.
value='';
else
value=$2;
shift;
fi
fi
case $parameter in
u ) url_file=$value;;
url-file ) url_file=$value;;
a ) authors_file=$value;;
authors-file ) authors_file=$value;;
d ) destination=$value;;
destination ) destination=$value;;
i ) ignore_file=$value;;
ignore-file ) ignore_file=$value;;
shared ) if [[ $value == '' ]]; then
gitinit_params="--shared";
else
gitinit_params="--shared=$value";
fi
;;
h ) echo -e $help | less >&2; exit;;
help ) echo -e $help | less >&2; exit;;
* ) # Pass any unknown parameters to git-svn directly.
if [[ $value == '' ]]; then
gitsvn_params="$gitsvn_params $flag_delimiter$parameter";
else
gitsvn_params="$gitsvn_params $flag_delimiter$parameter=$value";
fi;;
esac
# Remove the processed parameter.
shift;
done
# Check for required parameters.
if [[ $url_file == '' || $authors_file == '' ]]; then
echo -e $usage >&2;
exit 1;
fi
# Check for valid files.
if [[ ! -f $url_file ]]; then
echo "Specified URL file \"$url_file\" does not exist or is not a file." >&2;
echo -e $usage >&2;
exit 1;
fi
if [[ ! -f $authors_file ]]; then
echo "Specified authors file \"$authors_file\" does not exist or is not a file." >&2;
echo -e $usage >&2;
exit 1;
fi
# Process each URL in the repository list.
pwd=`pwd`;
tmp_destination="$pwd/tmp-git-repo";
mkdir -p $destination;
destination=`cd $destination; pwd`; #Absolute path.
# Ensure temporary repository location is empty.
if [[ -e $tmp_destination ]]; then
echo "Temporary repository location \"$tmp_destination\" already exists. Exiting." >&2;
exit 1;
fi
while read line
do
# Check for 2-field format: Name [tab] URL
name=`echo $line | awk '{print $1}'`;
url=`echo $line | awk '{print $2}'`;
# Check for simple 1-field format: URL
if [[ $url == '' ]]; then
url=$name;
name=`basename $url`;
fi
# Process each Subversion URL.
echo >&2;
echo "At $(date)..." >&2;
echo "Processing \"$name\" repository at $url..." >&2;
# Init the final bare repository.
mkdir $destination/$name.git;
cd $destination/$name.git;
git init --bare $gitinit_params;
git symbolic-ref HEAD refs/heads/trunk;
# Clone the original Subversion repository to a temp repository.
cd $pwd;
echo "- Cloning repository..." >&2;
git svn clone $url -A $authors_file --authors-prog=$dir/svn-lookup-author.sh --stdlayout --quiet $gitsvn_params $tmp_destination;
# Create .gitignore file.
echo "- Converting svn:ignore properties into a .gitignore file..." >&2;
if [[ $ignore_file != '' ]]; then
cp $ignore_file $tmp_destination/.gitignore;
fi
cd $tmp_destination;
git svn show-ignore --id trunk >> .gitignore;
git add .gitignore;
git commit --author="git-svn-migrate <[email protected]>" -m 'Convert svn:ignore properties to .gitignore.';
# Push to final bare repository and remove temp repository.
echo "- Pushing to new bare repository..." >&2;
git remote add bare $destination/$name.git;
git config remote.bare.push 'refs/remotes/*:refs/heads/*';
git push bare;
# Push the .gitignore commit that resides on master.
git push bare master:trunk;
cd $pwd;
rm -r $tmp_destination;
# Rename Subversion's "trunk" branch to Git's standard "master" branch.
cd $destination/$name.git;
git branch -m trunk master;
# Remove bogus branches of the form "name@REV".
git for-each-ref --format='%(refname)' refs/heads | grep '@[0-9][0-9]*' | cut -d / -f 3- |
while read ref
do
git branch -D "$ref";
done
# Convert git-svn tag branches to proper tags.
echo "- Converting svn tag directories to proper git tags..." >&2;
git for-each-ref --format='%(refname)' refs/heads/tags | cut -d / -f 4 |
while read ref
do
git tag -a "$ref" -m "Convert \"$ref\" to a proper git tag." "refs/heads/tags/$ref";
git branch -D "tags/$ref";
done
echo "- Conversion completed at $(date)." >&2;
done < $url_file