ruby-on-rails - 如何构建我的 RSpec 测试文件夹、文件和数据库?

标签 ruby-on-rails ruby rspec rspec-rails functional-testing

我已经使用 RoR 进行开发一年多了,但我才刚刚开始使用 RSpec 进行测试。

对于标准模型/ Controller 测试,我通常没有任何问题,但问题是我想测试一些复杂的功能流程,并且不知道如何构建我的测试文件夹/文件/数据库。

这是我的应用程序的基本结构:

class Customer
  has_one    :wallet
  has_many :orders    
  has_many :invoices, through: :orders
  has_many :invoice_summaries
end

class Wallet
  belongs_to :customer
end

class Order
  has_one    :invoice
  belongs_to :customer
end

class Invoice
  belongs_to :order
  belongs_to :invoice_summary
end

class InvoiceSummary
  belongs_to :customer
  has_many  :invoices
end

主要问题是我想模拟对象的生命周期,意思是:

  • 实例化将用于所有测试的客户和钱包(无需重新初始化)

  • 模拟时间流,创建和更新多个订单/发票对象和一些 invoice_summaries。

对于订单/发票/invoice_summaries 的创建和更新,我想要像这样的方法

def create_order_1
  # code specific to create my first order, return the created order
end

def create_order_2
  # code specific to create my second order, return the created order
end
.
.
.
def create_order_n
  # code specific to create my n-th order, return the created order
end

def bill_order(order_to_bill)
  # generic code to do the billing of the order passed as parameter
end

def cancel_order(order_to_cancel)
  # generic code to cancel the order passed as parameter
end

我已经找到了 gem Timecop用于模拟时间流。因此,我希望有一个易于理解的最终测试,看起来像

# Code for the initialization of customers and wallets object

describe "Wallet should be equal to 0 after first day" do
  Timecop.freeze(Time.new(2015,7,1))
  first_request = create_request_1
  first_request.customer.wallet.value.should? == 0
end

describe "Wallet should be equal to -30 after second day" do
  Timecop.freeze(Time.new(2015,7,2))
  bill_order(first_request)
  second_order = create_order_2
  first_request.customer.wallet.value.should? == -30
end

describe "Wallet should be equal to -20 after third day" do 
  Timecop.freeze(Time.new(2015,7,3))
  bill_order(second_request)
  cancel_order(first_request)
  first_request.customer.wallet.value.should? == -20
end

describe "Three first day invoice_summary should have 3 invoices" do
  Timecop.freeze(Time.new(2015,7,4))
  invoice_summary = InvoiceSummary.create(
      begin_date: Date.new(2015,7,1),
      end_date: Date.new(2015, 7,3)
  ) # real InvoiceSummary method
  invoice_summary.invoices.count.should? == 3
end

有没有人已经有这样的测试?在构建对象工厂、编写测试等方面是否有好的实践?

例如,有人告诉我将创建的客户/钱包放在 db/seed.rb 文件中是个好主意,但我真的不知道之后该怎么做。

最佳答案

完整回答你的问题可以写满书,所以我只能在这里概述一个答案。

关于创建要测试的对象,

  • db/seeds.rb 不是用于测试数据,而是用于静态数据,不被用户更改,这是应用程序运行所必需的,无论是在开发、测试还是生产中。此类数据的常见示例包括国家代码和用户角色。

  • 创建测试数据、夹具和工厂有两种通用方法。

    • Fixtures 是开箱即用的 Rails 方法。它们在创建测试数据库时创建一次。当您有很多测试时它们很快,因为它们只在测试套件开始时创建一次。但是,它们会扭曲测试,因为它们鼓励围绕现有固定装置编写测试,因此我强烈建议不要使用它们。
    • 工厂是对象创建实用程序。您在每个测试中创建所需的对象,并在最后删除或回滚它们。大多数使用工厂的 Rails 项目都使用 FactoryGirl。这就是我的建议。

    这两种方法都将对象创建代码放在其自己的文件中,该文件位于与您的规范不同的目录中,这可以防止它破坏您的规范文件。如果您在 SO 中搜索“fixtures or factory”,您会发现更多关于这两者的讨论。

如果您将所有重要值(在本例中包括日期和金额)放在可以看到的规范中并与您断言的结果进行比较,您的规范将更容易理解。 (否则您需要记住测试对象中的日期和金额才能理解规范。)您可以在测试 dateamount 参数中提供对象创建方法。那么你可能需要更少的方法。如果您使用 FactoryGirl,可能只是指定每个对象的 created_atamount 属性的问题。另请注意,Rails 具有类似 1.day.from_now 的方法;如果您以这种方式创建具有指定日期的对象,您可能不需要 timecop。

关于如何在文件系统中布置 RSpec 规范,在顶层,只需使布局与 Rails 应用程序的布局相同:

app/
  controllers/
    bars_controller.rb
    foos_controller.rb
  models/
    bar.rb
    foo.rb
    ...
  ...

spec/
  controllers/
    bars_controller_spec.rb
    foos_controller_spec.rb
    ...
  models/
    bar_spec.rb
    foo_spec.rb
    ...
  ...

如果您对单个类的规范太大,则表示该类太大。找到一些模式将其分解并单独测试各个部分。如果您真的无法拆分类(这种情况很少见),请将类的规范文件转换为规范文件目录,如我在 How to break down super long specs in RSpec? 中所述。

关于ruby-on-rails - 如何构建我的 RSpec 测试文件夹、文件和数据库?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31921261/

相关文章:

ruby-on-rails - 在 Rails 3 中测试子域约束路由

ruby - 我如何使用 rugged 检查我的 git 存储库中是否有未提交的更改?

ruby - 保留 HTML 页面的结构,删除所有文本节点

ruby-on-rails - Rails4 - 测试困惑..需要推荐

ruby-on-rails - 具有追随者/友谊的 RSpec 测试有很多通过用户

ruby-on-rails - Ruby on Rails:=>符号是什么意思?

ruby-on-rails - Rails 中的语句类过时了吗?

ruby-on-rails - 我们如何使用 rspec 测试交互组织者?

ruby-on-rails - rails link_to Controller Action

regex - 为什么在 Ruby 中 Regexp 对象被认为是 "falsy"?