web-dev-qa-db-ja.com

ウェブサイトのステージングのためのGitの受信後フック

ウェブサイトのステージング用にGitをセットアップして、git pullは、現在のバージョンをローカルで処理し、git Push to変更をリモートサーバーにプッシュします。希望どおりに機能するように設定しましたが、プッシュした後、手動でgit checkout -fまたはgit reset --hard HEADリモートサーバー上。

これらをサーバーの受信後フックとしてシェルスクリプトに入れてみましたが、効果がないようです。プッシュした後に「サーバーにプッシュされた変更」が表示されるため、スクリプトが実行されていることを知っています。受信後フックは次のとおりです。

#!/bin/sh
git reset --hard HEAD
echo "Changes pushed to server."
59
Matt

あなたの質問に対する答えはこちらです: http://toroid.org/ams/git-website-howto

要するに、やりたいことは、「切り離された作業ツリー」を裸のリポジトリに追加することです。通常、作業ツリーは.gitディレクトリを含むと考えます。ベアリポジトリには定義上ワークツリーがありませんが、ベアリポジトリとは異なるディレクトリにある限り作成できます。

Post-receiveフックは、リポジトリのHEADを作業ディレクトリに複製するための単純なgit checkout -fです。 Apacheはそれをドキュメントルートとして使用し、これで設定は完了です。ベアリポジトリにプッシュするたびに、Apacheはすぐにそれを提供し始めます。

通常、これを使用して、ステージングサーバーに自動的にプッシュし、「実際の」環境が私の​​変更を刺激するかどうかを確認します。ライブサーバーへの展開はまったく別の話です。 :-)

73
Paul

2015年3月に更新

変更をリモートリポジトリにプッシュするときのこのGit警告メッセージは何ですか? 」で述べたように、実際には直接を非ベアリポジトリにプッシュできます(Git 2.3 .0 +、2015年2月)と:

git config receive.denyCurrentBranch updateInstead

それに応じて作業ツリーを更新しますが、コミットされていない変更がある場合は更新を拒否します。

これにより、受信後のフックを回避できます。


(元の回答:2010年10月)

GitFAQnon-bare repo のこの更新後フックを推奨します:
(フックの実行で実際に何が起こっているのかを知る手がかりになるかもしれません。これはポスト受信ではなく、アップデート後のフックであることに注意してください)

#!/bin/sh
#
# This hook does two things:
#
#  1. update the "info" files that allow the list of references to be
#     queries over dumb transports such as http
#
#  2. if this repository looks like it is a non-bare repository, and
#     the checked-out branch is pushed to, then update the working copy.
#     This makes "Push" function somewhat similarly to darcs and bzr.
#
# To enable this hook, make this file executable by "chmod +x post-update".

git-update-server-info

is_bare=$(git-config --get --bool core.bare)

if [ -z "$is_bare" ]
then
    # for compatibility's sake, guess
    git_dir_full=$(cd $GIT_DIR; pwd)
    case $git_dir_full in */.git) is_bare=false;; *) is_bare=true;; esac
fi

update_wc() {
    ref=$1
    echo "Push to checked out branch $ref" >&2
    if [ ! -f $GIT_DIR/logs/HEAD ]
    then
        echo "E:Push to non-bare repository requires a HEAD reflog" >&2
        exit 1
    fi
    if (cd $GIT_WORK_TREE; git-diff-files -q --exit-code >/dev/null)
    then
        wc_dirty=0
    else
        echo "W:unstaged changes found in working copy" >&2
        wc_dirty=1
        desc="working copy"
    fi
    if git diff-index --cached HEAD@{1} >/dev/null
    then
        index_dirty=0
    else
        echo "W:uncommitted, staged changes found" >&2
        index_dirty=1
        if [ -n "$desc" ]
        then
            desc="$desc and index"
        else
            desc="index"
        fi
    fi
    if [ "$wc_dirty" -ne 0 -o "$index_dirty" -ne 0 ]
    then
        new=$(git rev-parse HEAD)
        echo "W:stashing dirty $desc - see git-stash(1)" >&2
        ( trap 'echo trapped $$; git symbolic-ref HEAD "'"$ref"'"' 2 3 13 15 ERR EXIT
        git-update-ref --no-deref HEAD HEAD@{1}
        cd $GIT_WORK_TREE
        git stash save "dirty $desc before update to $new";
        git-symbolic-ref HEAD "$ref"
        )
    fi

    # eye candy - show the WC updates :)
    echo "Updating working copy" >&2
    (cd $GIT_WORK_TREE
    git-diff-index -R --name-status HEAD >&2
    git-reset --hard HEAD)
}

