95
99

More than 5 years have passed since last update.

git-svnを利用した運用を考える

Posted at

最近、諸事情あってsubversionからgitへプロジェクトの半ばでバージョン管理を変更しました。その際にちょっと変わった形でgit-svnの運用を行う事になったのでやりかたに関して簡単にログを残しておきます。

プロジェクトを取り巻く事情

今まではSubversion(仮にメインとします)を用いた運用を行っていたのですが、気軽にtrunkに対してコミットする事が出来ない事情がありました。では、メインSubversion上にbranchをたてればいいじゃんという話になるのですが、ブランチすら気軽に切る事が出来ない状態でした。その上、メインのSubversionにコミットすることが可能なのはコミッターだけとなっておりコミッターの負担が増えつつありました。

今までの運用

今まではメインSubversionとは別に 同期しないSubversion(仮にローカルとします)を別途立てて、なんと手動でメイン<=>ローカル間の同期を取りながら運用を行ってきました。
Untitled.png
※破線の部分はVCSによって連携していない部分です。

当然ですがこの運用ははっきり言って色んな無理があり様々な問題が発生しました。
例えば、、、

  • メインからローカルへ ファイル単位で取り込みを行う必要があった
  • メインからローカルへ 取り込むタイミングが保証出来なかった
  • メインからローカルへ取り込みを行った際に どの変更を取り込んだかログが紛失してしまう
  • メインのブランチが増えた場合に 同期しなければいけないブランチが増える
  • ローカルからメインへコミットする際にコミッターがコミットを行うが、 ファイル単位でメール添付で行っておりミスがしやすい。
  • 何より色々めんどくさい!!

.. まぁ、正気の沙汰じゃないですね。

git-svnを利用した解決策を探る

まだコミットの粒度が大きい場合は、 コミッターの注意深い対応と開発者の空気を読んだ行動により、この運用でもなんとかなっていました。
しかしながら障害対応なども割り込んでくるため個別に行う必要がありカオスな状況になっていきます。
何よりこんなリスキーな運用を回しきれる訳がありませんし、自分がどこかで間違える自信があります。
そこでgit-svnを導入する事でローカルのSubversiongitに置き換えて同期させる方法を検討しました。

基本的には、git-svnを用いた通常のワークフローで問題なかったのですがいくつか問題がありました。

コミッターの制限

メインのSubversionへコミットが可能なのはコミッターのみでそれ以外のユーザがコミットをする事が禁止されています。

コメントの問題

ある特定の書式に従ってコミットコメントを書く必要があり、マージ運用を行った場合に問題が発生する可能性がありました。

gitの習熟度

全員で10名ほどのプロジェクトですが、数名をのぞきgitの習熟度がとても低い状態でした(というか知らない状態)。その状態でgitのみではなくgit-svnを利用出来る状態は危険な状態が予想されました。

というわけで解決策

これらの問題を考慮した上で以下のような方針で行うようにしました。
リポジトリの構成は以下のようになります。
Untitled(1).png

  1. コミッターのみgit-svnを利用する。
  2. その他の開発者はgitのみを利用する。
  3. 開発者はgitの共用リポジトリを参照し、それをcloneして開発を行う。
  4. 共用リポジトリはコミッターが所有する物とする。
  5. 共用リポジトリのmasterはメインSubversionのtrunkとなる。
  6. 共用リポジトリのsvn/から始まるブランチは、メインSubversionのブランチとなる
  7. master及びsvn/のリポジトリに対して開発者が直接コミットを行う事は出来ず、コミット可能なのはコミッターのみとする。

コミッターの流れ

最新化(メインSubversionからの変更履歴の取り込み)

  1. git svn rebase を用いて、メインSubversionの変更を取り込みます。
  2. git pushで共用リポジトリにメインSubversionの変更をpushする。

ソースの反映(Pull Requestの取り込み)

githubライクに開発者からPullRequestを受け取る事で、メインSubversionへのコミットを行います。

  1. git checkout masterでブランチ対象のリポジトリに移動します。
  2. git svn rebase でまずメインSubversionと同期します。
  3. git push によってメインSubversionと共用リポジトリの内容を同期させます。
  4. git fetchで共用リポジトリにpushされたブランチ内容を取得します。
  5. git merge --no-ff --no-commit BRANCH_NAME で変更内容をmasteに取り込みます。
  6. 取りこんだ内容が問題ないか動作確認を行います。
  7. 問題ないようであれば、git commitmasterにコミットを行います。この際にメインSubversionにコミットするコミットコメントを記述します。
  8. git svn dcommit -nで問題なくメインSubversionへコミットできるか確認を行います。
  9. git svn dcommit --interactiveでコミットされるログなどを確認します。
  10. 問題ないようであればgit svn dcommitで変更内容をメインSubversionへ送信します。
  11. 最後にgit pushで共用リポジトリにコミットした内容を反映させます。

ブランチマージ時のコンフリクト

上記5において、当然コンフリクトが発生する可能性があります。その場合はそのPullRequestの差し戻しを行います。この辺りは通常のgithubのPullRequestと同じ運用ですね。

開発者の流れ

開発者は共有gitのみを参照しながら作業します。

新規ブランチ作成からコミットまでの流れ

  1. git clone git-share local-git でローカルに共有gitをcloneします。
  2. git checkout masterでマスターブランチへ切り替えます
  3. git checkout -b issue/xxxで障害対応管理単位でブランチを切ります。
  4. 修正してコミットを繰り返します。
  5. 修正が完了したタイミングで、git push origin issue/xxx:issue/xxxでローカル変更を共用リポジトリにpushします。
  6. git request-pullで修正した内容を取り込んでもらうようにコミッターにメールを行います(当然口頭でも一言言いますが)

共有gitの内容をローカルへ

当然定期的にメインSubversionの内容をローカルへ反映させる必要があります。任意のタイミング以外でも、上記5のgit pushを行うタイミングの前などで以下の手順で最新化します。

  1. git checkout masterで共有gitと同期させているmasterへ移行します。
  2. git pull --rebaseで最新の変更を取り込みます。 # ローカルのmasterは共用リポジトリのmasterであり、メインSubversionのtrunkとなるため、 pull--rebaseを強制とします。

問題点

共有gitとメインSubversionの差分をコミッターしか反映出来ない

この点に関してはgitの習熟度に応じて各メンバーにgit-svnの利用を覚えてもらう事で、メインSubversionから共有gitへのルートを広げました。

最後にちょっとだけアクセントを

基本的にはこの構成で問題なかったのですが、共用リポジトリを見る方法を準備していませんでした。
そこで、今回はアトラシアン社のStashを導入しています。

今なら GitBucket という選択肢があったのですが当時はちょうどリリースする前で断念しました。

ただ、Stashのスターターライセンスは有料なだけあって、PullRequestや権限回りがしっかりしていたのでみんながGitのPush/Pull、PullRequestの概念になれるのにとても役に立ったと思います。

というわけですので、SVNからgitに移行しようとしていて躊躇している方は一度試してみてはいかがでしょうか。

95
99
2

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
95
99