#!/bin/bash
# -*- coding: utf-8 mode: sh -*- vim:sw=4:sts=4:et:ai:si:sta:fenc=utf-8

if [ $# -eq 1 -a "$1" == --nutools-completion ]; then
    echo '
function __pdev_completion() {
    local cur
    _get_comp_words_by_ref cur
    COMPREPLY=($(compgen -W "$(__pdev_branches)" "$cur"))
}
complete -F __pdev_completion pdev
'
    exit 0
fi

source "$(dirname "$0")/lib/ulib/ulib" || exit 1
urequire DEFAULTS ptools

function display_help() {
    uecho "$scriptname: basculer sur une branche de développement

USAGE
    $scriptname [FEATURE [SOURCE]]
    $scriptname -m|-l|-d [FEATURE]

- Vérifier l'existence de la branche develop. La créer si nécessaire en la
  basant sur [origin/]master.
- Vérifier s'il n'y a pas de modifications locales. Sinon, proposer de faire un
  commit ou un stash.
- Si FEATURE est spécifié, et si on n'est pas déjà sur cette branche, basculer
  vers cette nouvelle branche. S'il s'agit d'une nouvelle branche, la baser sur
  la branche SOURCE, qui vaut par défaut develop
- Si FEATURE n'est pas spécifié, basculer sur develop s'il s'agit de la seule
  solution, sinon afficher un menu pour choisir la branche de destination.

OPTIONS
    -C, --projdir PROJDIR
        Spécifier le répertoire de base du projet qui est dans git. Par défaut,
        on travaille dans le répertoire courant et on laisse git trouver le
        répertoire de base du projet. Avec cette option, le répertoire courant
        est modifié avant de lancer les commandes git.
    -O, --origin ORIGIN
        Spécifier le nom de l'origine. Par défaut, utiliser 'origin'
    -o, --offline
        En cas de création d'une branche, ne pas pousser vers l'origine; ne pas
        tenter le cas échéant de traquer la branche dans l'origine; ne pas
        supprimer une branche dans l'origine. Cette option est automatiquement
        activée si la variable UTOOLS_VCS_OFFLINE est définie.
    --online
        Annuler l'effet de la variable UTOOLS_VCS_OFFLINE: forcer le mode online
    --sync
        Faire un certain nombre d'opération pour 'corriger' le dépôt local: pour
        chacune des branches distantes, vérifier qu'il existe une branche locale
        qui la traque, et pour chaque feature branche locale, vérifier qu'il
        existe une branche distante associée. Cette option nécessite --online

    -s, --squash COMMIT_MSG
        Si la branche actuelle est une feature branch, la merger comme un seul
        commit avec le message COMMIT_MSG dans develop puis la supprimer. Puis
        basculer sur la branche develop.
        Cette option ne devrait pas être utilisée avec -k, puisque bien que les
        modifications soient mergées, la branche elle-même n'est pas considérée
        comme mergée. Les résultats sont donc indéfinis si la branche est mergée
        à plusieurs reprises.
    -b, --rebase
        Si la branche actuelle est une feature branch, lancer 'git rebase -i'
        sur la feature branch. Cela permet de réordonner les commits pour
        nettoyer l'historique avant de fusionner la branche avec -m
        Cette option devrait le cas échéant être utilisée immédiatement avant -m
        ou alors il faut forcer le push et communiquer avec l'équipe sur le fait
        que la branche de feature a été rebasée.
    -m, --merge
        Si la branche actuelle est une feature branch, la merger dans develop
        puis la supprimer. Puis basculer sur la branche develop.
    --merge-log
        Ajouter un résumé des modifications sur la feature branch en ajoutant le
        log en une ligne de chaque commit dans le message du merge. Cette option
        n'est en principe pas nécessaire puisque 'prel -um' intègre la liste des
        commits dans CHANGES.txt
    -k, --keep
        Avec l'option -m, ne pas supprimer une feature branch après l'avoir
        fusionnée dans develop. Cela permet d'intégrer les modifications petit à
        petit.
    --delete
        Supprimer une feature branch, à condition qu'elle aie déjà été
        entièrement fusionnée dans la branch develop
    --force-delete
        Supprimer une feature branch, même si elle n'a pas encore été fusionnée
        dans la branche develop

    -l, --log
    -d, --diff
        Afficher les modifications entre deux branches. L'option --log affiche
        les modifications dans l'ordre alors que --diff affiche les différences
        sous forme de diff. Les deux options peuvent être combinées et ont
        l'effet de 'git log -p'
        La branche comparée, s'il elle n'est pas spécifiée, est par défaut la
        branche courante. S'il s'agit d'une feature branch, elle est comparée à
        develop. S'il s'agit de la branche develop, elle est comparée à master."
}

projdir=
origin=origin
action=branch
squash_msg=
rebase_int=
merge_log=
merge_delete=1
force_delete=
log=
diff=
parse_opts "${PRETTYOPTS[@]}" \
    --help '$exit_with display_help' \
    -C:,--projdir: projdir= \
    -O:,--origin: origin= \
    -o,--offline UTOOLS_VCS_OFFLINE=1 \
    --online UTOOLS_VCS_OFFLINE= \
    --sync action=sync \
    -s:,--squash: '$action=squash; set@ squash_msg' \
    -b,--rebase '$action=none; rebase_int=1' \
    -m,--merge action=merge \
    --merge-log merge_log=1 \
    -k,--keep merge_delete= \
    --delete action=delete \
    --force-delete '$action=delete; force_delete=1' \
    -l,--log '$action=diff; log=1' \
    -d,--diff '$action=diff; diff=1' \
    @ args -- "$@" && set -- "${args[@]}" || die "$args"

if [ -n "$projdir" ]; then
    cd "$projdir" || die
fi

git_ensure_gitvcs

if [ "$action" == sync ]; then
    [ -z "$UTOOLS_VCS_OFFLINE" ] || die "L'option --sync nécessite le mode --online"

    setx -a branches=list_feature_branches "$origin"
    for branch in "${branches[@]}"; do
        git_track_branch "$branch" "$origin"
    done

    exit $?
fi

if ! git_have_branch develop; then
    estepn "Configuration de la branche develop"
    git_ensure_branch develop master "$origin"
    [ $? -eq 2 ] && die "Impossible de créer la branche develop. Veuillez vérifier que la branche master existe"
fi

setx branch=git_get_branch
if [ "$action" == branch ]; then
    feature="$1"
    source="${2:-develop}"

    if [ -z "$feature" ]; then
        setx -a branches=list_feature_branches "$origin"

        if [ ${#branches[*]} -eq 0 ]; then
            # En l'absence de feature branch, basculer sur develop
            feature=develop
        else
            array_ins branches develop
            default_branch="$branch"
            array_contains branches "$default_branch" || default_branch=develop
            simple_menu feature branches -d "$default_branch" \
                -t "Basculer vers une feature branch" \
                -m "Veuillez choisir la branche vers laquelle basculer"
        fi
    fi

    # On est peut-être déjà sur la bonne branche
    if git_is_branch "$feature"; then
        if [ -z "$UTOOLS_VCS_OFFLINE" ]; then
            git_track_branch "$feature" "$origin"
        fi
        git_fast_forward "$feature" "" "$origin"
        exit 0
    fi

    # Créer/basculer vers une feature branch
    git_ensure_cleancheckout
    is_any_branch "$feature" develop feature || die "$release: ce n'est pas une feature branch"
    r=0
    if git_have_branch "$feature"; then
        git checkout "$feature" || r=$?
    elif git_have_rbranch "$feature" "$origin"; then
        git checkout "$feature" || r=$?
    else
        estepn "\
Vous allez créer la nouvelle feature branch ${COULEUR_VERTE}$feature${COULEUR_NORMALE}
à partir de la branche source ${COULEUR_BLEUE}$source${COULEUR_NORMALE}"
        ask_yesno "Voulez-vous continuer?" O || die

        git_ensure_branch "$feature" "$source" "$origin"
        [ $? -eq 2 ] && die "Impossible de créer la branche $feature. Veuillez vérifier que la branche $source existe"
        git checkout "$feature" || r=$?
    fi
    if [ "$r" -eq 0 ]; then
        # éventuellement fast-forwarder automatiquement
        git_fast_forward "$feature" "" "$origin"
    fi

    exit $?
fi

feature="$1"
if [ -n "$feature" ]; then
    is_feature_branch "$feature" || die "$feature: ce n'est pas une feature branch"
    git_have_branch "$feature" || die "$feature: branche introuvable"
elif is_feature_branch "$branch"; then
    feature="$branch"
fi

if [ -n "$rebase_int" ]; then
    if [ "$branch" != "$feature" ]; then
        # s'assurer qu'on est sur la bonne branche
        git checkout "$feature" || die
        branch=feature
    fi
    setx base=git merge-base "$feature" develop
    git rebase -i "$base" || die
fi

if [ "$action" == squash -o "$action" == merge ]; then
    if [ -z "$feature" ]; then
        setx -a branches=list_feature_branches
        if [ ${#branches[*]} -eq 0 ]; then
            die "Aucune feature branch n'a été trouvée"
        elif [ ${#branches[*]} -eq 1 ]; then
            feature="${branches[0]}"
            estepn "Autosélection de $feature"
        else
            default_feature="$branch"
            array_contains branches "$default_feature" || default_feature="${branches[0]}"
            simple_menu feature branches -d "$default_feature" \
                -t "Choix de la feature branch" \
                -m "Veuillez choisir la branche"
        fi
    fi

    estepn "Intégration ${COULEUR_VERTE}$feature${COULEUR_NORMALE} --> ${COULEUR_BLEUE}develop${COULEUR_NORMALE}"
    ask_yesno "Voulez-vous continuer?" O || die

    # calculer la suite des modifications
    merge_msg="Intégration de la branche $feature"
    if [ -n "$merge_log" ]; then
        setx mergebase=git merge-base develop "$feature"
        setx modifs=git log --oneline "$mergebase..$feature"
        [ -n "$modifs" ] && merge_msg="$merge_msg
$modifs"
    fi

    git checkout develop || die
    if [ "$action" == squash ]; then
        [ -n "$squash_msg" ] || squash_msg="$merge_msg"
        git merge "$feature" --squash || die
        git commit -m "$squash_msg" || die
        # il faut forcer la suppression de la branche, puisqu'elle n'est pas
        # considérée comme mergée
        force_delete=1
    elif [ "$action" == merge ]; then
        git merge "$feature" -m "$merge_msg" --no-ff || die
    else
        die "$action: action imprévue"
    fi

    # mettre à jour la branche sur laquelle on se trouve
    setx branch=git_get_branch

    [ -n "$merge_delete" ] && action=delete
fi

if [ "$action" == delete ]; then
    if [ -z "$force_delete" ]; then
        # vérifier que la branche a été fusionnée
        git_is_merged "$feature" develop || die "Refus de supprimer la branche $feature: elle n'a pas été fusionnée dans develop"
    fi

    if [ "$branch" == "$feature" ]; then
        # si on est sur la branche en question, en sortir pour pouvoir la
        # supprimer
        git checkout develop || die
    fi

    estepi "Suppression de la branche locale"
    git branch -D "$feature"

    if git_have_remote "$origin"; then
        if [ -z "$UTOOLS_VCS_OFFLINE" ]; then
            estepi "Suppression de la branche distante"
            git push "$origin" ":$feature"
        elif git_have_rbranch "$feature" "$origin"; then
            eimportant "\
La branche $origin/$feature n'a plus lieu d'être, mais la configuration actuelle interdit de la supprimer.
Veuillez le faire manuellement avec la commande suivante:
    $(quoted_args git push "$origin" ":$feature")"
        fi
    fi
fi

if [ "$action" == diff ]; then
    if [ -n "$log" ]; then
        if [ -n "$feature" ]; then
            git log ${diff:+-p} develop.."$feature"
        else
            git log ${diff:+-p} master..develop
        fi
    elif [ -n "$diff" ]; then
        if [ -n "$feature" ]; then
            git diff develop.."$feature"
        else
            git diff master..develop
        fi
    fi
    exit $?
fi