if [ "$is_bare" = "false" ]
then
    active_branch=`git-symbolic-ref HEAD`
    export GIT_DIR=$(cd $GIT_DIR; pwd)
    GIT_WORK_TREE=${GIT_WORK_TREE-..}
    for ref
    do
        if [ "$ref" = "$active_branch" ]
        then
            update_wc $ref
        fi
    done
fi

これが機能するためには、次の構成設定のいずれかを使用して、現在のブランチに変更をプッシュすることを明確に許可する必要があります。

git config receive.denyCurrentBranch ignore

または

git config receive.denyCurrentBranch warn
15
VonC

私はまったく同じ問題を抱えていました。このリンクへの返信: http://toroid.org/ams/git-website-howto -次のコマンドで実行しました:

Sudo chmod +x hooks/post-receive

最初にSudoパーミッションが設定されていませんでした。

11
ido

VonCのスクリプトの修正バージョン、私のために動作します(絶対に保証なし)。

#!/bin/sh
#
# This hook does two things:
#
#  1. update the "info" files that allow the list of references to be
#     queries over dumb transports such as http
#
#  2. if this repository looks like it is a non-bare repository, and
#     the checked-out branch is pushed to, then update the working copy.
#     This makes "Push" function somewhat similarly to darcs and bzr.
#
# To enable this hook, make this file executable by "chmod +x post-update".

set -e

git update-server-info

is_bare=$(git config --get --bool core.bare)

if [ -z "${is_bare}" ]
then
    # for compatibility's sake, guess
    git_dir_full=$(cd $GIT_DIR; pwd)
    case $git_dir_full in */.git) is_bare=false;; *) is_bare=true;; esac
fi

update_wc() {
    ref=$1
    echo "Push to checked out branch $ref" >&2
    if [ ! -f ${GIT_DIR}/logs/HEAD ]
    then
        echo "E:Push to non-bare repository requires a HEAD reflog" >&2
        exit 1
    fi
    if (cd ${GIT_WORK_TREE}; git diff-files -q --exit-code >/dev/null)
    then
        wc_dirty=0
    else
        echo "W:unstaged changes found in working copy" >&2
        wc_dirty=1
        desc="working copy"
    fi
    if git diff-index --cached HEAD@{1} >/dev/null
    then
        index_dirty=0
    else
        echo "W:uncommitted, staged changes found" >&2
        index_dirty=1
        if [ -n "$desc" ]
        then
            desc="$desc and index"
        else
            desc="index"
        fi
    fi
    if [ "$wc_dirty" -ne 0 -o "$index_dirty" -ne 0 ]
    then
        new=$(git rev-parse HEAD)
        echo "W:stashing dirty $desc - see git-stash(1)" >&2
        ( trap 'echo trapped $$; git symbolic-ref HEAD "'"$ref"'"' 2 3 13 15 ERR EXIT
        git update-ref --no-deref HEAD HEAD@{1}
        cd ${GIT_WORK_TREE}
        git stash save "dirty $desc before update to $new";
        git symbolic-ref HEAD "$ref"
        )
    fi

    # eye candy - show the WC updates :)
    echo "Updating working copy" >&2
    (cd ${GIT_WORK_TREE}
    git diff-index -R --name-status HEAD >&2
    git reset --hard HEAD
    # need to touch some files or restart the application? do that here:
    # touch *.wsgi
    )

}

if [ x"${is_bare}" = x"false" ]
then
    active_branch=$(git symbolic-ref HEAD)
    export GIT_DIR=$(cd ${GIT_DIR}; pwd)
    GIT_WORK_TREE="${GIT_DIR}/.."
    for ref in $(cat)
    do
        if [ x"$ref" = x"${active_branch}" ]
        then
            update_wc $ref
        fi
    done
fi
6
Tronic

このgitデプロイメントを設定するためのシンプルなスクリプト:

受信後フックの準備:

echo '#!/bin/sh'        >  .git/hooks/post-receive
echo 'git checkout -f'  >> .git/hooks/post-receive
echo 'git reset --hard' >> .git/hooks/post-receive
chmod +x .git/hooks/post-receive

このリポジトリへのプッシュを許可しますが、それは裸ではありません:

git config receive.denycurrentbranch false
6
Honza

私は推測しているだけですが、これは許可の問題かもしれません(フルパスが必要ですか?cd?)。ログファイルで実際に何が起こっているかを確認します。

ただし、gitを介してファイルを公開することは、常に公開プロセスの1つのタスクにすぎません。通常、いくつかのファイルのコピー、他のファイルの削除、セットアップ、権限の更新、ドキュメントの生成などが必要です。

複雑なソリューションの場合、ビルドスクリプトはどのgitフックよりも優れている場合があります。これらのタスクを非常にうまく処理できるツール:

(これはあなたが期待している答えではないことを理解していますが、コメントとして投稿するには長すぎます)

1
takeshin