Class: Grift::MockMethod
- Inherits:
-
Object
- Object
- Grift::MockMethod
- Defined in:
- lib/grift/mock_method.rb,
lib/grift/mock_method/mock_executions.rb,
lib/grift/mock_method/mock_executions/mock_arguments.rb
Overview
A mock for a given class and method. This is the core of Grift. Mocking or spying usually returns a MockMethod.
Defined Under Namespace
Classes: MockExecutions
Instance Attribute Summary collapse
-
#klass ⇒ Object
readonly
Returns the value of attribute klass.
-
#method_access ⇒ Object
readonly
Returns the value of attribute method_access.
-
#method_name ⇒ Object
readonly
Returns the value of attribute method_name.
-
#true_method_cached ⇒ Object
readonly
Returns the value of attribute true_method_cached.
Class Method Summary collapse
-
.hash_key(klass, method_name) ⇒ String
Hashes the class and method for tracking mocks.
Instance Method Summary collapse
-
#calls ⇒ Array<Grift::MockMethod::MockExecutions::MockArguments>
Returns an array of the args used in each call to the mocked method This is syntactical sugar equivalent to calling
mock
and thencalls
. -
#count ⇒ Number
Returns the count of mock executions.
-
#empty? ⇒ Boolean
Returns true if there have been no calls to the mock.
-
#initialize(klass, method_name, watch: true) ⇒ Grift::MockMethod
constructor
A new instance of MockMethod.
-
#mock ⇒ Grift::MockMethod::MockExecutions
Gets the data for the mock results and calls for this mock.
-
#mock_clear ⇒ Grift::MockMethod::MockExecutions
Clears the mock execution and calls data for this mock, but keep the method mocked as before.
-
#mock_implementation ⇒ self
Accepts a block and mocks the method to execute that block instead of the original behavior whenever called while mocked.
-
#mock_implementation_n_times(n) ⇒ self
Accepts a number
n
and a block and mocks the method to execute that block instaead of the original behavior the nextn
times the method is called while mocked. -
#mock_implementation_once ⇒ self
Accepts a block and mocks the method to execute that block instead of the original behavior the next time the method is called while mocked.
-
#mock_reset ⇒ Grift::MockMethod::MockExecutions
Clears the mock execution and calls data for this mock, and mocks the method to return
nil
. -
#mock_restore(watch: false) ⇒ Grift::MockMethod::MockExecutions
Clears the mock execution and calls data for this mock, and restores the method to its original behavior.
-
#mock_return_value(return_value = nil) ⇒ self
Accepts a value and mocks the method to return that value instead of executing its original behavior while mocked.
-
#mock_return_value_n_times(n, return_value = nil) ⇒ self
Accepts a value and mocks the method to return that value
n
times instead of executing its original behavior while mocked. -
#mock_return_value_once(return_value = nil) ⇒ self
Accepts a value and mocks the method to return that value once instead of executing its original behavior while mocked.
-
#mock_return_values_in_order(return_values) ⇒ self
Accepts an array of values and mocks the method to return those values in order instead of executing its original behavior while mocked.
-
#results ⇒ Array
Returns an array of the results of each call to the mocked method This is syntactical sugar equivalent to calling
mock
and thenresults
. -
#to_s ⇒ String
String representation of the MockMethod.
Constructor Details
#initialize(klass, method_name, watch: true) ⇒ Grift::MockMethod
A new instance of MockMethod. Should be initialized via Grift.mock or Grift.spy_on
46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 |
# File 'lib/grift/mock_method.rb', line 46 def initialize(klass, method_name, watch: true) if Grift.restricted_method?(klass, method_name) raise(Grift::Error, "Cannot mock restricted method #{method_name} for class #{klass}") end @klass = klass @method_name = method_name @true_method_cached = false @mock_executions = MockExecutions.new @cache_method_name = :"#{CACHE_METHOD_PREFIX}_#{method_name}" # class methods are really instance methods of the singleton class @class_method = klass.singleton_class.method_defined?(method_name, true) || klass.singleton_class.private_method_defined?(method_name, true) @method_access, @inherited = method_access_definition raise(Grift::Error, "Cannot mock unknown method #{method_name} for class #{klass}") unless @method_access if class_instance.method_defined?(@cache_method_name) raise(Grift::Error, "Cannot mock already mocked method #{method_name} for class #{klass}") end watch_method if watch end |
Instance Attribute Details
#klass ⇒ Object (readonly)
Returns the value of attribute klass.
9 10 11 |
# File 'lib/grift/mock_method.rb', line 9 def klass @klass end |
#method_access ⇒ Object (readonly)
Returns the value of attribute method_access.
9 10 11 |
# File 'lib/grift/mock_method.rb', line 9 def method_access @method_access end |
#method_name ⇒ Object (readonly)
Returns the value of attribute method_name.
9 10 11 |
# File 'lib/grift/mock_method.rb', line 9 def method_name @method_name end |
#true_method_cached ⇒ Object (readonly)
Returns the value of attribute true_method_cached.
9 10 11 |
# File 'lib/grift/mock_method.rb', line 9 def true_method_cached @true_method_cached end |
Class Method Details
.hash_key(klass, method_name) ⇒ String
Hashes the class and method for tracking mocks.
26 27 28 |
# File 'lib/grift/mock_method.rb', line 26 def self.hash_key(klass, method_name) "#{klass}##{method_name}" end |
Instance Method Details
#calls ⇒ Array<Grift::MockMethod::MockExecutions::MockArguments>
Returns an array of the args used in each call to the mocked method This is syntactical sugar equivalent to calling mock
and then calls
.
472 473 474 |
# File 'lib/grift/mock_method.rb', line 472 def calls @mock_executions.calls end |
#count ⇒ Number
Returns the count of mock executions. This is syntactical sugar equivalent to calling mock
and then count
.
514 515 516 |
# File 'lib/grift/mock_method.rb', line 514 def count @mock_executions.count end |
#empty? ⇒ Boolean
Returns true if there have been no calls to the mock. This is syntactical sugar equivalent to calling mock
and then empty?
.
493 494 495 |
# File 'lib/grift/mock_method.rb', line 493 def empty? @mock_executions.empty? end |
#mock ⇒ Grift::MockMethod::MockExecutions
Gets the data for the mock results and calls for this mock.
88 89 90 |
# File 'lib/grift/mock_method.rb', line 88 def mock @mock_executions end |
#mock_clear ⇒ Grift::MockMethod::MockExecutions
Clears the mock execution and calls data for this mock, but keep the method mocked as before.
98 99 100 |
# File 'lib/grift/mock_method.rb', line 98 def mock_clear @mock_executions = MockExecutions.new end |
#mock_implementation ⇒ self
Accepts a block and mocks the method to execute that block instead of the original behavior whenever called while mocked.
152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 |
# File 'lib/grift/mock_method.rb', line 152 def mock_implementation(*) raise(Grift::Error, 'Must provide a block for the new implementation') unless block_given? premock_setup mock_executions = @mock_executions # required to access inside class instance block class_instance.remove_method(@method_name) if !@inherited && method_defined? class_instance.define_method @method_name do |*args, **kwargs| return_value = yield(*args, **kwargs) # record the args passed in the call to the method and the result mock_executions.store(args: args, result: return_value) return_value end class_instance.send(@method_access, @method_name) self end |
#mock_implementation_n_times(n) ⇒ self
Accepts a number n
and a block and mocks the method to execute that block instaead of the original behavior the next n
times the method is called while mocked. After the method has been called once, it will return to its original behavior. The method will continue to be watched.
IMPORANT: Calling #mock_clear clears the method call history. If it is called before the nth execution of the mocked method, the method will remain mocked for an additonal n
calls.
252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 |
# File 'lib/grift/mock_method.rb', line 252 def mock_implementation_n_times(n, *) raise(Grift::Error, 'Must provide a block for the new implementation') unless block_given? premock_setup # required to access inside class instance block mock_executions = @mock_executions clean_mock = lambda do unmock_method watch_method end class_instance.remove_method(@method_name) if !@inherited && method_defined? class_instance.define_method @method_name do |*args, **kwargs| return_value = yield(*args, **kwargs) # record the args passed in the call to the method and the result mock_executions.store(args: args, result: return_value) clean_mock.call if mock_executions.count == n return_value end class_instance.send(@method_access, @method_name) self end |
#mock_implementation_once ⇒ self
Accepts a block and mocks the method to execute that block instead of the original behavior the next time the method is called while mocked. After the method has been called once, it will return to its original behavior. The method will continue to be watched.
189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 |
# File 'lib/grift/mock_method.rb', line 189 def mock_implementation_once(*) raise(Grift::Error, 'Must provide a block for the new implementation') unless block_given? premock_setup # required to access inside class instance block mock_executions = @mock_executions clean_mock = lambda do unmock_method watch_method end class_instance.remove_method(@method_name) if !@inherited && method_defined? class_instance.define_method @method_name do |*args, **kwargs| return_value = yield(*args, **kwargs) # record the args passed in the call to the method and the result mock_executions.store(args: args, result: return_value) clean_mock.call return_value end class_instance.send(@method_access, @method_name) self end |
#mock_reset ⇒ Grift::MockMethod::MockExecutions
Clears the mock execution and calls data for this mock, and mocks the method to return nil
.
108 109 110 111 112 |
# File 'lib/grift/mock_method.rb', line 108 def mock_reset executions = mock_clear mock_return_value(nil) executions end |
#mock_restore(watch: false) ⇒ Grift::MockMethod::MockExecutions
Clears the mock execution and calls data for this mock, and restores the method to its original behavior. By default it also stops watching the method. This cleans up the mocking and restores expected behavior.
124 125 126 127 128 129 |
# File 'lib/grift/mock_method.rb', line 124 def mock_restore(watch: false) executions = mock_clear unmock_method if @true_method_cached watch_method if watch executions end |
#mock_return_value(return_value = nil) ⇒ self
Accepts a value and mocks the method to return that value instead of executing its original behavior while mocked.
295 296 297 298 299 300 301 302 303 304 305 306 307 308 |
# File 'lib/grift/mock_method.rb', line 295 def mock_return_value(return_value = nil) premock_setup mock_executions = @mock_executions # required to access inside class instance block class_instance.remove_method(@method_name) if !@inherited && method_defined? class_instance.define_method @method_name do |*args, **kwargs| # record the args passed in the call to the method and the result mock_executions.store(args: args, kwargs: kwargs, result: return_value) return_value end class_instance.send(@method_access, @method_name) self end |
#mock_return_value_n_times(n, return_value = nil) ⇒ self
Accepts a value and mocks the method to return that value n
times instead of executing its original behavior while mocked. After the method has been called n
times, it will return to its original behavior. The method will continue to be watched.
IMPORANT: Calling #mock_clear clears the method call history. If it is called before the nth execution of the mocked method, the method will remain mocked for an additional n
calls.
377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 |
# File 'lib/grift/mock_method.rb', line 377 def mock_return_value_n_times(n, return_value = nil) premock_setup # required to access mock inside class instance block mock_executions = @mock_executions clean_mock = lambda do unmock_method watch_method end class_instance.remove_method(@method_name) if !@inherited && method_defined? class_instance.define_method @method_name do |*args, **kwargs| # record the args passed in the call to the method and the result mock_executions.store(args: args, kwargs: kwargs, result: return_value) clean_mock.call if mock_executions.count == n return_value end class_instance.send(@method_access, @method_name) self end |
#mock_return_value_once(return_value = nil) ⇒ self
Accepts a value and mocks the method to return that value once instead of executing its original behavior while mocked. After the method has been called once, it will return to its original behavior. The method will continue to be watched.
325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 |
# File 'lib/grift/mock_method.rb', line 325 def mock_return_value_once(return_value = nil) premock_setup # required to access mock inside class instance block mock_executions = @mock_executions clean_mock = lambda do unmock_method watch_method end class_instance.remove_method(@method_name) if !@inherited && method_defined? class_instance.define_method @method_name do |*args, **kwargs| # record the args passed in the call to the method and the result mock_executions.store(args: args, kwargs: kwargs, result: return_value) clean_mock.call return_value end class_instance.send(@method_access, @method_name) self end |
#mock_return_values_in_order(return_values) ⇒ self
Accepts an array of values and mocks the method to return those values in order instead of executing its original behavior while mocked. After the method has been called enough times to return each of the values, it will return to its original behavior. The method continue to be watched.
417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 |
# File 'lib/grift/mock_method.rb', line 417 def mock_return_values_in_order(return_values) unless return_values.is_a?(Array) && !return_values.empty? raise(Grift::Error, 'Must provide a non-empty array for the return values') end premock_setup # required to access mock inside class instance block mock_executions = @mock_executions clean_mock = lambda do unmock_method watch_method end return_values_internal = return_values.dup class_instance.remove_method(@method_name) if !@inherited && method_defined? class_instance.define_method @method_name do |*args, **kwargs| # record the args passed in the call to the method and the result return_value = return_values_internal.shift mock_executions.store(args: args, kwargs: kwargs, result: return_value) clean_mock.call if return_values_internal.empty? return_value end class_instance.send(@method_access, @method_name) self end |
#results ⇒ Array
Returns an array of the results of each call to the mocked method This is syntactical sugar equivalent to calling mock
and then results
.
532 533 534 |
# File 'lib/grift/mock_method.rb', line 532 def results @mock_executions.results end |
#to_s ⇒ String
String representation of the MockMethod
454 455 456 |
# File 'lib/grift/mock_method.rb', line 454 def to_s Grift::MockMethod.hash_key(@klass, @method_name) end |