Why is my rake task running twice in my test?

ε祈祈猫儿з 提交于 2019-12-19 07:35:20

问题


I have a rake task test that I setup following the only examples I could find online.

It looks like this:

require 'test_helper'
require 'minitest/mock'
require 'rake'

class TestScrapeWelcome < ActiveSupport::TestCase
  def setup
    Rake.application.init
    Rake.application.load_rakefile

    @task = Rake::Task['scrape:scrape']
    @task.reenable
  end

  def teardown
    Rake::Task.clear
  end

  test "scraping text and sending to elasticsearch" do
    mocked_client = Minitest::Mock.new
    get_fixtures.each_with_index do |arg,i|
      mocked_client.expect :index, :return_value, [index: "test", type: 'welcome', id: i, body: arg]
    end
    Elasticsearch::Model.stub :client, mocked_client do
      @task.invoke
    end
    assert mocked_client.verify
  end

  private

  def get_fixtures
    (0..11).map { |i|
      File.read("test/fixtures/scrape/index_#{i}.json")
    }
  end

end

But after the task runs once it starts running again without me doing anything (puts prints before and after @task.invoke show that the task is only run the once).


回答1:


Turns out that rake is already required and initialized when the test runs so all of the following lines need to be removed or the task gets defined twice and runs twice even if you only invoke it once.

require 'minitest/mock'
require 'rake'
...
Rake.application.init
Rake.application.load_rakefile



回答2:


Updated answer for rails 5.1 (using minitest):

I found I needed the following to load tasks once and only once:

MyAppName::Application.load_tasks if Rake::Task.tasks.empty?

Alternatively add MyAppName::Application.load_tasks to your test_helper, if you don't mind tasks being loaded even when running individual tests that don't need them.

(Replace MyAppName with your application name)




回答3:


A solution that works for testing the tasks of a Gem that has been made a Railtie so it can add tasks to the Rails app:

Don't define the Railtie in test mode when you're also defining a Rails::Application class in spec_helper.rb (which allows your tests to call Rails.application.load_tasks). Otherwise the Rake file will be loaded once as a Railtie and once as an Engine:

class Railtie < Rails::Railtie
  rake_tasks do
    load 'tasks/mygem.rake'
  end
end unless Rails.env.test? # Without this condition tasks under test are run twice

Another solution would be to put a condition in the Rake file to skip the task definitions if the file has already been loaded.




回答4:


I've tried @iheggie answer but it worked in a way that indeed tests were run once but any other task was breaking with Don't know how to build task '<task_name_like_db_migrate>'.

I'm on Rails 3.2 still. It turned out that there were couple tasks loaded beforehand so the Rake::Task.tasks.empty? was never true and all other useful tasks were not loaded. I've fiddled with it and this version of it works for me right now:

Rake::Task.clear if Rails.env.test?
MyAppName::Application.load_tasks

Hope this helps anyone.



来源:https://stackoverflow.com/questions/31622304/why-is-my-rake-task-running-twice-in-my-test

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!