Can a watir browser object be re-used in a later Ruby process?

后端 未结 5 943
一向
一向 2021-01-14 05:39

So let\'s say pretty often a script runs that opens a browser and does web things:

require \'watir-webdriver\'

$browser = Watir::Browser.new(:firefox, :prof         


        
相关标签:
5条回答
  • 2021-01-14 06:16

    You can use DRb like this:
    browsers pool:

    require 'drb'
    require 'watir'
    
    browser = Watir::Browser.new :chrome
    DRb.start_service 'druby://127.0.0.1:9395', browser
    
    gets
    

    and then from test script use this browser:

    require 'drb'
    browser = DRbObject.new_with_uri 'druby://127.0.0.1:9395'
    browser.goto 'stackoverflow.com'
    
    0 讨论(0)
  • 2021-01-14 06:19

    Just an update on part 2 of my question.

    It seems one CAN serialize a Watir:Browser object with YAML, and because it's text-based the contents were quite interesting to me (e.g. some things I've only dreamed of tweaking hidden inside private elements of private classes...but that's a separate topic)

    Deserializing from YAML is still trouble. While I haven't tested beyond the first try it gives me some kind of reg exp parse error...not sure what that's about.

    (more on that at at how to serialize an object using TCPServer inside? )

    Meanwhile, even attempting to serialize with Marshal, which is also built-in to Ruby but stores in binary format, results in a very reasonable-sounding error about not being able to dump a TCPServer object (apparently contained within my Watir:Browser pointed to by $browser)

    All in all I'm not surprised at these results, but still pretty confident there is a way, until Watir arrives at something more native (like PersistentWebdriver or how it used to be in the days of jssh when you could simply attach to an already running browser with the right extension)

    Until then, if serialization + deserialization to a working object gets too thorny I'll resort to daemonizing a portion of my Ruby to keep objects persistent and spare the frequent and costly setup/teardowns. And I did take a gander at some established (unit testing) frameworks but none seem to fit well yet within my overall software structure--I'm not web testing after all.

    0 讨论(0)
  • 2021-01-14 06:24

    I'm pretty sure that at the point ruby exits, any handles or pointers to something like a browser object would become invalid. So re-using something in a later ruby process is likely not a good approach. In addition I might be wrong on this, but it does seem that webdriver is not very good at connecting to a running browser process. So for your approach to work it would really all need to be wrapped by some master process that was calling all the tests etc.. and hey wait a sec, that's starting to sound like a framework, which you might already (or perhaps should be) using in the first place.

    So a better solution is probably to look at whatever framework you are using to run your tests and investigate any capability for 'setup/teardown' actions (which can go by different names) which are run before and after either each test, groups of tests, or all tests. Going this way is good since most frameworks are designed to allow you to run any single test, or set of tests that you want to. And if your tests are well designed they can be run singly without having to expect the system was left in some perfect state by a prior test. Thus these sorts of setup/teardown actions are designed to work that way as well.

    As an example Cucumber has this at the feature level, with the idea of a 'background' which is basically intended as a way to dry out scenarios by defining common steps to run before each scenario in a feature file. (such as navigating to and logging into your site) This could include a call to a series of steps that would look to see if a browser object existed, and if not create one. However you'd need to put that in every feature file which starts to become rather non dry.

    Fortunately cucumber also allows a way to do this in one place via the use of Hooks. You can define hooks to run before steps, in the event of specific conditions, 'before' and 'after' each scenario, as well as code that runs once before any scenarios, and code defined to run 'at_exit' where you could close the browser after all scenarios have run.

    If I was using cucumber I'd look at the idea of a some code in env.rb that would run at the start to create a browser, complemented by at_exit code to close the browser. Then perhaps also code in a before hook which could check to see that the browser is still there and re-create it if needed, and maybe logout actions in a after hook. Leave stuff like logging in for the individual scenarios, or a background block if all scenarios in a feature login with the same sort of user.

    0 讨论(0)
  • 2021-01-14 06:33

    Not so much a solution but a workaround for part 1 of my question, using pkill. Posting here since it turned out to be a lot less trivial than I had hoped.

    After the ruby script exits, its spawned processes (which may not at all belong in the same PID tree anymore, like firefox-bin) have a predictable "session leader" which turned out to be the parent of the bash shell calling rubyprogram.rb in my case. Available as $PPID in Bash, for when you have to go higher than $$.

    Thus to really clean up unwanted heavyweight processes eg. after a ruby crash:

    #!/bin/bash
    # This is the script that wraps on top of Ruby scripts
    
    ./ruby_program_using_watirwebdriver_browser.rb  myparams &  # spawn ruby in background but keep going below:
    
    sleep 11 # give Ruby a chance to launch its web browser
    
    pstree -panu $$  # prints out a process tree starting under Bash, the parent of Ruby. Firefox may not show!
    
    wait  # now wait for Ruby to exit or crash
    
    pkill -s $PPID firefox-bin  # should only kill firefox-bin's caused above, not elsewhere on the system
    
    # Another way without pkill, will also print out what's getting killed if anything:
    awk '$7=="firefox-bin" && $3=="'$PPID'" {print $1}' <(ps x -o pid,pgid,sess,ppid,tty,time,comm) | xargs -rt kill
    

    OPTIONAL And since I use a dedicated Xvfb Xwindows server just for webdriving on DISPLAY :99, I can also count on xkill:

    timeout 1s  xwininfo -display :99 -root -all |awk '/("Navigator" "Firefox")/ {print $1}' |xargs -rt  xkill -display :99 -id  
    # the timeout is in case xkill decides to wait for user action, when window id was missing
    
    0 讨论(0)
  • 2021-01-14 06:36

    I guess, U cant just remember the variable from another process. But the solution might be creating a master process and process your script in loop in thread, periodically checking the browser running state. I'm using some thing similar in my acceptance tests on Cucumber + watir. So it will be some thing like that:

    require 'rubygems'
    require 'firewatir' # or watir
    @browser = FireWatir::Firefox.new
    
    t = Thread.new do
        @browser.goto "http://google.com"
        #call more browser actions here
    end
    while not_exit?
      if t.stop?
          # error occurred in thread, restart or exit
      end
      if browser_live?
          # browser was killed for a some reason
          # restart or exit
      end
    end
    @browser.close
    

    not_exit? - can be over TRAP for the ctrl+C

    browser_live? - you can check if firefox browser running with processes listings

    It is quite tricky but might work for you

    0 讨论(0)
提交回复
热议问题