diff options
Diffstat (limited to '')
-rwxr-xr-x | dot_local/lib/git-core/git-subrepo | 1897 |
1 files changed, 1897 insertions, 0 deletions
diff --git a/dot_local/lib/git-core/git-subrepo b/dot_local/lib/git-core/git-subrepo new file mode 100755 index 0000000..3865570 --- /dev/null +++ b/dot_local/lib/git-core/git-subrepo @@ -0,0 +1,1897 @@ +#!/usr/bin/env bash +# +# +# Copyright 2013-2020 - Ingy döt Net <ingy@ingy.net> +# + + +# Exit on any errors: +set -e + +# Import Bash+ helper functions: +SOURCE="$BASH_SOURCE" +while [[ -h $SOURCE ]]; do + DIR="$( cd -P "$( dirname "$SOURCE" )" && pwd )" + SOURCE="$(readlink "$SOURCE")" + [[ $SOURCE != /* ]] && SOURCE="$DIR/$SOURCE" +done +SOURCE_DIR="$(dirname "$SOURCE")" + +if [[ -z "$GIT_SUBREPO_ROOT" ]]; then + # If `make install` installation used: + source "${SOURCE_DIR}/git-subrepo.d/bash+.bash" +else + # If `source .rc` method used: + source "${SOURCE_DIR}/../ext/bashplus/lib/bash+.bash" +fi +bash+:import :std can + +VERSION=0.4.1 +REQUIRED_GIT_VERSION=2.7.0 +GIT_TMP="$(git rev-parse --git-common-dir 2> /dev/null || echo .git)/tmp" + +# `git rev-parse` turns this into a getopt parser and a command usage message: +GETOPT_SPEC="\ +git subrepo <command> <arguments> <options> + +Commands: + clone Clone a remote repository into a local subdirectory + init Turn a current subdirectory into a subrepo + pull Pull upstream changes to the subrepo + push Push local subrepo changes upstream + + fetch Fetch a subrepo's remote branch (and create a ref for it) + branch Create a branch containing the local subrepo commits + commit Commit a merged subrepo branch into the mainline + + status Get status of a subrepo (or all of them) + clean Remove branches, remotes and refs for a subrepo + config Set subrepo configuration properties + + help Documentation for git-subrepo (or specific command) + version Display git-subrepo version info + upgrade Upgrade the git-subrepo software itself + +See 'git help subrepo' for complete documentation and usage of each command. + +Options: +-- +h Show the command summary +help Help overview +version Print the git-subrepo version number + +a,all Perform command on all current subrepos +A,ALL Perform command on all subrepos and subsubrepos +b,branch= Specify the upstream branch to push/pull/fetch +e,edit Edit commit message +f,force Force certain operations +F,fetch Fetch the upstream content first +M,method= Method when you join, valid options are 'merge' or 'rebase' + Default is 'merge' +m,message= Specify a commit message +r,remote= Specify the upstream remote to push/pull/fetch +s,squash Squash commits on push +u,update Add the --branch and/or --remote overrides to .gitrepo + +q,quiet Show minimal output +v,verbose Show verbose output +d,debug Show the actual commands used +x,DEBUG Turn on -x Bash debugging +" + +#------------------------------------------------------------------------------ +# Top level function: +#------------------------------------------------------------------------------ +main() { + # Define global variables: + local command= # Subrepo subcommand to run + local command_arguments=() # Command args after getopt parsing + local commit_msg_args=() # Arguments to show in the commit msg + local subrepos=() # List of multiple subrepos + + local all_wanted=false # Apply command to all subrepos + local ALL_wanted=false # Apply command to all subrepos and subsubrepos + local force_wanted=false # Force certain operations + local fetch_wanted=false # Fetch requested before a command + local squash_wanted=false # Squash commits on push + local update_wanted=false # Update .gitrepo with --branch and/or --remote + + local quiet_wanted=false # Output should be quiet + local verbose_wanted=false # Output should be verbose + local debug_wanted=false # Show debug messages + + local subdir= # Subdirectory of the subrepo being used + local subref= # Valid git ref format of subdir + local gitrepo= # Path to .gitrepo file + local worktree= # Worktree created by 'git worktree' + local start_pwd=$(pwd) # Store the original directory + + local original_head_commit= # HEAD commit id at start of command + local original_head_branch= # HEAD ref at start of command + local upstream_head_commit= # HEAD commit id from a subrepo fetch + + local subrepo_remote= # Remote url for subrepo's upstream repo + local subrepo_branch= # Upstream branch to clone/push/pull + local subrepo_commit= # Upstream HEAD from previous clone/pull + local subrepo_parent= # Local commit from before previous clone/pull + local subrepo_former= # A retired gitrepo key that might still exist + + local refs_subrepo_branch= # A subrepo ref -> commit of branch/pull command + local refs_subrepo_commit= # A subrepo ref -> commit last merged + local refs_subrepo_fetch= # A subrepo ref -> FETCH_HEAD after fetch + local refs_subrepo_push= # A subrepo ref -> branch after push + + local override_remote= # Remote specified with -r + local override_branch= # Remote specified with -b + + local edit_wanted=false # Edit commit message using -e + local wanted_commit_message= # Custom commit message using -m + + local join_method= # Current join method (rebase/merge) + + local FAIL=true # Flag for RUN: fail on error + local OUT=false # Flag for RUN: put output in $output + local TTY=false # Flag for RUN: print output directly + local SAY=true # Flag for RUN: print command for verbose + local EXEC=false # Flag for RUN: run subprocess + local OK=true # Flag that commands have succeeded + local CODE=0 # Failure reason code + local INDENT= # Verbose indentation + + local git_version= # Git version in use + + # Check environment and parse CLI options: + assert-environment-ok + + # Parse and validate command options: + get-command-options "$@" + + # Make sure repo is in the proper state: + assert-repo-is-ready + + command-init + + if $all_wanted && [[ ! $command =~ ^(help|status)$ ]]; then + # Run the command on all subrepos + local args=( "${command_arguments[@]}" ) + get-all-subrepos + for subdir in ${subrepos[*]}; do + command-prepare + subrepo_remote= + subrepo_branch= + command_arguments=( "$subdir" "${args[@]}" ) + "command:$command" + done + else + # Run the command on a specific subrepo + command-prepare + "command:$command" + fi +} + +#------------------------------------------------------------------------------ +# API command functions. +# +# Most of these commands call a subrepo:$command function to do the actual +# work. The user facing output (via `say`) is done up here. The +# subrepo:* worker functions are meant to be called internally and don't print +# info to the user. +#------------------------------------------------------------------------------ + +# `git subrepo clone <url> [<subdir>]` command: +command:clone() { + command-setup +subrepo_remote subdir:guess-subdir + + # Clone (or reclone) the subrepo into the subdir: + local reclone_up_to_date=false + subrepo:clone + if "$reclone_up_to_date"; then + say "Subrepo '$subdir' is up to date." + return + fi + + # Successful command output: + local re= + $force_wanted && re=re + local remote="$subrepo_remote" + say "Subrepo '$remote' ($subrepo_branch) ${re}cloned into '$subdir'." +} + +# `git subrepo init <subdir>` command: +command:init() { + command-setup +subdir + local remote="${subrepo_remote:=none}" + local branch="${subrepo_branch:=master}" + + # Init new subrepo from the subdir: + subrepo:init + if OK; then + if [[ $remote == none ]]; then + say "Subrepo created from '$subdir' (with no remote)." + else + say "Subrepo created from '$subdir' with remote '$remote' ($branch)." + fi + else + die "Unknown init error code: '$CODE'" + fi + return 0 +} + +# `git subrepo pull <subdir>` command: +command:pull() { + command-setup +subdir + + subrepo:pull + if OK; then + say "Subrepo '$subdir' pulled from '$subrepo_remote' ($subrepo_branch)." + elif [[ $CODE -eq -1 ]]; then + say "Subrepo '$subdir' is up to date." + elif [[ $CODE -eq 1 ]]; then + error-join + return "$CODE" + else + die "Unknown pull error code: '$CODE'" + fi + return 0 +} + +# `git subrepo push <subdir>` command: +command:push() { + local branch= + command-setup +subdir branch + + subrepo:push + if OK; then + say "Subrepo '$subdir' pushed to '$subrepo_remote' ($subrepo_branch)." + elif [[ $CODE -eq -2 ]]; then + say "Subrepo '$subdir' has no new commits to push." + elif [[ $CODE -eq 1 ]]; then + error-join + return "$CODE" + else + die "Unknown push error code: '$CODE'" + fi + return 0 +} + +# `git subrepo fetch <subdir>` command +command:fetch() { + command-setup +subdir + if [[ $subrepo_remote == "none" ]]; then + say "Ignored '$subdir', no remote." + else + subrepo:fetch + say "Fetched '$subdir' from '$subrepo_remote' ($subrepo_branch)." + fi +} + +# `git subrepo branch <subdir>` command: +command:branch() { + command-setup +subdir + if $fetch_wanted; then + CALL subrepo:fetch + fi + + local branch="subrepo/$subref" + if $force_wanted; then + # We must make sure that the worktree is removed as well + worktree="$GIT_TMP/$branch" + git:delete-branch "$branch" + fi + + if git:branch-exists "$branch"; then + error "Branch '$branch' already exists. Use '--force' to override." + fi + + # Create the subrepo branch: + subrepo:branch + + say "Created branch '$branch' and worktree '$worktree'." +} + +# `git subrepo commit <subdir>` command +command:commit() { + command-setup +subdir subrepo_commit_ref + + if "$fetch_wanted"; then + CALL subrepo:fetch + fi + git:rev-exists "$refs_subrepo_fetch" || + error "Can't find ref '$refs_subrepo_fetch'. Try using -F." + upstream_head_commit="$(git rev-parse "$refs_subrepo_fetch")" + + [[ -n $subrepo_commit_ref ]] || + subrepo_commit_ref="subrepo/$subref" + subrepo:commit + + say "Subrepo commit '$subrepo_commit_ref' committed as" + say "subdir '$subdir/' to branch '$original_head_branch'." +} + +# `git subrepo status [<subdir>]` command: +command:status() { + subrepo:status | ${GIT_SUBREPO_PAGER} +} + +status-refs() { + local output= + while read line; do + [[ $line =~ ^([0-9a-f]+)\ refs/subrepo/$subref/([a-z]+) ]] || continue + local sha1=; sha1="$(git rev-parse --short "${BASH_REMATCH[1]}")" + local type="${BASH_REMATCH[2]}" + local ref="refs/subrepo/$subref/$type" + if [[ $type == branch ]]; then + output+=" Branch Ref: $sha1 ($ref)"$'\n' + elif [[ $type == commit ]]; then + output+=" Commit Ref: $sha1 ($ref)"$'\n' + elif [[ $type == fetch ]]; then + output+=" Fetch Ref: $sha1 ($ref)"$'\n' + elif [[ $type == pull ]]; then + output+=" Pull Ref: $sha1 ($ref)"$'\n' + elif [[ $type == push ]]; then + output+=" Push Ref: $sha1 ($ref)"$'\n' + fi + done < <(git show-ref) + if [[ -n $output ]]; then + printf " Refs:\n$output" + fi +} + +# `git subrepo clean <subdir>` command +command:clean() { + command-setup +subdir + local clean_list=() + subrepo:clean + for item in "${clean_list[@]}"; do + say "Removed $item." + done +} + +# Wrap git config $gitrepo +command:config() { + command-setup +subdir +config_option config_value + o "Update '$subdir' configuration with $config_option=$config_value" + + if [[ ! $config_option =~ ^(branch|cmdver|commit|method|remote|version)$ ]]; then + error "Option $config_option not recognized" + fi + + if [[ -z $config_value ]]; then + OUT=true RUN git config --file="$gitrepo" "subrepo.$config_option" + say "Subrepo '$subdir' option '$config_option' has value '$output'." + return + fi + + if ! $force_wanted; then + # Only allow changing method without force + if [[ ! $config_option == "method" ]]; then + error "This option is autogenerated, use '--force' to override." + fi + fi + + if [[ $config_option == "method" ]]; then + if [[ ! $config_value =~ ^(merge|rebase)$ ]]; then + error "Not a valid method. Valid options are 'merge' or 'rebase'." + fi + fi + + RUN git config --file="$gitrepo" "subrepo.$config_option" "$config_value" + say "Subrepo '$subdir' option '$config_option' set to '$config_value'." +} + + +# Launch the manpage viewer: +command:help() { + source "${SOURCE_DIR}/git-subrepo.d/help-functions.bash" + local cmd="${command_arguments[0]}" + if [[ -n $cmd ]]; then + if can "help:$cmd"; then + "help:$cmd" + echo + else + err "No help found for '$cmd'" + fi + elif $all_wanted; then + help:all + else + exec git help subrepo + fi + msg_ok=0 +} + +# Print version info. +# TODO: Add short commit id after version. +# Will need to get it from repo or make install can put it somewhere. +command:version() { + cat <<... +git-subrepo Version: $VERSION +Copyright 2013-2020 Ingy döt Net +https://github.com/ingydotnet/git-subrepo +$BASH_SOURCE +Git Version: $git_version + +... + : +} + +command:upgrade() { + local path="$0" + if [[ $path =~ ^/ && $path =~ ^(.*/git-subrepo)/lib/git-subrepo$ ]]; then + local subrepo_root="${BASH_REMATCH[1]}" + ( + o "Change directory to '$subrepo_root'." + cd "${BASH_REMATCH[1]}" + + local branch="$(git rev-parse --abbrev-ref HEAD)" + if [[ $branch != master ]]; then + error "git-subrepo repo is not on the 'master' branch" + fi + + o "'git pull' latest version." + RUN git pull --ff-only + + say "git-subrepo is up to date." + ) + else + die "\ + +Sorry. Your installation can't use the 'git subrepo upgrade' command. The +command only works if you installed git subrepo by adding +'/path/to/git-subrepo' to your PATH. + +If you used 'make install' to install git-subrepo, then just do this: + + cd /path/to/git-subrepo + git pull + make install + +" + fi +} + +#------------------------------------------------------------------------------ +# Subrepo command worker functions. +#------------------------------------------------------------------------------ + +# Clone by fetching remote content into our subdir: +subrepo:clone() { + re="$1" + + FAIL=false RUN git rev-parse HEAD + if ! OK; then + error "You can't clone into an empty repository" + fi + + # Turn off force unless really a reclone: + if $force_wanted && [[ ! -f $gitrepo ]]; then + force_wanted=false + fi + + if $force_wanted; then + o "--force indicates a reclone." + CALL subrepo:fetch + read-gitrepo-file + o "Check if we already are up to date." + if [[ $upstream_head_commit == $subrepo_commit ]]; then + reclone_up_to_date=true + return + fi + o "Remove the old subdir." + RUN git rm -r -- "$subdir" + else + assert-subdir-empty + if [[ -z $subrepo_branch ]]; then + o "Determine the upstream head branch." + get-upstream-head-branch + subrepo_branch="$output" + fi + + CALL subrepo:fetch + fi + + o "Make the directory '$subdir/' for the clone." + RUN mkdir -p -- "$subdir" + + o "Commit the new '$subdir/' content." + subrepo_commit_ref="$upstream_head_commit" + CALL subrepo:commit +} + +# Init a new subrepo from current repo: +subrepo:init() { + local branch_name="subrepo/${subref:??}" + # Check if subdir is proper candidate for this init: + assert-subdir-ready-for-init + + o "Put info into '$subdir/.gitrepo' file." + update-gitrepo-file + + o "Add the new '$subdir/.gitrepo' file." + # -f from pull request #219. TODO needs test. + RUN git add -f -- "$gitrepo" + + o "Commit new subrepo to the '$original_head_branch' branch." + subrepo_commit_ref="$original_head_commit" + RUN git commit -m "$(get-commit-message)" + + o "Create ref '$refs_subrepo_commit'." + git:make-ref "$refs_subrepo_commit" "$subrepo_commit_ref" +} + +# Properly merge a local subrepo branch with upstream and commit to mainline: +subrepo:pull() { + CALL subrepo:fetch + + # Check if we already are up to date + # If the -u flag is present, always perform the operation + if [[ $upstream_head_commit == $subrepo_commit ]] && ! $update_wanted; then + OK=false; CODE=-1; return + fi + + local branch_name="subrepo/$subref" + git:delete-branch "$branch_name" + + subrepo_commit_ref="$branch_name" + + o "Create subrepo branch '$branch_name'." + CALL subrepo:branch + cd "$worktree"; + + if [[ "$join_method" == "rebase" ]]; then + o "Rebase changes to $refs_subrepo_fetch" + FAIL=false OUT=true RUN git rebase "$refs_subrepo_fetch" "$branch_name" + if ! OK; then + say "The \"git rebase\" command failed:" + say + say " ${output//$'\n'/$'\n' }" + CODE=1 + return + fi + else + o "Merge in changes from $refs_subrepo_fetch" + FAIL=false OUT=true RUN git merge "$refs_subrepo_fetch" + if ! OK; then + say "The \"git merge\" command failed:" + say + say " ${output//$'\n'/$'\n' }" + CODE=1 + return + fi + fi + + o "Back to $start_pwd" + cd "$start_pwd"; + + o "Create ref '$refs_subrepo_branch' for branch '$branch_name'." + git:make-ref "$refs_subrepo_branch" "$branch_name" + + o "Commit the new '$subrepo_commit_ref' content." + CALL subrepo:commit +} + +# Push a properly merged subrepo branch upstream: +subrepo:push() { + local branch_name="$branch" + local new_upstream=false + local branch_created=false + + if [[ -z $branch_name ]]; then + FAIL=false OUT=false CALL subrepo:fetch + + if ! OK; then + # Check if we are pushing to a new upstream repo (or branch) and just + # push the commit directly. This is common after a `git subrepo init`: + # Force to case in + local re="(^|"$'\n'")fatal: couldn't find remote ref " + if [[ ${output,,} =~ $re ]]; then + o "Pushing to new upstream: $subrepo_remote ($subrepo_branch)." + new_upstream=true + else + error "Fetch for push failed: $output" + fi + else + # Check that we are up to date: + o "Check upstream head against .gitrepo commit." + if ! $force_wanted; then + if [[ $upstream_head_commit != $subrepo_commit ]]; then + error "There are new changes upstream, you need to pull first." + fi + fi + fi + + branch_name="subrepo/$subref" + git:delete-branch "$branch_name" + + if $squash_wanted; then + o "Squash commits" + subrepo_parent="HEAD^" + fi + + o "Create subrepo branch '$branch_name'." + CALL subrepo:branch "$branch_name" + cd "$worktree"; + + if [[ "$join_method" == "rebase" ]]; then + o "Rebase changes to $refs_subrepo_fetch" + FAIL=false OUT=true RUN git rebase "$refs_subrepo_fetch" "$branch_name" + if ! OK; then + say "The \"git rebase\" command failed:" + say + say " ${output//$'\n'/$'\n' }" + CODE=1 + return + fi + fi + branch_created=true + cd "$start_pwd" + else + if $squash_wanted; then + error "Squash option (-s) can't be used with branch parameter" + fi + fi + + o "Make sure that '$branch_name' exists." + git:branch-exists "$branch_name" || + error "No subrepo branch '$branch_name' to push." + + o "Check if we have something to push" + new_upstream_head_commit="$(git rev-parse "$branch_name")" + if ! $new_upstream; then + if [[ $upstream_head_commit == $new_upstream_head_commit ]]; then + OK=false + CODE=-2 + return + fi + fi + + if ! $force_wanted; then + o "Make sure '$branch_name' contains the '$refs_subrepo_fetch' HEAD." + if ! git:commit-in-rev-list "$upstream_head_commit" "$branch_name"; then + error "Can't commit: '$branch_name' doesn't contain upstream HEAD: " \ + "$upstream_head_commit" + fi + fi + + local force='' + "$force_wanted" && force=' --force' + + o "Push$force branch '$branch_name' to '$subrepo_remote' ($subrepo_branch)." + RUN git push$force "$subrepo_remote" "$branch_name":"$subrepo_branch" + + o "Create ref '$refs_subrepo_push' for branch '$branch_name'." + git:make-ref "$refs_subrepo_push" "$branch_name" + + if $branch_created; then + o "Remove branch '$branch_name'." + git:delete-branch "$branch_name" + fi + + o "Put updates into '$subdir/.gitrepo' file." + upstream_head_commit="$new_upstream_head_commit" + subrepo_commit_ref="$upstream_head_commit" + update-gitrepo-file + RUN git commit -m "$(get-commit-message)" +} + +# Fetch the subrepo's remote branch content: +subrepo:fetch() { + if [[ $subrepo_remote == none ]]; then + error "Can't fetch subrepo. Remote is 'none' in '$subdir/.gitrepo'." + fi + + o "Fetch the upstream: $subrepo_remote ($subrepo_branch)." + RUN git fetch --no-tags --quiet "$subrepo_remote" "$subrepo_branch" + OK || return + + o "Get the upstream subrepo HEAD commit." + OUT=true RUN git rev-parse FETCH_HEAD^0 + upstream_head_commit="$output" + + o "Create ref '$refs_subrepo_fetch'." + git:make-ref "$refs_subrepo_fetch" FETCH_HEAD^0 +} + +# Create a subrepo branch containing all changes +subrepo:branch() { + local branch="${1:-"subrepo/$subref"}" + o "Check if the '$branch' branch already exists." + git:branch-exists "$branch" && return + + local last_gitrepo_commit= + local first_gitrepo_commit= + + o "Subrepo parent: $subrepo_parent" + if [[ -n "$subrepo_parent" ]]; then + local prev_commit= + local ancestor= + o "Create new commits with parents into the subrepo fetch" + OUT=true RUN git rev-list --reverse --ancestry-path --topo-order "$subrepo_parent..HEAD" + local commit_list="$output" + for commit in $commit_list; do + o "Working on $commit" + + FAIL=false OUT=true RUN git config --blob \ + "$commit":"$subdir/.gitrepo" "subrepo.commit" + if [[ -z "$output" ]]; then + o "Ignore commit, no .gitrepo file" + continue + fi + + local gitrepo_commit="$output" + o ".gitrepo reference commit: $gitrepo_commit" + + + # Only include the commit if it's a child of the previous commit + # This way we create a single path between $subrepo_parent..HEAD + if [[ -n "$ancestor" ]]; then + local is_direct_child=$(git show -s --pretty=format:"%P" $commit | grep "$ancestor") + o "is child: $is_direct_child" + if [[ -z "$is_direct_child" ]]; then + o "Ignore $commit, it's not in the selected path" + continue + fi + fi + + # Remember the previous commit from the parent repo path + ancestor="$commit" + + o "Check for rebase" + if git:rev-exists "$refs_subrepo_fetch"; then + if ! git:commit-in-rev-list "$gitrepo_commit" "$refs_subrepo_fetch"; then + error "Local repository does not contain $gitrepo_commit. Try to 'git subrepo fetch $subref' or add the '-F' flag to always fetch the latest content." + fi + fi + + o "Find parents" + local first_parent= + [[ -n $prev_commit ]] && first_parent="-p $prev_commit" + local second_parent= + if [[ -z "$first_gitrepo_commit" ]]; then + first_gitrepo_commit="$gitrepo_commit" + second_parent="-p $gitrepo_commit" + fi + + if [[ "$join_method" != "rebase" ]]; then + # In the rebase case we don't create merge commits + if [[ "$gitrepo_commit" != "$last_gitrepo_commit" ]]; then + second_parent="-p $gitrepo_commit" + last_gitrepo_commit="$gitrepo_commit" + fi + fi + + o "Create a new commit $first_parent $second_parent" + FAIL=false RUN git cat-file -e "$commit":"$subdir" + if OK; then + o "Create with content" + local PREVIOUS_IFS=$IFS + IFS=$'\n' + local author_info=( $(git log -1 --format=%ad%n%ae%n%an "$commit") ) + IFS=$PREVIOUS_IFS + + # When we create new commits we leave the author information unchanged + # the committer will though be updated to the current user + # This should be analog how cherrypicking is handled allowing git + # to store both the original author but also the responsible committer + # that created the local version of the commit and pushed it. + prev_commit=$(git log -n 1 --format=%B "$commit" | + GIT_AUTHOR_DATE="${author_info[0]}" \ + GIT_AUTHOR_EMAIL="${author_info[1]}" \ + GIT_AUTHOR_NAME="${author_info[2]}" \ + git commit-tree -F - $first_parent $second_parent "$commit":"$subdir") + else + o "Create empty placeholder" + prev_commit=$(git commit-tree -m "EMPTY" \ + $first_parent $second_parent "4b825dc642cb6eb9a060e54bf8d69288fbee4904") + fi + done + + o "Create branch '$branch' for this new commit set $prev_commit." + RUN git branch "$branch" "$prev_commit" + else + o "No parent setting, use the subdir content." + RUN git branch "$branch" HEAD + TTY=true FAIL=false RUN git filter-branch -f --subdirectory-filter \ + "$subref" "$branch" + fi + + o "Remove the .gitrepo file from $first_gitrepo_commit..$branch" + local filter="$branch" + [[ -n "$first_gitrepo_commit" ]] && filter="$first_gitrepo_commit..$branch" + FAIL=false RUN git filter-branch -f --prune-empty --tree-filter \ + "rm -f .gitrepo" "$filter" + + git:create-worktree "$branch" + + o "Create ref '$refs_subrepo_branch'." + git:make-ref "$refs_subrepo_branch" "$branch" +} + +# Commit a merged subrepo branch: +subrepo:commit() { + o "Check that '$subrepo_commit_ref' exists." + git:rev-exists "$subrepo_commit_ref" || + error "Commit ref '$subrepo_commit_ref' does not exist." + + if ! "$force_wanted"; then + local upstream="$upstream_head_commit" + o "Make sure '$subrepo_commit_ref' contains the upstream HEAD." + if ! git:commit-in-rev-list "$upstream" "$subrepo_commit_ref"; then + error \ + "Can't commit: '$subrepo_commit_ref' doesn't contain upstream HEAD." + fi + fi + + if git ls-files -- "$subdir" | grep -q .; then + o "Remove old content of the subdir." + RUN git rm -r -- "$subdir" + fi + + o "Put remote subrepo content into '$subdir/'." + RUN git read-tree --prefix="$subdir" -u "$subrepo_commit_ref" + + o "Put info into '$subdir/.gitrepo' file." + update-gitrepo-file + RUN git add -f -- "$gitrepo" + + local commit_message + if [[ -n "$wanted_commit_message" ]]; then + commit_message="$wanted_commit_message" + else + commit_message="$(get-commit-message)" + fi + + local edit_flag= + $edit_wanted && edit_flag=--edit + + [[ -n $commit_message ]] || commit_message="$(get-commit-message)" + + local edit_flag= + $edit_wanted && edit_flag=--edit + + o "Commit to the '$original_head_branch' branch." + if [[ $original_head_commit != none ]]; then + RUN git commit $edit_flag -m "$commit_message" + else + # We had cloned into an empty repo, side effect of prior git reset --mixed + # command is that subrepo's history is now part of the index. Commit + # without that history. + OUT=true RUN git write-tree + OUT=true RUN git commit-tree $edit_flag -m "$commit_message" "$output" + RUN git reset --hard "$output" + fi + + # Clean up worktree to indicate that we are ready + git:remove-worktree + + o "Create ref '$refs_subrepo_commit'." + git:make-ref "$refs_subrepo_commit" "$subrepo_commit_ref" +} + +subrepo:status() { + if [[ ${#command_arguments[@]} -eq 0 ]]; then + get-all-subrepos + local count=${#subrepos[@]} + if ! "$quiet_wanted"; then + if [[ $count -eq 0 ]]; then + echo "No subrepos." + return + else + local s=; [[ $count -eq 1 ]] || s=s + echo "$count subrepo$s:" + echo + fi + fi + else + subrepos=("${command_arguments[@]}") + fi + + for subdir in "${subrepos[@]}"; do + check-and-normalize-subdir + encode-subdir + + if [[ ! -f $subdir/.gitrepo ]]; then + echo "'$subdir' is not a subrepo" + echo + continue + fi + + refs_subrepo_fetch="refs/subrepo/$subref/fetch" + upstream_head_commit="$( + git rev-parse --short "$refs_subrepo_fetch" 2> /dev/null || true + )" + subrepo_remote= + subrepo_branch= + + read-gitrepo-file + if $fetch_wanted; then + subrepo:fetch + fi + + if $quiet_wanted; then + echo "$subdir" + continue + fi + + echo "Git subrepo '$subdir':" + git:branch-exists "subrepo/$subref" && + echo " Subrepo Branch: subrepo/$subref" + local remote="subrepo/$subref" + FAIL=false OUT=true RUN git config "remote.$remote.url" + [[ -n $output ]] && + echo " Remote Name: subrepo/$subref" + echo " Remote URL: $subrepo_remote" + [[ -n $upstream_head_commit ]] && + echo " Upstream Ref: $upstream_head_commit" + echo " Tracking Branch: $subrepo_branch" + [[ -z $subrepo_commit ]] || + echo " Pulled Commit: $(git rev-parse --short $subrepo_commit)" + if [[ -n $subrepo_parent ]]; then + echo " Pull Parent: $(git rev-parse --short $subrepo_parent)" + # TODO Remove this eventually: + elif [[ -n $subrepo_former ]]; then + printf " Former Commit: $(git rev-parse --short $subrepo_former)" + echo " *** DEPRECATED ***" + fi + + # Grep for directory, branch can be in detached state due to conflicts + local _worktree=$(git worktree list | grep "$GIT_TMP/subrepo/$subdir") + if [[ -n $_worktree ]]; then + echo " Worktree: $_worktree" + fi + + if "$verbose_wanted"; then + status-refs + fi + + echo + done +} + +subrepo:clean() { + # Remove subrepo branches if exist: + local branch="subrepo/$subref" + local ref="refs/heads/$branch" + local worktree="$GIT_TMP/$branch" + + o "Clean $subdir" + git:remove-worktree + if [[ -e .git/$ref ]]; then + o "Remove branch '$branch'." + RUN git update-ref -d "$ref" + clean_list+=("branch '$branch'") + fi + + if "$force_wanted"; then + o "Remove all subrepo refs." + local suffix="" + if ! $all_wanted; then + suffix="$subref/" + fi + git show-ref | while read hash ref; do + if [[ "$ref" == refs/subrepo/$suffix* ]]; then + git update-ref -d "$ref" + fi + done + fi +} + +#------------------------------------------------------------------------------ +# Support functions: +#------------------------------------------------------------------------------ + + +# TODO: +# Collect original options and arguments into an array for commit message +# They should be normalized and pruned + +# Parse command line options: +get-command-options() { + [[ $# -eq 0 ]] && set -- --help + + [[ -n $GIT_SUBREPO_QUIET ]] && quiet_wanted=true + [[ -n $GIT_SUBREPO_VERBOSE ]] && verbose_wanted=true + [[ -n $GIT_SUBREPO_DEBUG ]] && debug_wanted=true + + eval "$( + echo "$GETOPT_SPEC" | + git rev-parse --parseopt -- "$@" || + echo exit $? + )" + + while [[ $# -gt 0 ]]; do + local option="$1"; shift + case "$option" in + --) break ;; + -a) all_wanted=true ;; + -A) ALL_wanted=true + all_wanted=true ;; + -b) subrepo_branch="$1" + override_branch="$1" + commit_msg_args+=("--branch=$1") + shift ;; + -e) edit_wanted=true ;; + -f) force_wanted=true + commit_msg_args+=("--force") ;; + -F) fetch_wanted=true ;; + -m) wanted_commit_message="$1" + shift;; + -M) join_method="$1" + shift;; + -M) join_method="$1" + shift;; + -r) subrepo_remote="$1" + override_remote="$1" + commit_msg_args+=("--remote=$1") + shift ;; + -s) squash_wanted=true ;; + -u) update_wanted=true + commit_msg_args+=("--update") ;; + -q) quiet_wanted=true ;; + -v) verbose_wanted=true ;; + -d) debug_wanted=true ;; + -x) set -x ;; + --version) + echo "$VERSION" + exit ;; + *) usage-error "Unexpected option: '$option'." ;; + esac + done + + # Set subrepo command: + command="$1"; shift + + # Make sure command exists: + can "command:$command" || + usage-error "'$command' is not a command. See 'git subrepo help'." + + command_arguments=("$@") + if [[ ${#command_arguments} -gt 0 ]]; then + local first="${command_arguments[0]}" + first="${first%/}" + command_arguments[0]="$first" + fi + commit_msg_args+=("${command_arguments[@]}") + + for option in all ALL edit fetch force squash; do + var="${option}_wanted" + if ${!var}; then + check_option $option + fi + done + + if [[ -n $override_branch ]]; then + check_option branch + fi + if [[ -n $override_remote ]]; then + check_option remote + fi + if [[ -n $wanted_commit_message ]]; then + check_option message + fi + if $update_wanted; then + check_option update + if [[ -z $subrepo_branch && -z $subrepo_remote ]]; then + usage-error "Can't use '--update' without '--branch' or '--remote'." + fi + fi +} + +options_help='all' +options_branch='all fetch force' +options_clean='ALL all force' +options_clone='branch edit force message method' +options_config='force' +options_commit='edit fetch force message' +options_fetch='all branch remote' +options_init='branch remote method' +options_pull='all branch edit force message remote update' +options_push='all branch force remote squash update' +options_status='ALL all fetch' +check_option() { + local var="options_${command//-/_}" + [[ ${!var} =~ $1 ]] || + usage-error "Invalid option '--$1' for '$command'." +} + +#------------------------------------------------------------------------------ +# Command argument validation: +#------------------------------------------------------------------------------ + +command-init() { + # Export variable to let other processes (possibly git hooks) know that they + # are running under git-subrepo. Set to current process pid, so it can be + # further verified if need be: + export GIT_SUBREPO_RUNNING="$$" + export GIT_SUBREPO_COMMAND="$command" + + : "${GIT_SUBREPO_PAGER:=${PAGER:-less}}" + if [[ $GIT_SUBREPO_PAGER == less ]]; then + GIT_SUBREPO_PAGER='less -FRX' + fi +} + +command-prepare() { + local output= + if git:rev-exists HEAD; then + git:get-head-branch-commit + fi + original_head_commit="${output:-none}" +} + +# Do the setup steps needed by most of the subrepo subcommands: +command-setup() { + get-params "$@" + + check-and-normalize-subdir + encode-subdir + gitrepo="$subdir/.gitrepo" + + if ! $force_wanted; then + o "Check for worktree with branch subrepo/$subdir" + local _worktree=$(git worktree list | grep "\[subrepo/$subdir\]" | cut -d ' ' -f1) + if [[ $command =~ ^(commit)$ && -z $_worktree ]]; then + error "There is no worktree available, use the branch command first" + elif [[ ! $command =~ ^(branch|clean|commit|push)$ && -n $_worktree ]]; then + if [[ -e $gitrepo ]]; then + error "There is already a worktree with branch subrepo/$subdir. +Use the --force flag to override this check or perform a subrepo clean +to remove the worktree." + else + error "There is already a worktree with branch subrepo/$subdir. +Use the --force flag to override this check or remove the worktree with +1. rm -rf $_worktree +2. git worktree prune +" + fi + fi + fi + + # Set refs_ variables: + refs_subrepo_branch="refs/subrepo/$subref/branch" + refs_subrepo_commit="refs/subrepo/$subref/commit" + refs_subrepo_fetch="refs/subrepo/$subref/fetch" + refs_subrepo_push="refs/subrepo/$subref/push" + + # Read/parse the .gitrepo file (unless clone/init; doesn't exist yet) + if [[ ! $command =~ ^(clone|init)$ ]]; then + read-gitrepo-file + fi + + true +} + +# Parse command line args according to a simple dsl spec: +get-params() { + local i=0 + local num=${#command_arguments[@]} + for arg in $@; do + local value="${command_arguments[i]}" + value="${value//%/%%}" + value="${value//\\/\\\\}" + # If arg starts with '+' then it is required + if [[ $arg == +* ]]; then + if [[ $i -ge $num ]]; then + usage-error "Command '$command' requires arg '${arg#+}'." + fi + printf -v ${arg#+} -- "$value" + # Look for function name after ':' to provide a default value + else + if [[ $i -lt $num ]]; then + printf -v ${arg%:*} -- "$value" + elif [[ $arg =~ : ]]; then + "${arg#*:}" + fi + fi + let i=$((i+1)) + done + + # Check for extra arguments: + if [[ $num -gt $i ]]; then + set -- ${command_arguments[@]} + for ((j = 1; j <= i; j++)); do shift; done + error "Unknown argument(s) '$*' for '$command' command." + fi +} + +check-and-normalize-subdir() { + # Sanity check subdir: + [[ -n $subdir ]] || + die "subdir not set" + [[ $subdir =~ ^/ || $subdir =~ ^[A-Z]: ]] && + usage-error "The subdir '$subdir' should not be absolute path." + subdir="${subdir#./}" + subdir="${subdir%/}" + [[ $subdir != *//* ]] || subdir=$(tr -s / <<< "$subdir") +} + +# Determine the correct subdir path to use: +guess-subdir() { + local dir="$subrepo_remote" + dir="${dir%.git}" + dir="${dir%/}" + dir="${dir##*/}" + [[ $dir =~ ^[-_a-zA-Z0-9]+$ ]] || + error "Can't determine subdir from '$subrepo_remote'." + subdir="$dir" + check-and-normalize-subdir + encode-subdir +} + +# Encode the subdir as a valid git ref format +# +# Input: env $subdir +# Output: env $subref +# +# For detail rules about valid git refs, see the manual of git-check-ref-format: +# URL: https://www.kernel.org/pub/software/scm/git/docs/git-check-ref-format.html +# Shell: git check-ref-format --help +# +encode-subdir() { + subref=$subdir + if [[ ! $subref ]] || git check-ref-format "subrepo/$subref"; then + return + fi + + ## 0. escape %, ensure the subref can be (almost) decoded back to subdir + subref=${subref//%/%25} + + ## 1. They can include slash / for hierarchical (directory) grouping, + ## but no slash-separated component can begin with a dot . or + ## end with the sequence .lock. + subref=/$subref/ + subref=${subref//\/.//%2e} + subref=${subref//.lock\//%2elock/} + subref=${subref#/} + subref=${subref%/} + + ## 2. They must contain at least one /. + ## Note: 'subrepo/' be will prefixed, so this is always true. + ## 3. They cannot have two consecutive dots .. anywhere. + subref=${subref//../%2e%2e} + subref=${subref//%2e./%2e%2e} + subref=${subref//.%2e/%2e%2e} + + ## 4. They cannot have ASCII control characters + ## (i.e. bytes whose values are lower than \040, or \177 DEL), space, + ## tilde ~, caret ^, or colon : anywhere. + ## 5. They cannot have question-mark ?, asterisk *, + ## or open bracket [ anywhere. + local i + for (( i = 1; i < 32; ++i )); do + # skip substitute NUL char (i=0), as bash will skip NUL in env + local x=$(printf "%02x" $i) + subref=${subref//$(printf "%b" "\x$x")/%$x} + done + subref=${subref//$'\177'/%7f} + subref=${subref// /%20} + subref=${subref//\~/%7e} + subref=${subref//^/%5e} + subref=${subref//:/%3a} + subref=${subref//\?/%3f} + subref=${subref//\*/%2a} + subref=${subref//\[/%5b} + subref=${subref//$'\n'/%0a} + + ## 6. They cannot begin or end with a slash / or contain multiple + ## consecutive slashes. + ## Note: This rule is not revertable. + [[ $subref != *//* ]] || subref=$(tr -s / <<< "$subref") + + ## 7. They cannot end with a dot .. + case "$subref" in + *.) subref=${subref%.} + subref+=%2e + ;; + esac + + ## 8. They cannot contain a sequence @\{. + subref=${subref//@\{/%40\{} + + ## 9. They cannot be the single character @. + ## Note: 'subrepo/' be will prefixed, so this is always true. + + ## 10. They cannot contain a \. + subref=${subref//\\/%5c} + + subref=$(git check-ref-format --normalize --allow-onelevel "$subref") || + error "Can't determine valid subref from '$subdir'." +} + +#------------------------------------------------------------------------------ +# State file (`.gitrepo`) functions: +#------------------------------------------------------------------------------ + +# Set subdir and gitrepo vars: +read-gitrepo-file() { + gitrepo="$subdir/.gitrepo" + + if [[ ! -f $gitrepo ]]; then + error "No '$gitrepo' file." + fi + + # Read .gitrepo values: + if [[ -z $subrepo_remote ]]; then + SAY=false OUT=true RUN git config --file="$gitrepo" subrepo.remote + subrepo_remote="$output" + fi + + if [[ -z $subrepo_branch ]]; then + SAY=false OUT=true RUN git config --file="$gitrepo" subrepo.branch + subrepo_branch="$output" + fi + + SAY=false OUT=true RUN git config --file="$gitrepo" subrepo.commit + subrepo_commit="$output" + + FAIL=false \ + SAY=false OUT=true RUN git config --file="$gitrepo" subrepo.parent + subrepo_parent="$output" + + FAIL=false \ + SAY=false OUT=true RUN git config --file="$gitrepo" subrepo.method + if [[ $output == "rebase" ]]; then + join_method="rebase" + else + # This is the default method + join_method="merge" + fi + + if [[ -z $subrepo_parent ]]; then + FAIL=false \ + SAY=false OUT=true RUN git config --file="$gitrepo" subrepo.former + subrepo_former="$output" + fi +} + + +# Update the subdir/.gitrepo state file: +update-gitrepo-file() { + local short_commit= + + local newfile=false + if [[ ! -e $gitrepo ]]; then + + FAIL=false RUN git cat-file -e "$original_head_commit":"$gitrepo" + + if OK; then + o "Try to recreate gitrepo file from $original_head_commit" + git cat-file -p "$original_head_commit":"$gitrepo" > "$gitrepo" + else + newfile=true + cat <<... > "$gitrepo" +; DO NOT EDIT (unless you know what you are doing) +; +; This subdirectory is a git "subrepo", and this file is maintained by the +; git-subrepo command. See https://github.com/git-commands/git-subrepo#readme +; +... + fi + fi + + + # TODO: only update remote and branch if supplied and $update_wanted + if $newfile || [[ $update_wanted && -n $override_remote ]]; then + RUN git config --file="$gitrepo" subrepo.remote "$subrepo_remote" + fi + + if $newfile || [[ $update_wanted && -n $override_branch ]]; then + RUN git config --file="$gitrepo" subrepo.branch "$subrepo_branch" + fi + + RUN git config --file="$gitrepo" subrepo.commit "$upstream_head_commit" + # Only write new parent when we are at the head of upstream + if [[ -n $upstream_head_commit && -n $subrepo_commit_ref ]]; then + OUT=true RUN git rev-parse "$subrepo_commit_ref" + o "$upstream_head_commit == $output" + if [[ $upstream_head_commit == $output ]]; then + RUN git config --file="$gitrepo" subrepo.parent "$original_head_commit" + fi + fi + + [[ -z $join_method ]] && join_method="merge" + RUN git config --file="$gitrepo" subrepo.method "$join_method" + + RUN git config --file="$gitrepo" subrepo.cmdver "$VERSION" + + RUN git add -f -- "$gitrepo" +} + +#------------------------------------------------------------------------------ +# Enviroment checks: +#------------------------------------------------------------------------------ + +# Check that system is ok for this command: +assert-environment-ok() { + type git &> /dev/null || + error "Can't find your 'git' command in '$PATH'." + + git_version=$(git --version | cut -d ' ' -f3) + + if [[ $( + printf "$REQUIRED_GIT_VERSION\n$git_version" | + sort -t. -k 1,1n -k 2,2n -k 3,3n | + head -n1 + ) == "$git_version" && + $git_version != "$REQUIRED_GIT_VERSION" + ]]; then + error "Requires git version $REQUIRED_GIT_VERSION or higher; "` + `"you have '$git_version'." + fi + + if [[ ${BASH_VERSINFO[0]} -lt 4 ]] ; then + echo "The git-subrepo command requires that 'Bash 4+' is installed." + echo "It doesn't need to be your shell, but it must be in your PATH." + if [[ $OSTYPE == darwin* ]]; then + echo "You appear to be on macOS." + echo "Try: 'brew install bash'." + echo "This will not change your user shell, it just installs 'Bash 5.x'." + fi + exit 1 + fi +} + +# Make sure git repo is ready: +assert-repo-is-ready() { + # Skip this for trivial info commands: + [[ $command =~ ^(help|version|upgrade)$ ]] && return + + # We must be inside a git repo: + git rev-parse --git-dir &> /dev/null || + error "Not inside a git repository." + + # Get the original branch and commit: + git:get-head-branch-name + original_head_branch="$output" + + # If a subrepo branch is currently checked out, then note it: + if [[ $original_head_branch =~ ^subrepo/(.*) ]]; then + error "Can't '$command' while subrepo branch is checked out." + fi + + # Make sure we are on a branch: + [[ $original_head_branch == HEAD || -z $original_head_branch ]] && + error "Must be on a branch to run this command." + + # In a work-tree: + SAY=false OUT=true RUN git rev-parse --is-inside-work-tree + [[ $output == true ]] || + error "Can't 'subrepo $command' outside a working tree." + + # HEAD exists: + [[ $command == clone ]] || + RUN git rev-parse --verify HEAD + + assert-working-copy-is-clean + + # For now, only support actions from top of repo: + if [[ -n "$(git rev-parse --show-prefix)" ]]; then + error "Need to run subrepo command from top level directory of the repo." + fi +} + +assert-working-copy-is-clean() { + # Repo is in a clean state: + if [[ $command =~ ^(clone|init|pull|push|branch|commit)$ ]]; then + # TODO: Should we check for untracked files? + local pwd=$(pwd) + o "Assert that working copy is clean: $pwd" + git update-index -q --ignore-submodules --refresh + git diff-files --quiet --ignore-submodules || + error "Can't $command subrepo. Unstaged changes. ($pwd)" + if [[ $command != clone ]] || git:rev-exists HEAD; then + git diff-index --quiet --ignore-submodules HEAD || + error "Can't $command subrepo. Working tree has changes. ($pwd)" + git diff-index --quiet --cached --ignore-submodules HEAD || + error "Can't $command subrepo. Index has changes. ($pwd)" + else + # Repo has no commits and we're cloning a subrepo. Working tree won't + # possibly have changes as there was nothing initial to change. + [[ -z $(git ls-files) ]] || + error "Can't $command subrepo. Index has changes. ($pwd)" + fi + fi +} + +# If subdir exists, make sure it is empty: +assert-subdir-ready-for-init() { + if [[ ! -e $subdir ]]; then + error "The subdir '$subdir' does not exist." + fi + if [[ -e $subdir/.gitrepo ]]; then + error "The subdir '$subdir' is already a subrepo." + fi + # Check that subdir is part of the repo + if [[ -z $(git log -1 -- $subdir) ]]; then + error "The subdir '$subdir' is not part of this repo." + fi +} + +# If subdir exists, make sure it is empty: +assert-subdir-empty() { + if [[ -e $subdir ]] && [[ -n $(ls -A $subdir) ]]; then + error "The subdir '$subdir' exists and is not empty." + fi +} + +#------------------------------------------------------------------------------ +# Getters of various information: +#------------------------------------------------------------------------------ + +# Find all the current subrepos by looking for all the subdirectories that +# contain a `.gitrepo` file. +get-all-subrepos() { + local paths=($(git ls-files | sed -n 's!/\.gitrepo$!!p' | sort)) + subrepos=() + local path + for path in "${paths[@]}"; do + add-subrepo "$path" + done +} + +add-subrepo() { + if ! $ALL_wanted; then + for path in "${subrepos[@]}"; do + [[ $1 =~ ^$path ]] && return + done + fi + subrepos+=("$1") +} + +# Determine the upstream's default head branch: +get-upstream-head-branch() { + OUT=true RUN git ls-remote $subrepo_remote + local remotes="$output" + [[ -n $remotes ]] || + error "Failed to 'git ls-remote $subrepo_remote'." + local commit="$( + echo "$remotes" | + grep HEAD | + cut -f1 + )" + local branch="$( + echo "$remotes" | + grep -E "$commit[[:space:]]+refs/heads/" | + grep -v HEAD | + head -n1 | + cut -f2 + )" + [[ $branch =~ refs/heads/ ]] || + error "Problem finding remote default head branch." + output="${branch#refs/heads/}" +} + +# Commit msg for an action commit: +# Don't use RUN here as it will pollute commit message +get-commit-message() { + local commit="none" + if git:rev-exists "$upstream_head_commit"; then + commit=$(git rev-parse --short "$upstream_head_commit") + fi + + local args=() debug_wanted=false + if $all_wanted; then + args+=("$subdir") + fi + args+=(${commit_msg_args[@]}) + + # Find the specific git-subrepo code used: + local command_remote='???' + local command_commit='???' + get-command-info + + local merged="none" + if git:rev-exists "$subrepo_commit_ref"; then + merged=$(git rev-parse --short "$subrepo_commit_ref") + fi + + local is_merge="" + if [[ $command != push ]]; then + if git:is_merge_commit "$subrepo_commit_ref"; then + is_merge=" (merge)" + fi + fi + + # TODO: Consider output for push! + + # Format subrepo commit message: + cat <<... +git subrepo $command$is_merge ${args[@]} + +subrepo: + subdir: "$subdir" + merged: "$merged" +upstream: + origin: "$subrepo_remote" + branch: "$subrepo_branch" + commit: "$commit" +git-subrepo: + version: "$VERSION" + origin: "$command_remote" + commit: "$command_commit" +... +} + +# Get location and version info about the git-subrepo command itself. This +# info goes into commit messages, so we can find out exactly how the commits +# were done. +get-command-info() { + local bin="$0" + if [[ $bin =~ / ]]; then + local lib="$(dirname "$bin")" + # XXX Makefile needs to install these symlinks: + # If `git-subrepo` was system-installed (`make install`): + if [[ -e $lib/git-subrepo.d/upstream ]] && + [[ -e $lib/git-subrepo.d/commit ]]; then + command_remote=$(readlink "$lib/git-subrepo.d/upstream") + command_commit=$(readlink "$lib/git-subrepo.d/commit") + elif [[ $lib =~ / ]]; then + lib="$(dirname "$lib")" + if [[ -d $lib/.git ]]; then + local remote="$( + GIT_DIR=$lib/.git git remote -v | + grep '^origin' | + head -n1 | + cut -f2 | + cut -d ' ' -f1 + )" + if [[ -n $remote ]]; then + command_remote="$remote" + else + local remote="$( + GIT_DIR=$lib/.git git remote -v | + head -n1 | + cut -f2 | + cut -d ' ' -f1 + )" + if [[ -n $remote ]]; then + command_remote="$remote" + fi + fi + local commit="$(GIT_DIR="$lib/.git" git rev-parse --short HEAD)" + if [[ -n $commit ]]; then + command_commit="$commit" + fi + fi + fi + fi +} + +#------------------------------------------------------------------------------ +# Instructional errors: +#------------------------------------------------------------------------------ + +error-join() { + cat <<... + +You will need to finish the $command by hand. A new working tree has been +created at $worktree so that you can resolve the conflicts +shown in the output above. + +This is the common conflict resolution workflow: + + 1. cd $worktree + 2. Resolve the conflicts (see "git status"). + 3. "git add" the resolved files. +... + + if [[ "$join_method" == "rebase" ]]; then + cat <<... + 4. git rebase --continue +... + else + cat <<... + 4. git commit +... + fi + + cat <<... + 5. If there are more conflicts, restart at step 2. + 6. cd $start_pwd +... + local branch_name="${branch:=subrepo/$subdir}" + if [[ "$command" == "push" ]]; then + cat <<... + 7. git subrepo push $subdir $branch_name +... + else + cat <<... + 7. git subrepo commit $subdir +... + fi + + if [[ "$command" == "pull" && "$join_method" == "rebase" ]]; then + cat <<... + +After you have performed the steps above you can push your local changes +without repeating the rebase by: + 1. git subrepo push $subdir $branch_name + +... + fi + +cat <<... +See "git help $join_method" for details. + +Alternatively, you can abort the $command and reset back to where you started: + + 1. git subrepo clean $subdir + +See "git help subrepo" for more help. + +... +} + +#------------------------------------------------------------------------------ +# Git command wrappers: +#------------------------------------------------------------------------------ + +git:branch-exists() { + git:rev-exists "refs/heads/$1" +} + +git:rev-exists() { + git rev-list "$1" -1 &> /dev/null +} + +git:ref-exists() { + test -n "$(git for-each-ref "$1")" +} + +git:get-head-branch-name() { + output= + local name="$(git symbolic-ref --short --quiet HEAD)" + [[ $name == HEAD ]] && return + output="$name" +} + +git:get-head-branch-commit() { + output="$(git rev-parse HEAD)" +} + +git:commit-in-rev-list() { + local commit="$1" + local list_head="$2" + git rev-list "$list_head" | grep -q "^$commit" +} + +git:make-ref() { + local ref_name="$1" + local commit="$(git rev-parse "$2")" + RUN git update-ref "$ref_name" "$commit" +} + +git:is_merge_commit() { + local commit="$1" + git show --summary "$commit" | grep -q ^Merge: +} + +git:create-worktree() { + local branch="$1" + worktree="$GIT_TMP/$branch" + RUN git worktree add "$worktree" "$branch" +} + +git:remove-worktree() { + o "Remove worktree: $worktree" + if [[ -d "$worktree" ]]; then + o "Check worktree for unsaved changes" + cd "$worktree" + assert-working-copy-is-clean + cd "$start_pwd" + + o "Clean up worktree $worktree" + rm -rf "$worktree" + RUN git worktree prune + fi +} + +git:delete-branch() { + local branch="$1" + o "Deleting old '$branch' branch." + # Remove worktree first, otherwise you can't delete the branch + git:remove-worktree + FAIL=false RUN git branch -D "$branch" +} + + +#------------------------------------------------------------------------------ +# Low level sugar commands: +#------------------------------------------------------------------------------ + +# Smart command runner: +RUN() { + $debug_wanted && $SAY && say '>>>' $* + if $EXEC; then + "$@" + return $? + fi + + OK=true + set +e + local rc= + local out= + if $debug_wanted && $TTY && interactive; then + "$@" + else + if $OUT; then + out="$("$@" 2>/dev/null)" + else + out="$("$@" 2>&1)" + fi + fi + rc=$? + set -e + + if [[ $rc -ne 0 ]]; then + OK=false + $FAIL && error "Command failed: '$*'.\n$out" + fi + output="$out" +} + + +interactive() { + if [[ -t 0 && -t 1 ]]; then + return 0 + else + return 1 + fi +} + +# Call a function with indent increased: +CALL() { + local INDENT=" $INDENT" + "$@" || true +} + +# Print verbose steps for commands with steps: +o() { + if $verbose_wanted; then + echo "$INDENT* $@" + fi +} + +# Print unless quiet mode: +say() { + $quiet_wanted || echo "$@" +} + +# Print to stderr: +err() { + echo "$@" >&2 +} + +# Check if OK: +OK() { + $OK +} + +# Nicely report common error messages: +usage-error() { + local msg="git-subrepo: $1" usage= + if [[ $GIT_SUBREPO_TEST_ERRORS != true ]]; then + source "${SOURCE_DIR}/git-subrepo.d/help-functions.bash" + if can "help:$command"; then + msg=$'\n'"$msg"$'\n'"$("help:$command")"$'\n' + fi + fi + echo "$msg" >&2 + exit 1 +} + +# Nicely report common error messages: +error() { + local msg="git-subrepo: $1" usage= + echo -e "$msg" >&2 + exit 1 +} + +# Start at the end: +[[ $BASH_SOURCE != "$0" ]] || main "$@" + +# Local Variables: +# tab-width: 2 +# sh-indentation: 2 +# sh-basic-offset: 2 +# End: +# vim: set ft=sh sw=2 lisp: |