howitzer 2.1.1 → 2.4.0

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.
Files changed (129) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG.md +44 -1
  3. data/LICENSE +1 -1
  4. data/README.md +19 -17
  5. data/generators/base_generator.rb +5 -7
  6. data/generators/config/config_generator.rb +3 -4
  7. data/generators/config/templates/boot.rb +2 -2
  8. data/generators/config/templates/capybara.rb +4 -5
  9. data/generators/config/templates/default.yml +21 -11
  10. data/generators/config/templates/drivers/browserstack.rb +31 -4
  11. data/generators/config/templates/drivers/crossbrowsertesting.rb +9 -11
  12. data/generators/config/templates/drivers/headless_chrome.rb +1 -1
  13. data/generators/config/templates/drivers/headless_firefox.rb +23 -0
  14. data/generators/config/templates/drivers/lambdatest.rb +30 -0
  15. data/generators/config/templates/drivers/sauce.rb +30 -2
  16. data/generators/config/templates/drivers/selenium.rb +1 -1
  17. data/generators/config/templates/drivers/selenium_grid.rb +3 -3
  18. data/generators/config/templates/drivers/testingbot.rb +29 -2
  19. data/generators/cucumber/templates/cuke_sniffer.rake +2 -2
  20. data/generators/cucumber/templates/env.rb +8 -0
  21. data/generators/cucumber/templates/hooks.rb +9 -3
  22. data/generators/cucumber/templates/transformers.rb +1 -1
  23. data/generators/prerequisites/templates/factory_bot.rb +2 -1
  24. data/generators/root/root_generator.rb +1 -1
  25. data/generators/root/templates/Gemfile.erb +4 -10
  26. data/generators/rspec/templates/spec_helper.rb +6 -5
  27. data/generators/turnip/templates/spec_helper.rb +6 -5
  28. data/generators/web/templates/example_page.rb +1 -1
  29. data/lib/howitzer/cache.rb +20 -19
  30. data/lib/howitzer/capybara_helpers.rb +59 -14
  31. data/lib/howitzer/email.rb +3 -2
  32. data/lib/howitzer/exceptions.rb +21 -21
  33. data/lib/howitzer/gmail_api/client.rb +13 -4
  34. data/lib/howitzer/log.rb +6 -6
  35. data/lib/howitzer/mail_adapters/gmail.rb +3 -0
  36. data/lib/howitzer/mail_adapters/mailgun.rb +3 -1
  37. data/lib/howitzer/mail_adapters/mailtrap.rb +3 -0
  38. data/lib/howitzer/mailgun_api/client.rb +3 -2
  39. data/lib/howitzer/mailgun_api/connector.rb +1 -0
  40. data/lib/howitzer/mailgun_api/response.rb +1 -2
  41. data/lib/howitzer/mailtrap_api/client.rb +1 -1
  42. data/lib/howitzer/meta/actions.rb +35 -0
  43. data/lib/howitzer/meta/element.rb +40 -0
  44. data/lib/howitzer/meta/entry.rb +62 -0
  45. data/lib/howitzer/meta/iframe.rb +41 -0
  46. data/lib/howitzer/meta/section.rb +30 -0
  47. data/lib/howitzer/meta.rb +11 -0
  48. data/lib/howitzer/utils/string_extensions.rb +6 -2
  49. data/lib/howitzer/version.rb +1 -1
  50. data/lib/howitzer/web/base_section.rb +1 -1
  51. data/lib/howitzer/web/capybara_context_holder.rb +1 -0
  52. data/lib/howitzer/web/capybara_methods_proxy.rb +15 -6
  53. data/lib/howitzer/web/element_dsl.rb +104 -44
  54. data/lib/howitzer/web/iframe_dsl.rb +4 -2
  55. data/lib/howitzer/web/page.rb +14 -14
  56. data/lib/howitzer/web/page_dsl.rb +18 -6
  57. data/lib/howitzer/web/page_validator.rb +27 -26
  58. data/lib/howitzer/web/section.rb +13 -2
  59. data/lib/howitzer/web/section_dsl.rb +65 -30
  60. data/lib/howitzer.rb +40 -0
  61. metadata +31 -157
  62. data/.coveralls.yml +0 -1
  63. data/.gitignore +0 -14
  64. data/.rspec +0 -3
  65. data/.rubocop.yml +0 -51
  66. data/.ruby-gemset +0 -1
  67. data/.travis.yml +0 -7
  68. data/Gemfile +0 -14
  69. data/ISSUE_TEMPLATE.md +0 -16
  70. data/MAINTENANCE.md +0 -32
  71. data/Rakefile +0 -38
  72. data/features/cli_help.feature +0 -31
  73. data/features/cli_new.feature +0 -389
  74. data/features/cli_unknown.feature +0 -17
  75. data/features/cli_update.feature +0 -218
  76. data/features/cli_version.feature +0 -14
  77. data/features/step_definitions/common_steps.rb +0 -34
  78. data/features/support/env.rb +0 -1
  79. data/generators/config/templates/drivers/phantomjs.rb +0 -19
  80. data/generators/config/templates/drivers/poltergeist.rb +0 -11
  81. data/generators/config/templates/drivers/webkit.rb +0 -6
  82. data/generators/root/templates/.gitignore +0 -21
  83. data/generators/root/templates/.rubocop.yml +0 -35
  84. data/generators/turnip/templates/.rspec +0 -1
  85. data/howitzer.gemspec +0 -39
  86. data/lib/howitzer/mail_adapters/debugmail.rb +0 -0
  87. data/spec/config/custom.yml +0 -9
  88. data/spec/spec_helper.rb +0 -73
  89. data/spec/support/generator_helper.rb +0 -21
  90. data/spec/support/logger_helper.rb +0 -13
  91. data/spec/support/shared_examples/capybara_context_holder.rb +0 -33
  92. data/spec/support/shared_examples/capybara_methods_proxy.rb +0 -94
  93. data/spec/support/shared_examples/dynamic_section_methods.rb +0 -35
  94. data/spec/support/shared_examples/element_dsl.rb +0 -242
  95. data/spec/unit/generators/base_generator_spec.rb +0 -283
  96. data/spec/unit/generators/config_generator_spec.rb +0 -59
  97. data/spec/unit/generators/cucumber_generator_spec.rb +0 -62
  98. data/spec/unit/generators/emails_generator_spec.rb +0 -35
  99. data/spec/unit/generators/prerequisites_generator_spec.rb +0 -53
  100. data/spec/unit/generators/root_generator_spec.rb +0 -75
  101. data/spec/unit/generators/rspec_generator_spec.rb +0 -36
  102. data/spec/unit/generators/tasks_generator_spec.rb +0 -31
  103. data/spec/unit/generators/turnip_generator_spec.rb +0 -52
  104. data/spec/unit/generators/web_generator_spec.rb +0 -52
  105. data/spec/unit/lib/cache_spec.rb +0 -85
  106. data/spec/unit/lib/capybara_helpers_spec.rb +0 -697
  107. data/spec/unit/lib/email_spec.rb +0 -186
  108. data/spec/unit/lib/gmail_api/client_spec.rb +0 -26
  109. data/spec/unit/lib/howitzer_spec.rb +0 -69
  110. data/spec/unit/lib/init_spec.rb +0 -2
  111. data/spec/unit/lib/log_spec.rb +0 -122
  112. data/spec/unit/lib/mail_adapters/abstract_spec.rb +0 -62
  113. data/spec/unit/lib/mail_adapters/gmail_spec.rb +0 -128
  114. data/spec/unit/lib/mail_adapters/mailgun_spec.rb +0 -158
  115. data/spec/unit/lib/mail_adapters/mailtrap_spec.rb +0 -130
  116. data/spec/unit/lib/mailgun_api/client_spec.rb +0 -80
  117. data/spec/unit/lib/mailgun_api/connector_spec.rb +0 -54
  118. data/spec/unit/lib/mailgun_api/response_spec.rb +0 -28
  119. data/spec/unit/lib/mailtrap_api/client_spec.rb +0 -67
  120. data/spec/unit/lib/utils/string_extensions_spec.rb +0 -77
  121. data/spec/unit/lib/web/base_section_spec.rb +0 -43
  122. data/spec/unit/lib/web/element_dsl_spec.rb +0 -22
  123. data/spec/unit/lib/web/iframe_dsl_spec.rb +0 -203
  124. data/spec/unit/lib/web/page_dsl_spec.rb +0 -74
  125. data/spec/unit/lib/web/page_spec.rb +0 -378
  126. data/spec/unit/lib/web/page_validator_spec.rb +0 -276
  127. data/spec/unit/lib/web/section_dsl_spec.rb +0 -165
  128. data/spec/unit/lib/web/section_spec.rb +0 -63
  129. data/spec/unit/version_spec.rb +0 -8
