質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.93%

railsをdockerを用いて開発する時のbundle installについて

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 1,243

kurukururin

score 21

現在、railsアプリケーションをdockerを用いて開発しようと思っています。

既にアプリケーションは途中まで開発していて、途中からのdocker導入となります。

一通りセットアップは済ませ、既存のアプリケーションをdocker上で動かすことは可能になりました。

しかし、色々なサイトを見ていてbundleに関するいくつかの疑問が浮上しました。

現在のDockerfileはこのようになっています。

FROM ruby:2.5.0
RUN apt-get update -qq && apt-get install -y nodejs postgresql-client
RUN mkdir /myapp
WORKDIR /myapp
COPY Gemfile /myapp/Gemfile
COPY Gemfile.lock /myapp/Gemfile.lock
RUN bundle install
COPY . /myapp

COPY entrypoint.sh /usr/bin/
RUN chmod +x /usr/bin/entrypoint.sh
ENTRYPOINT ["entrypoint.sh"]
EXPOSE 3000

CMD ["rails", "server", "-b", "0.0.0.0"]


また、docker-copmose.yml

version: "3"
services:
  backend:
    build:
      context: ./backend
    command: bash -c "rm -f tmp/pids/server.pid && bundle exec rails s -p 3000 -b '0.0.0.0'"
    volumes:
      - ./backend:/myapp
    ports:
      - "3000:3000"
    links:
      - db
  db:
    image: postgres:12
    ports:
      - "5432:5433"
    volumes:
      - postgresql-data:/var/lib/postgresql/data
volumes:
  postgresql-data:
    driver: local

まず、新たにgemを追加しようとします。

ホスト側のGemfileに
gem hogehoge
を追加し、
docker-compose build
を実行したとします。

この場合、Dockerfile内の

COPY Gemfile /myapp/Gemfile
COPY Gemfile.lock /myapp/Gemfile.lock


が変更を検知して、その次のRUN bundle installを実行すると思います。
この時、Dockerイメージ内にはCOPYされてきたGemfile内のパッケージはインストールされていない状態だと思うので、1から全てのgemをinstallすることになると思います。

その結果、Gemfile.lockがbundle installが実行された後の依存関係を新たに記載した状態になると思います。

しかし、この後の
COPY . /myappによって、ホスト側のbundle installしていない状態のGemfile.lockがCOPYされ、bundle install後のGemfil.lcokの内容が上書きされると思うのですが、どうなんでしょうか?

また、docker-compsoe.ymlのvolumes: ./backedn:/myappの記載によって上記と同じようにbundle installが済んでいない状態のホスト側のGemfile.lockがコンテナ内にマウントされて古いGemfile.lockが上書きされ、Gemfile.lock以外にもbundleディレクトリがまだホスト側では更新されていないので古いbundleディレクトリがコンテナ内にコピーされると思うのですが、このような考え方があっているのでしょうか?

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+1

はい、全体的にそのような認識で合っています。

COPY . /myappによって、ホスト側のbundle installしていない状態のGemfile.lockがCOPYされ、bundle install後のGemfil.lcokの内容が上書きされると思うのですが、どうなんでしょうか?

はい、上書きされてしまいます。
そのような動作を回避したい場合は、gem hogehoge を追加後、ホスト側では bundle install まで実行して Gemfile.lock を新しい内容に更新しておかなければいけません。

Gemfile.lock以外にもbundleディレクトリがまだホスト側では更新されていないので古いbundleディレクトリがコンテナ内にコピーされると思うのですが、

はい、コピーされることになります。
ただし、.dockerignore ファイルで、bundleディレクトリ を無視する設定がされているような場合は、Dockerイメージや立ち上げたコンテナに 古いbundleディレクトリ がコピーされることはありません。

例えば、以下のような内容の .dockerignore ファイルを作成しておくことで、
コピーやマウントする必要のないファイルを docker build や docker run の実行時に無視することができます。

.bundle
log
tmp
vendor/bundle

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/09/30 13:48

    御回答ありがとうございます!
    認識が合っているようでよかったです。

    その上で伺いたいのですが、ローカル環境で`bundle install`を行わずにGemfileとGemfile.lockをしっかり同期させ(古いGemfile.lockがコンテナにマウントされないようにする)るにはどのようにしたらいいのでしょうか。。

    bundleディレクトリに関してはdata volumeに入れることによってホストからコンテナへのマウントは避けられるということは分かりました。

    ホスト側でGemfileを更新後、`docker-compose run`や`docker-compose build`を打ってしまうとDockerfileによりGemfile.lockが上書きされてしまうのでどのようにすべきか悩んでいます。

    Gemfileを更新後起動しているコンテナの中に入ってdocker-compose exec web bundle installとして、コンテナ内でbundle installをしてからコンテナを再起動すると行った方法を思いついたのですがどうでしょうか。。

    キャンセル

  • 2019/09/30 15:23

    はい、まさにそのとおりで、docker-compose.yml にある volumes: の定義によって Gemfile.lock がコンテナにマウントされている状態であれば、docker-compose exec web bundle install などでコンテナ内で bundle install を実行することで、ローカル側の Gemfile.lock の内容も同期されるようになると思います。

    キャンセル

  • 2019/09/30 16:16

    自分の認識が合っていてよかったです。
    ありがとうございました!
    dockerでの開発の流れがだいたい理解できてきました

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.93%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る