jazz 0.1.1
Sign up to get free protection for your applications and to get access to all the features.
- data/LICENSE +20 -0
- data/README.rdoc +129 -0
- data/bin/jazz +230 -0
- data/doc/jasmine/files.html +460 -0
- data/doc/jasmine/index.html +322 -0
- data/doc/jasmine/symbols/_global_.html +1083 -0
- data/doc/jasmine/symbols/jasmine.Block.html +417 -0
- data/doc/jasmine/symbols/jasmine.Clock.html +678 -0
- data/doc/jasmine/symbols/jasmine.Env.html +1135 -0
- data/doc/jasmine/symbols/jasmine.EnvjsReporter.html +328 -0
- data/doc/jasmine/symbols/jasmine.JsApiReporter.html +822 -0
- data/doc/jasmine/symbols/jasmine.Matchers.html +1106 -0
- data/doc/jasmine/symbols/jasmine.MultiReporter.html +394 -0
- data/doc/jasmine/symbols/jasmine.NestedResults.html +710 -0
- data/doc/jasmine/symbols/jasmine.Reporter.html +574 -0
- data/doc/jasmine/symbols/jasmine.Runner.html +710 -0
- data/doc/jasmine/symbols/jasmine.Spec.html +1372 -0
- data/doc/jasmine/symbols/jasmine.Spy.html +855 -0
- data/doc/jasmine/symbols/jasmine.Suite.html +705 -0
- data/doc/jasmine/symbols/jasmine.XMLReporter.html +328 -0
- data/doc/jasmine/symbols/jasmine.html +1359 -0
- data/doc/jasmine/symbols/jasmine.util.html +535 -0
- data/doc/jasmine/symbols/src/lib_EnvjsReporter.js.html +149 -0
- data/doc/jasmine/symbols/src/lib_TrivialReporter.js.html +127 -0
- data/doc/jasmine/symbols/src/lib_XMLReporter.js.html +210 -0
- data/doc/jasmine/symbols/src/src_Block.js.html +35 -0
- data/doc/jasmine/symbols/src/src_Env.js.html +233 -0
- data/doc/jasmine/symbols/src/src_JsApiReporter.js.html +110 -0
- data/doc/jasmine/symbols/src/src_Matchers.js.html +399 -0
- data/doc/jasmine/symbols/src/src_MultiReporter.js.html +36 -0
- data/doc/jasmine/symbols/src/src_NestedResults.js.html +88 -0
- data/doc/jasmine/symbols/src/src_PrettyPrinter.js.html +128 -0
- data/doc/jasmine/symbols/src/src_Queue.js.html +119 -0
- data/doc/jasmine/symbols/src/src_Reporter.js.html +35 -0
- data/doc/jasmine/symbols/src/src_Reporters.js.html +51 -0
- data/doc/jasmine/symbols/src/src_Runner.js.html +75 -0
- data/doc/jasmine/symbols/src/src_Spec.js.html +228 -0
- data/doc/jasmine/symbols/src/src_Suite.js.html +77 -0
- data/doc/jasmine/symbols/src/src_WaitsBlock.js.html +21 -0
- data/doc/jasmine/symbols/src/src_WaitsForBlock.js.html +45 -0
- data/doc/jasmine/symbols/src/src_base.js.html +566 -0
- data/doc/jasmine/symbols/src/src_mock-timeout.js.html +185 -0
- data/doc/jasmine/symbols/src/src_util.js.html +75 -0
- data/lib/jazz.rb +0 -0
- data/lib/jazz/intro.js +46 -0
- data/lib/jazz/jasmine/EnvjsReporter.js +141 -0
- data/lib/jazz/jasmine/TrivialReporter.js +155 -0
- data/lib/jazz/jasmine/XMLReporter.js +241 -0
- data/lib/jazz/jasmine/consolex.js +28 -0
- data/lib/jazz/jasmine/jasmine-0.10.0.js +2526 -0
- data/lib/jazz/jasmine/jasmine.js +2526 -0
- data/lib/jazz/jasmine/json2.js +478 -0
- data/lib/jazz/options.rb +26 -0
- data/lib/jazz/outro.js +3 -0
- data/lib/jazz/qintro.js +113 -0
- data/lib/jazz/qoutro.js +3 -0
- data/lib/jazz/qunit/qunit.css +17 -0
- data/lib/jazz/qunit/qunit.js +1027 -0
- data/spec/example/hello_world_spec.js +29 -0
- data/spec/example/nested/HelloWorldSpec.js +13 -0
- data/spec/example/nested/SpecHelper.js +1 -0
- data/spec/example/plain_spec.js +8 -0
- data/spec/example/spec_helper.js +1 -0
- data/spec/spec_helper.js +18 -0
- data/spec/spec_runner.js +143 -0
- data/test/jazz_test.rb +7 -0
- data/test/test_helper.rb +10 -0
- data/vendor/jasmine/MIT.LICENSE +20 -0
- data/vendor/jasmine/README.markdown +479 -0
- data/vendor/jasmine/Rakefile +155 -0
- data/vendor/jasmine/Wakefile +68 -0
- data/vendor/jasmine/contrib/ruby/jasmine_runner.rb +334 -0
- data/vendor/jasmine/contrib/ruby/jasmine_spec_builder.rb +153 -0
- data/vendor/jasmine/contrib/ruby/run.html +47 -0
- data/vendor/jasmine/contrib/ruby/spec/jasmine_runner_spec.rb +71 -0
- data/vendor/jasmine/doc/files.html +460 -0
- data/vendor/jasmine/doc/index.html +322 -0
- data/vendor/jasmine/doc/symbols/_global_.html +1083 -0
- data/vendor/jasmine/doc/symbols/jasmine.Block.html +417 -0
- data/vendor/jasmine/doc/symbols/jasmine.Clock.html +678 -0
- data/vendor/jasmine/doc/symbols/jasmine.Env.html +1135 -0
- data/vendor/jasmine/doc/symbols/jasmine.EnvjsReporter.html +328 -0
- data/vendor/jasmine/doc/symbols/jasmine.JsApiReporter.html +822 -0
- data/vendor/jasmine/doc/symbols/jasmine.Matchers.html +1106 -0
- data/vendor/jasmine/doc/symbols/jasmine.MultiReporter.html +394 -0
- data/vendor/jasmine/doc/symbols/jasmine.NestedResults.html +710 -0
- data/vendor/jasmine/doc/symbols/jasmine.Reporter.html +574 -0
- data/vendor/jasmine/doc/symbols/jasmine.Runner.html +710 -0
- data/vendor/jasmine/doc/symbols/jasmine.Spec.html +1372 -0
- data/vendor/jasmine/doc/symbols/jasmine.Spy.html +855 -0
- data/vendor/jasmine/doc/symbols/jasmine.Suite.html +705 -0
- data/vendor/jasmine/doc/symbols/jasmine.XMLReporter.html +328 -0
- data/vendor/jasmine/doc/symbols/jasmine.html +1359 -0
- data/vendor/jasmine/doc/symbols/jasmine.util.html +535 -0
- data/vendor/jasmine/doc/symbols/src/lib_EnvjsReporter.js.html +149 -0
- data/vendor/jasmine/doc/symbols/src/lib_TrivialReporter.js.html +127 -0
- data/vendor/jasmine/doc/symbols/src/lib_XMLReporter.js.html +210 -0
- data/vendor/jasmine/doc/symbols/src/src_Block.js.html +35 -0
- data/vendor/jasmine/doc/symbols/src/src_Env.js.html +233 -0
- data/vendor/jasmine/doc/symbols/src/src_JsApiReporter.js.html +110 -0
- data/vendor/jasmine/doc/symbols/src/src_Matchers.js.html +399 -0
- data/vendor/jasmine/doc/symbols/src/src_MultiReporter.js.html +36 -0
- data/vendor/jasmine/doc/symbols/src/src_NestedResults.js.html +88 -0
- data/vendor/jasmine/doc/symbols/src/src_PrettyPrinter.js.html +128 -0
- data/vendor/jasmine/doc/symbols/src/src_Queue.js.html +119 -0
- data/vendor/jasmine/doc/symbols/src/src_Reporter.js.html +35 -0
- data/vendor/jasmine/doc/symbols/src/src_Reporters.js.html +51 -0
- data/vendor/jasmine/doc/symbols/src/src_Runner.js.html +75 -0
- data/vendor/jasmine/doc/symbols/src/src_Spec.js.html +228 -0
- data/vendor/jasmine/doc/symbols/src/src_Suite.js.html +77 -0
- data/vendor/jasmine/doc/symbols/src/src_WaitsBlock.js.html +21 -0
- data/vendor/jasmine/doc/symbols/src/src_WaitsForBlock.js.html +45 -0
- data/vendor/jasmine/doc/symbols/src/src_base.js.html +566 -0
- data/vendor/jasmine/doc/symbols/src/src_mock-timeout.js.html +185 -0
- data/vendor/jasmine/doc/symbols/src/src_util.js.html +75 -0
- data/vendor/jasmine/examples/html/example_suite.html +27 -0
- data/vendor/jasmine/examples/html/spec/example_suite.js +11 -0
- data/vendor/jasmine/examples/ruby/Rakefile +33 -0
- data/vendor/jasmine/examples/ruby/spec/example/example_spec.js +11 -0
- data/vendor/jasmine/examples/ruby/spec/jasmine_helper.rb +41 -0
- data/vendor/jasmine/examples/ruby/spec/jasmine_spec.rb +31 -0
- data/vendor/jasmine/examples/ruby/spec/saucelabs.yml +24 -0
- data/vendor/jasmine/geminstaller.yml +23 -0
- data/vendor/jasmine/images/fail-16.png +0 -0
- data/vendor/jasmine/images/fail.png +0 -0
- data/vendor/jasmine/images/go-16.png +0 -0
- data/vendor/jasmine/images/go.png +0 -0
- data/vendor/jasmine/images/pending-16.png +0 -0
- data/vendor/jasmine/images/pending.png +0 -0
- data/vendor/jasmine/images/question-bk.png +0 -0
- data/vendor/jasmine/images/questionbk-16.png +0 -0
- data/vendor/jasmine/images/spinner.gif +0 -0
- data/vendor/jasmine/lib/EnvjsReporter.js +141 -0
- data/vendor/jasmine/lib/TrivialReporter.js +155 -0
- data/vendor/jasmine/lib/XMLReporter.js +241 -0
- data/vendor/jasmine/lib/consolex.js +28 -0
- data/vendor/jasmine/lib/jasmine-0.10.0.js +2526 -0
- data/vendor/jasmine/lib/jasmine.css +86 -0
- data/vendor/jasmine/lib/json2.js +478 -0
- data/vendor/jasmine/spec/jasmine_helper.rb +44 -0
- data/vendor/jasmine/spec/jasmine_spec.rb +31 -0
- data/vendor/jasmine/spec/runner.html +43 -0
- data/vendor/jasmine/spec/runner.js +78 -0
- data/vendor/jasmine/spec/saucelabs.yml +24 -0
- data/vendor/jasmine/spec/suites/EnvSpec.js +71 -0
- data/vendor/jasmine/spec/suites/ExceptionsSpec.js +107 -0
- data/vendor/jasmine/spec/suites/JsApiReporterSpec.js +82 -0
- data/vendor/jasmine/spec/suites/MatchersSpec.js +589 -0
- data/vendor/jasmine/spec/suites/MockClockSpec.js +34 -0
- data/vendor/jasmine/spec/suites/MultiReporterSpec.js +30 -0
- data/vendor/jasmine/spec/suites/NestedResultsSpec.js +54 -0
- data/vendor/jasmine/spec/suites/PrettyPrintSpec.js +93 -0
- data/vendor/jasmine/spec/suites/QueueSpec.js +23 -0
- data/vendor/jasmine/spec/suites/ReporterSpec.js +60 -0
- data/vendor/jasmine/spec/suites/RunnerSpec.js +258 -0
- data/vendor/jasmine/spec/suites/SpecRunningSpec.js +1117 -0
- data/vendor/jasmine/spec/suites/SpecSpec.js +110 -0
- data/vendor/jasmine/spec/suites/SpySpec.js +187 -0
- data/vendor/jasmine/spec/suites/SuiteSpec.js +101 -0
- data/vendor/jasmine/spec/suites/TrivialReporterSpec.js +140 -0
- data/vendor/jasmine/spec/suites/UtilSpec.js +23 -0
- data/vendor/jasmine/src/Block.js +34 -0
- data/vendor/jasmine/src/Env.js +236 -0
- data/vendor/jasmine/src/JsApiReporter.js +103 -0
- data/vendor/jasmine/src/Matchers.js +326 -0
- data/vendor/jasmine/src/MultiReporter.js +28 -0
- data/vendor/jasmine/src/NestedResults.js +99 -0
- data/vendor/jasmine/src/PrettyPrinter.js +122 -0
- data/vendor/jasmine/src/Queue.js +210 -0
- data/vendor/jasmine/src/Reporter.js +27 -0
- data/vendor/jasmine/src/Reporters.js +43 -0
- data/vendor/jasmine/src/Runner.js +68 -0
- data/vendor/jasmine/src/Spec.js +254 -0
- data/vendor/jasmine/src/Suite.js +70 -0
- data/vendor/jasmine/src/WaitsBlock.js +13 -0
- data/vendor/jasmine/src/WaitsForBlock.js +38 -0
- data/vendor/jasmine/src/base.js +604 -0
- data/vendor/jasmine/src/mock-timeout.js +177 -0
- data/vendor/jasmine/src/util.js +67 -0
- data/vendor/jasmine/src/version.json +5 -0
- metadata +292 -0
@@ -0,0 +1,155 @@
|
|
1
|
+
require File.expand_path(File.join(File.dirname(__FILE__), "spec/jasmine_helper.rb"))
|
2
|
+
|
3
|
+
def jasmine_sources
|
4
|
+
sources = ["src/base.js", "src/util.js", "src/Env.js", "src/Reporter.js", "src/Block.js"]
|
5
|
+
sources += Dir.glob('src/*.js').reject{|f| f == 'src/base.js' || sources.include?(f)}.sort
|
6
|
+
sources
|
7
|
+
end
|
8
|
+
|
9
|
+
def jasmine_filename(version)
|
10
|
+
"jasmine-#{version['major']}.#{version['minor']}.#{version['build']}.js"
|
11
|
+
end
|
12
|
+
|
13
|
+
def version_hash
|
14
|
+
JSON.parse(File.new("src/version.json").read);
|
15
|
+
end
|
16
|
+
|
17
|
+
def start_jasmine_server(jasmine_includes = nil)
|
18
|
+
require File.expand_path(File.join(JasmineHelper.jasmine_root, "contrib/ruby/jasmine_spec_builder"))
|
19
|
+
|
20
|
+
puts "your tests are here:"
|
21
|
+
puts " http://localhost:8888/run.html"
|
22
|
+
|
23
|
+
Jasmine::SimpleServer.start(
|
24
|
+
8888,
|
25
|
+
lambda { JasmineHelper.specs },
|
26
|
+
JasmineHelper.dir_mappings,
|
27
|
+
:jasmine_files => jasmine_includes)
|
28
|
+
end
|
29
|
+
|
30
|
+
task :default => 'jasmine:dist'
|
31
|
+
|
32
|
+
namespace :jasmine do
|
33
|
+
|
34
|
+
desc 'Prepares for distribution'
|
35
|
+
task :dist => ['jasmine:build', 'jasmine:doc']
|
36
|
+
|
37
|
+
desc 'Check jasmine sources for coding problems'
|
38
|
+
task :lint do
|
39
|
+
passed = true
|
40
|
+
jasmine_sources.each do |src|
|
41
|
+
lines = File.read(src).split(/\n/)
|
42
|
+
lines.each_index do |i|
|
43
|
+
line = lines[i]
|
44
|
+
undefineds = line.scan(/.?undefined/)
|
45
|
+
if undefineds.include?(" undefined") || undefineds.include?("\tundefined")
|
46
|
+
puts "Dangerous undefined at #{src}:#{i}:\n > #{line}"
|
47
|
+
passed = false
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
unless passed
|
53
|
+
puts "Lint failed!"
|
54
|
+
exit 1
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
desc 'Builds lib/jasmine from source'
|
59
|
+
task :build => :lint do
|
60
|
+
puts 'Building Jasmine from source'
|
61
|
+
require 'json'
|
62
|
+
|
63
|
+
sources = jasmine_sources
|
64
|
+
version = version_hash
|
65
|
+
|
66
|
+
old_jasmine_files = Dir.glob('lib/jasmine*.js')
|
67
|
+
old_jasmine_files.each do |file|
|
68
|
+
File.delete(file)
|
69
|
+
end
|
70
|
+
|
71
|
+
jasmine = File.new("lib/#{jasmine_filename version}", 'w')
|
72
|
+
|
73
|
+
sources.each do |source_filename|
|
74
|
+
jasmine.puts(File.read(source_filename))
|
75
|
+
end
|
76
|
+
|
77
|
+
jasmine.puts %{
|
78
|
+
jasmine.version_= {
|
79
|
+
"major": #{version['major']},
|
80
|
+
"minor": #{version['minor']},
|
81
|
+
"build": #{version['build']},
|
82
|
+
"revision": #{Time.now.to_i}
|
83
|
+
};
|
84
|
+
}
|
85
|
+
|
86
|
+
jasmine.close
|
87
|
+
end
|
88
|
+
|
89
|
+
desc "Build jasmine documentation"
|
90
|
+
task :doc do
|
91
|
+
puts 'Creating Jasmine Documentation'
|
92
|
+
require 'rubygems'
|
93
|
+
#sudo gem install ragaskar-jsdoc_helper
|
94
|
+
require 'jsdoc_helper'
|
95
|
+
|
96
|
+
|
97
|
+
JsdocHelper::Rake::Task.new(:lambda_jsdoc) do |t|
|
98
|
+
t[:files] = jasmine_sources << 'lib/TrivialReporter.js'
|
99
|
+
t[:options] = "-a"
|
100
|
+
end
|
101
|
+
Rake::Task[:lambda_jsdoc].invoke
|
102
|
+
end
|
103
|
+
|
104
|
+
|
105
|
+
desc "Run jasmine tests of source via server"
|
106
|
+
task :server do
|
107
|
+
files = jasmine_sources + ['lib/TrivialReporter.js', 'lib/consolex.js']
|
108
|
+
jasmine_includes = lambda {
|
109
|
+
raw_jasmine_includes = files.collect { |f| File.expand_path(File.join(JasmineHelper.jasmine_root, f)) }
|
110
|
+
Jasmine.cachebust(raw_jasmine_includes).collect {|f| f.sub(JasmineHelper.jasmine_src_dir, "/src").sub(JasmineHelper.jasmine_lib_dir, "/lib") }
|
111
|
+
}
|
112
|
+
start_jasmine_server(jasmine_includes)
|
113
|
+
end
|
114
|
+
|
115
|
+
desc "Build jasmine and run tests via server"
|
116
|
+
task :server_build => 'jasmine:build' do
|
117
|
+
|
118
|
+
start_jasmine_server
|
119
|
+
end
|
120
|
+
|
121
|
+
namespace :test do
|
122
|
+
desc "Run continuous integration tests using a local Selenium runner"
|
123
|
+
task :ci => :'ci:local'
|
124
|
+
namespace :ci do
|
125
|
+
|
126
|
+
task :local => 'jasmine:build' do
|
127
|
+
require "spec"
|
128
|
+
require 'spec/rake/spectask'
|
129
|
+
Spec::Rake::SpecTask.new(:lambda_ci) do |t|
|
130
|
+
t.spec_opts = ["--color", "--format", "specdoc"]
|
131
|
+
t.spec_files = ["spec/jasmine_spec.rb"]
|
132
|
+
end
|
133
|
+
Rake::Task[:lambda_ci].invoke
|
134
|
+
end
|
135
|
+
|
136
|
+
desc "Run continuous integration tests using Sauce Labs 'Selenium in the Cloud'"
|
137
|
+
task :saucelabs => 'jasmine:build' do
|
138
|
+
ENV['SAUCELABS'] = 'true'
|
139
|
+
Rake::Task['jasmine:test:ci:local'].invoke
|
140
|
+
end
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
end
|
145
|
+
|
146
|
+
task :test do
|
147
|
+
system "echo jazz " + Dir["spec/**/*[Ss]pec.js"].join(" ")
|
148
|
+
system "jazz " + Dir["spec/**/*[Ss]pec.js"].join(" ")
|
149
|
+
end
|
150
|
+
|
151
|
+
task :default => :test
|
152
|
+
|
153
|
+
# Local Variables:
|
154
|
+
# mode:ruby
|
155
|
+
# End:
|
@@ -0,0 +1,68 @@
|
|
1
|
+
begin; require 'wake/event_handlers/em'; rescue LoadError; end
|
2
|
+
begin; require 'wake/deps'; rescue LoadError; end
|
3
|
+
|
4
|
+
# p "reload"
|
5
|
+
|
6
|
+
watch( %r(.*), :modified, lambda { |md| File.directory? md[0] } ) do |md|
|
7
|
+
raise Wake::Refresh
|
8
|
+
end
|
9
|
+
|
10
|
+
watch( %r(jasmine.wake), :modified ) do |md|
|
11
|
+
raise Wake::Refresh
|
12
|
+
end
|
13
|
+
|
14
|
+
map_to_test = lambda do |file, event|
|
15
|
+
case file
|
16
|
+
when %r(spec/(.*)([Ss]pec)\.js$)
|
17
|
+
# Run JS spec's using parallel HTML file if it exists
|
18
|
+
prefix = $~[1];
|
19
|
+
prefix.sub! %r(_$), ""
|
20
|
+
files = Dir[prefix+".*htm*"]
|
21
|
+
if html = files.detect { |f| f =~ %r(\.x?html?) }
|
22
|
+
event == :load ? nil : html
|
23
|
+
else
|
24
|
+
file
|
25
|
+
end
|
26
|
+
else; file
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
jazz = lambda do |*args|
|
31
|
+
files = []
|
32
|
+
# boy, clean this up, but call/splat are subtle
|
33
|
+
if Array === args[0]
|
34
|
+
args = args[0][0]
|
35
|
+
files = args.map { |pair| map_to_test.call( pair[0][0], pair[1] ) }
|
36
|
+
files.compact!
|
37
|
+
files.uniq!
|
38
|
+
else
|
39
|
+
(file, event) = *args
|
40
|
+
file = map_to_test.call file, event
|
41
|
+
if file
|
42
|
+
files = [ file ]
|
43
|
+
end
|
44
|
+
end
|
45
|
+
if !files.empty?
|
46
|
+
deps = ""
|
47
|
+
begin deps = "--deps #{db_path}"; rescue; end
|
48
|
+
cmd = "jazz #{deps} #{files.join(" ")}"
|
49
|
+
puts cmd
|
50
|
+
system cmd
|
51
|
+
# puts "exit status: #{$?.exitstatus}" if $?.exited? && $?.exitstatus != 0
|
52
|
+
if $?.signaled? && $?.termsig == 2
|
53
|
+
Process.kill 2, 0
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
watch( %r((^spec/.*[Ss]pec)\.js$), [ :load, :created, :modified ], nil, :batch => :js ) do |events|
|
59
|
+
jazz.call events
|
60
|
+
end
|
61
|
+
|
62
|
+
Signal.trap('QUIT') do
|
63
|
+
EM.stop
|
64
|
+
end
|
65
|
+
|
66
|
+
# Local Variables:
|
67
|
+
# mode:ruby
|
68
|
+
# End:
|
@@ -0,0 +1,334 @@
|
|
1
|
+
require 'socket'
|
2
|
+
require 'erb'
|
3
|
+
require 'json'
|
4
|
+
|
5
|
+
module Jasmine
|
6
|
+
def self.root
|
7
|
+
File.expand_path(File.join(File.dirname(__FILE__), '../..'))
|
8
|
+
end
|
9
|
+
|
10
|
+
# this seemingly-over-complex method is necessary to get an open port on at least some of our Macs
|
11
|
+
def self.open_socket_on_unused_port
|
12
|
+
infos = Socket::getaddrinfo("localhost", nil, Socket::AF_UNSPEC, Socket::SOCK_STREAM, 0, Socket::AI_PASSIVE)
|
13
|
+
families = Hash[*infos.collect { |af, *_| af }.uniq.zip([]).flatten]
|
14
|
+
|
15
|
+
return TCPServer.open('0.0.0.0', 0) if families.has_key?('AF_INET')
|
16
|
+
return TCPServer.open('::', 0) if families.has_key?('AF_INET6')
|
17
|
+
return TCPServer.open(0)
|
18
|
+
end
|
19
|
+
|
20
|
+
def self.find_unused_port
|
21
|
+
socket = open_socket_on_unused_port
|
22
|
+
port = socket.addr[1]
|
23
|
+
socket.close
|
24
|
+
port
|
25
|
+
end
|
26
|
+
|
27
|
+
def self.server_is_listening_on(hostname, port)
|
28
|
+
require 'socket'
|
29
|
+
begin
|
30
|
+
socket = TCPSocket.open(hostname, port)
|
31
|
+
rescue Errno::ECONNREFUSED
|
32
|
+
return false
|
33
|
+
end
|
34
|
+
socket.close
|
35
|
+
true
|
36
|
+
end
|
37
|
+
|
38
|
+
def self.wait_for_listener(port, name = "required process", seconds_to_wait = 10)
|
39
|
+
time_out_at = Time.now + seconds_to_wait
|
40
|
+
until server_is_listening_on "localhost", port
|
41
|
+
sleep 0.1
|
42
|
+
puts "Waiting for #{name} on #{port}..."
|
43
|
+
raise "#{name} didn't show up on port #{port} after #{seconds_to_wait} seconds." if Time.now > time_out_at
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
def self.kill_process_group(process_group_id, signal="TERM")
|
48
|
+
Process.kill signal, -process_group_id # negative pid means kill process group. (see man 2 kill)
|
49
|
+
end
|
50
|
+
|
51
|
+
def self.cachebust(files, root_dir="", replace=nil, replace_with=nil)
|
52
|
+
require 'digest/md5'
|
53
|
+
files.collect do |file_name|
|
54
|
+
real_file_name = replace && replace_with ? file_name.sub(replace, replace_with) : file_name
|
55
|
+
begin
|
56
|
+
digest = Digest::MD5.hexdigest(File.read("#{root_dir}#{real_file_name}"))
|
57
|
+
rescue
|
58
|
+
digest = "MISSING-FILE"
|
59
|
+
end
|
60
|
+
"#{file_name}?cachebust=#{digest}"
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
class RunAdapter
|
65
|
+
def initialize(spec_files_or_proc, options = {})
|
66
|
+
@spec_files_or_proc = Jasmine.files(spec_files_or_proc) || []
|
67
|
+
@jasmine_files = Jasmine.files(options[:jasmine_files]) || [
|
68
|
+
"/__JASMINE_ROOT__/lib/" + File.basename(Dir.glob("#{Jasmine.root}/lib/jasmine*.js").first),
|
69
|
+
"/__JASMINE_ROOT__/lib/TrivialReporter.js",
|
70
|
+
"/__JASMINE_ROOT__/lib/json2.js",
|
71
|
+
"/__JASMINE_ROOT__/lib/consolex.js",
|
72
|
+
]
|
73
|
+
@stylesheets = ["/__JASMINE_ROOT__/lib/jasmine.css"] + (Jasmine.files(options[:stylesheets]) || [])
|
74
|
+
@spec_helpers = Jasmine.files(options[:spec_helpers]) || []
|
75
|
+
end
|
76
|
+
|
77
|
+
def call(env)
|
78
|
+
run
|
79
|
+
end
|
80
|
+
|
81
|
+
def run
|
82
|
+
stylesheets = @stylesheets
|
83
|
+
spec_helpers = @spec_helpers
|
84
|
+
spec_files = @spec_files_or_proc
|
85
|
+
|
86
|
+
jasmine_files = @jasmine_files
|
87
|
+
jasmine_files = jasmine_files.call if jasmine_files.respond_to?(:call)
|
88
|
+
|
89
|
+
css_files = @stylesheets
|
90
|
+
|
91
|
+
|
92
|
+
body = ERB.new(File.read(File.join(File.dirname(__FILE__), "run.html"))).result(binding)
|
93
|
+
[
|
94
|
+
200,
|
95
|
+
{ 'Content-Type' => 'text/html' },
|
96
|
+
body
|
97
|
+
]
|
98
|
+
end
|
99
|
+
|
100
|
+
|
101
|
+
end
|
102
|
+
|
103
|
+
class Redirect
|
104
|
+
def initialize(url)
|
105
|
+
@url = url
|
106
|
+
end
|
107
|
+
|
108
|
+
def call(env)
|
109
|
+
[
|
110
|
+
302,
|
111
|
+
{ 'Location' => @url },
|
112
|
+
[]
|
113
|
+
]
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
117
|
+
class JsAlert
|
118
|
+
def call(env)
|
119
|
+
[
|
120
|
+
200,
|
121
|
+
{ 'Content-Type' => 'application/javascript' },
|
122
|
+
"document.write('<p>Couldn\\'t load #{env["PATH_INFO"]}!</p>');"
|
123
|
+
]
|
124
|
+
end
|
125
|
+
end
|
126
|
+
|
127
|
+
class FocusedSuite
|
128
|
+
def initialize(spec_files_or_proc, options)
|
129
|
+
@spec_files_or_proc = Jasmine.files(spec_files_or_proc) || []
|
130
|
+
@options = options
|
131
|
+
end
|
132
|
+
|
133
|
+
def call(env)
|
134
|
+
spec_files = @spec_files_or_proc
|
135
|
+
matching_specs = spec_files.select {|spec_file| spec_file =~ /#{Regexp.escape(env["PATH_INFO"])}/ }.compact
|
136
|
+
if !matching_specs.empty?
|
137
|
+
run_adapter = Jasmine::RunAdapter.new(matching_specs, @options)
|
138
|
+
run_adapter.run
|
139
|
+
else
|
140
|
+
[
|
141
|
+
200,
|
142
|
+
{ 'Content-Type' => 'application/javascript' },
|
143
|
+
"document.write('<p>Couldn\\'t find any specs matching #{env["PATH_INFO"]}!</p>');"
|
144
|
+
]
|
145
|
+
end
|
146
|
+
end
|
147
|
+
|
148
|
+
end
|
149
|
+
|
150
|
+
class SimpleServer
|
151
|
+
def self.start(port, spec_files_or_proc, mappings, options = {})
|
152
|
+
require 'thin'
|
153
|
+
config = {
|
154
|
+
'/__suite__' => Jasmine::FocusedSuite.new(spec_files_or_proc, options),
|
155
|
+
'/run.html' => Jasmine::Redirect.new('/'),
|
156
|
+
'/' => Jasmine::RunAdapter.new(spec_files_or_proc, options)
|
157
|
+
}
|
158
|
+
mappings.each do |from, to|
|
159
|
+
config[from] = Rack::File.new(to)
|
160
|
+
end
|
161
|
+
|
162
|
+
config["/__JASMINE_ROOT__"] = Rack::File.new(Jasmine.root)
|
163
|
+
|
164
|
+
app = Rack::Cascade.new([
|
165
|
+
Rack::URLMap.new(config),
|
166
|
+
JsAlert.new
|
167
|
+
])
|
168
|
+
|
169
|
+
begin
|
170
|
+
Thin::Server.start('0.0.0.0', port, app)
|
171
|
+
rescue RuntimeError => e
|
172
|
+
raise e unless e.message == 'no acceptor'
|
173
|
+
raise RuntimeError.new("A server is already running on port #{port}")
|
174
|
+
end
|
175
|
+
end
|
176
|
+
end
|
177
|
+
|
178
|
+
class SimpleClient
|
179
|
+
def initialize(selenium_host, selenium_port, selenium_browser_start_command, http_address)
|
180
|
+
require 'selenium/client'
|
181
|
+
@driver = Selenium::Client::Driver.new(
|
182
|
+
selenium_host,
|
183
|
+
selenium_port,
|
184
|
+
selenium_browser_start_command,
|
185
|
+
http_address
|
186
|
+
)
|
187
|
+
@http_address = http_address
|
188
|
+
end
|
189
|
+
|
190
|
+
def tests_have_finished?
|
191
|
+
@driver.get_eval("window.jasmine.getEnv().currentRunner.finished") == "true"
|
192
|
+
end
|
193
|
+
|
194
|
+
def connect
|
195
|
+
@driver.start
|
196
|
+
@driver.open("/")
|
197
|
+
end
|
198
|
+
|
199
|
+
def disconnect
|
200
|
+
@driver.stop
|
201
|
+
end
|
202
|
+
|
203
|
+
def run
|
204
|
+
until tests_have_finished? do
|
205
|
+
sleep 0.1
|
206
|
+
end
|
207
|
+
|
208
|
+
puts @driver.get_eval("window.results()")
|
209
|
+
failed_count = @driver.get_eval("window.jasmine.getEnv().currentRunner.results().failedCount").to_i
|
210
|
+
failed_count == 0
|
211
|
+
end
|
212
|
+
|
213
|
+
def eval_js(script)
|
214
|
+
escaped_script = "'" + script.gsub(/(['\\])/) { '\\' + $1 } + "'"
|
215
|
+
|
216
|
+
result = @driver.get_eval(" try { eval(#{escaped_script}, window); } catch(err) { window.eval(#{escaped_script}); }")
|
217
|
+
JSON.parse("[#{result}]")[0]
|
218
|
+
end
|
219
|
+
end
|
220
|
+
|
221
|
+
class Runner
|
222
|
+
def initialize(selenium_jar_path, spec_files, dir_mappings, options={})
|
223
|
+
@selenium_jar_path = selenium_jar_path
|
224
|
+
@spec_files = spec_files
|
225
|
+
@dir_mappings = dir_mappings
|
226
|
+
@options = options
|
227
|
+
|
228
|
+
@browser = options[:browser] ? options[:browser].delete(:browser) : 'firefox'
|
229
|
+
@selenium_pid = nil
|
230
|
+
@jasmine_server_pid = nil
|
231
|
+
@selenium_host = 'localhost'
|
232
|
+
@jasmine_server_port = Jasmine::find_unused_port
|
233
|
+
@selenium_server_port = Jasmine::find_unused_port
|
234
|
+
end
|
235
|
+
|
236
|
+
def start
|
237
|
+
start_jasmine_server
|
238
|
+
start_selenium_server
|
239
|
+
@client = Jasmine::SimpleClient.new(@selenium_host, @selenium_server_port, "*#{@browser}", "http://localhost:#{@jasmine_server_port}/")
|
240
|
+
@client.connect
|
241
|
+
end
|
242
|
+
|
243
|
+
def stop
|
244
|
+
@client.disconnect
|
245
|
+
stop_selenium_server
|
246
|
+
stop_jasmine_server
|
247
|
+
end
|
248
|
+
|
249
|
+
def start_jasmine_server
|
250
|
+
@jasmine_server_pid = fork do
|
251
|
+
Process.setpgrp
|
252
|
+
Jasmine::SimpleServer.start(@jasmine_server_port, @spec_files, @dir_mappings, @options)
|
253
|
+
exit! 0
|
254
|
+
end
|
255
|
+
puts "jasmine server started. pid is #{@jasmine_server_pid}"
|
256
|
+
Jasmine::wait_for_listener(@jasmine_server_port, "jasmine server")
|
257
|
+
end
|
258
|
+
|
259
|
+
def start_selenium_server
|
260
|
+
@selenium_pid = fork do
|
261
|
+
Process.setpgrp
|
262
|
+
exec "java -jar #{@selenium_jar_path} -port #{@selenium_server_port} > /dev/null 2>&1"
|
263
|
+
end
|
264
|
+
puts "selenium started. pid is #{@selenium_pid}"
|
265
|
+
Jasmine::wait_for_listener(@selenium_server_port, "selenium server")
|
266
|
+
end
|
267
|
+
|
268
|
+
def stop_jasmine_server
|
269
|
+
puts "shutting down Jasmine server..."
|
270
|
+
Jasmine::kill_process_group(@jasmine_server_pid) if @jasmine_server_pid
|
271
|
+
end
|
272
|
+
|
273
|
+
def stop_selenium_server
|
274
|
+
puts "shutting down Selenium server..."
|
275
|
+
Jasmine::kill_process_group(@selenium_pid) if @selenium_pid
|
276
|
+
end
|
277
|
+
|
278
|
+
def run
|
279
|
+
begin
|
280
|
+
start
|
281
|
+
puts "servers are listening on their ports -- running the test script..."
|
282
|
+
tests_passed = @client.run
|
283
|
+
ensure
|
284
|
+
stop
|
285
|
+
end
|
286
|
+
return tests_passed
|
287
|
+
end
|
288
|
+
|
289
|
+
def eval_js(script)
|
290
|
+
@client.eval_js(script)
|
291
|
+
end
|
292
|
+
end
|
293
|
+
|
294
|
+
class SauceLabsRunner < Runner
|
295
|
+
def initialize(spec_files, dir_mappings, options={})
|
296
|
+
@spec_files = spec_files
|
297
|
+
@dir_mappings = dir_mappings
|
298
|
+
@options = options
|
299
|
+
|
300
|
+
@browser = options[:browser] ? options[:browser].delete(:browser) : 'firefox'
|
301
|
+
@jasmine_server_pid = nil
|
302
|
+
@jasmine_server_port = Jasmine::find_unused_port
|
303
|
+
@saucelabs_config = SeleniumConfig.new(options[:saucelabs_config], options[:saucelabs_config_file], @jasmine_server_port)
|
304
|
+
end
|
305
|
+
|
306
|
+
def start_selenium_server
|
307
|
+
@sauce_tunnel = SauceTunnel.new(@saucelabs_config)
|
308
|
+
end
|
309
|
+
|
310
|
+
def start
|
311
|
+
start_jasmine_server
|
312
|
+
start_selenium_server
|
313
|
+
@client = Jasmine::SimpleClient.new(@saucelabs_config['selenium_server_address'],
|
314
|
+
4444,
|
315
|
+
@saucelabs_config['selenium_browser_key'],
|
316
|
+
"http://#{@saucelabs_config['application_address']}")
|
317
|
+
@client.connect
|
318
|
+
end
|
319
|
+
|
320
|
+
def stop
|
321
|
+
@client.disconnect
|
322
|
+
@sauce_tunnel.shutdown
|
323
|
+
stop_jasmine_server
|
324
|
+
end
|
325
|
+
|
326
|
+
end
|
327
|
+
|
328
|
+
def self.files(f)
|
329
|
+
result = f
|
330
|
+
result = result.call if result.respond_to?(:call)
|
331
|
+
result
|
332
|
+
end
|
333
|
+
|
334
|
+
end
|