@@ -29,10 +29,15 @@ module Howitzer
29
29
  # * `out` method extracts an instance variable from an original context if starts from @.
30
30
  # Otherwise it executes a method from an original context
31
31
 
32
- def method_missing(name, *args, &block)
32
+ def method_missing(name, *args, **options, &block)
33
33
  return super if name =~ /\A(?:be|have)_/
34
- return eval_in_out_context(*args, &block) if name == :out
35
- page_klass.given.send(name, *args, &block)
34
+ return eval_in_out_context(*args, **options, &block) if name == :out
35
+
36
+ if options.present?
37
+ page_klass.given.send(name, *args, **options, &block)
38
+ else
39
+ page_klass.given.send(name, *args, &block)
40
+ end
36
41
  end
37
42
 
38
43
  # Makes proxied methods to be evaludated and returned as a proc
@@ -44,11 +49,17 @@ module Howitzer
44
49
 
45
50
  private
46
51
 
47
- def eval_in_out_context(*args, &block)
52
+ def eval_in_out_context(*args, **options, &block)
48
53
  return nil if args.size.zero?
54
+
49
55
  name = args.shift
50
56
  return get_outer_instance_variable(name) if name.to_s.start_with?('@')
51
- outer_context.send(name, *args, &block)
57
+
58
+ if options.present?
59
+ outer_context.send(name, *args, **options, &block)
60
+ else
61
+ outer_context.send(name, *args, &block)
62
+ end
52
63
  end
