newbamboo-rspec 1.1.12
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/.autotest +4 -0
- data/History.txt +1263 -0
- data/License.txt +22 -0
- data/Manifest.txt +327 -0
- data/README.txt +57 -0
- data/Rakefile +104 -0
- data/TODO.txt +18 -0
- data/bin/autospec +4 -0
- data/bin/spec +4 -0
- data/examples/failing/README.txt +7 -0
- data/examples/failing/diffing_spec.rb +36 -0
- data/examples/failing/failing_autogenerated_docstrings_example.rb +19 -0
- data/examples/failing/failure_in_after.rb +10 -0
- data/examples/failing/failure_in_before.rb +10 -0
- data/examples/failing/mocking_example.rb +40 -0
- data/examples/failing/mocking_with_flexmock.rb +26 -0
- data/examples/failing/mocking_with_mocha.rb +25 -0
- data/examples/failing/mocking_with_rr.rb +27 -0
- data/examples/failing/partial_mock_example.rb +20 -0
- data/examples/failing/predicate_example.rb +34 -0
- data/examples/failing/raising_example.rb +47 -0
- data/examples/failing/spec_helper.rb +3 -0
- data/examples/failing/syntax_error_example.rb +7 -0
- data/examples/failing/team_spec.rb +44 -0
- data/examples/failing/timeout_behaviour.rb +7 -0
- data/examples/passing/autogenerated_docstrings_example.rb +25 -0
- data/examples/passing/before_and_after_example.rb +40 -0
- data/examples/passing/behave_as_example.rb +45 -0
- data/examples/passing/custom_expectation_matchers.rb +54 -0
- data/examples/passing/custom_formatter.rb +12 -0
- data/examples/passing/dynamic_spec.rb +9 -0
- data/examples/passing/file_accessor.rb +19 -0
- data/examples/passing/file_accessor_spec.rb +38 -0
- data/examples/passing/greeter_spec.rb +31 -0
- data/examples/passing/helper_method_example.rb +14 -0
- data/examples/passing/io_processor.rb +8 -0
- data/examples/passing/io_processor_spec.rb +21 -0
- data/examples/passing/legacy_spec.rb +11 -0
- data/examples/passing/mocking_example.rb +27 -0
- data/examples/passing/multi_threaded_example_group_runner.rb +26 -0
- data/examples/passing/nested_classes_example.rb +36 -0
- data/examples/passing/partial_mock_example.rb +29 -0
- data/examples/passing/pending_example.rb +20 -0
- data/examples/passing/predicate_example.rb +27 -0
- data/examples/passing/priority.txt +1 -0
- data/examples/passing/shared_example_group_example.rb +81 -0
- data/examples/passing/shared_stack_examples.rb +36 -0
- data/examples/passing/simple_matcher_example.rb +31 -0
- data/examples/passing/spec_helper.rb +3 -0
- data/examples/passing/stack.rb +36 -0
- data/examples/passing/stack_spec.rb +64 -0
- data/examples/passing/stack_spec_with_nested_example_groups.rb +67 -0
- data/examples/passing/stubbing_example.rb +69 -0
- data/examples/passing/yielding_example.rb +33 -0
- data/examples/ruby1.9.compatibility/access_to_constants_spec.rb +86 -0
- data/features/before_and_after_blocks/before_and_after_blocks.feature +168 -0
- data/features/example_groups/autogenerated_docstrings.feature +45 -0
- data/features/example_groups/example_group_with_should_methods.feature +17 -0
- data/features/example_groups/nested_groups.feature +17 -0
- data/features/example_groups/output.feature +20 -0
- data/features/interop/examples_and_tests_together.feature +31 -0
- data/features/interop/test_but_not_test_unit.feature +14 -0
- data/features/interop/test_case_with_should_methods.feature +17 -0
- data/features/mock_framework_integration/use_flexmock.feature +27 -0
- data/features/step_definitions/running_rspec.rb +54 -0
- data/features/support/env.rb +26 -0
- data/features/support/helpers/cmdline.rb +9 -0
- data/features/support/matchers/smart_match.rb +37 -0
- data/init.rb +9 -0
- data/lib/adapters/mock_frameworks/flexmock.rb +24 -0
- data/lib/adapters/mock_frameworks/mocha.rb +20 -0
- data/lib/adapters/mock_frameworks/rr.rb +22 -0
- data/lib/adapters/mock_frameworks/rspec.rb +20 -0
- data/lib/autotest/discover.rb +3 -0
- data/lib/autotest/rspec.rb +47 -0
- data/lib/spec/dsl/main.rb +87 -0
- data/lib/spec/dsl.rb +1 -0
- data/lib/spec/example/before_and_after_hooks.rb +99 -0
- data/lib/spec/example/errors.rb +30 -0
- data/lib/spec/example/example_group.rb +26 -0
- data/lib/spec/example/example_group_factory.rb +98 -0
- data/lib/spec/example/example_group_methods.rb +376 -0
- data/lib/spec/example/example_matcher.rb +44 -0
- data/lib/spec/example/example_methods.rb +137 -0
- data/lib/spec/example/module_reopening_fix.rb +21 -0
- data/lib/spec/example/pending.rb +18 -0
- data/lib/spec/example/shared_example_group.rb +59 -0
- data/lib/spec/example.rb +187 -0
- data/lib/spec/expectations/differs/default.rb +66 -0
- data/lib/spec/expectations/errors.rb +12 -0
- data/lib/spec/expectations/extensions/object.rb +63 -0
- data/lib/spec/expectations/extensions/string_and_symbol.rb +17 -0
- data/lib/spec/expectations/extensions.rb +2 -0
- data/lib/spec/expectations/handler.rb +32 -0
- data/lib/spec/expectations.rb +60 -0
- data/lib/spec/interop/test/unit/autorunner.rb +6 -0
- data/lib/spec/interop/test/unit/testcase.rb +73 -0
- data/lib/spec/interop/test/unit/testresult.rb +6 -0
- data/lib/spec/interop/test/unit/testsuite_adapter.rb +34 -0
- data/lib/spec/interop/test/unit/ui/console/testrunner.rb +61 -0
- data/lib/spec/interop/test.rb +12 -0
- data/lib/spec/matchers/be.rb +208 -0
- data/lib/spec/matchers/be_close.rb +21 -0
- data/lib/spec/matchers/change.rb +148 -0
- data/lib/spec/matchers/eql.rb +25 -0
- data/lib/spec/matchers/equal.rb +25 -0
- data/lib/spec/matchers/errors.rb +5 -0
- data/lib/spec/matchers/exist.rb +16 -0
- data/lib/spec/matchers/generated_descriptions.rb +48 -0
- data/lib/spec/matchers/has.rb +15 -0
- data/lib/spec/matchers/have.rb +150 -0
- data/lib/spec/matchers/include.rb +80 -0
- data/lib/spec/matchers/match.rb +22 -0
- data/lib/spec/matchers/match_array.rb +79 -0
- data/lib/spec/matchers/method_missing.rb +9 -0
- data/lib/spec/matchers/operator_matcher.rb +70 -0
- data/lib/spec/matchers/raise_error.rb +128 -0
- data/lib/spec/matchers/respond_to.rb +71 -0
- data/lib/spec/matchers/satisfy.rb +47 -0
- data/lib/spec/matchers/simple_matcher.rb +132 -0
- data/lib/spec/matchers/throw_symbol.rb +106 -0
- data/lib/spec/matchers/wrap_expectation.rb +55 -0
- data/lib/spec/matchers.rb +141 -0
- data/lib/spec/mocks/argument_expectation.rb +49 -0
- data/lib/spec/mocks/argument_matchers.rb +237 -0
- data/lib/spec/mocks/error_generator.rb +82 -0
- data/lib/spec/mocks/errors.rb +10 -0
- data/lib/spec/mocks/extensions/object.rb +3 -0
- data/lib/spec/mocks/extensions.rb +1 -0
- data/lib/spec/mocks/framework.rb +15 -0
- data/lib/spec/mocks/message_expectation.rb +331 -0
- data/lib/spec/mocks/methods.rb +51 -0
- data/lib/spec/mocks/mock.rb +55 -0
- data/lib/spec/mocks/order_group.rb +29 -0
- data/lib/spec/mocks/proxy.rb +230 -0
- data/lib/spec/mocks/space.rb +28 -0
- data/lib/spec/mocks/spec_methods.rb +55 -0
- data/lib/spec/mocks.rb +200 -0
- data/lib/spec/rake/spectask.rb +225 -0
- data/lib/spec/rake/verify_rcov.rb +52 -0
- data/lib/spec/ruby.rb +9 -0
- data/lib/spec/runner/backtrace_tweaker.rb +58 -0
- data/lib/spec/runner/class_and_arguments_parser.rb +14 -0
- data/lib/spec/runner/command_line.rb +35 -0
- data/lib/spec/runner/configuration.rb +166 -0
- data/lib/spec/runner/drb_command_line.rb +20 -0
- data/lib/spec/runner/example_group_runner.rb +70 -0
- data/lib/spec/runner/formatter/base_formatter.rb +79 -0
- data/lib/spec/runner/formatter/base_text_formatter.rb +127 -0
- data/lib/spec/runner/formatter/failing_example_groups_formatter.rb +27 -0
- data/lib/spec/runner/formatter/failing_examples_formatter.rb +20 -0
- data/lib/spec/runner/formatter/html_formatter.rb +340 -0
- data/lib/spec/runner/formatter/nested_text_formatter.rb +65 -0
- data/lib/spec/runner/formatter/profile_formatter.rb +51 -0
- data/lib/spec/runner/formatter/progress_bar_formatter.rb +34 -0
- data/lib/spec/runner/formatter/snippet_extractor.rb +52 -0
- data/lib/spec/runner/formatter/specdoc_formatter.rb +39 -0
- data/lib/spec/runner/formatter/text_mate_formatter.rb +16 -0
- data/lib/spec/runner/heckle_runner.rb +74 -0
- data/lib/spec/runner/heckle_runner_unsupported.rb +10 -0
- data/lib/spec/runner/option_parser.rb +204 -0
- data/lib/spec/runner/options.rb +320 -0
- data/lib/spec/runner/reporter.rb +167 -0
- data/lib/spec/runner/spec_parser.rb +73 -0
- data/lib/spec/runner.rb +88 -0
- data/lib/spec/version.rb +15 -0
- data/lib/spec.rb +7 -0
- data/resources/rake/examples.rake +7 -0
- data/resources/rake/examples_with_rcov.rake +9 -0
- data/resources/rake/failing_examples_with_html.rake +9 -0
- data/resources/rake/verify_rcov.rake +7 -0
- data/resources/spec/example_group_with_should_methods.rb +12 -0
- data/resources/spec/simple_spec.rb +8 -0
- data/resources/test/spec_and_test_together.rb +57 -0
- data/resources/test/spec_including_test_but_not_unit.rb +11 -0
- data/resources/test/test_case_with_should_methods.rb +30 -0
- data/rspec.gemspec +32 -0
- data/spec/README.jruby +15 -0
- data/spec/autotest/autotest_helper.rb +6 -0
- data/spec/autotest/autotest_matchers.rb +47 -0
- data/spec/autotest/discover_spec.rb +8 -0
- data/spec/autotest/failed_results_re_spec.rb +24 -0
- data/spec/autotest/rspec_spec.rb +114 -0
- data/spec/rspec_suite.rb +6 -0
- data/spec/ruby_forker.rb +13 -0
- data/spec/spec/dsl/main_spec.rb +88 -0
- data/spec/spec/example/example_group_class_definition_spec.rb +49 -0
- data/spec/spec/example/example_group_factory_spec.rb +224 -0
- data/spec/spec/example/example_group_methods_spec.rb +653 -0
- data/spec/spec/example/example_group_spec.rb +661 -0
- data/spec/spec/example/example_matcher_spec.rb +79 -0
- data/spec/spec/example/example_methods_spec.rb +204 -0
- data/spec/spec/example/helper_method_spec.rb +24 -0
- data/spec/spec/example/nested_example_group_spec.rb +71 -0
- data/spec/spec/example/pending_module_spec.rb +139 -0
- data/spec/spec/example/predicate_matcher_spec.rb +21 -0
- data/spec/spec/example/shared_example_group_spec.rb +257 -0
- data/spec/spec/example/subclassing_example_group_spec.rb +25 -0
- data/spec/spec/expectations/differs/default_spec.rb +127 -0
- data/spec/spec/expectations/extensions/object_spec.rb +45 -0
- data/spec/spec/expectations/fail_with_spec.rb +71 -0
- data/spec/spec/expectations/wrap_expectation_spec.rb +30 -0
- data/spec/spec/interop/test/unit/resources/spec_that_fails.rb +10 -0
- data/spec/spec/interop/test/unit/resources/spec_that_passes.rb +10 -0
- data/spec/spec/interop/test/unit/resources/spec_with_errors.rb +10 -0
- data/spec/spec/interop/test/unit/resources/spec_with_options_hash.rb +13 -0
- data/spec/spec/interop/test/unit/resources/test_case_that_fails.rb +10 -0
- data/spec/spec/interop/test/unit/resources/test_case_that_passes.rb +10 -0
- data/spec/spec/interop/test/unit/resources/test_case_with_errors.rb +10 -0
- data/spec/spec/interop/test/unit/resources/testsuite_adapter_spec_with_test_unit.rb +38 -0
- data/spec/spec/interop/test/unit/spec_spec.rb +48 -0
- data/spec/spec/interop/test/unit/test_unit_spec_helper.rb +18 -0
- data/spec/spec/interop/test/unit/testcase_spec.rb +49 -0
- data/spec/spec/interop/test/unit/testsuite_adapter_spec.rb +9 -0
- data/spec/spec/matchers/be_close_spec.rb +41 -0
- data/spec/spec/matchers/be_spec.rb +303 -0
- data/spec/spec/matchers/change_spec.rb +329 -0
- data/spec/spec/matchers/description_generation_spec.rb +172 -0
- data/spec/spec/matchers/eql_spec.rb +29 -0
- data/spec/spec/matchers/equal_spec.rb +29 -0
- data/spec/spec/matchers/exist_spec.rb +57 -0
- data/spec/spec/matchers/handler_spec.rb +111 -0
- data/spec/spec/matchers/has_spec.rb +63 -0
- data/spec/spec/matchers/have_spec.rb +399 -0
- data/spec/spec/matchers/include_spec.rb +88 -0
- data/spec/spec/matchers/match_array_spec.rb +83 -0
- data/spec/spec/matchers/match_spec.rb +37 -0
- data/spec/spec/matchers/matcher_methods_spec.rb +66 -0
- data/spec/spec/matchers/operator_matcher_spec.rb +191 -0
- data/spec/spec/matchers/raise_error_spec.rb +333 -0
- data/spec/spec/matchers/respond_to_spec.rb +116 -0
- data/spec/spec/matchers/satisfy_spec.rb +36 -0
- data/spec/spec/matchers/simple_matcher_spec.rb +93 -0
- data/spec/spec/matchers/throw_symbol_spec.rb +96 -0
- data/spec/spec/mocks/any_number_of_times_spec.rb +36 -0
- data/spec/spec/mocks/argument_expectation_spec.rb +23 -0
- data/spec/spec/mocks/at_least_spec.rb +97 -0
- data/spec/spec/mocks/at_most_spec.rb +93 -0
- data/spec/spec/mocks/bug_report_10260_spec.rb +8 -0
- data/spec/spec/mocks/bug_report_10263_spec.rb +24 -0
- data/spec/spec/mocks/bug_report_11545_spec.rb +32 -0
- data/spec/spec/mocks/bug_report_15719_spec.rb +30 -0
- data/spec/spec/mocks/bug_report_496.rb +19 -0
- data/spec/spec/mocks/bug_report_600_spec.rb +22 -0
- data/spec/spec/mocks/bug_report_7611_spec.rb +19 -0
- data/spec/spec/mocks/bug_report_7805_spec.rb +22 -0
- data/spec/spec/mocks/bug_report_8165_spec.rb +31 -0
- data/spec/spec/mocks/bug_report_8302_spec.rb +26 -0
- data/spec/spec/mocks/failing_argument_matchers_spec.rb +95 -0
- data/spec/spec/mocks/hash_including_matcher_spec.rb +90 -0
- data/spec/spec/mocks/hash_not_including_matcher_spec.rb +67 -0
- data/spec/spec/mocks/mock_ordering_spec.rb +84 -0
- data/spec/spec/mocks/mock_space_spec.rb +54 -0
- data/spec/spec/mocks/mock_spec.rb +579 -0
- data/spec/spec/mocks/multiple_return_value_spec.rb +113 -0
- data/spec/spec/mocks/nil_expectation_warning_spec.rb +53 -0
- data/spec/spec/mocks/null_object_mock_spec.rb +54 -0
- data/spec/spec/mocks/once_counts_spec.rb +53 -0
- data/spec/spec/mocks/options_hash_spec.rb +35 -0
- data/spec/spec/mocks/partial_mock_spec.rb +149 -0
- data/spec/spec/mocks/partial_mock_using_mocks_directly_spec.rb +66 -0
- data/spec/spec/mocks/passing_argument_matchers_spec.rb +145 -0
- data/spec/spec/mocks/precise_counts_spec.rb +52 -0
- data/spec/spec/mocks/record_messages_spec.rb +26 -0
- data/spec/spec/mocks/stub_spec.rb +194 -0
- data/spec/spec/mocks/stubbed_message_expectations_spec.rb +14 -0
- data/spec/spec/mocks/twice_counts_spec.rb +67 -0
- data/spec/spec/package/bin_spec_spec.rb +22 -0
- data/spec/spec/runner/class_and_argument_parser_spec.rb +23 -0
- data/spec/spec/runner/command_line_spec.rb +141 -0
- data/spec/spec/runner/configuration_spec.rb +301 -0
- data/spec/spec/runner/drb_command_line_spec.rb +97 -0
- data/spec/spec/runner/empty_file.txt +0 -0
- data/spec/spec/runner/example_group_runner_spec.rb +33 -0
- data/spec/spec/runner/examples.txt +2 -0
- data/spec/spec/runner/failed.txt +3 -0
- data/spec/spec/runner/formatter/base_formatter_spec.rb +23 -0
- data/spec/spec/runner/formatter/base_text_formatter_spec.rb +23 -0
- data/spec/spec/runner/formatter/failing_example_groups_formatter_spec.rb +45 -0
- data/spec/spec/runner/formatter/failing_examples_formatter_spec.rb +33 -0
- data/spec/spec/runner/formatter/html_formatted-1.8.4.html +365 -0
- data/spec/spec/runner/formatter/html_formatted-1.8.5-jruby.html +387 -0
- data/spec/spec/runner/formatter/html_formatted-1.8.5.html +371 -0
- data/spec/spec/runner/formatter/html_formatted-1.8.6-jruby.html +381 -0
- data/spec/spec/runner/formatter/html_formatted-1.8.6.html +379 -0
- data/spec/spec/runner/formatter/html_formatted-1.8.7.html +379 -0
- data/spec/spec/runner/formatter/html_formatted-1.9.1.html +379 -0
- data/spec/spec/runner/formatter/html_formatter_spec.rb +62 -0
- data/spec/spec/runner/formatter/nested_text_formatter_spec.rb +318 -0
- data/spec/spec/runner/formatter/profile_formatter_spec.rb +65 -0
- data/spec/spec/runner/formatter/progress_bar_formatter_spec.rb +145 -0
- data/spec/spec/runner/formatter/snippet_extractor_spec.rb +18 -0
- data/spec/spec/runner/formatter/spec_mate_formatter_spec.rb +104 -0
- data/spec/spec/runner/formatter/specdoc_formatter_spec.rb +159 -0
- data/spec/spec/runner/formatter/text_mate_formatted-1.8.4.html +365 -0
- data/spec/spec/runner/formatter/text_mate_formatted-1.8.6.html +373 -0
- data/spec/spec/runner/formatter/text_mate_formatted-1.8.7.html +373 -0
- data/spec/spec/runner/formatter/text_mate_formatted-1.9.1.html +373 -0
- data/spec/spec/runner/heckle_runner_spec.rb +78 -0
- data/spec/spec/runner/heckler_spec.rb +20 -0
- data/spec/spec/runner/noisy_backtrace_tweaker_spec.rb +45 -0
- data/spec/spec/runner/option_parser_spec.rb +396 -0
- data/spec/spec/runner/options_spec.rb +469 -0
- data/spec/spec/runner/output_one_time_fixture.rb +7 -0
- data/spec/spec/runner/output_one_time_fixture_runner.rb +8 -0
- data/spec/spec/runner/output_one_time_spec.rb +16 -0
- data/spec/spec/runner/quiet_backtrace_tweaker_spec.rb +62 -0
- data/spec/spec/runner/reporter_spec.rb +238 -0
- data/spec/spec/runner/resources/a_bar.rb +0 -0
- data/spec/spec/runner/resources/a_foo.rb +0 -0
- data/spec/spec/runner/resources/a_spec.rb +1 -0
- data/spec/spec/runner/resources/custom_example_group_runner.rb +14 -0
- data/spec/spec/runner/resources/utf8_encoded.rb +7 -0
- data/spec/spec/runner/spec.opts +2 -0
- data/spec/spec/runner/spec_drb.opts +1 -0
- data/spec/spec/runner/spec_parser/spec_parser_fixture.rb +70 -0
- data/spec/spec/runner/spec_parser_spec.rb +91 -0
- data/spec/spec/runner/spec_spaced.opts +2 -0
- data/spec/spec/runner_spec.rb +11 -0
- data/spec/spec/spec_classes.rb +133 -0
- data/spec/spec/spec_spec.rb +21 -0
- data/spec/spec.opts +6 -0
- data/spec/spec_helper.rb +107 -0
- metadata +388 -0
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
require 'optparse'
|
|
2
|
+
require 'stringio'
|
|
3
|
+
|
|
4
|
+
module Spec
|
|
5
|
+
module Runner
|
|
6
|
+
class OptionParser < ::OptionParser
|
|
7
|
+
class << self
|
|
8
|
+
def parse(args, err, out)
|
|
9
|
+
parser = new(err, out)
|
|
10
|
+
parser.parse(args)
|
|
11
|
+
parser.options
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
def spec_command?
|
|
15
|
+
$0.split('/').last == 'spec'
|
|
16
|
+
end
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
attr_reader :options
|
|
20
|
+
|
|
21
|
+
OPTIONS = {
|
|
22
|
+
:pattern => ["-p", "--pattern [PATTERN]","Limit files loaded to those matching this pattern. Defaults to '**/*_spec.rb'",
|
|
23
|
+
"Separate multiple patterns with commas.",
|
|
24
|
+
"Applies only to directories named on the command line (files",
|
|
25
|
+
"named explicitly on the command line will be loaded regardless)."],
|
|
26
|
+
:diff => ["-D", "--diff [FORMAT]","Show diff of objects that are expected to be equal when they are not",
|
|
27
|
+
"Builtin formats: unified|u|context|c",
|
|
28
|
+
"You can also specify a custom differ class",
|
|
29
|
+
"(in which case you should also specify --require)"],
|
|
30
|
+
:colour => ["-c", "--colour", "--color", "Show coloured (red/green) output"],
|
|
31
|
+
:example => ["-e", "--example [NAME|FILE_NAME]", "Execute example(s) with matching name(s). If the argument is",
|
|
32
|
+
"the path to an existing file (typically generated by a previous",
|
|
33
|
+
"run using --format failing_examples:file.txt), then the examples",
|
|
34
|
+
"on each line of thatfile will be executed. If the file is empty,",
|
|
35
|
+
"all examples will be run (as if --example was not specified).",
|
|
36
|
+
" ",
|
|
37
|
+
"If the argument is not an existing file, then it is treated as",
|
|
38
|
+
"an example name directly, causing RSpec to run just the example",
|
|
39
|
+
"matching that name"],
|
|
40
|
+
:specification => ["-s", "--specification [NAME]", "DEPRECATED - use -e instead", "(This will be removed when autotest works with -e)"],
|
|
41
|
+
:line => ["-l", "--line LINE_NUMBER", Integer, "Execute example group or example at given line.",
|
|
42
|
+
"(does not work for dynamically generated examples)"],
|
|
43
|
+
:format => ["-f", "--format FORMAT[:WHERE]","Specifies what format to use for output. Specify WHERE to tell",
|
|
44
|
+
"the formatter where to write the output. All built-in formats",
|
|
45
|
+
"expect WHERE to be a file name, and will write to $stdout if it's",
|
|
46
|
+
"not specified. The --format option may be specified several times",
|
|
47
|
+
"if you want several outputs",
|
|
48
|
+
" ",
|
|
49
|
+
"Builtin formats:",
|
|
50
|
+
"progress|p : Text-based progress bar",
|
|
51
|
+
"profile|o : Text-based progress bar with profiling of 10 slowest examples",
|
|
52
|
+
"specdoc|s : Code example doc strings",
|
|
53
|
+
"nested|n : Code example doc strings with nested groups intented",
|
|
54
|
+
"html|h : A nice HTML report",
|
|
55
|
+
"failing_examples|e : Write all failing examples - input for --example",
|
|
56
|
+
"failing_example_groups|g : Write all failing example groups - input for --example",
|
|
57
|
+
" ",
|
|
58
|
+
"FORMAT can also be the name of a custom formatter class",
|
|
59
|
+
"(in which case you should also specify --require to load it)"],
|
|
60
|
+
:require => ["-r", "--require FILE", "Require FILE before running specs",
|
|
61
|
+
"Useful for loading custom formatters or other extensions.",
|
|
62
|
+
"If this option is used it must come before the others"],
|
|
63
|
+
:backtrace => ["-b", "--backtrace", "Output full backtrace"],
|
|
64
|
+
:loadby => ["-L", "--loadby STRATEGY", "Specify the strategy by which spec files should be loaded.",
|
|
65
|
+
"STRATEGY can currently only be 'mtime' (File modification time)",
|
|
66
|
+
"By default, spec files are loaded in alphabetical order if --loadby",
|
|
67
|
+
"is not specified."],
|
|
68
|
+
:reverse => ["-R", "--reverse", "Run examples in reverse order"],
|
|
69
|
+
:timeout => ["-t", "--timeout FLOAT", "Interrupt and fail each example that doesn't complete in the",
|
|
70
|
+
"specified time"],
|
|
71
|
+
:heckle => ["-H", "--heckle CODE", "If all examples pass, this will mutate the classes and methods",
|
|
72
|
+
"identified by CODE little by little and run all the examples again",
|
|
73
|
+
"for each mutation. The intent is that for each mutation, at least",
|
|
74
|
+
"one example *should* fail, and RSpec will tell you if this is not the",
|
|
75
|
+
"case. CODE should be either Some::Module, Some::Class or",
|
|
76
|
+
"Some::Fabulous#method}"],
|
|
77
|
+
:dry_run => ["-d", "--dry-run", "Invokes formatters without executing the examples."],
|
|
78
|
+
:options_file => ["-O", "--options PATH", "Read options from a file"],
|
|
79
|
+
:generate_options => ["-G", "--generate-options PATH", "Generate an options file for --options"],
|
|
80
|
+
:runner => ["-U", "--runner RUNNER", "Use a custom Runner."],
|
|
81
|
+
:drb => ["-X", "--drb", "Run examples via DRb. (For example against script/spec_server)"],
|
|
82
|
+
:version => ["-v", "--version", "Show version"],
|
|
83
|
+
:help => ["-h", "--help", "You're looking at it"]
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
def initialize(err, out)
|
|
87
|
+
super()
|
|
88
|
+
@error_stream = err
|
|
89
|
+
@out_stream = out
|
|
90
|
+
@options = Options.new(@error_stream, @out_stream)
|
|
91
|
+
|
|
92
|
+
@file_factory = File
|
|
93
|
+
|
|
94
|
+
self.banner = "Usage: spec (FILE|DIRECTORY|GLOB)+ [options]"
|
|
95
|
+
self.separator ""
|
|
96
|
+
on(*OPTIONS[:pattern]) {|pattern| @options.filename_pattern = pattern}
|
|
97
|
+
on(*OPTIONS[:diff]) {|diff| @options.parse_diff(diff)}
|
|
98
|
+
on(*OPTIONS[:colour]) {@options.colour = true}
|
|
99
|
+
on(*OPTIONS[:example]) {|example| @options.parse_example(example)}
|
|
100
|
+
on(*OPTIONS[:specification]) {|example| @options.parse_example(example)}
|
|
101
|
+
on(*OPTIONS[:line]) {|line_number| @options.line_number = line_number.to_i}
|
|
102
|
+
on(*OPTIONS[:format]) {|format| @options.parse_format(format)}
|
|
103
|
+
on(*OPTIONS[:require]) {|requires| invoke_requires(requires)}
|
|
104
|
+
on(*OPTIONS[:backtrace]) {@options.backtrace_tweaker = NoisyBacktraceTweaker.new}
|
|
105
|
+
on(*OPTIONS[:loadby]) {|loadby| @options.loadby = loadby}
|
|
106
|
+
on(*OPTIONS[:reverse]) {@options.reverse = true}
|
|
107
|
+
on(*OPTIONS[:timeout]) {|timeout| @options.timeout = timeout.to_f}
|
|
108
|
+
on(*OPTIONS[:heckle]) {|heckle| @options.load_heckle_runner(heckle)}
|
|
109
|
+
on(*OPTIONS[:dry_run]) {@options.dry_run = true}
|
|
110
|
+
on(*OPTIONS[:options_file]) {|options_file| parse_options_file(options_file)}
|
|
111
|
+
on(*OPTIONS[:generate_options]) {|options_file|}
|
|
112
|
+
on(*OPTIONS[:runner]) {|runner| @options.user_input_for_runner = runner}
|
|
113
|
+
on(*OPTIONS[:drb]) {}
|
|
114
|
+
on(*OPTIONS[:version]) {parse_version}
|
|
115
|
+
on_tail(*OPTIONS[:help]) {parse_help}
|
|
116
|
+
end
|
|
117
|
+
|
|
118
|
+
def order!(argv, &blk)
|
|
119
|
+
@argv = argv.dup
|
|
120
|
+
@argv = (@argv.empty? && self.class.spec_command?) ? ['--help'] : @argv
|
|
121
|
+
@options.argv = @argv.dup
|
|
122
|
+
return if parse_generate_options
|
|
123
|
+
return if parse_drb
|
|
124
|
+
|
|
125
|
+
super(@argv) do |file|
|
|
126
|
+
@options.files << file
|
|
127
|
+
blk.call(file) if blk
|
|
128
|
+
end
|
|
129
|
+
|
|
130
|
+
@options
|
|
131
|
+
end
|
|
132
|
+
|
|
133
|
+
protected
|
|
134
|
+
def invoke_requires(requires)
|
|
135
|
+
requires.split(",").each do |file|
|
|
136
|
+
require file
|
|
137
|
+
end
|
|
138
|
+
end
|
|
139
|
+
|
|
140
|
+
def parse_options_file(options_file)
|
|
141
|
+
option_file_args = IO.readlines(options_file).map {|l| l.chomp.split " "}.flatten
|
|
142
|
+
@argv.push(*option_file_args)
|
|
143
|
+
# TODO - this is a brute force solution to http://rspec.lighthouseapp.com/projects/5645/tickets/293.
|
|
144
|
+
# Let's look for a cleaner way. Might not be one. But let's look. If not, perhaps
|
|
145
|
+
# this can be moved to a different method to indicate the special handling for drb?
|
|
146
|
+
parse_drb(@argv)
|
|
147
|
+
end
|
|
148
|
+
|
|
149
|
+
def parse_generate_options
|
|
150
|
+
# Remove the --generate-options option and the argument before writing to file
|
|
151
|
+
options_file = nil
|
|
152
|
+
['-G', '--generate-options'].each do |option|
|
|
153
|
+
if index = @argv.index(option)
|
|
154
|
+
@argv.delete_at(index)
|
|
155
|
+
options_file = @argv.delete_at(index)
|
|
156
|
+
end
|
|
157
|
+
end
|
|
158
|
+
|
|
159
|
+
if options_file
|
|
160
|
+
write_generated_options(options_file)
|
|
161
|
+
return true
|
|
162
|
+
else
|
|
163
|
+
return false
|
|
164
|
+
end
|
|
165
|
+
end
|
|
166
|
+
|
|
167
|
+
def write_generated_options(options_file)
|
|
168
|
+
File.open(options_file, 'w') do |io|
|
|
169
|
+
io.puts @argv.join("\n")
|
|
170
|
+
end
|
|
171
|
+
@out_stream.puts "\nOptions written to #{options_file}. You can now use these options with:"
|
|
172
|
+
@out_stream.puts "spec --options #{options_file}"
|
|
173
|
+
@options.examples_should_not_be_run
|
|
174
|
+
end
|
|
175
|
+
|
|
176
|
+
def parse_drb(argv = nil)
|
|
177
|
+
argv ||= @options.argv # TODO - see note about about http://rspec.lighthouseapp.com/projects/5645/tickets/293
|
|
178
|
+
is_drb = false
|
|
179
|
+
is_drb ||= argv.delete(OPTIONS[:drb][0])
|
|
180
|
+
is_drb ||= argv.delete(OPTIONS[:drb][1])
|
|
181
|
+
return false unless is_drb
|
|
182
|
+
@options.examples_should_not_be_run
|
|
183
|
+
DrbCommandLine.run(
|
|
184
|
+
self.class.parse(argv, @error_stream, @out_stream)
|
|
185
|
+
)
|
|
186
|
+
true
|
|
187
|
+
end
|
|
188
|
+
|
|
189
|
+
def parse_version
|
|
190
|
+
@out_stream.puts ::Spec::VERSION::SUMMARY
|
|
191
|
+
exit if stdout?
|
|
192
|
+
end
|
|
193
|
+
|
|
194
|
+
def parse_help
|
|
195
|
+
@out_stream.puts self
|
|
196
|
+
exit if stdout?
|
|
197
|
+
end
|
|
198
|
+
|
|
199
|
+
def stdout?
|
|
200
|
+
@out_stream == $stdout
|
|
201
|
+
end
|
|
202
|
+
end
|
|
203
|
+
end
|
|
204
|
+
end
|
|
@@ -0,0 +1,320 @@
|
|
|
1
|
+
module Spec
|
|
2
|
+
module Runner
|
|
3
|
+
class Options
|
|
4
|
+
FILE_SORTERS = {
|
|
5
|
+
'mtime' => lambda {|file_a, file_b| File.mtime(file_b) <=> File.mtime(file_a)}
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
EXAMPLE_FORMATTERS = { # Load these lazily for better speed
|
|
9
|
+
'specdoc' => ['spec/runner/formatter/specdoc_formatter', 'Formatter::SpecdocFormatter'],
|
|
10
|
+
's' => ['spec/runner/formatter/specdoc_formatter', 'Formatter::SpecdocFormatter'],
|
|
11
|
+
'nested' => ['spec/runner/formatter/nested_text_formatter', 'Formatter::NestedTextFormatter'],
|
|
12
|
+
'n' => ['spec/runner/formatter/nested_text_formatter', 'Formatter::NestedTextFormatter'],
|
|
13
|
+
'html' => ['spec/runner/formatter/html_formatter', 'Formatter::HtmlFormatter'],
|
|
14
|
+
'h' => ['spec/runner/formatter/html_formatter', 'Formatter::HtmlFormatter'],
|
|
15
|
+
'progress' => ['spec/runner/formatter/progress_bar_formatter', 'Formatter::ProgressBarFormatter'],
|
|
16
|
+
'p' => ['spec/runner/formatter/progress_bar_formatter', 'Formatter::ProgressBarFormatter'],
|
|
17
|
+
'failing_examples' => ['spec/runner/formatter/failing_examples_formatter', 'Formatter::FailingExamplesFormatter'],
|
|
18
|
+
'e' => ['spec/runner/formatter/failing_examples_formatter', 'Formatter::FailingExamplesFormatter'],
|
|
19
|
+
'failing_example_groups' => ['spec/runner/formatter/failing_example_groups_formatter', 'Formatter::FailingExampleGroupsFormatter'],
|
|
20
|
+
'g' => ['spec/runner/formatter/failing_example_groups_formatter', 'Formatter::FailingExampleGroupsFormatter'],
|
|
21
|
+
'profile' => ['spec/runner/formatter/profile_formatter', 'Formatter::ProfileFormatter'],
|
|
22
|
+
'o' => ['spec/runner/formatter/profile_formatter', 'Formatter::ProfileFormatter'],
|
|
23
|
+
'textmate' => ['spec/runner/formatter/text_mate_formatter', 'Formatter::TextMateFormatter']
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
attr_accessor(
|
|
27
|
+
:filename_pattern,
|
|
28
|
+
:backtrace_tweaker,
|
|
29
|
+
:context_lines,
|
|
30
|
+
:diff_format,
|
|
31
|
+
:dry_run,
|
|
32
|
+
:profile,
|
|
33
|
+
:heckle_runner,
|
|
34
|
+
:line_number,
|
|
35
|
+
:loadby,
|
|
36
|
+
:reporter,
|
|
37
|
+
:reverse,
|
|
38
|
+
:timeout,
|
|
39
|
+
:verbose,
|
|
40
|
+
:user_input_for_runner,
|
|
41
|
+
:error_stream,
|
|
42
|
+
:output_stream,
|
|
43
|
+
# TODO: BT - Figure out a better name
|
|
44
|
+
:argv
|
|
45
|
+
)
|
|
46
|
+
attr_reader :colour, :differ_class, :files, :examples, :example_groups
|
|
47
|
+
|
|
48
|
+
def initialize(error_stream, output_stream)
|
|
49
|
+
@error_stream = error_stream
|
|
50
|
+
@output_stream = output_stream
|
|
51
|
+
@filename_pattern = "**/*_spec.rb"
|
|
52
|
+
@backtrace_tweaker = QuietBacktraceTweaker.new
|
|
53
|
+
@examples = []
|
|
54
|
+
@colour = false
|
|
55
|
+
@profile = false
|
|
56
|
+
@dry_run = false
|
|
57
|
+
@reporter = Reporter.new(self)
|
|
58
|
+
@context_lines = 3
|
|
59
|
+
@diff_format = :unified
|
|
60
|
+
@files = []
|
|
61
|
+
@example_groups = []
|
|
62
|
+
@result = nil
|
|
63
|
+
@examples_run = false
|
|
64
|
+
@examples_should_be_run = nil
|
|
65
|
+
@user_input_for_runner = nil
|
|
66
|
+
@after_suite_parts = []
|
|
67
|
+
end
|
|
68
|
+
|
|
69
|
+
def add_example_group(example_group)
|
|
70
|
+
@example_groups << example_group
|
|
71
|
+
end
|
|
72
|
+
|
|
73
|
+
def remove_example_group(example_group)
|
|
74
|
+
@example_groups.delete(example_group)
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
def run_examples
|
|
78
|
+
return true unless examples_should_be_run?
|
|
79
|
+
success = true
|
|
80
|
+
begin
|
|
81
|
+
runner = custom_runner || ExampleGroupRunner.new(self)
|
|
82
|
+
|
|
83
|
+
unless @files_loaded
|
|
84
|
+
runner.load_files(files_to_load)
|
|
85
|
+
@files_loaded = true
|
|
86
|
+
end
|
|
87
|
+
|
|
88
|
+
# TODO - this has to happen after the files get loaded,
|
|
89
|
+
# otherwise the before_suite_parts are not populated
|
|
90
|
+
# from the configuration. There is no spec for this
|
|
91
|
+
# directly, but features/before_and_after_blocks/before_and_after_blocks.story
|
|
92
|
+
# will fail if this happens before the files are loaded.
|
|
93
|
+
before_suite_parts.each do |part|
|
|
94
|
+
part.call
|
|
95
|
+
end
|
|
96
|
+
|
|
97
|
+
if example_groups.empty?
|
|
98
|
+
true
|
|
99
|
+
else
|
|
100
|
+
set_spec_from_line_number if line_number
|
|
101
|
+
success = runner.run
|
|
102
|
+
@examples_run = true
|
|
103
|
+
heckle if heckle_runner
|
|
104
|
+
success
|
|
105
|
+
end
|
|
106
|
+
ensure
|
|
107
|
+
after_suite_parts.each do |part|
|
|
108
|
+
part.call(success)
|
|
109
|
+
end
|
|
110
|
+
end
|
|
111
|
+
end
|
|
112
|
+
|
|
113
|
+
def before_suite_parts
|
|
114
|
+
Spec::Example::BeforeAndAfterHooks.before_suite_parts
|
|
115
|
+
end
|
|
116
|
+
|
|
117
|
+
def after_suite_parts
|
|
118
|
+
Spec::Example::BeforeAndAfterHooks.after_suite_parts
|
|
119
|
+
end
|
|
120
|
+
|
|
121
|
+
def examples_run?
|
|
122
|
+
@examples_run
|
|
123
|
+
end
|
|
124
|
+
|
|
125
|
+
def examples_should_not_be_run
|
|
126
|
+
@examples_should_be_run = false
|
|
127
|
+
end
|
|
128
|
+
|
|
129
|
+
def predicate_matchers
|
|
130
|
+
# TODO - don't like this dependency - perhaps store these in here instead?
|
|
131
|
+
Spec::Runner.configuration.predicate_matchers
|
|
132
|
+
end
|
|
133
|
+
|
|
134
|
+
def mock_framework
|
|
135
|
+
# TODO - don't like this dependency - perhaps store this in here instead?
|
|
136
|
+
Spec::Runner.configuration.mock_framework
|
|
137
|
+
end
|
|
138
|
+
|
|
139
|
+
def colour=(colour)
|
|
140
|
+
@colour = colour
|
|
141
|
+
if @colour && RUBY_PLATFORM =~ /mswin|mingw/ ;\
|
|
142
|
+
begin ;\
|
|
143
|
+
replace_output = @output_stream.equal?($stdout) ;\
|
|
144
|
+
require 'rubygems' ;\
|
|
145
|
+
require 'Win32/Console/ANSI' ;\
|
|
146
|
+
@output_stream = $stdout if replace_output ;\
|
|
147
|
+
rescue LoadError ;\
|
|
148
|
+
warn "You must 'gem install win32console' to use colour on Windows" ;\
|
|
149
|
+
@colour = false ;\
|
|
150
|
+
end
|
|
151
|
+
end
|
|
152
|
+
end
|
|
153
|
+
|
|
154
|
+
def parse_diff(format)
|
|
155
|
+
case format
|
|
156
|
+
when :context, 'context', 'c'
|
|
157
|
+
@diff_format = :context
|
|
158
|
+
default_differ
|
|
159
|
+
when :unified, 'unified', 'u', '', nil
|
|
160
|
+
@diff_format = :unified
|
|
161
|
+
default_differ
|
|
162
|
+
else
|
|
163
|
+
@diff_format = :custom
|
|
164
|
+
self.differ_class = load_class(format, 'differ', '--diff')
|
|
165
|
+
end
|
|
166
|
+
end
|
|
167
|
+
|
|
168
|
+
def parse_example(example)
|
|
169
|
+
if(File.file?(example))
|
|
170
|
+
@examples = [File.open(example).read.split("\n")].flatten
|
|
171
|
+
else
|
|
172
|
+
@examples = [example]
|
|
173
|
+
end
|
|
174
|
+
end
|
|
175
|
+
|
|
176
|
+
def parse_format(format_arg)
|
|
177
|
+
format, where = ClassAndArgumentsParser.parse(format_arg)
|
|
178
|
+
unless where
|
|
179
|
+
raise "When using several --format options only one of them can be without a file" if @out_used
|
|
180
|
+
where = @output_stream
|
|
181
|
+
@out_used = true
|
|
182
|
+
end
|
|
183
|
+
@format_options ||= []
|
|
184
|
+
@format_options << [format, where]
|
|
185
|
+
end
|
|
186
|
+
|
|
187
|
+
def formatters
|
|
188
|
+
@format_options ||= [['progress', @output_stream]]
|
|
189
|
+
@formatters ||= load_formatters(@format_options, EXAMPLE_FORMATTERS)
|
|
190
|
+
end
|
|
191
|
+
|
|
192
|
+
def load_formatters(format_options, formatters)
|
|
193
|
+
format_options.map do |format, where|
|
|
194
|
+
formatter_type = if formatters[format]
|
|
195
|
+
require formatters[format][0]
|
|
196
|
+
eval(formatters[format][1], binding, __FILE__, __LINE__)
|
|
197
|
+
else
|
|
198
|
+
load_class(format, 'formatter', '--format')
|
|
199
|
+
end
|
|
200
|
+
formatter_type.new(self, where)
|
|
201
|
+
end
|
|
202
|
+
end
|
|
203
|
+
|
|
204
|
+
def load_heckle_runner(heckle)
|
|
205
|
+
suffix = [/mswin/, /java/].detect{|p| p =~ RUBY_PLATFORM} ? '_unsupported' : ''
|
|
206
|
+
require "spec/runner/heckle_runner#{suffix}"
|
|
207
|
+
@heckle_runner = HeckleRunner.new(heckle)
|
|
208
|
+
end
|
|
209
|
+
|
|
210
|
+
def number_of_examples
|
|
211
|
+
return examples.size unless examples.empty?
|
|
212
|
+
@example_groups.inject(0) {|sum, group| sum + group.number_of_examples}
|
|
213
|
+
end
|
|
214
|
+
|
|
215
|
+
def files_to_load
|
|
216
|
+
result = []
|
|
217
|
+
sorted_files.each do |file|
|
|
218
|
+
if File.directory?(file)
|
|
219
|
+
filename_pattern.split(",").each do |pattern|
|
|
220
|
+
result += Dir[File.expand_path("#{file}/#{pattern.strip}")]
|
|
221
|
+
end
|
|
222
|
+
elsif File.file?(file)
|
|
223
|
+
result << file
|
|
224
|
+
else
|
|
225
|
+
raise "File or directory not found: #{file}"
|
|
226
|
+
end
|
|
227
|
+
end
|
|
228
|
+
result
|
|
229
|
+
end
|
|
230
|
+
|
|
231
|
+
def dry_run?
|
|
232
|
+
@dry_run == true
|
|
233
|
+
end
|
|
234
|
+
|
|
235
|
+
protected
|
|
236
|
+
def examples_should_be_run?
|
|
237
|
+
return @examples_should_be_run unless @examples_should_be_run.nil?
|
|
238
|
+
@examples_should_be_run = true
|
|
239
|
+
end
|
|
240
|
+
|
|
241
|
+
def differ_class=(klass)
|
|
242
|
+
return unless klass
|
|
243
|
+
@differ_class = klass
|
|
244
|
+
Spec::Expectations.differ = self.differ_class.new(self)
|
|
245
|
+
end
|
|
246
|
+
|
|
247
|
+
def load_class(name, kind, option)
|
|
248
|
+
if name =~ /\A(?:::)?([A-Z]\w*(?:::[A-Z]\w*)*)\z/
|
|
249
|
+
arg = $2 == "" ? nil : $2
|
|
250
|
+
[$1, arg]
|
|
251
|
+
else
|
|
252
|
+
m = "#{name.inspect} is not a valid class name"
|
|
253
|
+
@error_stream.puts m
|
|
254
|
+
raise m
|
|
255
|
+
end
|
|
256
|
+
begin
|
|
257
|
+
eval(name, binding, __FILE__, __LINE__)
|
|
258
|
+
rescue NameError => e
|
|
259
|
+
@error_stream.puts "Couldn't find #{kind} class #{name}"
|
|
260
|
+
@error_stream.puts "Make sure the --require option is specified *before* #{option}"
|
|
261
|
+
if $_spec_spec ; raise e ; else exit(1) ; end
|
|
262
|
+
end
|
|
263
|
+
end
|
|
264
|
+
|
|
265
|
+
def custom_runner
|
|
266
|
+
return nil unless custom_runner?
|
|
267
|
+
klass_name, arg = ClassAndArgumentsParser.parse(user_input_for_runner)
|
|
268
|
+
runner_type = load_class(klass_name, 'example group runner', '--runner')
|
|
269
|
+
return runner_type.new(self, arg)
|
|
270
|
+
end
|
|
271
|
+
|
|
272
|
+
def custom_runner?
|
|
273
|
+
return user_input_for_runner ? true : false
|
|
274
|
+
end
|
|
275
|
+
|
|
276
|
+
def heckle
|
|
277
|
+
heckle_runner = self.heckle_runner
|
|
278
|
+
self.heckle_runner = nil
|
|
279
|
+
heckle_runner.heckle_with
|
|
280
|
+
end
|
|
281
|
+
|
|
282
|
+
def sorted_files
|
|
283
|
+
return sorter ? files.sort(&sorter) : files
|
|
284
|
+
end
|
|
285
|
+
|
|
286
|
+
def sorter
|
|
287
|
+
FILE_SORTERS[loadby]
|
|
288
|
+
end
|
|
289
|
+
|
|
290
|
+
def default_differ
|
|
291
|
+
require 'spec/expectations/differs/default'
|
|
292
|
+
self.differ_class = Spec::Expectations::Differs::Default
|
|
293
|
+
end
|
|
294
|
+
|
|
295
|
+
def set_spec_from_line_number
|
|
296
|
+
if examples.empty?
|
|
297
|
+
if files.length == 1
|
|
298
|
+
if File.directory?(files[0])
|
|
299
|
+
error_stream.puts "You must specify one file, not a directory when using the --line option"
|
|
300
|
+
exit(1) if stderr?
|
|
301
|
+
else
|
|
302
|
+
example = SpecParser.new(self).spec_name_for(files[0], line_number)
|
|
303
|
+
@examples = [example]
|
|
304
|
+
end
|
|
305
|
+
else
|
|
306
|
+
error_stream.puts "Only one file can be specified when using the --line option: #{files.inspect}"
|
|
307
|
+
exit(3) if stderr?
|
|
308
|
+
end
|
|
309
|
+
else
|
|
310
|
+
error_stream.puts "You cannot use both --line and --example"
|
|
311
|
+
exit(4) if stderr?
|
|
312
|
+
end
|
|
313
|
+
end
|
|
314
|
+
|
|
315
|
+
def stderr?
|
|
316
|
+
@error_stream == $stderr
|
|
317
|
+
end
|
|
318
|
+
end
|
|
319
|
+
end
|
|
320
|
+
end
|
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
module Spec
|
|
2
|
+
module Runner
|
|
3
|
+
class Reporter
|
|
4
|
+
attr_reader :options, :example_groups
|
|
5
|
+
|
|
6
|
+
def initialize(options)
|
|
7
|
+
@options = options
|
|
8
|
+
@options.reporter = self
|
|
9
|
+
clear
|
|
10
|
+
end
|
|
11
|
+
|
|
12
|
+
def add_example_group(example_group)
|
|
13
|
+
formatters.each do |f|
|
|
14
|
+
f.add_example_group(example_group)
|
|
15
|
+
end
|
|
16
|
+
example_groups << example_group
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
def example_started(example)
|
|
20
|
+
formatters.each{|f| f.example_started(example)}
|
|
21
|
+
end
|
|
22
|
+
|
|
23
|
+
def example_finished(example, error=nil)
|
|
24
|
+
@examples << example
|
|
25
|
+
|
|
26
|
+
if error.nil?
|
|
27
|
+
example_passed(example)
|
|
28
|
+
elsif Spec::Example::ExamplePendingError === error
|
|
29
|
+
example_pending(example, error.pending_caller, error.message)
|
|
30
|
+
else
|
|
31
|
+
example_failed(example, error)
|
|
32
|
+
end
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
def failure(example, error)
|
|
36
|
+
backtrace_tweaker.tweak_backtrace(error)
|
|
37
|
+
failure = Failure.new(example, error)
|
|
38
|
+
@failures << failure
|
|
39
|
+
formatters.each do |f|
|
|
40
|
+
f.example_failed(example, @failures.length, failure)
|
|
41
|
+
end
|
|
42
|
+
end
|
|
43
|
+
alias_method :example_failed, :failure
|
|
44
|
+
|
|
45
|
+
def start(number_of_examples)
|
|
46
|
+
clear
|
|
47
|
+
@start_time = Time.new
|
|
48
|
+
formatters.each{|f| f.start(number_of_examples)}
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
def end
|
|
52
|
+
@end_time = Time.new
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
# Dumps the summary and returns the total number of failures
|
|
56
|
+
def dump
|
|
57
|
+
formatters.each{|f| f.start_dump}
|
|
58
|
+
dump_pending
|
|
59
|
+
dump_failures
|
|
60
|
+
formatters.each do |f|
|
|
61
|
+
f.dump_summary(duration, @examples.length, @failures.length, @pending_count)
|
|
62
|
+
f.close
|
|
63
|
+
end
|
|
64
|
+
@failures.length
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
private
|
|
68
|
+
|
|
69
|
+
def formatters
|
|
70
|
+
@options.formatters
|
|
71
|
+
end
|
|
72
|
+
|
|
73
|
+
def backtrace_tweaker
|
|
74
|
+
@options.backtrace_tweaker
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
def clear
|
|
78
|
+
@example_groups = []
|
|
79
|
+
@failures = []
|
|
80
|
+
@pending_count = 0
|
|
81
|
+
@examples = []
|
|
82
|
+
@start_time = nil
|
|
83
|
+
@end_time = nil
|
|
84
|
+
end
|
|
85
|
+
|
|
86
|
+
def dump_failures
|
|
87
|
+
return if @failures.empty?
|
|
88
|
+
@failures.inject(1) do |index, failure|
|
|
89
|
+
formatters.each{|f| f.dump_failure(index, failure)}
|
|
90
|
+
index + 1
|
|
91
|
+
end
|
|
92
|
+
end
|
|
93
|
+
|
|
94
|
+
def dump_pending
|
|
95
|
+
formatters.each{|f| f.dump_pending}
|
|
96
|
+
end
|
|
97
|
+
|
|
98
|
+
def duration
|
|
99
|
+
return @end_time - @start_time unless (@end_time.nil? or @start_time.nil?)
|
|
100
|
+
return "0.0"
|
|
101
|
+
end
|
|
102
|
+
|
|
103
|
+
def example_passed(example)
|
|
104
|
+
formatters.each{|f| f.example_passed(example)}
|
|
105
|
+
end
|
|
106
|
+
|
|
107
|
+
EXAMPLE_PENDING_DEPRECATION_WARNING = <<-WARNING
|
|
108
|
+
DEPRECATION NOTICE: RSpec's formatters have changed example_pending
|
|
109
|
+
to accept three arguments instead of just two. Please see the rdoc
|
|
110
|
+
for Spec::Runner::Formatter::BaseFormatter#example_pending
|
|
111
|
+
for more information.
|
|
112
|
+
|
|
113
|
+
Please update any custom formatters to accept the third argument
|
|
114
|
+
to example_pending. Support for example_pending with two arguments
|
|
115
|
+
and this warning message will be removed after the RSpec 1.1.5 release.
|
|
116
|
+
WARNING
|
|
117
|
+
|
|
118
|
+
def example_pending(example, pending_caller, message="Not Yet Implemented")
|
|
119
|
+
@pending_count += 1
|
|
120
|
+
formatters.each do |formatter|
|
|
121
|
+
if formatter_uses_deprecated_example_pending_method?(formatter)
|
|
122
|
+
Kernel.warn EXAMPLE_PENDING_DEPRECATION_WARNING
|
|
123
|
+
formatter.example_pending(example, message)
|
|
124
|
+
else
|
|
125
|
+
formatter.example_pending(example, message, pending_caller)
|
|
126
|
+
end
|
|
127
|
+
end
|
|
128
|
+
end
|
|
129
|
+
|
|
130
|
+
def formatter_uses_deprecated_example_pending_method?(formatter)
|
|
131
|
+
formatter.method(:example_pending).arity == 2
|
|
132
|
+
end
|
|
133
|
+
|
|
134
|
+
class Failure
|
|
135
|
+
attr_reader :example, :exception
|
|
136
|
+
|
|
137
|
+
def initialize(example, exception)
|
|
138
|
+
@example = example
|
|
139
|
+
@exception = exception
|
|
140
|
+
end
|
|
141
|
+
|
|
142
|
+
def header
|
|
143
|
+
if expectation_not_met?
|
|
144
|
+
"'#{example_name}' FAILED"
|
|
145
|
+
elsif pending_fixed?
|
|
146
|
+
"'#{example_name}' FIXED"
|
|
147
|
+
else
|
|
148
|
+
"#{@exception.class.name} in '#{example_name}'"
|
|
149
|
+
end
|
|
150
|
+
end
|
|
151
|
+
|
|
152
|
+
def pending_fixed?
|
|
153
|
+
@exception.is_a?(Spec::Example::PendingExampleFixedError)
|
|
154
|
+
end
|
|
155
|
+
|
|
156
|
+
def expectation_not_met?
|
|
157
|
+
@exception.is_a?(Spec::Expectations::ExpectationNotMetError)
|
|
158
|
+
end
|
|
159
|
+
|
|
160
|
+
protected
|
|
161
|
+
def example_name
|
|
162
|
+
@example.full_description
|
|
163
|
+
end
|
|
164
|
+
end
|
|
165
|
+
end
|
|
166
|
+
end
|
|
167
|
+
end
|