2016-06-23 43 views
6

Ho un modello ProjectKeyword in cui io uso tipo di dati jsonb nella colonna :segemnted_dataRuby on Rails jsonb valore della colonna di default

class ProjectKeyword < ApplicationRecord 
    belongs_to :project 
    belongs_to :keyword 
    has_many :project_keyword_dimensions 
    has_many :dimensions, through: :project_keyword_dimensions 

    validates :project_id, :keyword_id, presence: true 
end 

migrazione

class AddSegemtnedDataToProjectKeywords < ActiveRecord::Migration[5.0] 
    def change 
    add_column :project_keywords, :segmented_data, :jsonb, default: '{}' 
    add_index :project_keywords, :segmented_data, using: :gin 
    end 
end 

Il mio problema è quando creo nuovi project_keyword istanza di default valore di segmented_data è una stringa non un hash e non posso aggiornare questo campo o unire con un altro hash Ad esempio

[12] pry(#)> new_pr_keyword = ProjectKeyword.new(project_id: 1671333, keyword_id: 39155) 
=> #<ProjectKeyword:0x007fd997641090 id: nil, project_id: 1671333, keyword_id: 39155, segmented_data: "{}"> 
[13] pry(#)> new_pr_keyword.save! 
=> true 
[14] pry(#)> new_pr_keyword.segmented_data.update({'new_data' => 'some_data'}) 
NoMethodError: undefined method `update' for "{}":String 
from (pry):14:in `block (3 levels) in <top (required)>' 

Ma quando si assegna il valore hash al campo segmented_data prima dell'aggiornamento, il metodo update funziona correttamente.

Per esempio

[15] pry(#)> new_pr_keyword.segmented_data = {'new_data' => 'some_data'} 
=> {"new_data"=>"some_data"} 
[16] pry(#)> new_pr_keyword.save! 
=> true 
[17] pry(#)> new_pr_keyword.segmented_data.update({'new_data_2' => 'some_data_2'}) 
=> {"new_data"=>"some_data", "new_data_2"=>"some_data_2"} 
[18] pry(#)> new_pr_keyword.save! 
=> true 

La domanda è: come fare valore predefinito di segmented_data per essere un non di classe Hash una stringa in modo metodo di aggiornamento allora funzionerà subito su questo campo, dopo aver oggetto appena creato.

+0

hai provato questo http://stackoverflow.com/a/20746242/3884750? –

+0

Sì, ho provato, non funziona –

risposta

12

Questo ha funzionato per me su diversi progetti:

add_column :project_keywords, :segmented_data, :jsonb, default: {} 

(non una stringa, un hash rubino)

Mi sembra di ricordare questo non funziona on Rails 3, ma dovrebbe andare bene in Rails 4.

+0

Fine in Rails 4 - E anche in Rails 5. –