53
64
 
54
65
  def get_outer_instance_variable(name)
@@ -58,9 +69,10 @@ module Howitzer
58
69
  attr_accessor :page_klass, :outer_context
59
70
  end
60
71
 
61
- def self.included(base) #:nodoc:
72
+ def self.included(base) # :nodoc:
62
73
  base.extend(ClassMethods)
63
74
  end
75
+
64
76
  # This module holds page dsl class methods
65
77
  module ClassMethods
66
78
  # Allows to execute page methods in context of the page.
@@ -6,7 +6,7 @@ module Howitzer
6
6
  module PageValidator
7
7
  @validations = {}
8
8
 
9
- def self.included(base) #:nodoc:
9
+ def self.included(base) # :nodoc:
10
10
  base.extend(ClassMethods)
11
11
  end
12
12
 
@@ -21,17 +21,19 @@ module Howitzer
21
21
 
22
22
  def check_validations_are_defined!
23
23
  return if self.class.validations.present?
24
+
24
25
  raise Howitzer::NoValidationError, "No any page validation was found for '#{self.class.name}' page"
25
26
  end
26
27
 
27
28
  # This module holds page validation class methods
28
29
  module ClassMethods
29
30
  # Adds validation to validation list for current page
30
- # @param name [Symbol, String] a validation type. Possible values [:url, :element_presence, :title]
31
- # @param value [Symbol, String, Regexp]
31
+ # @param type [Symbol, String] a validation type. Possible values [:url, :element_presence, :title]
32
+ # @param pattern_or_element_name [Symbol, String, Regexp]
32
33
  # For :url and :title validation types must be <b>Regexp</b>
33
- # For :element_presence must be one of element names described for page
34
- # @param additional_value [Object, nil] any value required to pass for a labmda selector
34
+ # For :element_presence must be one of element names described for the page
35
+ # @param args [Array] any arguments required to pass for a lambda selector (:element_presence type only)
36
+ # @param options [Hash] keyword arguments required to pass for a lambda selector (:element_presence type only)
35
37
  # @raise [Howitzer::UnknownValidationError] if unknown validation type
36
38
  # @raise [Howitzer::UndefinedElementError] if :element_presence validations refers to undefined element name
37
39
  # @example
@@ -44,12 +46,23 @@ module Howitzer
44
46
  # end
45
47
  # @example
46
48
  # class HomePage < Howitzer::Web::Page
47
- # validate :element_presence, :menu_item, 'Logout'
48
- # element :menu_item, :xpath, ->(name) { ".//a[.='#{name}']" }
49
+ # validate :element_presence, :menu_item, lambda_args(text: 'Logout')
50
+ # element :menu_item, :xpath, ->(text:) { ".//a[.='#{text}']" }
49
51
  # end
50
52
 
