Создать действие в контроллере — это опустить поле ссылки

#ruby-on-rails

Вопрос:

У меня есть проблема с моим приложением rails, когда я создаю клиента для среды, само действие, если опустить параметр environment_id в состоянии вставки.

Это моя customer.rb модель:

 class Customer < ActiveRecord::Base
  belongs_to :environment, inverse_of: :customers

  has_many :all_services, class_name: 'Service'
  has_many :services, inverse_of: :customer

  has_paper_trail ignore: %i[created_at updated_at]
end
 

Это моя environment.rb модель:

 class Environment < ActiveRecord::Base
  has_many :all_customers, class_name: 'Customer', dependent: :destroy
  has_many :customers, inverse_of: :environment

  has_many :all_services, class_name: 'Service', dependent: :destroy
  has_many :services, inverse_of: :environment

  has_many :all_versions, class_name: 'Version', dependent: :destroy
  has_many :versions, inverse_of: :environment

  has_many :all_role_permissions, class_name: 'RolePermission', dependent: :destroy
  has_many :role_permissions, inverse_of: :environment

  has_paper_trail ignore: %i[created_at updated_at]
end
 

Это действие customer_controller.rb создать:

 def create
 if customer_params.permitted?
  render json: Customer.create!(customer_params), status: :ok
 else
  render json: { message: Api::V1::INVALID_PARAMETERS }, status: :bad_request
 end
end

def customer_params
 params.require(:customer).permit(:environment_id, :full_name, :document_type, :document_value, :customer_type)
end
 

А это журнал сервера:

 Started POST "/api/v1/customers" for 127.0.0.1 at 2021-06-02 18:18:44  0100
Processing by Api::V1::CustomersController#create as HTML
  Parameters: {"full_name"=>"cvbcv", "document_type"=>"bcvbc", "document_value"=>"vbcvbcv", "customer_type"=>"bcvbcvb", "environment_id"=>1, "customer"=>{"full_name"=>"cvbcv", "document_type"=>"bcvbc", "document_value"=>"vbcvbcv", "customer_type"=>"bcvbcvb", "environment_id"=>1}}
  User Load (0.7ms)  SELECT "users".* FROM "users" WHERE "users"."uid" = $1 LIMIT $2  [["uid", "test1@test.cl"], ["LIMIT", 1]]
   (0.5ms)  BEGIN
  ↳ app/services/api/customers.rb:20:in `create_customer'
  Environment Load (0.5ms)  SELECT "environments".* FROM "environments" WHERE "environments"."id" = $1 LIMIT $2  [["id", 1], ["LIMIT", 1]]
  ↳ app/services/api/customers.rb:20:in `create_customer'
  Customer Create (1.4ms)  INSERT INTO "customers" ("full_name", "document_type", "document_value", "customer_type", "created_at", "updated_at") VALUES ($1, $2, $3, $4, $5, $6) RETURNING "id"  [["full_name", "cvbcv"], ["document_type", "bcvbc"], ["document_value", "vbcvbcv"], ["customer_type", "bcvbcvb"], ["created_at", "2021-06-02 17:18:44.378509"], ["updated_at", "2021-06-02 17:18:44.378509"]]
  ↳ app/services/api/customers.rb:20:in `create_customer'
   (1.2ms)  ROLLBACK
  ↳ app/services/api/customers.rb:20:in `create_customer'
Completed 500 Internal Server Error in 22ms (ActiveRecord: 4.3ms | Allocations: 6384)


  
ActiveRecord::NotNullViolation (PG::NotNullViolation: ERROR:  null value in column "environment_id" violates not-null constraint
DETAIL:  Failing row contains (28, cvbcv, bcvbc, vbcvbcv, bcvbcvb, null, 2021-06-02 17:18:44.378509, 2021-06-02 17:18:44.378509).
):

 

Я перепробовал все, что мог знать, я трижды проверил миграции и схему.
Я также проверил версию active record gem из предыдущих проектов, и все в порядке.

Одна вещь, которая может иметь значение, это то, что у проекта есть интерфейс в Angular.

Ответ №1:

Rails проверяет наличие среды в базе данных, потому что клиент belongs_to к ней.

Запрос ниже в вашем журнале пытается загрузить среду с ID=1 :

 Environment Load (0.5ms)  SELECT "environments".* FROM "environments" WHERE "environments"."id" = $1 LIMIT $2  [["id", 1], ["LIMIT", 1]]
 

После этого Rails пытается связаться с INSERT новым клиентом без ссылки на среду, потому что кажется, что среда с идентификатором=1 не существует в базе данных.

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

1. > Rails проверяет наличие среды в базе данных, потому что Клиент принадлежит к ней. это неточно. Rails ничего не проверил, поэтому возникает исключение db-exception.

2. @TimKretschmer согласен, тогда почему возникает запрос на загрузку среды?

Ответ №2:

существует множество способов ее решения. проблема: значение environment_id равно нулю и уже не равно нулю, определенному в базе данных postgres.

  • вы можете добавить a required: true в ассоциацию, которая затем проверяет наличие rails. в вашем случае возникнет ошибка проверки.
  • вы можете добавить validates :enviroment_id, presence: true , который будет делать то же самое
  • вы объединяете идентификатор в параметры

пример, чтобы всегда объединять идентификатор среды в параметры

 def customer_params
 params.require(:customer).permit(:full_name, :document_type, :document_value, :customer_type).merge(environment_id: GET_ME_THE_ENV_ID)
end