Repository: nvie/gitflow Branch: develop Commit: d2eee63886e2 Files: 18 Total size: 83.3 KB Directory structure: gitextract_ltzdd6d0/ ├── .gitignore ├── .gitmodules ├── AUTHORS ├── Changes.mdown ├── LICENSE ├── Makefile ├── README.mdown ├── bump-version ├── contrib/ │ ├── gitflow-installer.sh │ └── msysgit-install.cmd ├── git-flow ├── git-flow-feature ├── git-flow-hotfix ├── git-flow-init ├── git-flow-release ├── git-flow-support ├── git-flow-version └── gitflow-common ================================================ FILE CONTENTS ================================================ ================================================ FILE: .gitignore ================================================ debian/files debian/*.substvars debian/*.debhelper.log debian/*/* ================================================ FILE: .gitmodules ================================================ [submodule "shFlags"] path = shFlags url = git://github.com/nvie/shFlags.git ================================================ FILE: AUTHORS ================================================ Authors are (ordered by first commit date): - Vincent Driessen - Benedikt Böhm - Daniel Truemper - Jason L. Shiffer - Randy Merrill - Rick Osborne - Mark Derricutt - Nowell Strite - Felipe Talavera - Guillaume-Jean Herbiet - Joseph A. Levin - Jannis Leidel - Konstantin Tjuterev - Kiall Mac Innes - Jon Bernard - Olivier Mengué - Emre Berge Ergenekon - Eric Holmes - Vedang Manerikar - Myke Hines Portions derived from other open source works are clearly marked. ================================================ FILE: Changes.mdown ================================================ 0.4.2: ----- Release date: **not yet** * `git flow init` now detects situations where origin already has gitflow branches set up, and behaves accordingly (thanks Emre Berge Ergenekon). * `git flow feature finish` can now be called without a feature branch name(prefix) argument and will finish the current branch, if on any. * `git flow feature pull` now has a `-r` flag, to support `pull --rebase` semantics (thanks Vedang Manerikar). * Various minor bug fixes related to internal argument passing. * Improved some documentation. * Better support for Windows and BSD users. * Add package installer for the Windows platform. 0.4.1: ----- Release date: **2011/02/04** * New option `-d` added to `git flow init`, to initialize with defaults without asking for input interactively. Ideal for creating git-flow enabled repos in custom scripts. * The parsing issues related to git-flow feature's flags are now dealt with on all known platforms. (Fixed #54, #62, #86, #97) * Escape queries for detecting branch/tag names. (Fixed #91) 0.4: --- Release date: **2010/10/18** * The flag parsing issues of git-flow subcommands are solved for most platforms. * `git flow {feature,hotfix,release} finish` now takes a `-k` flag, to keep the branch around after finishing. * `git flow release finish` takes a `-n` flag, to skip tagging. * For consistency, `git flow {release,hotfix}` now, too, have a `publish` and `track` subcommand, just like `feature`. * Various minor fixes. 0.3: ---- Release date: **2010/07/22** * New subcommands for `git flow feature`: - **checkout**: For easily checking out features by their short name. Even allows unique prefixes as arguments (see below). - **pull**: This subcommand allows you to painlessly work on a feature branch together with another peer. This is especially valuable for doing peer reviews of other people's code. For more detailed info, see the [commit log][1]. * Easier addressing of branch names by using name prefixes. For example, when using: git flow feature finish fo this automatically finishes the feature branch `foobar` if that's the only feature branch name starting with `fo`. * No force flag anymore for new feature branches `git flow feature start` lost its `-f` (force) flag. You now don't have to be in a clean repo anymore to start a new feature branch. This avoids the manual `git stash`, `git flow feature start`, `git stash pop` cycle. * You can use `git-flow` in stand-alone repo's now. This means it does not assume you have an `origin` repository. (Thanks [Mark][2].) * No commands fetch from `origin` by default anymore. There were some issues related to disabling this flag on some platforms. * Init guesses branch names you may want to use for `develop` and `master`. * Added super-easy installation script. (Thanks [Rick][3].) * Added BSD license. [1]: http://github.com/nvie/gitflow/commit/f68d405cc3a11e9df3671f567658a6ab6ed8e0a1 [2]: http://github.com/talios [3]: http://github.com/rickosborne Older versions -------------- No change history is recorded for pre-0.3 releases. ================================================ FILE: LICENSE ================================================ Copyright 2010 Vincent Driessen. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. The views and conclusions contained in the software and documentation are those of the authors and should not be interpreted as representing official policies, either expressed or implied, of Vincent Driessen. ================================================ FILE: Makefile ================================================ # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # prefix=/usr/local # files that need mode 755 EXEC_FILES=git-flow # files that need mode 644 SCRIPT_FILES =git-flow-init SCRIPT_FILES+=git-flow-feature SCRIPT_FILES+=git-flow-hotfix SCRIPT_FILES+=git-flow-release SCRIPT_FILES+=git-flow-support SCRIPT_FILES+=git-flow-version SCRIPT_FILES+=gitflow-common SCRIPT_FILES+=gitflow-shFlags all: @echo "usage: make install" @echo " make uninstall" install: @test -f gitflow-shFlags || (echo "Run 'git submodule init && git submodule update' first." ; exit 1 ) install -d -m 0755 $(prefix)/bin install -m 0755 $(EXEC_FILES) $(prefix)/bin install -m 0644 $(SCRIPT_FILES) $(prefix)/bin uninstall: test -d $(prefix)/bin && \ cd $(prefix)/bin && \ rm -f $(EXEC_FILES) $(SCRIPT_FILES) ================================================ FILE: README.mdown ================================================ > [!IMPORTANT] > > # ⚠️ git-flow has moved to git-flow-next! > > **This repository is no longer maintained.** The wonderful team at [Tower](https://www.git-tower.com/) has created [**git-flow-next**](https://git-flow.sh) — a modern, fully customizable evolution of git-flow, built on a generic branch dependency model. It’s fully backward compatible with git-flow, open source, and actively maintained. > > ### Get started with git-flow-next: > > - **Website:** https://git-flow.sh > - **GitHub:** https://github.com/gittower/git-flow-next > - **Evolution Story:** Read about the journey from git-flow → git-flow-avh → git-flow-next in [their blog post](https://git-flow.sh/blog) > > ### Why git-flow-next? > > git-flow-next builds upon the foundation laid by the original git-flow, offering: > > - Full customization of branch names and workflow > - Modern implementation with active maintenance > - Backward compatibility with existing git-flow workflows > - Upcoming features like stacked branches and topic branch syncing > > ### Thank you ❤️ > > To everyone who has used, contributed to, and supported git-flow over the past 15+ years — thank you! Your feedback, contributions, and adoption made git-flow one of the most widely-used Git workflow tools. Special thanks to Peter van der Does for maintaining git-flow-avh, and to the folks at Tower for carrying the torch forward with git-flow-next. --- (Below, you’ll find the original documentation for historical reference.) --- git-flow ======== A collection of Git extensions to provide high-level repository operations for Vincent Driessen's [branching model](http://nvie.com/git-model "original blog post"). Getting started --------------- For the best introduction to get started with `git flow`, please read Jeff Kreeftmeijer's blog post: [http://jeffkreeftmeijer.com/2010/why-arent-you-using-git-flow/](http://jeffkreeftmeijer.com/2010/why-arent-you-using-git-flow/) Or have a look at one of these screen casts: * [How to use a scalable Git branching model called git-flow](http://buildamodule.com/video/change-management-and-version-control-deploying-releases-features-and-fixes-with-git-how-to-use-a-scalable-git-branching-model-called-gitflow) (by Build a Module) * [A short introduction to git-flow](http://vimeo.com/16018419) (by Mark Derricutt) * [On the path with git-flow](http://codesherpas.com/screencasts/on_the_path_gitflow.mov) (by Dave Bock) Installing git-flow ------------------- See the Wiki for up-to-date [Installation Instructions](https://github.com/nvie/gitflow/wiki/Installation). Integration with your shell --------------------------- For those who use the [Bash](http://www.gnu.org/software/bash/) or [ZSH](http://www.zsh.org) shell, please check out the excellent work on the [git-flow-completion](http://github.com/bobthecow/git-flow-completion) project by [bobthecow](http://github.com/bobthecow). It offers tab-completion for all git-flow subcommands and branch names. FAQ --- See the [FAQ](http://github.com/nvie/gitflow/wiki/FAQ) section of the project Wiki. License terms ------------- git-flow is published under the liberal terms of the BSD License, see the [LICENSE](LICENSE) file. Although the BSD License does not require you to share any modifications you make to the source code, you are very much encouraged and invited to contribute back your modifications to the community, preferably in a Github fork, of course. ### Initialization To initialize a new repo with the basic branch structure, use: git flow init [-d] This will then interactively prompt you with some questions on which branches you would like to use as development and production branches, and how you would like your prefixes be named. You may simply press Return on any of those questions to accept the (sane) default suggestions. The ``-d`` flag will accept all defaults. ### Creating feature/release/hotfix/support branches * To list/start/finish feature branches, use: git flow feature git flow feature start [] git flow feature finish For feature branches, the `` arg must be a commit on `develop`. * To push/pull a feature branch to the remote repository, use: git flow feature publish git flow feature pull * To list/start/finish release branches, use: git flow release git flow release start [] git flow release finish For release branches, the `` arg must be a commit on `develop`. * To list/start/finish hotfix branches, use: git flow hotfix git flow hotfix start [] git flow hotfix finish For hotfix branches, the `` arg must be a commit on `master`. * To list/start support branches, use: git flow support git flow support start For support branches, the `` arg must be a commit on `master`. ================================================ FILE: bump-version ================================================ #!/bin/sh usage() { echo "usage: bump-version " } if [ $# -ne 1 ]; then usage exit 1 fi if ! sed 's/^GITFLOW_VERSION=.*$/GITFLOW_VERSION='$1'/g' git-flow-version > .git-flow-version.new; then echo "Could not replace GITFLOW_VERSION variable." >&2 exit 2 fi mv .git-flow-version.new git-flow-version git add git-flow-version git commit -m "Bumped version number to $1" git-flow-version ================================================ FILE: contrib/gitflow-installer.sh ================================================ #!/bin/bash # git-flow make-less installer for *nix systems, by Rick Osborne # Based on the git-flow core Makefile: # http://github.com/nvie/gitflow/blob/master/Makefile # Licensed under the same restrictions as git-flow: # http://github.com/nvie/gitflow/blob/develop/LICENSE # Does this need to be smarter for each host OS? if [ -z "$INSTALL_PREFIX" ] ; then INSTALL_PREFIX="/usr/local/bin" fi if [ -z "$REPO_NAME" ] ; then REPO_NAME="gitflow" fi if [ -z "$REPO_HOME" ] ; then REPO_HOME="http://github.com/nvie/gitflow.git" fi EXEC_FILES="git-flow" SCRIPT_FILES="git-flow-init git-flow-feature git-flow-hotfix git-flow-release git-flow-support git-flow-version gitflow-common gitflow-shFlags" SUBMODULE_FILE="gitflow-shFlags" echo "### gitflow no-make installer ###" case "$1" in uninstall) echo "Uninstalling git-flow from $INSTALL_PREFIX" if [ -d "$INSTALL_PREFIX" ] ; then for script_file in $SCRIPT_FILES $EXEC_FILES ; do echo "rm -vf $INSTALL_PREFIX/$script_file" rm -vf "$INSTALL_PREFIX/$script_file" done else echo "The '$INSTALL_PREFIX' directory was not found." echo "Do you need to set INSTALL_PREFIX ?" fi exit ;; help) echo "Usage: [environment] gitflow-installer.sh [install|uninstall]" echo "Environment:" echo " INSTALL_PREFIX=$INSTALL_PREFIX" echo " REPO_HOME=$REPO_HOME" echo " REPO_NAME=$REPO_NAME" exit ;; *) echo "Installing git-flow to $INSTALL_PREFIX" if [ -d "$REPO_NAME" -a -d "$REPO_NAME/.git" ] ; then echo "Using existing repo: $REPO_NAME" else echo "Cloning repo from GitHub to $REPO_NAME" git clone "$REPO_HOME" "$REPO_NAME" fi if [ -f "$REPO_NAME/$SUBMODULE_FILE" ] ; then echo "Submodules look up to date" else echo "Updating submodules" lastcwd=$PWD cd "$REPO_NAME" git submodule init git submodule update cd "$lastcwd" fi install -v -d -m 0755 "$INSTALL_PREFIX" for exec_file in $EXEC_FILES ; do install -v -m 0755 "$REPO_NAME/$exec_file" "$INSTALL_PREFIX" done for script_file in $SCRIPT_FILES ; do install -v -m 0644 "$REPO_NAME/$script_file" "$INSTALL_PREFIX" done exit ;; esac ================================================ FILE: contrib/msysgit-install.cmd ================================================ @echo off setlocal if not "%~1"=="" set GIT_HOME=%~f1 if "%GIT_HOME%"=="" call :FindGitHome "git.cmd" if exist "%GIT_HOME%" goto :GitHomeOK echo MsysGit installation directory not found.>&2 echo Try to give the directory name on the command line:>&2 echo %0 "%ProgramFiles%\Git" endlocal exit /B 1 :GitHomeOK set ERR=0 echo Installing gitflow into "%GIT_HOME%"... call :ChkGetopt getopt.exe || set ERR=1 if %ERR%==1 goto :End echo getopt.exe... Found if not exist "%GIT_HOME%\bin\git-flow" goto :Install echo GitFlow is already installed.>&2 set /p mychoice="Do you want to replace it [y/n]" if "%mychoice%"=="y" goto :DeleteOldFiles goto :Abort :DeleteOldFiles echo Deleting old files... for /F %%i in ("%GIT_HOME%\git-flow*" "%GIT_HOME%\gitflow-*") do if exist "%%~fi" del /F /Q "%%~fi" :Install echo Copying files... ::goto :EOF xcopy "%~dp0\..\git-flow" "%GIT_HOME%\bin" /Y /R /F if errorlevel 4 if not errorlevel 5 goto :AccessDenied if errorlevel 1 set ERR=1 xcopy "%~dp0\..\git-flow*" "%GIT_HOME%\bin" /Y /R /F || set ERR=1 xcopy "%~dp0\..\gitflow-*" "%GIT_HOME%\bin" /Y /R /F || set ERR=1 xcopy "%~dp0\..\shFlags\src\shflags" "%GIT_HOME%\bin\gitflow-shFlags" /Y /R /F || set ERR=1 if %ERR%==1 choice /T 30 /C Y /D Y /M "Some unexpected errors happened. Sorry, you'll have to fix them by yourself." :End endlocal & exit /B %ERR% goto :EOF :AccessDenied set ERR=1 echo. echo You should run this script with "Full Administrator" rights:>&2 echo - Right-click with Shift on the script from the Explorer>&2 echo - Select "Run as administrator">&2 choice /T 30 /C YN /D Y /N >nul goto :End :Abort echo Installation canceled.>&2 set ERR=1 goto :End :ChkGetopt :: %1 is getopt.exe if exist "%GIT_HOME%\bin\%1" goto :EOF if exist "%USERPROFILE%\bin\%1" goto :EOF if exist "%~f$PATH:1" goto :EOF echo %GIT_HOME%\bin\%1 not found.>&2 echo You have to install this file manually. See the GitFlow README. exit /B 1 :FindGitHome setlocal set GIT_CMD_DIR=%~dp$PATH:1 if "%GIT_CMD_DIR%"=="" endlocal & goto :EOF endlocal & set GIT_HOME=%GIT_CMD_DIR:~0,-5% goto :EOF ================================================ FILE: git-flow ================================================ #!/bin/sh # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # # set this to workaround expr problems in shFlags on freebsd if uname -s | egrep -iq 'bsd'; then export EXPR_COMPAT=1; fi # enable debug mode if [ "$DEBUG" = "yes" ]; then set -x fi # The sed expression here replaces all backslashes by forward slashes. # This helps our Windows users, while not bothering our Unix users. export GITFLOW_DIR=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')") usage() { echo "usage: git flow " echo echo "Available subcommands are:" echo " init Initialize a new git repo with support for the branching model." echo " feature Manage your feature branches." echo " release Manage your release branches." echo " hotfix Manage your hotfix branches." echo " support Manage your support branches." echo " version Shows version information." echo echo "Try 'git flow help' for details." } main() { if [ $# -lt 1 ]; then usage exit 1 fi # load common functionality . "$GITFLOW_DIR/gitflow-common" # This environmental variable fixes non-POSIX getopt style argument # parsing, effectively breaking git-flow subcommand parsing on several # Linux platforms. export POSIXLY_CORRECT=1 # use the shFlags project to parse the command line arguments . "$GITFLOW_DIR/gitflow-shFlags" FLAGS_PARENT="git flow" # allow user to request git action logging DEFINE_boolean show_commands false 'show actions taken (git commands)' g # do actual parsing FLAGS "$@" || exit $? eval set -- "${FLAGS_ARGV}" # sanity checks SUBCOMMAND="$1"; shift if [ ! -e "$GITFLOW_DIR/git-flow-$SUBCOMMAND" ]; then usage exit 1 fi # run command . "$GITFLOW_DIR/git-flow-$SUBCOMMAND" FLAGS_PARENT="git flow $SUBCOMMAND" # test if the first argument is a flag (i.e. starts with '-') # in that case, we interpret this arg as a flag for the default # command SUBACTION="default" if [ "$1" != "" ] && { ! echo "$1" | grep -q "^-"; } then SUBACTION="$1"; shift fi if ! type "cmd_$SUBACTION" >/dev/null 2>&1; then warn "Unknown subcommand: '$SUBACTION'" usage exit 1 fi # run the specified action if [ $SUBACTION != "help" ] && [ $SUBCOMMAND != "init" ] ; then init fi cmd_$SUBACTION "$@" } main "$@" ================================================ FILE: git-flow-feature ================================================ # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # init() { require_git_repo require_gitflow_initialized gitflow_load_settings PREFIX=$(git config --get gitflow.prefix.feature) } usage() { echo "usage: git flow feature [list] [-v]" echo " git flow feature start [-F] []" echo " git flow feature finish [-rFkDS] []" echo " git flow feature publish " echo " git flow feature track " echo " git flow feature diff []" echo " git flow feature rebase [-i] []" echo " git flow feature checkout []" echo " git flow feature pull [-r] []" } cmd_default() { cmd_list "$@" } cmd_list() { DEFINE_boolean verbose false 'verbose (more) output' v parse_args "$@" local feature_branches local current_branch local short_names feature_branches=$(echo "$(git_local_branches)" | grep "^$PREFIX") if [ -z "$feature_branches" ]; then warn "No feature branches exist." warn "" warn "You can start a new feature branch:" warn "" warn " git flow feature start []" warn "" exit 0 fi current_branch=$(git branch --no-color | grep '^\* ' | grep -v 'no branch' | sed 's/^* //g') short_names=$(echo "$feature_branches" | sed "s ^$PREFIX g") # determine column width first local width=0 local branch for branch in $short_names; do local len=${#branch} width=$(max $width $len) done width=$(($width+3)) local branch for branch in $short_names; do local fullname=$PREFIX$branch local base=$(git merge-base "$fullname" "$DEVELOP_BRANCH") local develop_sha=$(git rev-parse "$DEVELOP_BRANCH") local branch_sha=$(git rev-parse "$fullname") if [ "$fullname" = "$current_branch" ]; then printf "* " else printf " " fi if flag verbose; then printf "%-${width}s" "$branch" if [ "$branch_sha" = "$develop_sha" ]; then printf "(no commits yet)" elif [ "$base" = "$branch_sha" ]; then printf "(is behind develop, may ff)" elif [ "$base" = "$develop_sha" ]; then printf "(based on latest develop)" else printf "(may be rebased)" fi else printf "%s" "$branch" fi echo done } cmd_help() { usage exit 0 } require_name_arg() { if [ "$NAME" = "" ]; then warn "Missing argument " usage exit 1 fi } expand_nameprefix_arg() { require_name_arg local expanded_name local exitcode expanded_name=$(gitflow_resolve_nameprefix "$NAME" "$PREFIX") exitcode=$? case $exitcode in 0) NAME=$expanded_name BRANCH=$PREFIX$NAME ;; *) exit 1 ;; esac } use_current_feature_branch_name() { local current_branch=$(git_current_branch) if startswith "$current_branch" "$PREFIX"; then BRANCH=$current_branch NAME=${BRANCH#$PREFIX} else warn "The current HEAD is no feature branch." warn "Please specify a argument." exit 1 fi } expand_nameprefix_arg_or_current() { if [ "$NAME" != "" ]; then expand_nameprefix_arg require_branch "$PREFIX$NAME" else use_current_feature_branch_name fi } name_or_current() { if [ -z "$NAME" ]; then use_current_feature_branch_name fi } parse_args() { # parse options FLAGS "$@" || exit $? eval set -- "${FLAGS_ARGV}" # read arguments into global variables NAME=$1 BRANCH=$PREFIX$NAME } parse_remote_name() { # parse options FLAGS "$@" || exit $? eval set -- "${FLAGS_ARGV}" # read arguments into global variables REMOTE=$1 NAME=$2 BRANCH=$PREFIX$NAME } cmd_start() { DEFINE_boolean fetch false 'fetch from origin before performing local operation' F parse_args "$@" BASE=${2:-$DEVELOP_BRANCH} require_name_arg # sanity checks require_branch_absent "$BRANCH" # update the local repo with remote changes, if asked if flag fetch; then git_do fetch -q "$ORIGIN" "$DEVELOP_BRANCH" fi # if the origin branch counterpart exists, assert that the local branch # isn't behind it (to avoid unnecessary rebasing) if git_branch_exists "$ORIGIN/$DEVELOP_BRANCH"; then require_branches_equal "$DEVELOP_BRANCH" "$ORIGIN/$DEVELOP_BRANCH" fi # create branch if ! git_do checkout -b "$BRANCH" "$BASE"; then die "Could not create feature branch '$BRANCH'" fi echo echo "Summary of actions:" echo "- A new branch '$BRANCH' was created, based on '$BASE'" echo "- You are now on branch '$BRANCH'" echo "" echo "Now, start committing on your feature. When done, use:" echo "" echo " git flow feature finish $NAME" echo } cmd_finish() { DEFINE_boolean fetch false "fetch from $ORIGIN before performing finish" F DEFINE_boolean rebase false "rebase instead of merge" r DEFINE_boolean keep false "keep branch after performing finish" k DEFINE_boolean force_delete false "force delete feature branch after finish" D DEFINE_boolean squash false "squash feature during merge" S parse_args "$@" expand_nameprefix_arg_or_current # sanity checks require_branch "$BRANCH" # detect if we're restoring from a merge conflict if [ -f "$DOT_GIT_DIR/.gitflow/MERGE_BASE" ]; then # # TODO: detect that we're working on the correct branch here! # The user need not necessarily have given the same $NAME twice here # (although he/she should). # # TODO: git_is_clean_working_tree() should provide an alternative # exit code for "unmerged changes in working tree", which we should # actually be testing for here if git_is_clean_working_tree; then FINISH_BASE=$(cat "$DOT_GIT_DIR/.gitflow/MERGE_BASE") # Since the working tree is now clean, either the user did a # succesfull merge manually, or the merge was cancelled. # We detect this using git_is_branch_merged_into() if git_is_branch_merged_into "$BRANCH" "$FINISH_BASE"; then rm -f "$DOT_GIT_DIR/.gitflow/MERGE_BASE" helper_finish_cleanup exit 0 else # If the user cancelled the merge and decided to wait until later, # that's fine. But we have to acknowledge this by removing the # MERGE_BASE file and continuing normal execution of the finish rm -f "$DOT_GIT_DIR/.gitflow/MERGE_BASE" fi else echo echo "Merge conflicts not resolved yet, use:" echo " git mergetool" echo " git commit" echo echo "You can then complete the finish by running it again:" echo " git flow feature finish $NAME" echo exit 1 fi fi # sanity checks require_clean_working_tree # update local repo with remote changes first, if asked if has "$ORIGIN/$BRANCH" $(git_remote_branches); then if flag fetch; then git_do fetch -q "$ORIGIN" "$BRANCH" git_do fetch -q "$ORIGIN" "$DEVELOP_BRANCH" fi fi if has "$ORIGIN/$BRANCH" $(git_remote_branches); then require_branches_equal "$BRANCH" "$ORIGIN/$BRANCH" fi if has "$ORIGIN/$DEVELOP_BRANCH" $(git_remote_branches); then require_branches_equal "$DEVELOP_BRANCH" "$ORIGIN/$DEVELOP_BRANCH" fi # if the user wants to rebase, do that first if flag rebase; then if ! git flow feature rebase "$NAME" "$DEVELOP_BRANCH"; then warn "Finish was aborted due to conflicts during rebase." warn "Please finish the rebase manually now." warn "When finished, re-run:" warn " git flow feature finish '$NAME' '$DEVELOP_BRANCH'" exit 1 fi fi # merge into BASE git_do checkout "$DEVELOP_BRANCH" if [ "$(git rev-list -n2 "$DEVELOP_BRANCH..$BRANCH" | wc -l)" -eq 1 ]; then git_do merge --ff "$BRANCH" else if noflag squash; then git_do merge --no-ff "$BRANCH" else git_do merge --squash "$BRANCH" git_do commit git_do merge "$BRANCH" fi fi if [ $? -ne 0 ]; then # oops.. we have a merge conflict! # write the given $DEVELOP_BRANCH to a temporary file (we need it later) mkdir -p "$DOT_GIT_DIR/.gitflow" echo "$DEVELOP_BRANCH" > "$DOT_GIT_DIR/.gitflow/MERGE_BASE" echo echo "There were merge conflicts. To resolve the merge conflict manually, use:" echo " git mergetool" echo " git commit" echo echo "You can then complete the finish by running it again:" echo " git flow feature finish $NAME" echo exit 1 fi # when no merge conflict is detected, just clean up the feature branch helper_finish_cleanup } helper_finish_cleanup() { # sanity checks require_branch "$BRANCH" require_clean_working_tree # delete branch if flag fetch; then git_do push "$ORIGIN" ":refs/heads/$BRANCH" fi if noflag keep; then if flag force_delete; then git_do branch -D "$BRANCH" else git_do branch -d "$BRANCH" fi fi echo echo "Summary of actions:" echo "- The feature branch '$BRANCH' was merged into '$DEVELOP_BRANCH'" #echo "- Merge conflicts were resolved" # TODO: Add this line when it's supported if flag keep; then echo "- Feature branch '$BRANCH' is still available" else echo "- Feature branch '$BRANCH' has been removed" fi echo "- You are now on branch '$DEVELOP_BRANCH'" echo } cmd_publish() { parse_args "$@" expand_nameprefix_arg # sanity checks require_clean_working_tree require_branch "$BRANCH" git_do fetch -q "$ORIGIN" require_branch_absent "$ORIGIN/$BRANCH" # create remote branch git_do push "$ORIGIN" "$BRANCH:refs/heads/$BRANCH" git_do fetch -q "$ORIGIN" # configure remote tracking git_do config "branch.$BRANCH.remote" "$ORIGIN" git_do config "branch.$BRANCH.merge" "refs/heads/$BRANCH" git_do checkout "$BRANCH" echo echo "Summary of actions:" echo "- A new remote branch '$BRANCH' was created" echo "- The local branch '$BRANCH' was configured to track the remote branch" echo "- You are now on branch '$BRANCH'" echo } cmd_track() { parse_args "$@" require_name_arg # sanity checks require_clean_working_tree require_branch_absent "$BRANCH" git_do fetch -q "$ORIGIN" require_branch "$ORIGIN/$BRANCH" # create tracking branch git_do checkout -b "$BRANCH" "$ORIGIN/$BRANCH" echo echo "Summary of actions:" echo "- A new remote tracking branch '$BRANCH' was created" echo "- You are now on branch '$BRANCH'" echo } cmd_diff() { parse_args "$@" if [ "$NAME" != "" ]; then expand_nameprefix_arg BASE=$(git merge-base "$DEVELOP_BRANCH" "$BRANCH") git diff "$BASE..$BRANCH" else if ! git_current_branch | grep -q "^$PREFIX"; then die "Not on a feature branch. Name one explicitly." fi BASE=$(git merge-base "$DEVELOP_BRANCH" HEAD) git diff "$BASE" fi } cmd_checkout() { parse_args "$@" if [ "$NAME" != "" ]; then expand_nameprefix_arg git_do checkout "$BRANCH" else die "Name a feature branch explicitly." fi } cmd_co() { # Alias for checkout cmd_checkout "$@" } cmd_rebase() { DEFINE_boolean interactive false 'do an interactive rebase' i parse_args "$@" expand_nameprefix_arg_or_current warn "Will try to rebase '$NAME'..." require_clean_working_tree require_branch "$BRANCH" git_do checkout -q "$BRANCH" local OPTS= if flag interactive; then OPTS="$OPTS -i" fi git_do rebase $OPTS "$DEVELOP_BRANCH" } avoid_accidental_cross_branch_action() { local current_branch=$(git_current_branch) if [ "$BRANCH" != "$current_branch" ]; then warn "Trying to pull from '$BRANCH' while currently on branch '$current_branch'." warn "To avoid unintended merges, git-flow aborted." return 1 fi return 0 } cmd_pull() { #DEFINE_string prefix false 'alternative remote feature branch name prefix' p DEFINE_boolean rebase false "pull with rebase" r parse_remote_name "$@" if [ -z "$REMOTE" ]; then die "Name a remote explicitly." fi name_or_current # To avoid accidentally merging different feature branches into each other, # die if the current feature branch differs from the requested $NAME # argument. local current_branch=$(git_current_branch) if startswith "$current_branch" "$PREFIX"; then # we are on a local feature branch already, so $BRANCH must be equal to # the current branch avoid_accidental_cross_branch_action || die fi require_clean_working_tree if git_branch_exists "$BRANCH"; then # Again, avoid accidental merges avoid_accidental_cross_branch_action || die # we already have a local branch called like this, so simply pull the # remote changes in if flag rebase; then if ! git_do pull --rebase -q "$REMOTE" "$BRANCH"; then warn "Pull was aborted. There might be conflicts during rebase or '$REMOTE' might be inaccessible." exit 1 fi else git_do pull -q "$REMOTE" "$BRANCH" || die "Failed to pull from remote '$REMOTE'." fi echo "Pulled $REMOTE's changes into $BRANCH." else # setup the local branch clone for the first time git_do fetch -q "$REMOTE" "$BRANCH" || die "Fetch failed." # stores in FETCH_HEAD git_do branch --no-track "$BRANCH" FETCH_HEAD || die "Branch failed." git_do checkout -q "$BRANCH" || die "Checking out new local branch failed." echo "Created local branch $BRANCH based on $REMOTE's $BRANCH." fi } ================================================ FILE: git-flow-hotfix ================================================ # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # init() { require_git_repo require_gitflow_initialized gitflow_load_settings VERSION_PREFIX=$(eval "echo `git config --get gitflow.prefix.versiontag`") PREFIX=$(git config --get gitflow.prefix.hotfix) } usage() { echo "usage: git flow hotfix [list] [-v]" echo " git flow hotfix start [-F] []" echo " git flow hotfix finish [-Fsumpk] " echo " git flow hotfix publish " echo " git flow hotfix track " } cmd_default() { cmd_list "$@" } cmd_list() { DEFINE_boolean verbose false 'verbose (more) output' v parse_args "$@" local hotfix_branches local current_branch local short_names hotfix_branches=$(echo "$(git_local_branches)" | grep "^$PREFIX") if [ -z "$hotfix_branches" ]; then warn "No hotfix branches exist." warn "" warn "You can start a new hotfix branch:" warn "" warn " git flow hotfix start []" warn "" exit 0 fi current_branch=$(git branch --no-color | grep '^\* ' | grep -v 'no branch' | sed 's/^* //g') short_names=$(echo "$hotfix_branches" | sed "s ^$PREFIX g") # determine column width first local width=0 local branch for branch in $short_names; do local len=${#branch} width=$(max $width $len) done width=$(($width+3)) local branch for branch in $short_names; do local fullname=$PREFIX$branch local base=$(git merge-base "$fullname" "$MASTER_BRANCH") local master_sha=$(git rev-parse "$MASTER_BRANCH") local branch_sha=$(git rev-parse "$fullname") if [ "$fullname" = "$current_branch" ]; then printf "* " else printf " " fi if flag verbose; then printf "%-${width}s" "$branch" if [ "$branch_sha" = "$master_sha" ]; then printf "(no commits yet)" else local tagname=$(git name-rev --tags --no-undefined --name-only "$base") local nicename if [ "$tagname" != "" ]; then nicename=$tagname else nicename=$(git rev-parse --short "$base") fi printf "(based on $nicename)" fi else printf "%s" "$branch" fi echo done } cmd_help() { usage exit 0 } parse_args() { # parse options FLAGS "$@" || exit $? eval set -- "${FLAGS_ARGV}" # read arguments into global variables VERSION=$1 BRANCH=$PREFIX$VERSION } require_version_arg() { if [ "$VERSION" = "" ]; then warn "Missing argument " usage exit 1 fi } require_base_is_on_master() { if ! git branch --no-color --contains "$BASE" 2>/dev/null \ | sed 's/[* ] //g' \ | grep -q "^$MASTER_BRANCH\$"; then die "fatal: Given base '$BASE' is not a valid commit on '$MASTER_BRANCH'." fi } require_no_existing_hotfix_branches() { local hotfix_branches=$(echo "$(git_local_branches)" | grep "^$PREFIX") local first_branch=$(echo ${hotfix_branches} | head -n1) first_branch=${first_branch#$PREFIX} [ -z "$hotfix_branches" ] || \ die "There is an existing hotfix branch ($first_branch). Finish that one first." } cmd_start() { DEFINE_boolean fetch false "fetch from $ORIGIN before performing finish" F parse_args "$@" BASE=${2:-$MASTER_BRANCH} require_version_arg require_base_is_on_master require_no_existing_hotfix_branches # sanity checks require_clean_working_tree require_branch_absent "$BRANCH" require_tag_absent "$VERSION_PREFIX$VERSION" if flag fetch; then git_do fetch -q "$ORIGIN" "$MASTER_BRANCH" fi if has "$ORIGIN/$MASTER_BRANCH" $(git_remote_branches); then require_branches_equal "$MASTER_BRANCH" "$ORIGIN/$MASTER_BRANCH" fi # create branch git_do checkout -b "$BRANCH" "$BASE" echo echo "Summary of actions:" echo "- A new branch '$BRANCH' was created, based on '$BASE'" echo "- You are now on branch '$BRANCH'" echo echo "Follow-up actions:" echo "- Bump the version number now!" echo "- Start committing your hot fixes" echo "- When done, run:" echo echo " git flow hotfix finish '$VERSION'" echo } cmd_publish() { parse_args "$@" require_version_arg # sanity checks require_clean_working_tree require_branch "$BRANCH" git_do fetch -q "$ORIGIN" require_branch_absent "$ORIGIN/$BRANCH" # create remote branch git_do push "$ORIGIN" "$BRANCH:refs/heads/$BRANCH" git_do fetch -q "$ORIGIN" # configure remote tracking git config "branch.$BRANCH.remote" "$ORIGIN" git config "branch.$BRANCH.merge" "refs/heads/$BRANCH" git_do checkout "$BRANCH" echo echo "Summary of actions:" echo "- A new remote branch '$BRANCH' was created" echo "- The local branch '$BRANCH' was configured to track the remote branch" echo "- You are now on branch '$BRANCH'" echo } cmd_track() { parse_args "$@" require_version_arg # sanity checks require_clean_working_tree require_branch_absent "$BRANCH" git_do fetch -q "$ORIGIN" require_branch "$ORIGIN/$BRANCH" # create tracking branch git_do checkout -b "$BRANCH" "$ORIGIN/$BRANCH" echo echo "Summary of actions:" echo "- A new remote tracking branch '$BRANCH' was created" echo "- You are now on branch '$BRANCH'" echo } cmd_finish() { DEFINE_boolean fetch false "fetch from $ORIGIN before performing finish" F DEFINE_boolean sign false "sign the release tag cryptographically" s DEFINE_string signingkey "" "use the given GPG-key for the digital signature (implies -s)" u DEFINE_string message "" "use the given tag message" m DEFINE_string messagefile "" "use the contents of the given file as tag message" f DEFINE_boolean push false "push to $ORIGIN after performing finish" p DEFINE_boolean keep false "keep branch after performing finish" k DEFINE_boolean notag false "don't tag this release" n parse_args "$@" require_version_arg # handle flags that imply other flags if [ "$FLAGS_signingkey" != "" ]; then FLAGS_sign=$FLAGS_TRUE fi # sanity checks require_branch "$BRANCH" require_clean_working_tree if flag fetch; then git_do fetch -q "$ORIGIN" "$MASTER_BRANCH" || \ die "Could not fetch $MASTER_BRANCH from $ORIGIN." git_do fetch -q "$ORIGIN" "$DEVELOP_BRANCH" || \ die "Could not fetch $DEVELOP_BRANCH from $ORIGIN." fi if has "$ORIGIN/$MASTER_BRANCH" $(git_remote_branches); then require_branches_equal "$MASTER_BRANCH" "$ORIGIN/$MASTER_BRANCH" fi if has "$ORIGIN/$DEVELOP_BRANCH" $(git_remote_branches); then require_branches_equal "$DEVELOP_BRANCH" "$ORIGIN/$DEVELOP_BRANCH" fi # try to merge into master # in case a previous attempt to finish this release branch has failed, # but the merge into master was successful, we skip it now if ! git_is_branch_merged_into "$BRANCH" "$MASTER_BRANCH"; then git_do checkout "$MASTER_BRANCH" || \ die "Could not check out $MASTER_BRANCH." git_do merge --no-ff "$BRANCH" || \ die "There were merge conflicts." # TODO: What do we do now? fi if noflag notag; then # try to tag the release # in case a previous attempt to finish this release branch has failed, # but the tag was set successful, we skip it now local tagname=$VERSION_PREFIX$VERSION if ! git_tag_exists "$tagname"; then local opts="-a" flag sign && opts="$opts -s" [ "$FLAGS_signingkey" != "" ] && opts="$opts -u '$FLAGS_signingkey'" [ "$FLAGS_message" != "" ] && opts="$opts -m '$FLAGS_message'" [ "$FLAGS_messagefile" != "" ] && opts="$opts -F '$FLAGS_messagefile'" eval git_do tag $opts "$VERSION_PREFIX$VERSION" "$BRANCH" || \ die "Tagging failed. Please run finish again to retry." fi fi # try to merge into develop # in case a previous attempt to finish this release branch has failed, # but the merge into develop was successful, we skip it now if ! git_is_branch_merged_into "$BRANCH" "$DEVELOP_BRANCH"; then git_do checkout "$DEVELOP_BRANCH" || \ die "Could not check out $DEVELOP_BRANCH." # TODO: Actually, accounting for 'git describe' pays, so we should # ideally git merge --no-ff $tagname here, instead! git_do merge --no-ff "$BRANCH" || \ die "There were merge conflicts." # TODO: What do we do now? fi # delete branch if noflag keep; then git_do branch -d "$BRANCH" fi if flag push; then git_do push "$ORIGIN" "$DEVELOP_BRANCH" || \ die "Could not push to $DEVELOP_BRANCH from $ORIGIN." git_do push "$ORIGIN" "$MASTER_BRANCH" || \ die "Could not push to $MASTER_BRANCH from $ORIGIN." if noflag notag; then git_do push --tags "$ORIGIN" || \ die "Could not push tags to $ORIGIN." fi fi echo echo "Summary of actions:" echo "- Latest objects have been fetched from '$ORIGIN'" echo "- Hotfix branch has been merged into '$MASTER_BRANCH'" if noflag notag; then echo "- The hotfix was tagged '$VERSION_PREFIX$VERSION'" fi echo "- Hotfix branch has been back-merged into '$DEVELOP_BRANCH'" if flag keep; then echo "- Hotfix branch '$BRANCH' is still available" else echo "- Hotfix branch '$BRANCH' has been deleted" fi if flag push; then echo "- '$DEVELOP_BRANCH', '$MASTER_BRANCH' and tags have been pushed to '$ORIGIN'" fi echo } ================================================ FILE: git-flow-init ================================================ # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # usage() { echo "usage: git flow init [-fd]" } parse_args() { # parse options FLAGS "$@" || exit $? eval set -- "${FLAGS_ARGV}" } # Default entry when no SUBACTION is given cmd_default() { DEFINE_boolean force false 'force setting of gitflow branches, even if already configured' f DEFINE_boolean defaults false 'use default branch naming conventions' d parse_args "$@" if ! git rev-parse --git-dir >/dev/null 2>&1; then git_do init else # assure that we are not working in a repo with local changes git_repo_is_headless || require_clean_working_tree fi # running git flow init on an already initialized repo is fine if gitflow_is_initialized && ! flag force; then warn "Already initialized for gitflow." warn "To force reinitialization, use: git flow init -f" exit 0 fi local branch_count local answer if flag defaults; then warn "Using default branch names." fi # add a master branch if no such branch exists yet local master_branch if gitflow_has_master_configured && ! flag force; then master_branch=$(git config --get gitflow.branch.master) else # Two cases are distinguished: # 1. A fresh git repo (without any branches) # We will create a new master/develop branch for the user # 2. Some branches do already exist # We will disallow creation of new master/develop branches and # rather allow to use existing branches for git-flow. local default_suggestion local should_check_existence branch_count=$(git_local_branches | wc -l) if [ "$branch_count" -eq 0 ]; then echo "No branches exist yet. Base branches must be created now." should_check_existence=NO default_suggestion=$(git config --get gitflow.branch.master || echo master) else echo echo "Which branch should be used for bringing forth production releases?" git_local_branches | sed 's/^.*$/ - &/g' should_check_existence=YES default_suggestion= for guess in $(git config --get gitflow.branch.master) \ 'production' 'main' 'master'; do if git_local_branch_exists "$guess"; then default_suggestion="$guess" break fi done fi printf "Branch name for production releases: [$default_suggestion] " if noflag defaults; then read answer else printf "\n" fi master_branch=${answer:-$default_suggestion} # check existence in case of an already existing repo if [ "$should_check_existence" = "YES" ]; then # if no local branch exists and a remote branch of the same # name exists, checkout that branch and use it for master if ! git_local_branch_exists "$master_branch" && \ git_remote_branch_exists "origin/$master_branch"; then git_do branch "$master_branch" "origin/$master_branch" >/dev/null 2>&1 elif ! git_local_branch_exists "$master_branch"; then die "Local branch '$master_branch' does not exist." fi fi # store the name of the master branch git_do config gitflow.branch.master "$master_branch" fi # add a develop branch if no such branch exists yet local develop_branch if gitflow_has_develop_configured && ! flag force; then develop_branch=$(git config --get gitflow.branch.develop) else # Again, the same two cases as with the master selection are # considered (fresh repo or repo that contains branches) local default_suggestion local should_check_existence branch_count=$(git_local_branches | grep -v "^${master_branch}\$" | wc -l) if [ "$branch_count" -eq 0 ]; then should_check_existence=NO default_suggestion=$(git config --get gitflow.branch.develop || echo develop) else echo echo "Which branch should be used for integration of the \"next release\"?" git_local_branches | grep -v "^${master_branch}\$" | sed 's/^.*$/ - &/g' should_check_existence=YES default_suggestion= for guess in $(git config --get gitflow.branch.develop) \ 'develop' 'int' 'integration' 'master'; do if git_local_branch_exists "$guess" && [ "$guess" != "$master_branch" ]; then default_suggestion="$guess" break fi done if [ -z $default_suggestion ]; then should_check_existence=NO default_suggestion=$(git config --get gitflow.branch.develop || echo develop) fi fi printf "Branch name for \"next release\" development: [$default_suggestion] " if noflag defaults; then read answer else printf "\n" fi develop_branch=${answer:-$default_suggestion} if [ "$master_branch" = "$develop_branch" ]; then die "Production and integration branches should differ." fi # check existence in case of an already existing repo if [ "$should_check_existence" = "YES" ]; then git_local_branch_exists "$develop_branch" || \ die "Local branch '$develop_branch' does not exist." fi # store the name of the develop branch git_do config gitflow.branch.develop "$develop_branch" fi # Creation of HEAD # ---------------- # We create a HEAD now, if it does not exist yet (in a fresh repo). We need # it to be able to create new branches. local created_gitflow_branch=0 if ! git rev-parse --quiet --verify HEAD >/dev/null 2>&1; then git_do symbolic-ref HEAD "refs/heads/$master_branch" git_do commit --allow-empty --quiet -m "Initial commit" created_gitflow_branch=1 fi # Creation of master # ------------------ # At this point, there always is a master branch: either it existed already # (and was picked interactively as the production branch) or it has just # been created in a fresh repo # Creation of develop # ------------------- # The develop branch possibly does not exist yet. This is the case when, # in a git init'ed repo with one or more commits, master was picked as the # default production branch and develop was "created". We should create # the develop branch now in that case (we base it on master, of course) if ! git_local_branch_exists "$develop_branch"; then if git_remote_branch_exists "origin/$develop_branch"; then git_do branch "$develop_branch" "origin/$develop_branch" >/dev/null 2>&1 else git_do branch --no-track "$develop_branch" "$master_branch" fi created_gitflow_branch=1 fi # assert the gitflow repo has been correctly initialized gitflow_is_initialized # switch to develop branch if its newly created if [ $created_gitflow_branch -eq 1 ]; then git_do checkout -q "$develop_branch" fi # finally, ask the user for naming conventions (branch and tag prefixes) if flag force || \ ! git config --get gitflow.prefix.feature >/dev/null 2>&1 || ! git config --get gitflow.prefix.release >/dev/null 2>&1 || ! git config --get gitflow.prefix.hotfix >/dev/null 2>&1 || ! git config --get gitflow.prefix.support >/dev/null 2>&1 || ! git config --get gitflow.prefix.versiontag >/dev/null 2>&1; then echo echo "How to name your supporting branch prefixes?" fi local prefix # Feature branches if ! git config --get gitflow.prefix.feature >/dev/null 2>&1 || flag force; then default_suggestion=$(git config --get gitflow.prefix.feature || echo feature/) printf "Feature branches? [$default_suggestion] " if noflag defaults; then read answer else printf "\n" fi [ "$answer" = "-" ] && prefix= || prefix=${answer:-$default_suggestion} git_do config gitflow.prefix.feature "$prefix" fi # Release branches if ! git config --get gitflow.prefix.release >/dev/null 2>&1 || flag force; then default_suggestion=$(git config --get gitflow.prefix.release || echo release/) printf "Release branches? [$default_suggestion] " if noflag defaults; then read answer else printf "\n" fi [ "$answer" = "-" ] && prefix= || prefix=${answer:-$default_suggestion} git_do config gitflow.prefix.release "$prefix" fi # Hotfix branches if ! git config --get gitflow.prefix.hotfix >/dev/null 2>&1 || flag force; then default_suggestion=$(git config --get gitflow.prefix.hotfix || echo hotfix/) printf "Hotfix branches? [$default_suggestion] " if noflag defaults; then read answer else printf "\n" fi [ "$answer" = "-" ] && prefix= || prefix=${answer:-$default_suggestion} git_do config gitflow.prefix.hotfix "$prefix" fi # Support branches if ! git config --get gitflow.prefix.support >/dev/null 2>&1 || flag force; then default_suggestion=$(git config --get gitflow.prefix.support || echo support/) printf "Support branches? [$default_suggestion] " if noflag defaults; then read answer else printf "\n" fi [ "$answer" = "-" ] && prefix= || prefix=${answer:-$default_suggestion} git_do config gitflow.prefix.support "$prefix" fi # Version tag prefix if ! git config --get gitflow.prefix.versiontag >/dev/null 2>&1 || flag force; then default_suggestion=$(git config --get gitflow.prefix.versiontag || echo "") printf "Version tag prefix? [$default_suggestion] " if noflag defaults; then read answer else printf "\n" fi [ "$answer" = "-" ] && prefix= || prefix=${answer:-$default_suggestion} git_do config gitflow.prefix.versiontag "$prefix" fi # TODO: what to do with origin? } cmd_help() { usage exit 0 } ================================================ FILE: git-flow-release ================================================ # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # init() { require_git_repo require_gitflow_initialized gitflow_load_settings VERSION_PREFIX=$(eval "echo `git config --get gitflow.prefix.versiontag`") PREFIX=$(git config --get gitflow.prefix.release) } usage() { echo "usage: git flow release [list] [-v]" echo " git flow release start [-F] []" echo " git flow release finish [-FsumpkS] " echo " git flow release publish " echo " git flow release track " } cmd_default() { cmd_list "$@" } cmd_list() { DEFINE_boolean verbose false 'verbose (more) output' v parse_args "$@" local release_branches local current_branch local short_names release_branches=$(echo "$(git_local_branches)" | grep "^$PREFIX") if [ -z "$release_branches" ]; then warn "No release branches exist." warn "" warn "You can start a new release branch:" warn "" warn " git flow release start []" warn "" exit 0 fi current_branch=$(git branch --no-color | grep '^\* ' | grep -v 'no branch' | sed 's/^* //g') short_names=$(echo "$release_branches" | sed "s ^$PREFIX g") # determine column width first local width=0 local branch for branch in $short_names; do local len=${#branch} width=$(max $width $len) done width=$(($width+3)) local branch for branch in $short_names; do local fullname=$PREFIX$branch local base=$(git merge-base "$fullname" "$DEVELOP_BRANCH") local develop_sha=$(git rev-parse "$DEVELOP_BRANCH") local branch_sha=$(git rev-parse "$fullname") if [ "$fullname" = "$current_branch" ]; then printf "* " else printf " " fi if flag verbose; then printf "%-${width}s" "$branch" if [ "$branch_sha" = "$develop_sha" ]; then printf "(no commits yet)" else local nicename=$(git rev-parse --short "$base") printf "(based on $nicename)" fi else printf "%s" "$branch" fi echo done } cmd_help() { usage exit 0 } parse_args() { # parse options FLAGS "$@" || exit $? eval set -- "${FLAGS_ARGV}" # read arguments into global variables VERSION=$1 BRANCH=$PREFIX$VERSION } require_version_arg() { if [ "$VERSION" = "" ]; then warn "Missing argument " usage exit 1 fi } require_base_is_on_develop() { if ! git_do branch --no-color --contains "$BASE" 2>/dev/null \ | sed 's/[* ] //g' \ | grep -q "^$DEVELOP_BRANCH\$"; then die "fatal: Given base '$BASE' is not a valid commit on '$DEVELOP_BRANCH'." fi } require_no_existing_release_branches() { local release_branches=$(echo "$(git_local_branches)" | grep "^$PREFIX") local first_branch=$(echo ${release_branches} | head -n1) first_branch=${first_branch#$PREFIX} [ -z "$release_branches" ] || \ die "There is an existing release branch ($first_branch). Finish that one first." } cmd_start() { DEFINE_boolean fetch false "fetch from $ORIGIN before performing finish" F parse_args "$@" BASE=${2:-$DEVELOP_BRANCH} require_version_arg require_base_is_on_develop require_no_existing_release_branches # sanity checks require_clean_working_tree require_branch_absent "$BRANCH" require_tag_absent "$VERSION_PREFIX$VERSION" if flag fetch; then git_do fetch -q "$ORIGIN" "$DEVELOP_BRANCH" fi if has "$ORIGIN/$DEVELOP_BRANCH" $(git_remote_branches); then require_branches_equal "$DEVELOP_BRANCH" "$ORIGIN/$DEVELOP_BRANCH" fi # create branch git_do checkout -b "$BRANCH" "$BASE" echo echo "Summary of actions:" echo "- A new branch '$BRANCH' was created, based on '$BASE'" echo "- You are now on branch '$BRANCH'" echo echo "Follow-up actions:" echo "- Bump the version number now!" echo "- Start committing last-minute fixes in preparing your release" echo "- When done, run:" echo echo " git flow release finish '$VERSION'" echo } cmd_finish() { DEFINE_boolean fetch false "fetch from $ORIGIN before performing finish" F DEFINE_boolean sign false "sign the release tag cryptographically" s DEFINE_string signingkey "" "use the given GPG-key for the digital signature (implies -s)" u DEFINE_string message "" "use the given tag message" m DEFINE_string messagefile "" "use the contents of the given file as a tag message" f DEFINE_boolean push false "push to $ORIGIN after performing finish" p DEFINE_boolean keep false "keep branch after performing finish" k DEFINE_boolean notag false "don't tag this release" n DEFINE_boolean squash false "squash release during merge" S parse_args "$@" require_version_arg # handle flags that imply other flags if [ "$FLAGS_signingkey" != "" ]; then FLAGS_sign=$FLAGS_TRUE fi # sanity checks require_branch "$BRANCH" require_clean_working_tree if flag fetch; then git_do fetch -q "$ORIGIN" "$MASTER_BRANCH" || \ die "Could not fetch $MASTER_BRANCH from $ORIGIN." git_do fetch -q "$ORIGIN" "$DEVELOP_BRANCH" || \ die "Could not fetch $DEVELOP_BRANCH from $ORIGIN." fi if has "$ORIGIN/$MASTER_BRANCH" $(git_remote_branches); then require_branches_equal "$MASTER_BRANCH" "$ORIGIN/$MASTER_BRANCH" fi if has "$ORIGIN/$DEVELOP_BRANCH" $(git_remote_branches); then require_branches_equal "$DEVELOP_BRANCH" "$ORIGIN/$DEVELOP_BRANCH" fi # try to merge into master # in case a previous attempt to finish this release branch has failed, # but the merge into master was successful, we skip it now if ! git_is_branch_merged_into "$BRANCH" "$MASTER_BRANCH"; then git_do checkout "$MASTER_BRANCH" || \ die "Could not check out $MASTER_BRANCH." if noflag squash; then git_do merge --no-ff "$BRANCH" || \ die "There were merge conflicts." # TODO: What do we do now? else git_do merge --squash "$BRANCH" || \ die "There were merge conflicts." git_do commit fi fi if noflag notag; then # try to tag the release # in case a previous attempt to finish this release branch has failed, # but the tag was set successful, we skip it now local tagname=$VERSION_PREFIX$VERSION if ! git_tag_exists "$tagname"; then local opts="-a" flag sign && opts="$opts -s" [ "$FLAGS_signingkey" != "" ] && opts="$opts -u '$FLAGS_signingkey'" [ "$FLAGS_message" != "" ] && opts="$opts -m '$FLAGS_message'" [ "$FLAGS_messagefile" != "" ] && opts="$opts -F '$FLAGS_messagefile'" eval git_do tag $opts "$tagname" "$BRANCH" || \ die "Tagging failed. Please run finish again to retry." fi fi # try to merge into develop # in case a previous attempt to finish this release branch has failed, # but the merge into develop was successful, we skip it now if ! git_is_branch_merged_into "$BRANCH" "$DEVELOP_BRANCH"; then git_do checkout "$DEVELOP_BRANCH" || \ die "Could not check out $DEVELOP_BRANCH." # TODO: Actually, accounting for 'git describe' pays, so we should # ideally git merge --no-ff $tagname here, instead! if noflag squash; then git_do merge --no-ff "$BRANCH" || \ die "There were merge conflicts." # TODO: What do we do now? else git_do merge --squash "$BRANCH" || \ die "There were merge conflicts." # TODO: What do we do now? git_do commit fi fi # delete branch if noflag keep; then if [ "$BRANCH" = "$(git_current_branch)" ]; then git_do checkout "$MASTER_BRANCH" fi git_do branch -d "$BRANCH" fi if flag push; then git_do push "$ORIGIN" "$DEVELOP_BRANCH" || \ die "Could not push to $DEVELOP_BRANCH from $ORIGIN." git_do push "$ORIGIN" "$MASTER_BRANCH" || \ die "Could not push to $MASTER_BRANCH from $ORIGIN." if noflag notag; then git_do push --tags "$ORIGIN" || \ die "Could not push tags to $ORIGIN." fi git_do push "$ORIGIN" :"$BRANCH" || \ die "Could not delete the remote $BRANCH in $ORIGIN." fi echo echo "Summary of actions:" echo "- Latest objects have been fetched from '$ORIGIN'" echo "- Release branch has been merged into '$MASTER_BRANCH'" if noflag notag; then echo "- The release was tagged '$tagname'" fi echo "- Release branch has been back-merged into '$DEVELOP_BRANCH'" if flag keep; then echo "- Release branch '$BRANCH' is still available" else echo "- Release branch '$BRANCH' has been deleted" fi if flag push; then echo "- '$DEVELOP_BRANCH', '$MASTER_BRANCH' and tags have been pushed to '$ORIGIN'" echo "- Release branch '$BRANCH' in '$ORIGIN' has been deleted." fi echo } cmd_publish() { parse_args "$@" require_version_arg # sanity checks require_clean_working_tree require_branch "$BRANCH" git_do fetch -q "$ORIGIN" require_branch_absent "$ORIGIN/$BRANCH" # create remote branch git_do push "$ORIGIN" "$BRANCH:refs/heads/$BRANCH" git_do fetch -q "$ORIGIN" # configure remote tracking git_do config "branch.$BRANCH.remote" "$ORIGIN" git_do config "branch.$BRANCH.merge" "refs/heads/$BRANCH" git_do checkout "$BRANCH" echo echo "Summary of actions:" echo "- A new remote branch '$BRANCH' was created" echo "- The local branch '$BRANCH' was configured to track the remote branch" echo "- You are now on branch '$BRANCH'" echo } cmd_track() { parse_args "$@" require_version_arg # sanity checks require_clean_working_tree require_branch_absent "$BRANCH" git_do fetch -q "$ORIGIN" require_branch "$ORIGIN/$BRANCH" # create tracking branch git_do checkout -b "$BRANCH" "$ORIGIN/$BRANCH" echo echo "Summary of actions:" echo "- A new remote tracking branch '$BRANCH' was created" echo "- You are now on branch '$BRANCH'" echo } ================================================ FILE: git-flow-support ================================================ # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # init() { require_git_repo require_gitflow_initialized gitflow_load_settings VERSION_PREFIX=$(eval "echo `git config --get gitflow.prefix.versiontag`") PREFIX=$(git config --get gitflow.prefix.support) } warn "note: The support subcommand is still very EXPERIMENTAL!" warn "note: DO NOT use it in a production situation." usage() { echo "usage: git flow support [list] [-v]" echo " git flow support start [-F] " } cmd_default() { cmd_list "$@" } cmd_list() { DEFINE_boolean verbose false 'verbose (more) output' v parse_args "$@" local support_branches local current_branch local short_names support_branches=$(echo "$(git_local_branches)" | grep "^$PREFIX") if [ -z "$support_branches" ]; then warn "No support branches exist." warn "" warn "You can start a new support branch:" warn "" warn " git flow support start " warn "" exit 0 fi current_branch=$(git branch --no-color | grep '^\* ' | grep -v 'no branch' | sed 's/^* //g') short_names=$(echo "$support_branches" | sed "s ^$PREFIX g") # determine column width first local width=0 local branch for branch in $short_names; do local len=${#branch} width=$(max $width $len) done width=$(($width+3)) local branch for branch in $short_names; do local fullname=$PREFIX$branch local base=$(git merge-base "$fullname" "$MASTER_BRANCH") local master_sha=$(git rev-parse "$MASTER_BRANCH") local branch_sha=$(git rev-parse "$fullname") if [ "$fullname" = "$current_branch" ]; then printf "* " else printf " " fi if flag verbose; then printf "%-${width}s" "$branch" if [ "$branch_sha" = "$master_sha" ]; then printf "(no commits yet)" else local tagname=$(git name-rev --tags --no-undefined --name-only "$base") local nicename if [ "$tagname" != "" ]; then nicename=$tagname else nicename=$(git rev-parse --short "$base") fi printf "(based on $nicename)" fi else printf "%s" "$branch" fi echo done } cmd_help() { usage exit 0 } parse_args() { # parse options FLAGS "$@" || exit $? eval set -- "${FLAGS_ARGV}" # read arguments into global variables VERSION=$1 BASE=$2 BRANCH=$PREFIX$VERSION } require_version_arg() { if [ "$VERSION" = "" ]; then warn "Missing argument " usage exit 1 fi } require_base_arg() { if [ "$BASE" = "" ]; then warn "Missing argument " usage exit 1 fi } require_base_is_on_master() { if ! git branch --no-color --contains "$BASE" 2>/dev/null \ | sed 's/[* ] //g' \ | grep -q "^$MASTER_BRANCH\$"; then die "fatal: Given base '$BASE' is not a valid commit on '$MASTER_BRANCH'." fi } cmd_start() { DEFINE_boolean fetch false "fetch from $ORIGIN before performing finish" F parse_args "$@" require_version_arg require_base_arg require_base_is_on_master # sanity checks require_clean_working_tree # fetch remote changes if flag fetch; then git_do fetch -q "$ORIGIN" "$BASE" fi require_branch_absent "$BRANCH" # create branch git_do checkout -b "$BRANCH" "$BASE" echo echo "Summary of actions:" echo "- A new branch '$BRANCH' was created, based on '$BASE'" echo "- You are now on branch '$BRANCH'" echo } ================================================ FILE: git-flow-version ================================================ # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # GITFLOW_VERSION=0.4.2-pre usage() { echo "usage: git flow version" } cmd_default() { echo "$GITFLOW_VERSION" } cmd_help() { usage exit 0 } ================================================ FILE: gitflow-common ================================================ # # git-flow -- A collection of Git extensions to provide high-level # repository operations for Vincent Driessen's branching model. # # Original blog post presenting this model is found at: # http://nvie.com/git-model # # Feel free to contribute to this project at: # http://github.com/nvie/gitflow # # Copyright 2010 Vincent Driessen. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # 1. Redistributions of source code must retain the above copyright notice, # this list of conditions and the following disclaimer. # # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY VINCENT DRIESSEN ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO # EVENT SHALL VINCENT DRIESSEN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, # INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, # EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # # The views and conclusions contained in the software and documentation are # those of the authors and should not be interpreted as representing official # policies, either expressed or implied, of Vincent Driessen. # # # Common functionality # # shell output warn() { echo "$@" >&2; } die() { warn "$@"; exit 1; } escape() { echo "$1" | sed 's/\([\.\$\*]\)/\\\1/g' } # set logic has() { local item=$1; shift echo " $@ " | grep -q " $(escape $item) " } # basic math min() { [ "$1" -le "$2" ] && echo "$1" || echo "$2"; } max() { [ "$1" -ge "$2" ] && echo "$1" || echo "$2"; } # basic string matching startswith() { [ "$1" != "${1#$2}" ]; } endswith() { [ "$1" != "${1%$2}" ]; } # convenience functions for checking shFlags flags flag() { local FLAG; eval FLAG='$FLAGS_'$1; [ $FLAG -eq $FLAGS_TRUE ]; } noflag() { local FLAG; eval FLAG='$FLAGS_'$1; [ $FLAG -ne $FLAGS_TRUE ]; } # # Git specific common functionality # git_do() { # equivalent to git, used to indicate actions that make modifications if flag show_commands; then echo "git $@" >&2 fi git "$@" } git_local_branches() { git branch --no-color | sed 's/^[* ] //'; } git_remote_branches() { git branch -r --no-color | sed 's/^[* ] //'; } git_all_branches() { ( git branch --no-color; git branch -r --no-color) | sed 's/^[* ] //'; } git_all_tags() { git tag; } git_current_branch() { git branch --no-color | grep '^\* ' | grep -v 'no branch' | sed 's/^* //g' } git_is_clean_working_tree() { if ! git diff --no-ext-diff --ignore-submodules --quiet --exit-code; then return 1 elif ! git diff-index --cached --quiet --ignore-submodules HEAD --; then return 2 else return 0 fi } git_repo_is_headless() { ! git rev-parse --quiet --verify HEAD >/dev/null 2>&1 } git_local_branch_exists() { has $1 $(git_local_branches) } git_remote_branch_exists() { has $1 $(git_remote_branches) } git_branch_exists() { has $1 $(git_all_branches) } git_tag_exists() { has $1 $(git_all_tags) } # # git_compare_branches() # # Tests whether branches and their "origin" counterparts have diverged and need # merging first. It returns error codes to provide more detail, like so: # # 0 Branch heads point to the same commit # 1 First given branch needs fast-forwarding # 2 Second given branch needs fast-forwarding # 3 Branch needs a real merge # 4 There is no merge base, i.e. the branches have no common ancestors # git_compare_branches() { local commit1=$(git rev-parse "$1") local commit2=$(git rev-parse "$2") if [ "$commit1" != "$commit2" ]; then local base=$(git merge-base "$commit1" "$commit2") if [ $? -ne 0 ]; then return 4 elif [ "$commit1" = "$base" ]; then return 1 elif [ "$commit2" = "$base" ]; then return 2 else return 3 fi else return 0 fi } # # git_is_branch_merged_into() # # Checks whether branch $1 is succesfully merged into $2 # git_is_branch_merged_into() { local subject=$1 local base=$2 local all_merges="$(git branch --no-color --contains $subject | sed 's/^[* ] //')" has $base $all_merges } # # gitflow specific common functionality # # check if this repo has been inited for gitflow gitflow_has_master_configured() { local master=$(git config --get gitflow.branch.master) [ "$master" != "" ] && git_local_branch_exists "$master" } gitflow_has_develop_configured() { local develop=$(git config --get gitflow.branch.develop) [ "$develop" != "" ] && git_local_branch_exists "$develop" } gitflow_has_prefixes_configured() { git config --get gitflow.prefix.feature >/dev/null 2>&1 && \ git config --get gitflow.prefix.release >/dev/null 2>&1 && \ git config --get gitflow.prefix.hotfix >/dev/null 2>&1 && \ git config --get gitflow.prefix.support >/dev/null 2>&1 && \ git config --get gitflow.prefix.versiontag >/dev/null 2>&1 } gitflow_is_initialized() { gitflow_has_master_configured && \ gitflow_has_develop_configured && \ [ "$(git config --get gitflow.branch.master)" != \ "$(git config --get gitflow.branch.develop)" ] && \ gitflow_has_prefixes_configured } # loading settings that can be overridden using git config gitflow_load_settings() { export DOT_GIT_DIR=$(git rev-parse --git-dir 2>/dev/null) export MASTER_BRANCH=$(git config --get gitflow.branch.master) export DEVELOP_BRANCH=$(git config --get gitflow.branch.develop) export ORIGIN=$(git config --get gitflow.origin || echo origin) } # # gitflow_resolve_nameprefix # # Inputs: # $1 = name prefix to resolve # $2 = branch prefix to use # # Searches branch names from git_local_branches() to look for a unique # branch name whose name starts with the given name prefix. # # There are multiple exit codes possible: # 0: The unambiguous full name of the branch is written to stdout # (success) # 1: No match is found. # 2: Multiple matches found. These matches are written to stderr # gitflow_resolve_nameprefix() { local name=$1 local prefix=$2 local matches local num_matches # first, check if there is a perfect match if git_local_branch_exists "$prefix$name"; then echo "$name" return 0 fi matches=$(echo "$(git_local_branches)" | grep "^$(escape "$prefix$name")") num_matches=$(echo "$matches" | wc -l) if [ -z "$matches" ]; then # no prefix match, so take it literally warn "No branch matches prefix '$name'" return 1 else if [ $num_matches -eq 1 ]; then echo "${matches#$prefix}" return 0 else # multiple matches, cannot decide warn "Multiple branches match prefix '$name':" for match in $matches; do warn "- $match" done return 2 fi fi } # # Assertions for use in git-flow subcommands # require_git_repo() { if ! git rev-parse --git-dir >/dev/null 2>&1; then die "fatal: Not a git repository" fi } require_gitflow_initialized() { if ! gitflow_is_initialized; then die "fatal: Not a gitflow-enabled repo yet. Please run \"git flow init\" first." fi } require_clean_working_tree() { git_is_clean_working_tree local result=$? if [ $result -eq 1 ]; then die "fatal: Working tree contains unstaged changes. Aborting." fi if [ $result -eq 2 ]; then die "fatal: Index contains uncommited changes. Aborting." fi } require_local_branch() { if ! git_local_branch_exists $1; then die "fatal: Local branch '$1' does not exist and is required." fi } require_remote_branch() { if ! has $1 $(git_remote_branches); then die "Remote branch '$1' does not exist and is required." fi } require_branch() { if ! has $1 $(git_all_branches); then die "Branch '$1' does not exist and is required." fi } require_branch_absent() { if has $1 $(git_all_branches); then die "Branch '$1' already exists. Pick another name." fi } require_tag_absent() { for tag in $(git_all_tags); do if [ "$1" = "$tag" ]; then die "Tag '$1' already exists. Pick another name." fi done } require_branches_equal() { require_local_branch "$1" require_remote_branch "$2" git_compare_branches "$1" "$2" local status=$? if [ $status -gt 0 ]; then warn "Branches '$1' and '$2' have diverged." if [ $status -eq 1 ]; then die "And branch '$1' may be fast-forwarded." elif [ $status -eq 2 ]; then # Warn here, since there is no harm in being ahead warn "And local branch '$1' is ahead of '$2'." else die "Branches need merging first." fi fi }