#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