Контроллер записи и код модели Ruby на основе указанных спецификаций - PullRequest
0 голосов
/ 12 апреля 2019

Я должен написать код Ruby на основе предоставленной спецификации и выполнить тесты, указанные в спецификации, при выполнении кода.Ниже приведена спецификация:

describe Product, type: :model do
  let!(:product) { Product.create(name: 'Hammer', category: 'Tools', price: 10.99) }

  describe ".search" do
    subject { Product.search(term) }

    context "has result" do
      context "term in name" do
        let(:term) { "hammer" }
        it { is_expected.to eq([product]) }
      end

      context "term in category" do
        let(:term) { "tool" }
        it { is_expected.to eq([product]) }
      end
    end

    context "has no result" do
      let(:term) { "nail" }
      it { is_expected.to eq([]) }
    end
  end
end

describe ProductsController, type: :controller do
  let!(:product) { Product.create(name: 'Blue shoes', category: 'Footwear', price: 10.99) }
  let!(:response) { RequestHelpers.execute('GET', ProductsController, 'search', {term: 'Shoe', format: :json}) }

  describe "#search" do
    it "returns json" do
      first_product = JSON.parse(response.body).first
      expect(first_product['name']).to eq 'Blue shoes' 
      expect(first_product['category']).to eq 'Footwear'
      expect(first_product['price']).to eq '10.99'
    end
  end
end

Я написал следующий код:

class ProductsController < ActionController::Base
  def create
    @product = Product.new(product_params)
    render json: @product
  end

  def search
    @products = Product.search(params[:search])
    render json: @products
  end

  private
  def product_params
    params.require(:product).permit(:name, :category, :price)
  end
end

class Product < ActiveRecord::Base
  validates_presence_of :name, :category, :price

  def self.search(search)
    where('name LIKE ?', "%#{search}%")
  end

end

Я ожидаю, что все тесты пройдут

Однако, когда я запустлюкод, я получаю трассировку стека ниже:

Test Results:
 Log
-- create_table(:products)
   -> 0.0047s
 Product
 .search
 has result
 term in name
 example at ./spec.rb:10
 term in category
 example at ./spec.rb:15
Test Failed
expected: [#<Product id: 1, name: "Hammer", category: "Tools", price: 0.1099e2, created_at: "2019-04-12 19:01:03", updated_at: "2019-04-12 19:01:03">]
     got: #<ActiveRecord::Relation []>

(compared using ==)

Diff:
@@ -1,2 +1,2 @@
-[#<Product id: 1, name: "Hammer", category: "Tools", price: 0.1099e2, created_at: "2019-04-12 19:01:03", updated_at: "2019-04-12 19:01:03">]
+[]
 has no result
 example at ./spec.rb:21
 ProductsController
 #search
 returns json

Где я могу ошибаться в моей реализации кода?

1 Ответ

1 голос
/ 12 апреля 2019

Ваш синтаксис:

find(:all, :conditions => ['name LIKE ?', "%#{search}%"])

, кажется, устарел / устарел (см. в этом вопросе ).Вместо этого, возможно, попробуйте что-то вроде:

class Product < ActiveRecord::Base
  validates_presence_of :name, :category, :price

  def self.search(search)
    if search
      where('name LIKE ?', "%#{search}%")
    else
      all
    end
  end

end

Этот синтаксис where может требовать или не нуждаться в некоторых играх, поскольку я не проверял это.Кто-то, скорее всего, укажет, если это ошибка.

Кроме того, вы определяете метод search как:

def self.search(search)
  ...
end

, что означает, что требуется аргумент search.Но затем вы проверяете наличие search здесь:

if search
  where('name LIKE ?', "%#{search}%")
else
  all
end

, которое на самом деле не имеет смысла, так как требуется search.Вы должны (1) удалить условие if или (2) сделать необязательное search, выполнив:

def self.search(search=nil)
  ...
end

На основании ваших правок ваш тест "термин в категории" не выполнен, потому что вы 'выполняется только поиск в поле name, здесь:

where('name LIKE ?', "%#{search}%")

У вас нет продукта с name подобным «инструментом», поэтому ваш результат - возврат пустого набора результатов - вот чтоошибка говорит вам.

Попробуйте что-то вроде:

where('name LIKE :search OR category LIKE :search', search: "%#{search}%")

Опять же, вам, возможно, придется поиграть с этим синтаксисом.

...