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

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

ただいまの
回答率

88.78%

【Rails】データベースの型の変更(Postgresqlの本番環境のみ)

解決済

回答 2

投稿

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

yuki0920

score 31

実現したいこと

Railsによるアプリケーション開発で、カラムのデータ型の変更を行いたいです。
(postsテーブルのrateカラムをstringからfloat型に変更)

データベースをdevelopment環境とproduction環境でそれぞれ別のものを利用しているためか、この点が
ネックとなり型の変更を行えておりproduction環境でのみエラーが出てマイグレーションの実行ができておりません。

production環境のみマイグレーションを適用するする方法があれば教えていただきたいです。

環境

言語:Ruby 2.5.3
フレームワーク:Rails 5.2.3
データベース(development):Mysql
データベース(production):Postgresql
デプロイ:Heroku

試したこととエラー内容

1.下記マイグレーーションの実行

下記マイグレーションの実行により、development環境のデータ型の変更は完了しました。

class ChangeDataRateToPost < ActiveRecord::Migration[5.2]
  def change
    add_column :posts, :new_rate, :float
  end
end

2.エラー内容

production環境でのマイグレーションの実行時に下記のエラーが出ています。

$ heroku run rails db:migrate
Running rails db:migrate on ⬢ myapp... up, run.4444 (Hobby)
D, [2019-05-18T13:20:05.582744 #4] DEBUG -- :    (1.0ms)  SELECT pg_try_advisory_lock(6921524284597967520)
D, [2019-05-18T13:20:05.603255 #4] DEBUG -- :    (3.4ms)  SELECT "schema_migrations"."version" FROM "schema_migrations" ORDER BY "schema_migrations"."version" ASC
I, [2019-05-18T13:20:05.604879 #4]  INFO -- : Migrating to ChangeDataRateToPost (20190518120113)
D, [2019-05-18T13:20:05.607894 #4] DEBUG -- :    (0.8ms)  BEGIN
== 20190518120113 ChangeDataRateToPost: migrating =============================
-- add_column(:posts, :new_rate, :float)
D, [2019-05-18T13:20:05.615485 #4] DEBUG -- :    (7.1ms)  ALTER TABLE "posts" ADD "new_rate" float
   -> 0.0075s
== 20190518120113 ChangeDataRateToPost: migrated (0.0076s) ====================

D, [2019-05-18T13:20:05.622064 #4] DEBUG -- :   ActiveRecord::SchemaMigration Create (1.2ms)  INSERT INTO "schema_migrations" ("version") VALUES ($1) RETURNING "version"  [["version", "20190518120113"]]
D, [2019-05-18T13:20:05.624289 #4] DEBUG -- :    (1.9ms)  COMMIT
I, [2019-05-18T13:20:05.624393 #4]  INFO -- : Migrating to RenameRateToPosts (20190518124329)
D, [2019-05-18T13:20:05.625879 #4] DEBUG -- :    (0.7ms)  BEGIN
== 20190518124329 RenameRateToPosts: migrating ================================
-- change_column(:posts, :rate, :float)
D, [2019-05-18T13:20:05.628843 #4] DEBUG -- :    (2.5ms)  ALTER TABLE "posts" ALTER COLUMN "rate" TYPE float
D, [2019-05-18T13:20:05.633732 #4] DEBUG -- :    (4.7ms)  ROLLBACK
D, [2019-05-18T13:20:05.636263 #4] DEBUG -- :    (2.3ms)  SELECT pg_advisory_unlock(6921524284597967520)
rails aborted!
StandardError: An error has occurred, this and all later migrations canceled:

PG::DatatypeMismatch: ERROR:  column "rate" cannot be cast automatically to type double precision
HINT:  You might need to specify "USING rate::double precision".
: ALTER TABLE "posts" ALTER COLUMN "rate" TYPE float
/app/vendor/bundle/ruby/2.5.0/gems/activerecord-5.2.3/lib/active_record/connection_adapters/postgresql/database_statements.rb:75:in `async_exec'
/app/vendor/bundle/ruby/2.5.0/gems/activerecord-5.2.3/lib/active_record/connection_adapters/postgresql/database_statements.rb:75:in `block (2 levels) in execute'
/app/vendor/bundle/ruby/2.5.0/gems/activesupport-5.2.3/lib/active_support/dependencies/interlock.rb:48:in `block in permit_concurrent_loads'
/app/vendor/bundle/ruby/2.5.0/gems/activesupport-5.2.3/lib/active_support/concurrency/share_lock.rb:187:in `yield_shares'
/app/vendor/bundle/ruby/2.5.0/gems/activesupport-5.2.3/lib/active_support/dependencies/interlock.rb:47:in `permit_concurrent_loads'
/app/vendor/bundle/ruby/2.5.0/gems/activerecord-5.2.3/lib/active_record/connection_adapters/postgresql/database_statements.rb:74:in `block in execute'

(字数制限のため中略)

/app/bin/rails:9:in `<main>'

Caused by:
ActiveRecord::StatementInvalid: PG::DatatypeMismatch: ERROR:  column "rate" cannot be cast automatically to type double precision
HINT:  You might need to specify "USING rate::double precision".
: ALTER TABLE "posts" ALTER COLUMN "rate" TYPE float
/app/vendor/bundle/ruby/2.5.0/gems/activerecord-5.2.3/lib/active_record/connection_adapters/postgresql/database_statements.rb:75:in `async_exec'


(字数制限のため中略)

/app/bin/rails:9:in `<main>'

Caused by:
PG::DatatypeMismatch: ERROR:  column "rate" cannot be cast automatically to type double precision
HINT:  You might need to specify "USING rate::double precision".
/app/vendor/bundle/ruby/2.5.0/gems/activerecord-5.2.3/lib/active_record/connection_adapters/postgresql/database_statements.rb:75:in `async_exec'

(字数制限のため中略)

/app/bin/rails:9:in `<main>'
Tasks: TOP => db:migrate
(See full trace by running task with --trace)

想定していること

エラー文などにより、production環境のPostgresqlのデータ型を変更する際には、castするオプションを利用することが実行可能であることがわかりました。
development環境のマイグレーションはすでに実行済みのためエラーが出てしまいそうだと感じています。

加えて、Postgresqlのデータ型の変更には下記のマイグレーションが必要そうです。しかし、development環境のmysqlはcastが利用できません。

def up
  change_column :posts, :rate, 'float USING CAST(rate AS float)'
end

def down
  change_column :posts, :rate, :string
end
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

+1

production環境のみマイグレーションを適用するする方法があれば教えていただきたいです。

とのことなので、下記でいかがでしょうか?

rake db:migrate RAILS_ENV=production

参考:http://railsdoc.com/rake

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/05/19 22:07

    ご回答ありがとうございます。

    質問の内容が不明瞭で誤解を与えてしまっており恐縮なのですが、本番環境はherokuを利用しており、

    `rake db:migrate RAILS_ENV=production`は利用できませんでした。

    ただ今自己解決できましたので、上記に解決法を記載します。

    キャンセル

  • 2019/05/19 22:10

    こちらこそきちんと読み込めていなくて失礼しました。
    お返事いただきありがとうございます!

    キャンセル

check解決した方法

0

下記の実行で完了できました!

class ChangeDataRateToPost < ActiveRecord::Migration[5.2]
  def up
    # 環境ごとにマイグレーションを分ける
    if Rails.env.development? || Rails.env.test?
      change_column :posts, :rate, :float
    else Rails.env.production?
      # 本番環境はusingオプションを追加
      change_column :posts, :rate, 'float USING CAST(rate AS float)'
    end
  end

  def down
    change_column :posts, :rate, :string
  end


参考
PostgreSQL vs. Rails migration: How to change columns from string to integer

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

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

関連した質問

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