summaryrefslogtreecommitdiff
path: root/git-rewrite-branch
blob: a2b3c811609a6459fee7e08694840a91ae0cba88 (plain)
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
#!/bin/bash -euE

# when $gitmode is true, $ibranch's commits are used as IDs
gitmode=true
tag='git-rewrite-id'

ibranch=''
obranch=''
wbranch=''

usage() {
	echo 'malformed call to internal function'
}

verbose() {
	: "$*"
	#echo "$*"
}

################################################################################

id2commit() {
	[[ $# = 2 ]] || { usage; return 1; }
	local branch=$1
	local id=$2
	if [[ $branch == $ibranch ]] && $gitmode; then
		printf '%s\n' "$id"
	else
		git log "$branch" --pretty=format:'%H' --grep "${tag}: ${id}"
	fi
}

commit2id() {
	[[ $# = 2 ]] || { usage; return 1; }
	local branch=$1
	local commit=$2
	if [[ $branch == $ibranch ]] && $gitmode; then
		printf '%s\n' "$commit"
	else
		git log "$branch" -n1 --pretty=formtat:'%B' "$commit" | sed -n "s|^\s*${tag}: ||p"
	fi
}

# commit2commit
c2c() {
	[[ $# = 3 ]] || { usage; return 1; }
	local from=$1
	local to=$2
	local commit=$3
	if [[ "$from" == "$to" ]]; then
		# optimization
		# also, properly normalizes $ibranch when $gitmode=true
		git log "$commit" -n1 --pretty=format:'%H'
	else
		id2commit "$to" "$(commit2id "$from" "$commit")"
	fi
}

################################################################################

main() {
	# Parse command line arguments #########################################
	if [[ $1 = '--svn' ]]; then
		gitmode=false
		tag='git-svn-id'
		shift
	fi

	if [[ $# < 3 ]]; then
		usage
		exit 1
	fi

	ibranch=$1
	obranch=$2
	shift 2

	local filters=();
	if $gitmode; then
		filters=(--msg-filter "git-rewrite-branch--appendid '${tag}'")
	fi
	filters+=("$@")

	# Main #################################################################

	if git checkout "$obranch" -- 2>/dev/null; then
		# obranch exists, update it
		echo 'Updating existing rewritten branch...'

		local icommit="$(c2c "$ibranch" "$ibranch" "$ibranch")"
		local ocommit="$(c2c "$obranch" "$ibranch" "$obranch")"
		if [[ "$icommit" == "$ocommit" ]]; then
			echo "Nothing to do"
			return 0
		fi

		# Just to be safe
		git branch -D "$obranch.tmp" 2>/dev/null || true

		wbranch="$obranch.tmp"
		local common="$(c2c "$obranch" "$ibranch" "$obranch")"
		if c2c "$ibranch" "$ibranch" "${common}^" &>/dev/null; then
			revlist="$(c2c "$obranch" "$ibranch" "$obranch")^..${wbranch}"
		else
			# There is only one commit from $ibranch in $obranch
			revlist="$wbranch"
		fi
	else
		# obranch does not exist, create it
		echo 'Creating new rewritten branch...'
		wbranch=$obranch
		revlist=$wbranch
	fi

	git checkout "$ibranch"
	git checkout -b "$wbranch"
	git filter-branch -f "${filters[@]}" "$revlist"

	if [[ "$obranch" != "$wbranch" ]]; then
		# rebase the changes in wbranch onto obranch
		echo 'Rebasing rewrites onto existing branch...'
		local wcommit="$(c2c "$wbranch" "$ibranch" "$wbranch")"
		if [[ "$wcommit" == "$ocommit" ]]; then
			echo "Nothing to do"
			return 0
		fi

		local commonish="$(c2c "$obranch" "$wbranch" "$obranch")"
		cmd=(git rebase --onto "$obranch" "$commonish" "$wbranch")

		verbose
		verbose "  o---o---o $obranch"
		verbose '          :'
		verbose "  o---o---o---o---o $ibranch"
		verbose '       \  :'
		verbose '        `-C---o---o '"$wbranch"
		verbose
		verbose "  C = $commonish"
		verbose
		verbose "  ${cmd[*]}"
		verbose

		"${cmd[@]}"
		git checkout "$obranch"
		git branch -d "$wbranch"
	fi
}

main "$@"