Десериализация ActiveRecord из JSON

#ruby-on-rails #json #deserialization #mass-assignment

#ruby-on-rails #json #десериализация #массовое присвоение

Вопрос:

Я хотел бы сохранить результат запроса в redis, используя сериализацию JSON, и запросить его обратно.

Получение результатов запроса в json довольно просто:

 JSON.generate(Model.all.collect {|item| item.attributes})
  

Однако я не нашел правильного способа десериализовать его обратно в ActiveRecord. Самый простой способ:

 JSON.parse(@json_string).collect {|item| Model.new.from_json(item)}
  

Выдает ошибку:

 WARNING: Can't mass-assign protected attributes: id
  

Таким образом, идентификатор становится пустым. Я думал просто использовать OpenStruct для представлений вместо ActiveRecord, но я уверен, что есть лучший способ.

Ответ №1:

Вы можете создать экземпляр нового объекта из JSON, а затем присвоить идентификатор впоследствии. Вероятно, лучше всего создать свой собственный метод для этого:

 class Model
  def self.from_json_with_id(params = {})
    params = JSON.parse(params)
    model = new(params.reject {|k,v| k == "id"})
    model.id = params["id"]
    model
  end
end
  

Или, может быть, просто переопределите from_json() метод.

Комментарии:

1. Хотя я удивлен, для этого нет встроенной поддержки. Звучит неплохо. Спасибо

2. Просто предупреждаю, я это делал. Возможно, вы захотите использовать allocate вместо new , если хотите, чтобы ActiveRecord понимал, что это не новый объект в базе данных

Ответ №2:

Почему не так:

 JSON.parse(@json_string).each do |item|
     item.delete(:id) # I tested it in my case it also works without this line
     object=Model.create(item)
end
  

Если хост, создавший JSON, добавляет корень JSON, вам, возможно, придется использовать item[1] вместо item .

Комментарии:

1. Элементы json были получены из базы данных в первую очередь. Использование model.create попытается создать элемент в базе данных и завершится неудачей, поскольку элемент уже существует.