51
- def validate(name, value, additional_value = nil)
52
- validate_by_type(name, value, additional_value)
53
+ def validate(type, pattern_or_element_name, *args, **options)
54
+ case type.to_s.to_sym
55
+ when :url, :title
56
+ if args.present? || options.present?
57
+ raise ArgumentError, "Additional arguments and options are not supported by '#{type}' the validator"
58
+ end
59
+
60
+ send("validate_by_#{type}", pattern_or_element_name)
61
+ when :element_presence
62
+ validate_by_element_presence(pattern_or_element_name, *args, **options)
63
+ else
64
+ raise Howitzer::UnknownValidationError, "unknown '#{type}' validation type"
65
+ end
53
66
  end
54
67
 
55
68
  # Check whether current page is opened or no
@@ -60,6 +73,7 @@ module Howitzer
60
73
 
61
74
  def opened?(sync: true)
62
75
  return validations.all? { |(_, validation)| validation.call(self, sync) } if validations.present?
76
+
63
77
  raise Howitzer::NoValidationError, "No any page validation was found for '#{name}' page"
64
78
  end
65
79
 
@@ -78,17 +92,17 @@ module Howitzer
78
92
 
79
93
  private
80
94
 
81
- def validate_element(element_name, value = nil)
95
+ def validate_by_element_presence(element_name, *args, **options)
82
96
  validations[:element_presence] =
83
97
  lambda do |web_page, sync|
84
98
  if element_name.present? && !private_method_defined?("#{element_name}_element")
85
99
  raise(Howitzer::UndefinedElementError, ':element_presence validation refers to ' \
86
- "undefined '#{element_name}' element on '#{name}' page.")
100
+ "undefined '#{element_name}' element on '#{name}' page.")
87
101
  end
88
102
  if sync
89
- web_page.instance.public_send(*["has_#{element_name}_element?", value].compact)
103
+ web_page.instance.public_send("has_#{element_name}_element?", *args, **options)
90
104
  else
91
- !web_page.instance.public_send(*["has_no_#{element_name}_element?", value].compact)
105
+ !web_page.instance.public_send("has_no_#{element_name}_element?", *args, **options)
92
106
  end
93
107
  end
94
108
  end
@@ -102,19 +116,6 @@ module Howitzer
102
116
  validations[:title] =
103
117
  ->(web_page, sync) { sync ? web_page.instance.has_title?(pattern) : pattern === web_page.instance.title }
104
118
  end
105
-
106
- def validate_by_type(type, value, additional_value)
107
- case type.to_s.to_sym
108
- when :url
109
- validate_by_url(value)
110
- when :element_presence
111
- validate_element(value, additional_value)
112
- when :title
113
- validate_by_title(value)
114
- else
115
- raise Howitzer::UnknownValidationError, "unknown '#{type}' validation type"
116
- end
117
- end
118
119
  end
119
120
  end
120
121
  end
@@ -1,25 +1,36 @@
1
1
  require 'howitzer/web/base_section'
2
+ require 'howitzer/meta'
2
3
 
3
4
  module Howitzer
4
5
  module Web
5
6
  # This class uses for named sections which possible to reuse in different pages
6
7
  class Section < BaseSection
8
+ # Provides access to meta information about entities in section
9
+ # @return [Meta::Entry]
10
+ def meta
11
+ @meta ||= Meta::Entry.new(self)
12
+ end
13
+
7
14
  class << self
8
15
  protected
9
16
 
10
17
  # DSL method which specifies section container selector represented by HTML element.
11
18
  # Any elements described in sections will start in this HTML element.
12
19
  # @param args [Array] original Capybara arguments. For details, see `Capybara::Node::Finders#all.
20
+ # @param options [Array] original Capybara options. For details, see `Capybara::Node::Finders#all.
13
21
  # @raise [ArgumentError] if no arguments were passed
14
22
  # @example
15
23
  # class MenuSection < Howitzer::Web::Section
16
- # me :xpath, ".//*[@id='panel']"
24
+ # me :xpath, ".//*[@id='panel']",
17
25
  # end
18
26
  # @!visibility public
19
27
 
20
- def me(*args)
28
+ def me(*args, **options)
21
29
  raise ArgumentError, 'Finder arguments are missing' if args.blank?
30
+
22
31
  @default_finder_args = args
32
+ @default_finder_options = options
33
+ self
23
34
  end
24
35
  end
25
36
  end
@@ -5,7 +5,7 @@ module Howitzer
5
5
  module SectionDsl
6
6
  include CapybaraContextHolder
7
7
 
