Dmitry Marakasov | 6912ea9 | 2013-05-21 00:24:34 +0400 | [diff] [blame] | 1 | #!/bin/sh |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 2 | # |
| 3 | # git-subtree.sh: split/join git repositories in subdirectories of this one |
| 4 | # |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 5 | # Copyright (C) 2009 Avery Pennarun <apenwarr@gmail.com> |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 6 | # |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 7 | if test $# -eq 0 |
| 8 | then |
| 9 | set -- -h |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 10 | fi |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 11 | OPTS_SPEC="\ |
Avery Pennarun | f4f2955 | 2009-05-30 01:10:14 -0400 | [diff] [blame] | 12 | git subtree add --prefix=<prefix> <commit> |
Anthony Baire | 1c3e0f0 | 2013-11-27 19:34:09 +0100 | [diff] [blame] | 13 | git subtree add --prefix=<prefix> <repository> <ref> |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 14 | git subtree merge --prefix=<prefix> <commit> |
Anthony Baire | 1c3e0f0 | 2013-11-27 19:34:09 +0100 | [diff] [blame] | 15 | git subtree pull --prefix=<prefix> <repository> <ref> |
| 16 | git subtree push --prefix=<prefix> <repository> <ref> |
Denton Liu | 77128ed | 2019-03-11 02:47:17 -0700 | [diff] [blame] | 17 | git subtree split --prefix=<prefix> <commit> |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 18 | -- |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 19 | h,help show the help |
| 20 | q quiet |
Avery Pennarun | 942dce5 | 2009-04-26 18:06:08 -0400 | [diff] [blame] | 21 | d show debug messages |
Jakub Suder | 6e25f79 | 2010-01-12 22:38:21 +0100 | [diff] [blame] | 22 | P,prefix= the name of the subdir to split out |
Jakub Suder | 2da0969 | 2010-01-09 19:55:35 +0100 | [diff] [blame] | 23 | m,message= use the given message as the commit message for the merge commit |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 24 | options for 'split' |
Avery Pennarun | d0eb1b1 | 2009-04-26 08:59:12 -0400 | [diff] [blame] | 25 | annotate= add a prefix to commit message of new commits |
Avery Pennarun | 43a3951 | 2009-05-30 01:05:43 -0400 | [diff] [blame] | 26 | b,branch= create a new branch from the split subtree |
| 27 | ignore-joins ignore prior --rejoin commits |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 28 | onto= try connecting new tree to an existing one |
| 29 | rejoin merge the new branch back into HEAD |
Danny Lin | 6ccc71a | 2015-05-07 15:52:33 +0800 | [diff] [blame] | 30 | options for 'add', 'merge', and 'pull' |
Avery Pennarun | 8e79043 | 2009-05-30 00:48:07 -0400 | [diff] [blame] | 31 | squash merge subtree changes as a single commit |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 32 | " |
Cole Stanfield | 11f1511 | 2010-09-13 13:16:57 -0600 | [diff] [blame] | 33 | eval "$(echo "$OPTS_SPEC" | git rev-parse --parseopt -- "$@" || echo exit $?)" |
Avery Pennarun | 9c632ea | 2010-06-24 01:53:05 -0400 | [diff] [blame] | 34 | |
Avery Pennarun | df2302d | 2010-06-24 16:57:58 -0400 | [diff] [blame] | 35 | PATH=$PATH:$(git --exec-path) |
Avery Pennarun | 33aaa69 | 2009-08-26 10:41:03 -0400 | [diff] [blame] | 36 | . git-sh-setup |
Avery Pennarun | 9c632ea | 2010-06-24 01:53:05 -0400 | [diff] [blame] | 37 | |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 38 | require_work_tree |
| 39 | |
| 40 | quiet= |
Avery Pennarun | 43a3951 | 2009-05-30 01:05:43 -0400 | [diff] [blame] | 41 | branch= |
Avery Pennarun | 942dce5 | 2009-04-26 18:06:08 -0400 | [diff] [blame] | 42 | debug= |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 43 | command= |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 44 | onto= |
| 45 | rejoin= |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 46 | ignore_joins= |
Avery Pennarun | d0eb1b1 | 2009-04-26 08:59:12 -0400 | [diff] [blame] | 47 | annotate= |
Avery Pennarun | 8e79043 | 2009-05-30 00:48:07 -0400 | [diff] [blame] | 48 | squash= |
Jakub Suder | 2da0969 | 2010-01-09 19:55:35 +0100 | [diff] [blame] | 49 | message= |
Jeff King | cba5e28 | 2014-03-17 15:59:47 -0400 | [diff] [blame] | 50 | prefix= |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 51 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 52 | debug () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 53 | if test -n "$debug" |
| 54 | then |
Danny Lin | 2ded109 | 2015-05-08 08:56:59 +0800 | [diff] [blame] | 55 | printf "%s\n" "$*" >&2 |
Avery Pennarun | 942dce5 | 2009-04-26 18:06:08 -0400 | [diff] [blame] | 56 | fi |
| 57 | } |
| 58 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 59 | say () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 60 | if test -z "$quiet" |
| 61 | then |
Danny Lin | 2ded109 | 2015-05-08 08:56:59 +0800 | [diff] [blame] | 62 | printf "%s\n" "$*" >&2 |
| 63 | fi |
| 64 | } |
| 65 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 66 | progress () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 67 | if test -z "$quiet" |
| 68 | then |
Danny Lin | 2ded109 | 2015-05-08 08:56:59 +0800 | [diff] [blame] | 69 | printf "%s\r" "$*" >&2 |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 70 | fi |
| 71 | } |
| 72 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 73 | assert () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 74 | if ! "$@" |
| 75 | then |
Avery Pennarun | 2573354 | 2009-04-24 14:24:38 -0400 | [diff] [blame] | 76 | die "assertion failed: " "$@" |
| 77 | fi |
| 78 | } |
| 79 | |
Denton Liu | 77128ed | 2019-03-11 02:47:17 -0700 | [diff] [blame] | 80 | ensure_single_rev () { |
| 81 | if test $# -ne 1 |
| 82 | then |
| 83 | die "You must provide exactly one revision. Got: '$@'" |
| 84 | fi |
| 85 | } |
Avery Pennarun | 2573354 | 2009-04-24 14:24:38 -0400 | [diff] [blame] | 86 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 87 | while test $# -gt 0 |
| 88 | do |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 89 | opt="$1" |
| 90 | shift |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 91 | |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 92 | case "$opt" in |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 93 | -q) |
| 94 | quiet=1 |
| 95 | ;; |
| 96 | -d) |
| 97 | debug=1 |
| 98 | ;; |
| 99 | --annotate) |
| 100 | annotate="$1" |
| 101 | shift |
| 102 | ;; |
| 103 | --no-annotate) |
| 104 | annotate= |
| 105 | ;; |
| 106 | -b) |
| 107 | branch="$1" |
| 108 | shift |
| 109 | ;; |
| 110 | -P) |
| 111 | prefix="${1%/}" |
| 112 | shift |
| 113 | ;; |
| 114 | -m) |
| 115 | message="$1" |
| 116 | shift |
| 117 | ;; |
| 118 | --no-prefix) |
| 119 | prefix= |
| 120 | ;; |
| 121 | --onto) |
| 122 | onto="$1" |
| 123 | shift |
| 124 | ;; |
| 125 | --no-onto) |
| 126 | onto= |
| 127 | ;; |
| 128 | --rejoin) |
| 129 | rejoin=1 |
| 130 | ;; |
| 131 | --no-rejoin) |
| 132 | rejoin= |
| 133 | ;; |
| 134 | --ignore-joins) |
| 135 | ignore_joins=1 |
| 136 | ;; |
| 137 | --no-ignore-joins) |
| 138 | ignore_joins= |
| 139 | ;; |
| 140 | --squash) |
| 141 | squash=1 |
| 142 | ;; |
| 143 | --no-squash) |
| 144 | squash= |
| 145 | ;; |
| 146 | --) |
| 147 | break |
| 148 | ;; |
| 149 | *) |
| 150 | die "Unexpected option: $opt" |
| 151 | ;; |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 152 | esac |
| 153 | done |
| 154 | |
| 155 | command="$1" |
| 156 | shift |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 157 | |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 158 | case "$command" in |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 159 | add|merge|pull) |
| 160 | default= |
| 161 | ;; |
| 162 | split|push) |
| 163 | default="--default HEAD" |
| 164 | ;; |
| 165 | *) |
| 166 | die "Unknown command '$command'" |
| 167 | ;; |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 168 | esac |
| 169 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 170 | if test -z "$prefix" |
| 171 | then |
Avery Pennarun | 9a8821f | 2009-04-24 22:57:14 -0400 | [diff] [blame] | 172 | die "You must provide the --prefix option." |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 173 | fi |
Win Treese | ec54f0d | 2010-02-05 22:02:43 -0500 | [diff] [blame] | 174 | |
Avery Pennarun | 77ba305 | 2010-02-08 15:00:42 -0500 | [diff] [blame] | 175 | case "$command" in |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 176 | add) |
| 177 | test -e "$prefix" && |
| 178 | die "prefix '$prefix' already exists." |
| 179 | ;; |
| 180 | *) |
| 181 | test -e "$prefix" || |
| 182 | die "'$prefix' does not exist; use 'git subtree add'" |
| 183 | ;; |
Avery Pennarun | 77ba305 | 2010-02-08 15:00:42 -0500 | [diff] [blame] | 184 | esac |
Win Treese | ec54f0d | 2010-02-05 22:02:43 -0500 | [diff] [blame] | 185 | |
Avery Pennarun | 6f2012c | 2009-10-02 15:22:15 -0400 | [diff] [blame] | 186 | dir="$(dirname "$prefix/.")" |
Avery Pennarun | 9a8821f | 2009-04-24 22:57:14 -0400 | [diff] [blame] | 187 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 188 | if test "$command" != "pull" && |
| 189 | test "$command" != "add" && |
| 190 | test "$command" != "push" |
| 191 | then |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 192 | revs=$(git rev-parse $default --revs-only "$@") || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 193 | dirs=$(git rev-parse --no-revs --no-flags "$@") || exit $? |
Denton Liu | 77128ed | 2019-03-11 02:47:17 -0700 | [diff] [blame] | 194 | ensure_single_rev $revs |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 195 | if test -n "$dirs" |
| 196 | then |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 197 | die "Error: Use --prefix instead of bare filenames." |
| 198 | fi |
Avery Pennarun | 9a8821f | 2009-04-24 22:57:14 -0400 | [diff] [blame] | 199 | fi |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 200 | |
| 201 | debug "command: {$command}" |
| 202 | debug "quiet: {$quiet}" |
| 203 | debug "revs: {$revs}" |
| 204 | debug "dir: {$dir}" |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 205 | debug "opts: {$*}" |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 206 | debug |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 207 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 208 | cache_setup () { |
Avery Pennarun | 2573354 | 2009-04-24 14:24:38 -0400 | [diff] [blame] | 209 | cachedir="$GIT_DIR/subtree-cache/$$" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 210 | rm -rf "$cachedir" || |
| 211 | die "Can't delete old cachedir: $cachedir" |
| 212 | mkdir -p "$cachedir" || |
| 213 | die "Can't create new cachedir: $cachedir" |
| 214 | mkdir -p "$cachedir/notree" || |
| 215 | die "Can't create new cachedir: $cachedir/notree" |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 216 | debug "Using cachedir: $cachedir" >&2 |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 217 | } |
| 218 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 219 | cache_get () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 220 | for oldrev in "$@" |
| 221 | do |
| 222 | if test -r "$cachedir/$oldrev" |
| 223 | then |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 224 | read newrev <"$cachedir/$oldrev" |
| 225 | echo $newrev |
| 226 | fi |
| 227 | done |
| 228 | } |
| 229 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 230 | cache_miss () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 231 | for oldrev in "$@" |
| 232 | do |
| 233 | if ! test -r "$cachedir/$oldrev" |
| 234 | then |
Jesse Greenwald | 915b989 | 2010-11-09 22:18:36 -0600 | [diff] [blame] | 235 | echo $oldrev |
| 236 | fi |
| 237 | done |
| 238 | } |
| 239 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 240 | check_parents () { |
Strain, Roger L | 315a84f | 2018-09-28 13:35:39 -0500 | [diff] [blame] | 241 | missed=$(cache_miss "$1") |
| 242 | local indent=$(($2 + 1)) |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 243 | for miss in $missed |
| 244 | do |
| 245 | if ! test -r "$cachedir/notree/$miss" |
| 246 | then |
Jesse Greenwald | 915b989 | 2010-11-09 22:18:36 -0600 | [diff] [blame] | 247 | debug " incorrect order: $miss" |
Strain, Roger L | 315a84f | 2018-09-28 13:35:39 -0500 | [diff] [blame] | 248 | process_split_commit "$miss" "" "$indent" |
Jesse Greenwald | 915b989 | 2010-11-09 22:18:36 -0600 | [diff] [blame] | 249 | fi |
| 250 | done |
| 251 | } |
| 252 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 253 | set_notree () { |
Jesse Greenwald | 915b989 | 2010-11-09 22:18:36 -0600 | [diff] [blame] | 254 | echo "1" > "$cachedir/notree/$1" |
| 255 | } |
| 256 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 257 | cache_set () { |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 258 | oldrev="$1" |
| 259 | newrev="$2" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 260 | if test "$oldrev" != "latest_old" && |
| 261 | test "$oldrev" != "latest_new" && |
| 262 | test -e "$cachedir/$oldrev" |
| 263 | then |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 264 | die "cache for $oldrev already exists!" |
| 265 | fi |
| 266 | echo "$newrev" >"$cachedir/$oldrev" |
| 267 | } |
| 268 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 269 | rev_exists () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 270 | if git rev-parse "$1" >/dev/null 2>&1 |
| 271 | then |
Avery Pennarun | 43a3951 | 2009-05-30 01:05:43 -0400 | [diff] [blame] | 272 | return 0 |
| 273 | else |
| 274 | return 1 |
| 275 | fi |
| 276 | } |
| 277 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 278 | rev_is_descendant_of_branch () { |
Jakub Suder | 0a56294 | 2010-01-09 19:56:05 +0100 | [diff] [blame] | 279 | newrev="$1" |
| 280 | branch="$2" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 281 | branch_hash=$(git rev-parse "$branch") |
| 282 | match=$(git rev-list -1 "$branch_hash" "^$newrev") |
Jakub Suder | 0a56294 | 2010-01-09 19:56:05 +0100 | [diff] [blame] | 283 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 284 | if test -z "$match" |
| 285 | then |
Jakub Suder | 0a56294 | 2010-01-09 19:56:05 +0100 | [diff] [blame] | 286 | return 0 |
| 287 | else |
| 288 | return 1 |
| 289 | fi |
| 290 | } |
| 291 | |
Avery Pennarun | b9de535 | 2009-04-25 00:06:45 -0400 | [diff] [blame] | 292 | # if a commit doesn't have a parent, this might not work. But we only want |
| 293 | # to remove the parent from the rev-list, and since it doesn't exist, it won't |
| 294 | # be there anyway, so do nothing in that case. |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 295 | try_remove_previous () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 296 | if rev_exists "$1^" |
| 297 | then |
Avery Pennarun | b9de535 | 2009-04-25 00:06:45 -0400 | [diff] [blame] | 298 | echo "^$1^" |
| 299 | fi |
| 300 | } |
| 301 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 302 | find_latest_squash () { |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 303 | debug "Looking for latest squash ($dir)..." |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 304 | dir="$1" |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 305 | sq= |
| 306 | main= |
| 307 | sub= |
Avery Pennarun | 6f2012c | 2009-10-02 15:22:15 -0400 | [diff] [blame] | 308 | git log --grep="^git-subtree-dir: $dir/*\$" \ |
Stephen R Guglielmo | 8841b52 | 2018-02-23 15:41:25 -0500 | [diff] [blame] | 309 | --no-show-signature --pretty=format:'START %H%n%s%n%n%b%nEND%n' HEAD | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 310 | while read a b junk |
| 311 | do |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 312 | debug "$a $b $junk" |
| 313 | debug "{{$sq/$main/$sub}}" |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 314 | case "$a" in |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 315 | START) |
| 316 | sq="$b" |
| 317 | ;; |
| 318 | git-subtree-mainline:) |
| 319 | main="$b" |
| 320 | ;; |
| 321 | git-subtree-split:) |
| 322 | sub="$(git rev-parse "$b^0")" || |
| 323 | die "could not rev-parse split hash $b from commit $sq" |
| 324 | ;; |
| 325 | END) |
| 326 | if test -n "$sub" |
| 327 | then |
| 328 | if test -n "$main" |
| 329 | then |
| 330 | # a rejoin commit? |
| 331 | # Pretend its sub was a squash. |
| 332 | sq="$sub" |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 333 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 334 | debug "Squash found: $sq $sub" |
| 335 | echo "$sq" "$sub" |
| 336 | break |
| 337 | fi |
| 338 | sq= |
| 339 | main= |
| 340 | sub= |
| 341 | ;; |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 342 | esac |
| 343 | done |
| 344 | } |
| 345 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 346 | find_existing_splits () { |
Avery Pennarun | 8b4a77f | 2009-04-24 16:48:08 -0400 | [diff] [blame] | 347 | debug "Looking for prior splits..." |
| 348 | dir="$1" |
| 349 | revs="$2" |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 350 | main= |
| 351 | sub= |
Strain, Roger L | dd21d43 | 2018-09-28 13:35:38 -0500 | [diff] [blame] | 352 | local grep_format="^git-subtree-dir: $dir/*\$" |
| 353 | if test -n "$ignore_joins" |
| 354 | then |
| 355 | grep_format="^Add '$dir/' from commit '" |
| 356 | fi |
| 357 | git log --grep="$grep_format" \ |
Stephen R Guglielmo | 8841b52 | 2018-02-23 15:41:25 -0500 | [diff] [blame] | 358 | --no-show-signature --pretty=format:'START %H%n%s%n%n%b%nEND%n' $revs | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 359 | while read a b junk |
| 360 | do |
Avery Pennarun | 8b4a77f | 2009-04-24 16:48:08 -0400 | [diff] [blame] | 361 | case "$a" in |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 362 | START) |
| 363 | sq="$b" |
| 364 | ;; |
| 365 | git-subtree-mainline:) |
| 366 | main="$b" |
| 367 | ;; |
| 368 | git-subtree-split:) |
| 369 | sub="$(git rev-parse "$b^0")" || |
| 370 | die "could not rev-parse split hash $b from commit $sq" |
| 371 | ;; |
| 372 | END) |
| 373 | debug " Main is: '$main'" |
| 374 | if test -z "$main" -a -n "$sub" |
| 375 | then |
| 376 | # squash commits refer to a subtree |
| 377 | debug " Squash: $sq from $sub" |
| 378 | cache_set "$sq" "$sub" |
| 379 | fi |
| 380 | if test -n "$main" -a -n "$sub" |
| 381 | then |
| 382 | debug " Prior: $main -> $sub" |
| 383 | cache_set $main $sub |
| 384 | cache_set $sub $sub |
| 385 | try_remove_previous "$main" |
| 386 | try_remove_previous "$sub" |
| 387 | fi |
| 388 | main= |
| 389 | sub= |
| 390 | ;; |
Avery Pennarun | 8b4a77f | 2009-04-24 16:48:08 -0400 | [diff] [blame] | 391 | esac |
| 392 | done |
| 393 | } |
| 394 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 395 | copy_commit () { |
Avery Pennarun | f96bc79 | 2009-05-30 00:47:59 -0400 | [diff] [blame] | 396 | # We're going to set some environment vars here, so |
Avery Pennarun | fd9500e | 2009-04-24 14:45:02 -0400 | [diff] [blame] | 397 | # do it in a subshell to get rid of them safely later |
Avery Pennarun | a64f3a7 | 2009-04-26 16:53:57 -0400 | [diff] [blame] | 398 | debug copy_commit "{$1}" "{$2}" "{$3}" |
Stephen R Guglielmo | 8841b52 | 2018-02-23 15:41:25 -0500 | [diff] [blame] | 399 | git log -1 --no-show-signature --pretty=format:'%an%n%ae%n%aD%n%cn%n%ce%n%cD%n%B' "$1" | |
Avery Pennarun | fd9500e | 2009-04-24 14:45:02 -0400 | [diff] [blame] | 400 | ( |
| 401 | read GIT_AUTHOR_NAME |
| 402 | read GIT_AUTHOR_EMAIL |
| 403 | read GIT_AUTHOR_DATE |
| 404 | read GIT_COMMITTER_NAME |
| 405 | read GIT_COMMITTER_EMAIL |
| 406 | read GIT_COMMITTER_DATE |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 407 | export GIT_AUTHOR_NAME \ |
| 408 | GIT_AUTHOR_EMAIL \ |
| 409 | GIT_AUTHOR_DATE \ |
| 410 | GIT_COMMITTER_NAME \ |
| 411 | GIT_COMMITTER_EMAIL \ |
| 412 | GIT_COMMITTER_DATE |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 413 | ( |
| 414 | printf "%s" "$annotate" |
| 415 | cat |
| 416 | ) | |
Avery Pennarun | fd9500e | 2009-04-24 14:45:02 -0400 | [diff] [blame] | 417 | git commit-tree "$2" $3 # reads the rest of stdin |
| 418 | ) || die "Can't copy commit $1" |
| 419 | } |
| 420 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 421 | add_msg () { |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 422 | dir="$1" |
| 423 | latest_old="$2" |
| 424 | latest_new="$3" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 425 | if test -n "$message" |
| 426 | then |
Jakub Suder | 2da0969 | 2010-01-09 19:55:35 +0100 | [diff] [blame] | 427 | commit_message="$message" |
| 428 | else |
| 429 | commit_message="Add '$dir/' from commit '$latest_new'" |
| 430 | fi |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 431 | cat <<-EOF |
Jakub Suder | 2da0969 | 2010-01-09 19:55:35 +0100 | [diff] [blame] | 432 | $commit_message |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 433 | |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 434 | git-subtree-dir: $dir |
| 435 | git-subtree-mainline: $latest_old |
| 436 | git-subtree-split: $latest_new |
| 437 | EOF |
| 438 | } |
| 439 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 440 | add_squashed_msg () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 441 | if test -n "$message" |
| 442 | then |
Jakub Suder | 2da0969 | 2010-01-09 19:55:35 +0100 | [diff] [blame] | 443 | echo "$message" |
| 444 | else |
| 445 | echo "Merge commit '$1' as '$2'" |
| 446 | fi |
| 447 | } |
| 448 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 449 | rejoin_msg () { |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 450 | dir="$1" |
| 451 | latest_old="$2" |
| 452 | latest_new="$3" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 453 | if test -n "$message" |
| 454 | then |
Jakub Suder | 2da0969 | 2010-01-09 19:55:35 +0100 | [diff] [blame] | 455 | commit_message="$message" |
| 456 | else |
| 457 | commit_message="Split '$dir/' into commit '$latest_new'" |
| 458 | fi |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 459 | cat <<-EOF |
Jakub Suder | 1262916 | 2010-01-12 22:38:34 +0100 | [diff] [blame] | 460 | $commit_message |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 461 | |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 462 | git-subtree-dir: $dir |
Avery Pennarun | 8b4a77f | 2009-04-24 16:48:08 -0400 | [diff] [blame] | 463 | git-subtree-mainline: $latest_old |
| 464 | git-subtree-split: $latest_new |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 465 | EOF |
| 466 | } |
| 467 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 468 | squash_msg () { |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 469 | dir="$1" |
| 470 | oldsub="$2" |
| 471 | newsub="$3" |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 472 | newsub_short=$(git rev-parse --short "$newsub") |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 473 | |
| 474 | if test -n "$oldsub" |
| 475 | then |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 476 | oldsub_short=$(git rev-parse --short "$oldsub") |
| 477 | echo "Squashed '$dir/' changes from $oldsub_short..$newsub_short" |
| 478 | echo |
Stephen R Guglielmo | 8841b52 | 2018-02-23 15:41:25 -0500 | [diff] [blame] | 479 | git log --no-show-signature --pretty=tformat:'%h %s' "$oldsub..$newsub" |
| 480 | git log --no-show-signature --pretty=tformat:'REVERT: %h %s' "$newsub..$oldsub" |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 481 | else |
| 482 | echo "Squashed '$dir/' content from commit $newsub_short" |
| 483 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 484 | |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 485 | echo |
| 486 | echo "git-subtree-dir: $dir" |
| 487 | echo "git-subtree-split: $newsub" |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 488 | } |
| 489 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 490 | toptree_for_commit () { |
Avery Pennarun | 210d083 | 2009-04-24 21:49:19 -0400 | [diff] [blame] | 491 | commit="$1" |
Stephen R Guglielmo | 8841b52 | 2018-02-23 15:41:25 -0500 | [diff] [blame] | 492 | git rev-parse --verify "$commit^{tree}" || exit $? |
Avery Pennarun | 210d083 | 2009-04-24 21:49:19 -0400 | [diff] [blame] | 493 | } |
| 494 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 495 | subtree_for_commit () { |
Avery Pennarun | 210d083 | 2009-04-24 21:49:19 -0400 | [diff] [blame] | 496 | commit="$1" |
| 497 | dir="$2" |
| 498 | git ls-tree "$commit" -- "$dir" | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 499 | while read mode type tree name |
| 500 | do |
| 501 | assert test "$name" = "$dir" |
| 502 | assert test "$type" = "tree" -o "$type" = "commit" |
| 503 | test "$type" = "commit" && continue # ignore submodules |
Avery Pennarun | 768d6d1 | 2009-04-24 17:53:10 -0400 | [diff] [blame] | 504 | echo $tree |
| 505 | break |
| 506 | done |
| 507 | } |
| 508 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 509 | tree_changed () { |
Avery Pennarun | 768d6d1 | 2009-04-24 17:53:10 -0400 | [diff] [blame] | 510 | tree=$1 |
| 511 | shift |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 512 | if test $# -ne 1 |
| 513 | then |
Avery Pennarun | 768d6d1 | 2009-04-24 17:53:10 -0400 | [diff] [blame] | 514 | return 0 # weird parents, consider it changed |
| 515 | else |
Avery Pennarun | 210d083 | 2009-04-24 21:49:19 -0400 | [diff] [blame] | 516 | ptree=$(toptree_for_commit $1) |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 517 | if test "$ptree" != "$tree" |
| 518 | then |
Avery Pennarun | 768d6d1 | 2009-04-24 17:53:10 -0400 | [diff] [blame] | 519 | return 0 # changed |
| 520 | else |
| 521 | return 1 # not changed |
| 522 | fi |
| 523 | fi |
| 524 | } |
| 525 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 526 | new_squash_commit () { |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 527 | old="$1" |
| 528 | oldsub="$2" |
| 529 | newsub="$3" |
| 530 | tree=$(toptree_for_commit $newsub) || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 531 | if test -n "$old" |
| 532 | then |
| 533 | squash_msg "$dir" "$oldsub" "$newsub" | |
| 534 | git commit-tree "$tree" -p "$old" || exit $? |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 535 | else |
| 536 | squash_msg "$dir" "" "$newsub" | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 537 | git commit-tree "$tree" || exit $? |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 538 | fi |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 539 | } |
| 540 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 541 | copy_or_skip () { |
Avery Pennarun | d691265 | 2009-04-24 22:05:30 -0400 | [diff] [blame] | 542 | rev="$1" |
| 543 | tree="$2" |
| 544 | newparents="$3" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 545 | assert test -n "$tree" |
Avery Pennarun | d691265 | 2009-04-24 22:05:30 -0400 | [diff] [blame] | 546 | |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 547 | identical= |
Avery Pennarun | 49cf822 | 2009-04-26 17:07:16 -0400 | [diff] [blame] | 548 | nonidentical= |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 549 | p= |
Avery Pennarun | a64f3a7 | 2009-04-26 16:53:57 -0400 | [diff] [blame] | 550 | gotparents= |
Strain, Roger L | 68f8ff8 | 2018-09-28 13:35:40 -0500 | [diff] [blame] | 551 | copycommit= |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 552 | for parent in $newparents |
| 553 | do |
Avery Pennarun | d691265 | 2009-04-24 22:05:30 -0400 | [diff] [blame] | 554 | ptree=$(toptree_for_commit $parent) || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 555 | test -z "$ptree" && continue |
| 556 | if test "$ptree" = "$tree" |
| 557 | then |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 558 | # an identical parent could be used in place of this rev. |
Strain, Roger L | 68f8ff8 | 2018-09-28 13:35:40 -0500 | [diff] [blame] | 559 | if test -n "$identical" |
| 560 | then |
| 561 | # if a previous identical parent was found, check whether |
| 562 | # one is already an ancestor of the other |
| 563 | mergebase=$(git merge-base $identical $parent) |
| 564 | if test "$identical" = "$mergebase" |
| 565 | then |
| 566 | # current identical commit is an ancestor of parent |
| 567 | identical="$parent" |
| 568 | elif test "$parent" != "$mergebase" |
| 569 | then |
| 570 | # no common history; commit must be copied |
| 571 | copycommit=1 |
| 572 | fi |
| 573 | else |
| 574 | # first identical parent detected |
| 575 | identical="$parent" |
| 576 | fi |
Avery Pennarun | 49cf822 | 2009-04-26 17:07:16 -0400 | [diff] [blame] | 577 | else |
| 578 | nonidentical="$parent" |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 579 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 580 | |
Avery Pennarun | a64f3a7 | 2009-04-26 16:53:57 -0400 | [diff] [blame] | 581 | # sometimes both old parents map to the same newparent; |
| 582 | # eliminate duplicates |
| 583 | is_new=1 |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 584 | for gp in $gotparents |
| 585 | do |
| 586 | if test "$gp" = "$parent" |
| 587 | then |
Avery Pennarun | a64f3a7 | 2009-04-26 16:53:57 -0400 | [diff] [blame] | 588 | is_new= |
| 589 | break |
| 590 | fi |
| 591 | done |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 592 | if test -n "$is_new" |
| 593 | then |
Avery Pennarun | a64f3a7 | 2009-04-26 16:53:57 -0400 | [diff] [blame] | 594 | gotparents="$gotparents $parent" |
Avery Pennarun | d691265 | 2009-04-24 22:05:30 -0400 | [diff] [blame] | 595 | p="$p -p $parent" |
| 596 | fi |
| 597 | done |
Dave Ware | 933cfeb | 2016-01-15 13:41:43 +1300 | [diff] [blame] | 598 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 599 | if test -n "$identical" && test -n "$nonidentical" |
| 600 | then |
Dave Ware | 933cfeb | 2016-01-15 13:41:43 +1300 | [diff] [blame] | 601 | extras=$(git rev-list --count $identical..$nonidentical) |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 602 | if test "$extras" -ne 0 |
| 603 | then |
Dave Ware | 933cfeb | 2016-01-15 13:41:43 +1300 | [diff] [blame] | 604 | # we need to preserve history along the other branch |
| 605 | copycommit=1 |
| 606 | fi |
| 607 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 608 | if test -n "$identical" && test -z "$copycommit" |
| 609 | then |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 610 | echo $identical |
| 611 | else |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 612 | copy_commit "$rev" "$tree" "$p" || exit $? |
Avery Pennarun | 96db2c0 | 2009-04-24 22:36:06 -0400 | [diff] [blame] | 613 | fi |
Avery Pennarun | d691265 | 2009-04-24 22:05:30 -0400 | [diff] [blame] | 614 | } |
| 615 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 616 | ensure_clean () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 617 | if ! git diff-index HEAD --exit-code --quiet 2>&1 |
| 618 | then |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 619 | die "Working tree has modifications. Cannot add." |
| 620 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 621 | if ! git diff-index --cached HEAD --exit-code --quiet 2>&1 |
| 622 | then |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 623 | die "Index has modifications. Cannot add." |
| 624 | fi |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 625 | } |
| 626 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 627 | ensure_valid_ref_format () { |
Anthony Baire | 1c3e0f0 | 2013-11-27 19:34:09 +0100 | [diff] [blame] | 628 | git check-ref-format "refs/heads/$1" || |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 629 | die "'$1' does not look like a ref" |
Anthony Baire | 1c3e0f0 | 2013-11-27 19:34:09 +0100 | [diff] [blame] | 630 | } |
| 631 | |
Strain, Roger L | 565e4b7 | 2018-09-28 13:35:37 -0500 | [diff] [blame] | 632 | process_split_commit () { |
| 633 | local rev="$1" |
| 634 | local parents="$2" |
Strain, Roger L | 315a84f | 2018-09-28 13:35:39 -0500 | [diff] [blame] | 635 | local indent=$3 |
| 636 | |
| 637 | if test $indent -eq 0 |
| 638 | then |
| 639 | revcount=$(($revcount + 1)) |
| 640 | else |
| 641 | # processing commit without normal parent information; |
| 642 | # fetch from repo |
Roger Strain | 19ad68d | 2018-10-12 08:52:18 -0500 | [diff] [blame] | 643 | parents=$(git rev-parse "$rev^@") |
Strain, Roger L | 315a84f | 2018-09-28 13:35:39 -0500 | [diff] [blame] | 644 | extracount=$(($extracount + 1)) |
| 645 | fi |
| 646 | |
| 647 | progress "$revcount/$revmax ($createcount) [$extracount]" |
| 648 | |
Strain, Roger L | 565e4b7 | 2018-09-28 13:35:37 -0500 | [diff] [blame] | 649 | debug "Processing commit: $rev" |
| 650 | exists=$(cache_get "$rev") |
| 651 | if test -n "$exists" |
| 652 | then |
| 653 | debug " prior: $exists" |
| 654 | return |
| 655 | fi |
| 656 | createcount=$(($createcount + 1)) |
| 657 | debug " parents: $parents" |
Strain, Roger L | 315a84f | 2018-09-28 13:35:39 -0500 | [diff] [blame] | 658 | check_parents "$parents" "$indent" |
Strain, Roger L | 565e4b7 | 2018-09-28 13:35:37 -0500 | [diff] [blame] | 659 | newparents=$(cache_get $parents) |
| 660 | debug " newparents: $newparents" |
| 661 | |
| 662 | tree=$(subtree_for_commit "$rev" "$dir") |
| 663 | debug " tree is: $tree" |
| 664 | |
Strain, Roger L | 565e4b7 | 2018-09-28 13:35:37 -0500 | [diff] [blame] | 665 | # ugly. is there no better way to tell if this is a subtree |
| 666 | # vs. a mainline commit? Does it matter? |
| 667 | if test -z "$tree" |
| 668 | then |
| 669 | set_notree "$rev" |
| 670 | if test -n "$newparents" |
| 671 | then |
| 672 | cache_set "$rev" "$rev" |
| 673 | fi |
| 674 | return |
| 675 | fi |
| 676 | |
| 677 | newrev=$(copy_or_skip "$rev" "$tree" "$newparents") || exit $? |
| 678 | debug " newrev is: $newrev" |
| 679 | cache_set "$rev" "$newrev" |
| 680 | cache_set latest_new "$newrev" |
| 681 | cache_set latest_old "$rev" |
| 682 | } |
| 683 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 684 | cmd_add () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 685 | if test -e "$dir" |
| 686 | then |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 687 | die "'$dir' already exists. Cannot add." |
| 688 | fi |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 689 | |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 690 | ensure_clean |
David A. Greene | 10a4958 | 2013-02-04 22:06:03 -0600 | [diff] [blame] | 691 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 692 | if test $# -eq 1 |
| 693 | then |
| 694 | git rev-parse -q --verify "$1^{commit}" >/dev/null || |
| 695 | die "'$1' does not refer to a commit" |
David A. Greene | 10a4958 | 2013-02-04 22:06:03 -0600 | [diff] [blame] | 696 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 697 | cmd_add_commit "$@" |
| 698 | |
| 699 | elif test $# -eq 2 |
| 700 | then |
| 701 | # Technically we could accept a refspec here but we're |
| 702 | # just going to turn around and add FETCH_HEAD under the |
| 703 | # specified directory. Allowing a refspec might be |
| 704 | # misleading because we won't do anything with any other |
| 705 | # branches fetched via the refspec. |
| 706 | ensure_valid_ref_format "$2" |
| 707 | |
| 708 | cmd_add_repository "$@" |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 709 | else |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 710 | say "error: parameters were '$@'" |
| 711 | die "Provide either a commit or a repository and commit." |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 712 | fi |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 713 | } |
| 714 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 715 | cmd_add_repository () { |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 716 | echo "git fetch" "$@" |
| 717 | repository=$1 |
| 718 | refspec=$2 |
| 719 | git fetch "$@" || exit $? |
| 720 | revs=FETCH_HEAD |
| 721 | set -- $revs |
| 722 | cmd_add_commit "$@" |
| 723 | } |
| 724 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 725 | cmd_add_commit () { |
Denton Liu | 77128ed | 2019-03-11 02:47:17 -0700 | [diff] [blame] | 726 | rev=$(git rev-parse $default --revs-only "$@") || exit $? |
| 727 | ensure_single_rev $rev |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 728 | |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 729 | debug "Adding $dir as '$rev'..." |
| 730 | git read-tree --prefix="$dir" $rev || exit $? |
Avery Pennarun | 227f781 | 2009-08-26 10:43:43 -0400 | [diff] [blame] | 731 | git checkout -- "$dir" || exit $? |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 732 | tree=$(git write-tree) || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 733 | |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 734 | headrev=$(git rev-parse HEAD) || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 735 | if test -n "$headrev" && test "$headrev" != "$rev" |
| 736 | then |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 737 | headp="-p $headrev" |
| 738 | else |
| 739 | headp= |
| 740 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 741 | |
| 742 | if test -n "$squash" |
| 743 | then |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 744 | rev=$(new_squash_commit "" "" "$rev") || exit $? |
Jakub Suder | 2da0969 | 2010-01-09 19:55:35 +0100 | [diff] [blame] | 745 | commit=$(add_squashed_msg "$rev" "$dir" | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 746 | git commit-tree "$tree" $headp -p "$rev") || exit $? |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 747 | else |
James Denholm | 6f1871f | 2014-05-13 14:08:58 +1000 | [diff] [blame] | 748 | revp=$(peel_committish "$rev") && |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 749 | commit=$(add_msg "$dir" $headrev "$rev" | |
| 750 | git commit-tree "$tree" $headp -p "$revp") || exit $? |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 751 | fi |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 752 | git reset "$commit" || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 753 | |
Avery Pennarun | d713e2d | 2009-05-30 04:11:43 -0400 | [diff] [blame] | 754 | say "Added dir '$dir'" |
Avery Pennarun | eb7b590 | 2009-04-24 23:28:30 -0400 | [diff] [blame] | 755 | } |
| 756 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 757 | cmd_split () { |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 758 | debug "Splitting $dir..." |
| 759 | cache_setup || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 760 | |
| 761 | if test -n "$onto" |
| 762 | then |
Avery Pennarun | 847e868 | 2009-04-24 21:35:50 -0400 | [diff] [blame] | 763 | debug "Reading history for --onto=$onto..." |
Avery Pennarun | 33ff583 | 2009-04-24 17:05:14 -0400 | [diff] [blame] | 764 | git rev-list $onto | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 765 | while read rev |
| 766 | do |
Avery Pennarun | 33ff583 | 2009-04-24 17:05:14 -0400 | [diff] [blame] | 767 | # the 'onto' history is already just the subdir, so |
| 768 | # any parent we find there can be used verbatim |
Avery Pennarun | 2c71b7c | 2009-04-24 17:42:33 -0400 | [diff] [blame] | 769 | debug " cache: $rev" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 770 | cache_set "$rev" "$rev" |
Avery Pennarun | 33ff583 | 2009-04-24 17:05:14 -0400 | [diff] [blame] | 771 | done |
| 772 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 773 | |
Strain, Roger L | dd21d43 | 2018-09-28 13:35:38 -0500 | [diff] [blame] | 774 | unrevs="$(find_existing_splits "$dir" "$revs")" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 775 | |
Avery Pennarun | 1f73862 | 2009-04-26 15:54:42 -0400 | [diff] [blame] | 776 | # We can't restrict rev-list to only $dir here, because some of our |
| 777 | # parents have the $dir contents the root, and those won't match. |
| 778 | # (and rev-list --follow doesn't seem to solve this) |
Jesse Greenwald | 6f4f84f | 2010-11-09 08:34:49 -0600 | [diff] [blame] | 779 | grl='git rev-list --topo-order --reverse --parents $revs $unrevs' |
Avery Pennarun | 942dce5 | 2009-04-26 18:06:08 -0400 | [diff] [blame] | 780 | revmax=$(eval "$grl" | wc -l) |
| 781 | revcount=0 |
| 782 | createcount=0 |
Strain, Roger L | 315a84f | 2018-09-28 13:35:39 -0500 | [diff] [blame] | 783 | extracount=0 |
Avery Pennarun | 942dce5 | 2009-04-26 18:06:08 -0400 | [diff] [blame] | 784 | eval "$grl" | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 785 | while read rev parents |
| 786 | do |
Strain, Roger L | 315a84f | 2018-09-28 13:35:39 -0500 | [diff] [blame] | 787 | process_split_commit "$rev" "$parents" 0 |
Avery Pennarun | 2573354 | 2009-04-24 14:24:38 -0400 | [diff] [blame] | 788 | done || exit $? |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 789 | |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 790 | latest_new=$(cache_get latest_new) |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 791 | if test -z "$latest_new" |
| 792 | then |
Avery Pennarun | e25a6bf | 2009-04-24 14:52:27 -0400 | [diff] [blame] | 793 | die "No new revisions were found" |
| 794 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 795 | |
| 796 | if test -n "$rejoin" |
| 797 | then |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 798 | debug "Merging split branch into HEAD..." |
| 799 | latest_old=$(cache_get latest_old) |
| 800 | git merge -s ours \ |
David Aguilar | 0f12c7d | 2016-07-25 21:14:15 -0700 | [diff] [blame] | 801 | --allow-unrelated-histories \ |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 802 | -m "$(rejoin_msg "$dir" "$latest_old" "$latest_new")" \ |
| 803 | "$latest_new" >&2 || exit $? |
Avery Pennarun | b77172f | 2009-04-24 15:48:41 -0400 | [diff] [blame] | 804 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 805 | if test -n "$branch" |
| 806 | then |
| 807 | if rev_exists "refs/heads/$branch" |
| 808 | then |
| 809 | if ! rev_is_descendant_of_branch "$latest_new" "$branch" |
| 810 | then |
Jakub Suder | 0a56294 | 2010-01-09 19:56:05 +0100 | [diff] [blame] | 811 | die "Branch '$branch' is not an ancestor of commit '$latest_new'." |
| 812 | fi |
| 813 | action='Updated' |
| 814 | else |
| 815 | action='Created' |
| 816 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 817 | git update-ref -m 'subtree split' \ |
| 818 | "refs/heads/$branch" "$latest_new" || exit $? |
Jakub Suder | 0a56294 | 2010-01-09 19:56:05 +0100 | [diff] [blame] | 819 | say "$action branch '$branch'" |
Avery Pennarun | 43a3951 | 2009-05-30 01:05:43 -0400 | [diff] [blame] | 820 | fi |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 821 | echo "$latest_new" |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 822 | exit 0 |
| 823 | } |
| 824 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 825 | cmd_merge () { |
Denton Liu | 77128ed | 2019-03-11 02:47:17 -0700 | [diff] [blame] | 826 | rev=$(git rev-parse $default --revs-only "$@") || exit $? |
| 827 | ensure_single_rev $rev |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 828 | ensure_clean |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 829 | |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 830 | if test -n "$squash" |
| 831 | then |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 832 | first_split="$(find_latest_squash "$dir")" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 833 | if test -z "$first_split" |
| 834 | then |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 835 | die "Can't squash-merge: '$dir' was never added." |
| 836 | fi |
| 837 | set $first_split |
| 838 | old=$1 |
| 839 | sub=$2 |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 840 | if test "$sub" = "$rev" |
| 841 | then |
Avery Pennarun | eb4fb91 | 2009-05-30 03:33:17 -0400 | [diff] [blame] | 842 | say "Subtree is already at commit $rev." |
| 843 | exit 0 |
| 844 | fi |
Avery Pennarun | 1cc2cff | 2009-05-30 03:18:27 -0400 | [diff] [blame] | 845 | new=$(new_squash_commit "$old" "$sub" "$rev") || exit $? |
| 846 | debug "New squash commit: $new" |
| 847 | rev="$new" |
| 848 | fi |
Pelle Wessman | 448e71e | 2010-05-07 21:21:25 +0200 | [diff] [blame] | 849 | |
| 850 | version=$(git version) |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 851 | if test "$version" \< "git version 1.7" |
| 852 | then |
| 853 | if test -n "$message" |
| 854 | then |
| 855 | git merge -s subtree --message="$message" "$rev" |
Pelle Wessman | 448e71e | 2010-05-07 21:21:25 +0200 | [diff] [blame] | 856 | else |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 857 | git merge -s subtree "$rev" |
Pelle Wessman | 448e71e | 2010-05-07 21:21:25 +0200 | [diff] [blame] | 858 | fi |
Avery Pennarun | 349a70d | 2010-02-06 15:05:17 -0500 | [diff] [blame] | 859 | else |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 860 | if test -n "$message" |
| 861 | then |
| 862 | git merge -Xsubtree="$prefix" \ |
| 863 | --message="$message" "$rev" |
Pelle Wessman | 448e71e | 2010-05-07 21:21:25 +0200 | [diff] [blame] | 864 | else |
| 865 | git merge -Xsubtree="$prefix" $rev |
| 866 | fi |
Avery Pennarun | 349a70d | 2010-02-06 15:05:17 -0500 | [diff] [blame] | 867 | fi |
Avery Pennarun | 0ca71b3 | 2009-04-24 14:13:34 -0400 | [diff] [blame] | 868 | } |
| 869 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 870 | cmd_pull () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 871 | if test $# -ne 2 |
| 872 | then |
| 873 | die "You must provide <repository> <ref>" |
Anthony Baire | 1c3e0f0 | 2013-11-27 19:34:09 +0100 | [diff] [blame] | 874 | fi |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 875 | ensure_clean |
Anthony Baire | 1c3e0f0 | 2013-11-27 19:34:09 +0100 | [diff] [blame] | 876 | ensure_valid_ref_format "$2" |
Avery Pennarun | e31d1e2 | 2009-10-02 18:23:54 -0400 | [diff] [blame] | 877 | git fetch "$@" || exit $? |
| 878 | revs=FETCH_HEAD |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 879 | set -- $revs |
| 880 | cmd_merge "$@" |
| 881 | } |
| 882 | |
David Aguilar | d7fd792 | 2016-07-27 17:16:50 -0700 | [diff] [blame] | 883 | cmd_push () { |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 884 | if test $# -ne 2 |
| 885 | then |
| 886 | die "You must provide <repository> <ref>" |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 887 | fi |
Anthony Baire | 1c3e0f0 | 2013-11-27 19:34:09 +0100 | [diff] [blame] | 888 | ensure_valid_ref_format "$2" |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 889 | if test -e "$dir" |
| 890 | then |
| 891 | repository=$1 |
| 892 | refspec=$2 |
| 893 | echo "git push using: " "$repository" "$refspec" |
| 894 | localrev=$(git subtree split --prefix="$prefix") || die |
| 895 | git push "$repository" "$localrev":"refs/heads/$refspec" |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 896 | else |
David Aguilar | 6ae6a23 | 2016-07-27 17:16:49 -0700 | [diff] [blame] | 897 | die "'$dir' must already exist. Try 'git subtree add'." |
Wayne Walter | c00d1d1 | 2010-02-13 14:32:21 -0500 | [diff] [blame] | 898 | fi |
Avery Pennarun | 13648af | 2009-04-24 23:41:19 -0400 | [diff] [blame] | 899 | } |
| 900 | |
| 901 | "cmd_$command" "$@" |