如何设置名称与数据库模式不同的属性

How to set attributes with different names than a DB schema

我是新手 Ruby 开发人员。我无法弄清楚如何创建一个 ActiveRecord 模型,其属性名称与数据库模式中定义的不同

考虑以下架构

  create_table "sync_tasks", force: :cascade do |t|
    t.string "name"
    t.string "path"
    t.string "task_type"
    t.string "status"
    t.boolean "async", default: false
    t.boolean "direct_download", default: true
    t.datetime "created_at", null: false
    t.datetime "completed_at"
    t.datetime "updated_at", null: false
  end

我有以下有效负载

{
    "name" : "Sync /var/www/",
    "path" : "/var/www",
    "directDownload": true,
    "async" : false,
    "taskType" : "directory"
}

并尝试创建我的模型

class SyncTask < ApplicationRecord
  TYPE_DB='db'
  TYPE_FILE='file'
  TYPE_DIRECTORY='directory'

  def initialize(params)
 #   super
    @task_type = params[:taskType]
    @direct_download = params[:directDownload]
    @path = params[:path]
    @status = params[:status]
    @async = params[:async]
  end
end

当我尝试保存时出现错误

<NoMethodError: undefined method `[]' for nil:NilClass>

我也无法访问那样的字段

  new_task = SyncTask.new(allowed_task_params)
  new_task.task_type

它抛出以下错误

#<NoMethodError: undefined method `task_type' for #<SyncTask not initialized>>

如果我取消对 super 调用的注释,它会给出另一个错误

#<ActiveModel::UnknownAttributeError: unknown attribute 'taskType' for SyncTask.>

我做错了什么?如何使用不同的属性名称并自行初始化模型?

谢谢

您可以 transform the keys ,例如:

=> payload = { "name": "Sync /var/www/", "path": "/var/www", "directDownload": true, "taskType": "directory" }
=> h = payload.transform_keys { |key| key.to_s.underscore }  # only since v4.0.2
=> h = Hash[payload.map { |(k, v)| [k.to_s.underscore, v] }] # before v.4.0.2
#> {"name"=>"Sync /var/www/", "path"=>"/var/www", "direct_download"=>true, "task_type"=>"directory"}
=> new_task = SyncTask.new(h)

您不应在 AR 模型上使用初始化方法。如果你仍然需要使用 initialize,请使用 after_initialize 钩子。因为用initialize还要声明super,所以最好用callback。