web-dev-qa-db-ja.com

ActiveRecordはYAMLの代わりにJSONを使用してシリアル化します

シリアル化された列を使用するモデルがあります。

class Form < ActiveRecord::Base
  serialize :options, Hash
end

このシリアル化でYAMLの代わりにJSONを使用する方法はありますか?

55
Toby Hede

Rails 3.1では、

class Form < ActiveRecord::Base
  serialize :column, JSON
end

役立つことを願っています

156
Justas L.

Rails 3.1では、serializeでカスタムコーダーを使用できます。

class ColorCoder
  # Called to deserialize data to Ruby object.
  def load(data)
  end

  # Called to convert from Ruby object to serialized data.
  def dump(obj)
  end
end

class Fruits < ActiveRecord::Base
  serialize :color, ColorCoder.new
end

お役に立てれば。

参照:

serializeの定義: https://github.com/Rails/rails/blob/master/activerecord/lib/active_record/base.rb#L556

Railsに同梱されているデフォルトのYAMLコーダー: https://github.com/Rails/rails/blob/master/activerecord/lib/active_record/coders/yaml_column.rb

そして、これがloadの呼び出しが発生する場所です。 https://github.com/Rails/rails/blob/master/activerecord/lib/active_record/attribute_methods/read.rb#L132

58
balu

更新

より適切なRails> = 3.1の回答については、midの高評価の回答を参照してください。これはRails <3.1。

おそらくこれがあなたが探しているものです。

Form.find(:first).to_json

更新

1) 'json'をインストールします gem

gem install json

2)JsonWrapperクラスを作成する

# lib/json_wrapper.rb

require 'json'
class JsonWrapper
  def initialize(attribute)
    @attribute = attribute.to_s
  end

  def before_save(record)
    record.send("#{@attribute}=", JsonWrapper.encrypt(record.send("#{@attribute}")))
  end

  def after_save(record)
    record.send("#{@attribute}=", JsonWrapper.decrypt(record.send("#{@attribute}")))
  end

  def self.encrypt(value)
    value.to_json
  end

  def self.decrypt(value)
    JSON.parse(value) rescue value
  end
end

3)モデルコールバックを追加します。

#app/models/user.rb

class User < ActiveRecord::Base
    before_save      JsonWrapper.new( :name )
    after_save       JsonWrapper.new( :name )

    def after_find
      self.name = JsonWrapper.decrypt self.name
    end
end

4)テストしてください!

User.create :name => {"a"=>"b", "c"=>["d", "e"]}

PS:

まったく乾燥していませんが、ベストを尽くしました。誰でもafter_findUserモデルでは、素晴らしいでしょう。

11
St.Woland

私の要件はこの段階で多くのコードを再利用する必要がなかったので、私の蒸留されたコードは上記の答えのバリエーションです:

  require "json/ext"

  before_save :json_serialize  
  after_save  :json_deserialize


  def json_serialize    
    self.options = self.options.to_json
  end

  def json_deserialize    
    self.options = JSON.parse(options)
  end

  def after_find 
    json_deserialize        
  end  

乾杯、最後は非常に簡単です!

8
Toby Hede

serialize :attr, JSON使用composed_ofメソッドは次のように機能します。

  composed_of :auth,
              :class_name => 'ActiveSupport::JSON',
              :mapping => %w(url to_json),
              :constructor => Proc.new { |url| ActiveSupport::JSON.decode(url) }

ここで、urlはjsonを使用してシリアル化される属性で、authはモデルで使用可能な新しいメソッドであり、その値をjson形式でurl属性に保存します。 (まだ完全にはテストされていませんが、機能しているようです)

3
caribu

デフォルトを使用する独自のYAMLコーダーを作成しました。クラスは次のとおりです。

class JSONColumn
  def initialize(default={})
    @default = default
  end

  # this might be the database default and we should plan for empty strings or nils
  def load(s)
    s.present? ? JSON.load(s) : @default.clone
  end

  # this should only be nil or an object that serializes to JSON (like a hash or array)
  def dump(o)
    JSON.dump(o || @default)
  end
end

loaddumpはインスタンスメソッドであるため、モデル定義のserializeへの2番目の引数としてインスタンスを渡す必要があります。以下に例を示します。

class Person < ActiveRecord::Base
  validate :name, :pets, :presence => true
  serialize :pets, JSONColumn.new([])
end

新しいインスタンスを作成し、インスタンスをロードして、IRBにインスタンスをダンプしようとしましたが、すべて正常に動作しているようでした。私も ブログ投稿 について書いた。

3
Benjamin Atkin

より簡単な解決策は、 このブログ投稿Michael Rykovによる で説明されているcomposed_ofを使用することです。このソリューションは、使用するコールバックの数が少ないため、気に入っています。

その要点は次のとおりです。

composed_of :settings, :class_name => 'Settings', :mapping => %w(settings to_json),
                       :constructor => Settings.method(:from_json),
                       :converter   => Settings.method(:from_json)

after_validation do |u|
  u.settings = u.settings if u.settings.dirty? # Force to serialize
end
1
Aleran

アレラン、このメソッドをRails 3?で使用しましたか?私の理解では、彼はあなたがSettingsクラスを定義する必要がないと言っていますが、これを試してみると、Settingが定義されていないことを教え続けます。それと、さらに説明する必要がありましたか?ありがとう。

0
Andrew Lank