8
- def self.included(base) #:nodoc:
8
+ def self.included(base) # :nodoc:
9
9
  base.extend(ClassMethods)
10
10
  end
11
11
 
@@ -17,8 +17,9 @@ module Howitzer
17
17
 
18
18
  # Instantiates an anynomous or named section and executes block code in the section scope
19
19
 
20
- def initialize(name, *args, &block)
20
+ def initialize(name, *args, **options, &block)
21
21
  @args = args
22
+ @options = options
22
23
  self.section_class =
23
24
  if block
24
25
  Class.new(Howitzer::Web::BaseSection)
@@ -29,29 +30,40 @@ module Howitzer
29
30
  end
30
31
 
31
32
  # # Defines an element on the section
32
- # # @see Howitzer::PageDsl::ClassMethods#element
33
+ # # @see Howitzer::Web::ElementDsl::ClassMethods#element
33
34
 
34
- def element(*args)
35
- section_class.send(:element, *args)
35
+ def element(*args, **options)
36
+ section_class.send(:element, *args, **options)
36
37
  end
37
38
 
38
39
  # Delegates a section describing to the section class
39
40
 
40
- def section(name, *args, &block)
41
- section_class.send(:section, name, *args, &block)
41
+ def section(name, *args, **options, &block)
42
+ section_class.send(:section, name, *args, **options, &block)
42
43
  end
43
44
 
44
- # Returns a selector for the section.
45
+ # Returns selector arguments for the section.
45
46
  # @note If anonymous section uses, then inline selector should be specified.
46
47
  # Otherwise arguments should be defined with `.me` dsl method in named section
47
48
  # @return [Array]
48
49
  # @raise [ArgumentError] when finder arguments were not specified
49
50
 
50
51
  def finder_args
51
- @finder_args ||= begin
52
- return @args if @args.present?
53
- section_class.default_finder_args || raise(ArgumentError, 'Missing finder arguments')
54
- end
52
+ return @args if @args.present?
53
+
54
+ @finder_args ||= (section_class.default_finder_args || raise(ArgumentError, 'Missing finder arguments'))
55
+ end
56
+
57
+ # Returns selector options for the section.
58
+ # @note If anonymous section uses, then inline selector should be specified.
59
+ # Otherwise arguments should be defined with `.me` dsl method in named section
60
+ # @return [Array]
61
+ # @raise [ArgumentError] when finder arguments were not specified
62
+
63
+ def finder_options
64
+ @options if @args.present? # it is ok to have blank options, so we rely here on the argments
65
+
66
+ @finder_options ||= (section_class.default_finder_options || {})
55
67
  end
56
68
  end
57
69
 
@@ -74,6 +86,9 @@ module Howitzer
74
86
  # @param args [Array] original Capybara arguments. For details, see `Capybara::Node::Finders#all.
75
87
  # (In most cases should be ommited for named sections because the section selector is specified
76
88
  # with `#me` method. But must be specified for anonymous sections)
89
+ # @param options [Hash] original Capybara options. For details, see `Capybara::Node::Finders#all.
90
+ # (In most cases should be ommited for named sections because the section selector is specified
91
+ # with `#me` method. But may be specified for anonymous sections)
77
92
  # @param block [Proc] this block can contain nested sections and elements
78
93
  # @example Named section
79
94
  # class MenuSection < Howitzer::Web::Section
@@ -109,37 +124,57 @@ module Howitzer
109
124
  # HomePage.on { info_panel_section.edit_info(title: 'Test Title') }
110
125
  # @!visibility public
111
126
 
112
- def section(name, *args, &block)
113
- scope = SectionScope.new(name, *args, &block)
114
- define_section_method(scope.section_class, name, scope.finder_args)
115
- define_sections_method(scope.section_class, name, scope.finder_args)
116
- define_has_section_method(name, scope.finder_args)
117
- define_has_no_section_method(name, scope.finder_args)
127
+ def section(name, *args, **options, &block)
128
+ scope = SectionScope.new(name, *args, **options, &block)
129
+ define_section_method(scope.section_class, name, *scope.finder_args, **scope.finder_options)
130
+ define_sections_method(scope.section_class, name, *scope.finder_args, **scope.finder_options)
131
+ define_has_section_method(name, *scope.finder_args, **scope.finder_options)
132
+ define_has_no_section_method(name, *scope.finder_args, **scope.finder_options)
118
133
  end
119
134
 
120
135
  private
121
136
 
122
- def define_section_method(klass, name, args)
123
- define_method("#{name}_section") do
124
- klass.new(self, capybara_context.find(*args))
137
+ def define_section_method(klass, name, *args, **options)
138
+ define_method("#{name}_section") do |**block_options|
139
+ kwdargs = options.transform_keys(&:to_sym).merge(block_options.transform_keys(&:to_sym))
140
+ if kwdargs.present?
141
+ klass.new(self, capybara_context.find(*args, **kwdargs))
142
+ else
143
+ klass.new(self, capybara_context.find(*args))
144
+ end
125
145
  end
126
146
  end
127
147
 
128
- def define_sections_method(klass, name, args)
129
- define_method("#{name}_sections") do
130
- capybara_context.all(*args).map { |el| klass.new(self, el) }
148
+ def define_sections_method(klass, name, *args, **options)
149
+ define_method("#{name}_sections") do |**block_options|
150
+ kwdargs = options.transform_keys(&:to_sym).merge(block_options.transform_keys(&:to_sym))
151
+ if kwdargs.present?
152
+ capybara_context.all(*args, **kwdargs).map { |el| klass.new(self, el) }
153
+ else
154
+ capybara_context.all(*args).map { |el| klass.new(self, el) }
155
+ end
131
156
  end
132
157
  end
133
158
 
134
- def define_has_section_method(name, args)
135
- define_method("has_#{name}_section?") do
136
- capybara_context.has_selector?(*args)
159
+ def define_has_section_method(name, *args, **options)
160
+ define_method("has_#{name}_section?") do |**block_options|
161
+ kwdargs = options.transform_keys(&:to_sym).merge(block_options.transform_keys(&:to_sym))
162
+ if kwdargs.present?
163
+ capybara_context.has_selector?(*args, **kwdargs)
164
+ else
165
+ capybara_context.has_selector?(*args)
166
+ end
137
167
  end
138
168
  end
139
169
 
140
- def define_has_no_section_method(name, args)
141
- define_method("has_no_#{name}_section?") do
142
- capybara_context.has_no_selector?(*args)
170
+ def define_has_no_section_method(name, *args, **options)
171
+ define_method("has_no_#{name}_section?") do |**block_options|
172
+ kwdargs = options.transform_keys(&:to_sym).merge(block_options.transform_keys(&:to_sym))
173
+ if kwdargs.present?
174
+ capybara_context.has_no_selector?(*args, **kwdarg)
175
+ else
176
+ capybara_context.has_no_selector?(*args)
177
+ end
143
178
  end
144
179
  end
145
180
  end
data/lib/howitzer.rb CHANGED
@@ -29,6 +29,45 @@ module Howitzer
29
29
  ::SexySettings::Base.instance.all['mailgun_idle_timeout']
30
30
  end
31
31
 
32
+ # @return active session name
33
+
34
+ def session_name
35
+ @session_name ||= 'default'
36
+ end
37
+
38
+ # Sets new session name
39
+ #
40
+ # @param name [String] string identifier for the session
41
+ #
42
+ # @example Executing code in another browser
43
+ # Howitzer.session_name = 'browser2'
44
+ # LoginPage.on do
45
+ # expect(title).to eq('Login Page')
46
+ # end
47
+ #
48
+ # # Switching back to main browser
49
+ # Howitzer.session_name = 'default'
50
+
51
+ def session_name=(name)
52
+ @session_name = name
53
+ Capybara.session_name = @session_name
54
+ end
55
+
56
+ # Yield a block using a specific session name
57
+ #
58
+ # @param name [String] string identifier for the session
59
+ #
60
+ # @example Opening page in another browser
61
+ # Howitzer.using_session('browser2') do
62
+ # LoginPage.on do
63
+ # expect(title).to eq('Login Page')
64
+ # end
65
+ # end
66
+
67
+ def using_session(name, &block)
68
+ Capybara.using_session(name, &block)
69
+ end
70
+
32
71
  attr_accessor :current_rake_task
33
72
  end
34
73
 
@@ -60,6 +99,7 @@ module Howitzer
60
99
 
61
100
  def self.sexy_setting!(name)
62
101
  return Howitzer.public_send(name) if Howitzer.respond_to?(name)
102
+
63
103
  raise UndefinedSexySettingError,
64
104
  "Undefined '#{name}' setting. Please add the setting to config/default.yml:\n #{name}: some_value\n"
65
105
  end