2017-09-17 1 views
0

Ich bin ein Neuling Ruby-Entwickler. Ich kann nicht herausfinden, wie ein ActiveRecord Modell mit unterschiedlichen Attributen Namen erstellen, als in einem DB-Schema definiertSo legen Sie Attribute mit anderen Namen als ein DB-Schema fest

das folgende Schema

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 

und ich habe die folgende Nutzlast

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

Und versuche Betrachten um mein Modell so zu erstellen

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 

Wenn ich versuche, es zu speichern, einen Fehler wirft

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

auch ich nicht in der Lage bin Feld zuzugreifen, wie die

new_task = SyncTask.new(allowed_task_params) 
    new_task.task_type 

Es führt den folgenden Fehler

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

Falls ich Kommentar- der Superaufruf gibt einen anderen Fehler

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

Was mache ich falsch? Wie kann ich verschiedene Attributnamen verwenden und das Modell selbst initialisieren?

Dank

Antwort

1

Sie können transform the keys, zum Beispiel:

=> 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) 

Sie sollten nicht die Methode initialize auf AR-Modellen verwendet werden. Wenn Sie immer noch initialize verwenden müssen, verwenden Sie after_initialize Haken. Da wir bei der Initialisierung das Super deklarieren müssen, ist es am besten, den Rückruf zu verwenden.

+0

Es gibt keine Möglichkeit, benutzerdefinierte Initializer überhaupt bereitzustellen? Ich habe nicht erwartet, dass ein solches Verhalten von solch dynamischen Sprache wie Ruby – bxfvgekd

+0

Es funktioniert, aber ich mag diese Lösung nicht, wirklich schlechte Funktionalität. – bxfvgekd

+0

In diesem Fall sind Sie abhängig von Modellimplementierungsdetails und Sie können nicht verschiedene Schichten Ihrer Anwendung trennen, wirklich enttäuscht – bxfvgekd

Verwandte Themen