繁体   English   中英

等待Websocket Rails事件在测试环境中完成

[英]Wait for Websocket Rails event to finish in test environment

我在Rails应用程序中使用Websocket Rails gem发送事件,该事件触发页面的刷新。 一切工作正常,但是在测试中(我正在使用Rspec + Capybara + Selenium),我似乎无法找到一种方法来等待页面响应Websocket事件。 我正在通过以下方式触发测试中的事件:

WebsocketRails[@michael.id].trigger "refresh_task", {task: @michaels_task, modified_by: @archer}

该事件绑定如下。 ajax调用返回一个实际上是另一个ajax调用的脚本。

channel.bind('refresh_task', function(task_user) {
        var task = task_user["task"];
        var modified_by = task_user["modified_by"];
        if ($('span[data-mahi-task-id="' + task.id +'"]').length > 0) {//if the task is currently visible
            if ($('span[data-mahi-task-id="' + task.id +'"] div.modal').length > 0 ) { //the edit is open, so prompt the user 
                refresh_task = confirm(modified_by.name + " has modified this task, do you want to re-load? (if you don't re-load you risk overwriting their changes)");
            } else {
                refresh_task = true;
            }
            if (refresh_task == true) {
                //hide the modal if it exists
                $('span[data-mahi-task-id="' + task.id +'"] div.modal').modal('hide');

                //grey out the task while we load => this will get removed in ready.js
                $('span[data-mahi-task-id="' + task.id +'"]').addClass('ajax-disabled');
                // fake a "cancel" button on the update user task which will trigger a re-load of the task
                $.ajax({
                  url: "/users/" + task.created_by_id + "/tasks/" + task.id,
                  data: {task: task, update_details_cancel_button: "", _method:'put'}, //need to teel rails this is a put - not all browsers support put
                  dataType: 'script', //what format we expect the response to be - i.e. we want to execupte the resulting script
                  complete : ajaxComplete(task,modified_by),
                  method: 'POST' //not all browsers support put so send a POST with the data _method:'put' as above
                });
            }
        }
    });

一旦ajax启动,我就可以告诉它运行并等待它完成,但是我需要我的测试才能在触发事件和以某种方式启动ajax之间等待?

我通过编写一个接受代码块的辅助方法解决了这一问题。 helper方法记录当前时间,然后向ajax start事件添加一个函数,该事件在sessionStorage变量中记录ajax启动的时间。 然后,我可以运行传递给该方法的代码块,并等待直到ajax启动,如下所示:

def wait_for_ajax_to_start
  start_time = Time.now.to_f * 1000 #number of milliseconds
  #attach a hander on ajaxstart to store the time ajax started in a sessionStorage variable called 'ajaxStarted'
  page.evaluate_script('$(document).ajaxStart(function() {sessionStorage.ajaxStarted = $.now()})')

  #run the code block passed to this method
  yield

  #wait until the ajax has started or timeout
  Timeout.timeout(Capybara.default_wait_time) do #timeout after the default wait time
    loop until (page.evaluate_script('sessionStorage.ajaxStarted') != nil) && (page.evaluate_script('sessionStorage.ajaxStarted').to_i >= start_time)
  end
end

我这样调用辅助方法:

wait_for_ajax_to_start {
  WebsocketRails[@michael.id].trigger "refresh_task", {task: @michaels_task, modified_by: @archer}  
}

到目前为止,这似乎对我有用。

您的测试应等待刷新导致页面上的可见更改

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM