jasmine/contrib/ruby/jasmine_runner.rb

183 lines
4.7 KiB
Ruby
Raw Normal View History

require 'socket'
require 'erb'
module Jasmine
2009-07-13 22:08:11 +00:00
# this seemingly-over-complex method is necessary to get an open port on at least some of our Macs
2009-07-13 22:07:45 +00:00
def self.open_socket_on_unused_port
infos = Socket::getaddrinfo("localhost", nil, Socket::AF_UNSPEC, Socket::SOCK_STREAM, 0, Socket::AI_PASSIVE)
families = Hash[*infos.collect { |af, *_| af }.uniq.zip([]).flatten]
return TCPServer.open('0.0.0.0', 0) if families.has_key?('AF_INET')
return TCPServer.open('::', 0) if families.has_key?('AF_INET6')
return TCPServer.open(0)
end
def self.find_unused_port
2009-07-13 22:07:45 +00:00
socket = open_socket_on_unused_port
port = socket.addr[1]
socket.close
port
end
class RunAdapter
def initialize(spec_files)
p "spec_files: #{spec_files}"
@spec_files = spec_files
end
def call(env)
spec_files = @spec_files
body = ERB.new(File.read(File.join(File.dirname(__FILE__), "run.html"))).result(binding)
[
200,
{ 'Content-Type' => 'text/html' },
body
]
end
end
class SimpleServer
def self.start(port, spec_dir, mappings)
2009-07-10 21:35:24 +00:00
require 'thin'
config = {
'/run.html' => Jasmine::RunAdapter.new(spec_dir)
}
mappings.each do |from, to|
config[from] = Rack::File.new(to)
end
app = Rack::URLMap.new(config)
server_port = Jasmine::find_unused_port
Thin::Server.start('0.0.0.0', port, app)
end
end
class SimpleClient
def initialize(selenium_host, selenium_port, selenium_browser_start_command, http_address)
require 'selenium'
@driver = Selenium::Client::Driver.new(
selenium_host,
selenium_port,
selenium_browser_start_command,
http_address
)
@http_address = http_address
end
def tests_have_finished?
@driver.get_eval("window.jasmine.getEnv().currentRunner.finished") == "true"
end
2009-07-10 17:14:27 +00:00
def connect
@driver.start
@driver.open("/run.html")
2009-07-10 17:14:27 +00:00
end
def disconnect
@driver.stop
end
2009-07-10 17:14:27 +00:00
def run
until tests_have_finished? do
sleep 0.1
end
puts @driver.get_eval("window.getResults()")
failed_count = @driver.get_eval("window.jasmine.getEnv().currentRunner.getResults().failedCount").to_i
failed_count == 0
end
2009-07-10 17:14:27 +00:00
def eval_js(script)
escaped_script = "'" + script.gsub(/(['\\])/) { '\\' + $1 } + "'"
2009-07-10 21:35:24 +00:00
result = @driver.get_eval("window.eval(#{escaped_script})")
JSON.parse("[#{result}]")[0]
2009-07-10 17:14:27 +00:00
end
end
class Runner
def initialize(selenium_jar_path, spec_files, dir_mappings)
@selenium_jar_path = selenium_jar_path
@spec_files = spec_files
@dir_mappings = dir_mappings
2009-07-10 17:14:27 +00:00
@selenium_pid = nil
@jasmine_server_pid = nil
end
2009-07-10 17:14:27 +00:00
def start
start_servers
@client = Jasmine::SimpleClient.new("localhost", @selenium_server_port, "*firefox", "http://localhost:#{@jasmine_server_port}/")
@client.connect
end
def stop
@client.disconnect
stop_servers
end
def server_is_listening_on(hostname, port)
require 'socket'
begin
2009-07-10 17:14:27 +00:00
socket = TCPSocket.open(hostname, port)
rescue Errno::ECONNREFUSED
return false
end
socket.close
true
end
def wait_for_listener(port, name = "required process", seconds_to_wait = 10)
seconds_waited = 0
until server_is_listening_on "localhost", port
sleep 1
puts "Waiting for #{name} on #{port}..."
raise "#{name} didn't show up on port #{port} after #{seconds_to_wait} seconds." if (seconds_waited += 1) >= seconds_to_wait
end
end
def start_servers
@jasmine_server_port = Jasmine::find_unused_port
@selenium_server_port = Jasmine::find_unused_port
@selenium_pid = fork do
exec "java -jar #{@selenium_jar_path} -port #{@selenium_server_port}"
2009-07-10 17:14:27 +00:00
end
puts "selenium started. pid is #{@selenium_pid}"
2009-07-10 17:14:27 +00:00
@jasmine_server_pid = fork do
Jasmine::SimpleServer.start(@jasmine_server_port, @spec_files, @dir_mappings)
sleep(100000)
2009-07-10 17:14:27 +00:00
end
puts "jasmine server started. pid is #{@jasmine_server_pid}"
2009-07-10 17:14:27 +00:00
wait_for_listener(@selenium_server_port, "selenium server")
wait_for_listener(@jasmine_server_port, "jasmine server")
end
2009-07-10 17:14:27 +00:00
def stop_servers
puts "shutting down the servers..."
Process.kill 15, @selenium_pid if @selenium_pid
Process.kill 15, @jasmine_server_pid if @jasmine_server_pid
end
2009-07-10 17:14:27 +00:00
def run
begin
start
puts "servers are listening on their ports -- running the test script..."
2009-07-10 17:14:27 +00:00
tests_passed = @client.run
ensure
2009-07-10 17:14:27 +00:00
stop
end
return tests_passed
end
2009-07-10 17:14:27 +00:00
def eval_js(script)
@client.eval_js(script)
end
end
end