179 lines
5.2 KiB
Bash
179 lines
5.2 KiB
Bash
#!/bin/bash
|
|
_dir=$( dirname $(readlink -e -- "${BASH_SOURCE}"))
|
|
source $_dir/io.sh || exit 255
|
|
source $_dir/basic.sh || exit 255
|
|
source $_dir/pkg.sh || exit 255
|
|
|
|
########################################################
|
|
# Shorthands for git
|
|
########################################################
|
|
|
|
git.clone_quick() {
|
|
require git
|
|
git clone --depth=1 --single-branch $*
|
|
}
|
|
|
|
git.is_repo() {
|
|
require git
|
|
[ "$1" ] || die "Path is not specified" 101
|
|
require_dir "$1/"
|
|
check_dir "$1/.git"
|
|
}
|
|
|
|
git.require_repo() {
|
|
require git
|
|
git.is_repo "$1" || die "'$1' is not git repository!" 10
|
|
}
|
|
|
|
git.cfg() {
|
|
require git
|
|
[ "$1" ] || die "Key is not specified" 101
|
|
if [[ "$2" ]]; then
|
|
git config --global --replace-all "$1" "$2"
|
|
else
|
|
echo $(git config --global --get-all "$1")
|
|
fi
|
|
}
|
|
|
|
git.set_user() {
|
|
require git
|
|
[ "$1" ] || die "git.set_user: Repo is not specified" 100
|
|
git.cfg "$1" "user.name" "$2"
|
|
git.cfg "$1" "user.email" "$3"
|
|
success "User set to '$name <$email>' in ${FWHITE}$1"
|
|
}
|
|
|
|
git.fetch() {
|
|
require git
|
|
if [ "$1" ]; then
|
|
if git.remote_branch_exists "origin/$1"; then
|
|
git fetch origin "refs/heads/$1:refs/remotes/origin/$1" --progress --prune --quiet 2>&1 || die "Could not fetch $1 from origin" 12
|
|
else
|
|
warn "Tried to fetch branch 'origin/$1' but it does not exist."
|
|
fi
|
|
else
|
|
git fetch origin --progress --prune --quiet 2>&1 || exit 12
|
|
fi
|
|
}
|
|
|
|
git.reset() {
|
|
require git
|
|
git reset --hard HEAD
|
|
git clean -fd
|
|
}
|
|
|
|
git.clone() {
|
|
require git
|
|
git clone $* 2>&1
|
|
}
|
|
|
|
git.co() {
|
|
require git
|
|
git checkout $* 2>&1
|
|
}
|
|
|
|
git.is_it_current_branch() {
|
|
require git
|
|
[ "$1" ] || die "git.is_it_current_branch: Branch is not specified" 19
|
|
[[ "$(git.current_branch)" = "$1" ]]
|
|
}
|
|
|
|
git.pull() {
|
|
require git
|
|
[ "$1" ] && BRANCH=$1 || BRANCH=$(git.current_branch)
|
|
# note "Updating branch $BRANCH..."
|
|
git pull origin "refs/heads/$BRANCH:refs/remotes/origin/$BRANCH" --prune --force --quiet 2>&1 || exit 13
|
|
git pull origin --tags --force --quiet 2>&1 || exit 13
|
|
# [ "$1" ] || die "git.pull: Branch is not specified" 19
|
|
# if [ "$1" ]; then
|
|
# note "Updating branch $1..."
|
|
# git pull origin "refs/heads/$1:refs/remotes/origin/$1" --prune --force --quiet 2>&1 || exit 13
|
|
# else
|
|
# note "Updating current branch..."
|
|
# git pull
|
|
# fi
|
|
}
|
|
|
|
git.current_branch() {
|
|
require git
|
|
git branch --show-current || exit 18
|
|
}
|
|
|
|
git.local_branch_exists() {
|
|
require git
|
|
[ -n "$(git for-each-ref --format='%(refname:short)' refs/heads/$1)" ]
|
|
}
|
|
|
|
git.update_refs() {
|
|
require git
|
|
info "Updating local refs..."
|
|
git remote update origin --prune 1>/dev/null 2>&1 || exit 18
|
|
}
|
|
|
|
git.delete_remote_branch() {
|
|
require git
|
|
[ "$1" ] || die "git.remote_branch_exists: Branch is not specified" 19
|
|
if git.remote_branch_exists "origin/$1"; then
|
|
git push origin :"$1" # || die "Could not delete the remote $1 in $ORIGIN"
|
|
return 0
|
|
else
|
|
warn "Trying to delete the remote branch $1, but it does not exists in origin"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
git.is_clean_worktree() {
|
|
require git
|
|
git rev-parse --verify HEAD >/dev/null || exit 18
|
|
git update-index -q --ignore-submodules --refresh
|
|
git diff-files --quiet --ignore-submodules || return 1
|
|
git diff-index --quiet --ignore-submodules --cached HEAD -- || return 2
|
|
return 0
|
|
}
|
|
|
|
git.is_branch_merged_into() {
|
|
require git
|
|
[ "$1" ] || die "git.remote_branch_exists: Branch1 is not specified" 19
|
|
[ "$2" ] || die "git.remote_branch_exists: Branch2 is not specified" 19
|
|
git.update_refs
|
|
local merge_hash=$(git merge-base "$1"^{} "$2"^{})
|
|
local base_hash=$(git rev-parse "$1"^{})
|
|
[ "$merge_hash" = "$base_hash" ]
|
|
}
|
|
|
|
git.remote_branch_exists() {
|
|
require git
|
|
[ "$1" ] || die "git.remote_branch_exists: Branch is not specified" 19
|
|
git.update_refs
|
|
[ -n "$(git for-each-ref --format='%(refname:short)' refs/remotes/$1)" ]
|
|
}
|
|
|
|
git.new_branch() {
|
|
require git
|
|
[ "$1" ] || die "git.new_branch: Branch is not specified" 19
|
|
if [ "$2" ] && ! git.local_branch_exists "$2" && git.remote_branch_exists "origin/$2"; then
|
|
git.co -b "$1" origin/"$2"
|
|
else
|
|
git.co -b "$1" "$2"
|
|
fi
|
|
}
|
|
|
|
git.require_clean_worktree() {
|
|
require git
|
|
if ! git.is_clean_worktree; then
|
|
warn "Your working tree is dirty! Look at this:"
|
|
git status -bs
|
|
_T="What should you do now?\n"
|
|
_T="${_T}\t${BOLD}${FWHITE}0.${RESET} try to continue as is\t- errors may occur!\n"
|
|
_T="${_T}\t${BOLD}${FWHITE}1.${RESET} hard reset\t\t\t- clear current changes and new files\n"
|
|
_T="${_T}\t${BOLD}${FWHITE}2.${RESET} stash changes (default)\t- save all changes in safe to apply them later via 'git stash pop'\n"
|
|
_T="${_T}\t${BOLD}${FWHITE}3.${RESET} cancel\n"
|
|
ask "${_T}${BOLD}${FWHITE}Your choice [0-3]" reset_answer
|
|
case $reset_answer in
|
|
1 ) warn "Clearing your work..." && git.reset ;;
|
|
3 ) exit ;;
|
|
* ) git stash -a -u -m "WIP before switch to $branch_task" ;;
|
|
esac
|
|
fi
|
|
}
|