2009-07-10 01:25:04 +00:00
|
|
|
require 'socket'
|
2009-07-09 18:17:58 +00:00
|
|
|
require 'erb'
|
|
|
|
|
|
|
|
module Jasmine
|
2009-07-10 01:25:04 +00:00
|
|
|
def self.find_unused_port
|
|
|
|
socket = TCPserver.open(0)
|
|
|
|
port = socket.addr[1]
|
|
|
|
socket.close
|
|
|
|
port
|
|
|
|
end
|
|
|
|
|
2009-07-09 18:17:58 +00:00
|
|
|
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
|
2009-07-10 01:25:04 +00:00
|
|
|
def self.start(port, spec_dir, mappings)
|
2009-07-10 21:35:24 +00:00
|
|
|
require 'thin'
|
2009-07-09 18:17:58 +00:00
|
|
|
|
|
|
|
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)
|
|
|
|
|
2009-07-10 01:25:04 +00:00
|
|
|
server_port = Jasmine::find_unused_port
|
|
|
|
Thin::Server.start('0.0.0.0', port, app)
|
2009-07-09 18:17:58 +00:00
|
|
|
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
|
2009-07-09 18:17:58 +00:00
|
|
|
@driver.start
|
2009-07-10 01:25:04 +00:00
|
|
|
@driver.open("/run.html")
|
2009-07-10 17:14:27 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def disconnect
|
|
|
|
@driver.stop
|
|
|
|
end
|
2009-07-09 18:17:58 +00:00
|
|
|
|
2009-07-10 17:14:27 +00:00
|
|
|
def run
|
2009-07-09 18:17:58 +00:00
|
|
|
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
|
2009-07-09 18:17:58 +00:00
|
|
|
end
|
2009-07-10 01:14:10 +00:00
|
|
|
|
|
|
|
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
|
2009-07-10 01:14:10 +00:00
|
|
|
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'
|
2009-07-10 01:14:10 +00:00
|
|
|
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}"
|
|
|
|
end
|
|
|
|
puts "selenium started. pid is #{@selenium_pid}"
|
2009-07-10 01:25:04 +00:00
|
|
|
|
2009-07-10 17:14:27 +00:00
|
|
|
@jasmine_server_pid = fork do
|
|
|
|
Jasmine::SimpleServer.start(@jasmine_server_port, @spec_files, @dir_mappings)
|
|
|
|
end
|
|
|
|
puts "jasmine server started. pid is #{@jasmine_server_pid}"
|
2009-07-10 01:14:10 +00:00
|
|
|
|
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 01:14:10 +00:00
|
|
|
|
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 01:14:10 +00:00
|
|
|
|
2009-07-10 17:14:27 +00:00
|
|
|
def run
|
|
|
|
begin
|
|
|
|
start
|
2009-07-10 01:14:10 +00:00
|
|
|
puts "servers are listening on their ports -- running the test script..."
|
2009-07-10 17:14:27 +00:00
|
|
|
tests_passed = @client.run
|
2009-07-10 01:14:10 +00:00
|
|
|
ensure
|
2009-07-10 17:14:27 +00:00
|
|
|
stop
|
2009-07-10 01:14:10 +00:00
|
|
|
end
|
|
|
|
return tests_passed
|
|
|
|
end
|
2009-07-10 17:14:27 +00:00
|
|
|
|
|
|
|
def eval_js(script)
|
|
|
|
@client.eval_js(script)
|
|
|
|
end
|
2009-07-10 01:14:10 +00:00
|
|
|
end
|
2009-07-09 18:17:58 +00:00
|
|
|
end
|