Skip to content

Commit b0eccc3

Browse files
committed
! Fix keyword arguments contract when used with optional positional arguments
1 parent 8be37e8 commit b0eccc3

File tree

3 files changed

+70
-27
lines changed

3 files changed

+70
-27
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
Feature: KeywordArgs when used with optional positional arguments
2+
3+
Checks that the argument is an options hash, and all required keyword arguments are present, and all values pass their respective contracts
4+
5+
```ruby
6+
Contract Any, KeywordArgs[:number => Num, :description => Optional[String]] => Any
7+
```
8+
9+
Background:
10+
Given a file named "keyword_args_with_optional_positional_args_usage.rb" with:
11+
"""ruby
12+
require "contracts"
13+
C = Contracts
14+
15+
class Example
16+
include Contracts::Core
17+
18+
Contract C::Any, String, C::KeywordArgs[b: C::Optional[String]] => Symbol
19+
def foo(output, a = 'a', b: 'b')
20+
p [a, b]
21+
output
22+
end
23+
end
24+
"""
25+
26+
Scenario: Accepts arguments when only require arguments filled and valid
27+
Given a file named "accepts_all_filled_valid_args.rb" with:
28+
"""ruby
29+
require "./keyword_args_with_optional_positional_args_usage"
30+
puts Example.new.foo(:output)
31+
"""
32+
When I run `ruby accepts_all_filled_valid_args.rb`
33+
Then output should contain:
34+
"""
35+
["a", "b"]
36+
output
37+
"""
38+
39+
Scenario: Accepts arguments when all filled and valid
40+
Given a file named "accepts_all_filled_valid_args.rb" with:
41+
"""ruby
42+
require "./keyword_args_with_optional_positional_args_usage"
43+
puts Example.new.foo(:output, 'c', b: 'd')
44+
"""
45+
When I run `ruby accepts_all_filled_valid_args.rb`
46+
Then output should contain:
47+
"""
48+
["c", "d"]
49+
output
50+
"""

lib/contracts.rb

+6-24
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,7 @@ class Contract < Contracts::Decorator
5252
end
5353
end
5454

55-
attr_reader :args_contracts, :ret_contract, :klass, :method
55+
attr_reader :args_contracts, :kargs_contract, :ret_contract, :klass, :method
5656

5757
def initialize(klass, method, *contracts)
5858
super(klass, method)
@@ -69,13 +69,18 @@ def initialize(klass, method, *contracts)
6969

7070
# internally we just convert that return value syntax back to an array
7171
@args_contracts = contracts[0, contracts.size - 1] + contracts[-1].keys
72+
# Extract contract for keyword arguments
73+
@kargs_contract = args_contracts.find { |c| c.is_a?(Contracts::Builtin::KeywordArgs) }
74+
args_contracts.delete(kargs_contract) if kargs_contract
7275

7376
@ret_contract = contracts[-1].values[0]
7477

7578
@args_validators = args_contracts.map do |contract|
7679
Contract.make_validator(contract)
7780
end
7881

82+
@kargs_validator = kargs_contract ? Contract.make_validator(kargs_contract) : nil
83+
7984
@args_contract_index = args_contracts.index do |contract|
8085
contract.is_a? Contracts::Args
8186
end
@@ -93,16 +98,6 @@ def initialize(klass, method, *contracts)
9398

9499
# ====
95100

96-
# == @has_options_contract
97-
last_contract = args_contracts.last
98-
penultimate_contract = args_contracts[-2]
99-
@has_options_contract = if @has_proc_contract
100-
penultimate_contract.is_a?(Contracts::Builtin::KeywordArgs)
101-
else
102-
last_contract.is_a?(Contracts::Builtin::KeywordArgs)
103-
end
104-
# ===
105-
106101
@klass, @method = klass, method
107102
end
108103

@@ -255,19 +250,6 @@ def maybe_append_block! args, blk
255250
true
256251
end
257252

258-
# Same thing for when we have named params but didn't pass any in.
259-
# returns true if it appended nil
260-
def maybe_append_options! args, kargs, blk
261-
return false unless @has_options_contract
262-
263-
if @has_proc_contract && args_contracts[-2].is_a?(Contracts::Builtin::KeywordArgs)
264-
args.insert(-2, kargs)
265-
elsif args_contracts[-1].is_a?(Contracts::Builtin::KeywordArgs)
266-
args << kargs
267-
end
268-
true
269-
end
270-
271253
# Used to determine type of failure exception this contract should raise in case of failure
272254
def failure_exception
273255
if pattern_match?

lib/contracts/call_with.rb

+14-3
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,20 @@ def call_with_inner(returns, this, *args, **kargs, &blk)
1212
# Explicitly append blk=nil if nil != Proc contract violation anticipated
1313
nil_block_appended = maybe_append_block!(args, blk)
1414

15-
# Explicitly append options={} if Hash contract is present
16-
kargs_appended = maybe_append_options!(args, kargs, blk)
15+
if @kargs_validator && !@kargs_validator[kargs]
16+
data = {
17+
arg: kargs,
18+
contract: kargs_contract,
19+
class: klass,
20+
method: method,
21+
contracts: self,
22+
arg_pos: :keyword,
23+
total_args: args.size,
24+
return_value: false,
25+
}
26+
return ParamContractError.new("as return value", data) if returns
27+
return unless Contract.failure_callback(data)
28+
end
1729

1830
# Loop forward validating the arguments up to the splat (if there is one)
1931
(@args_contract_index || args.size).times do |i|
@@ -84,7 +96,6 @@ def call_with_inner(returns, this, *args, **kargs, &blk)
8496
# If we put the block into args for validating, restore the args
8597
# OR if we added a fake nil at the end because a block wasn't passed in.
8698
args.slice!(-1) if blk || nil_block_appended
87-
args.slice!(-1) if kargs_appended
8899
result = if method.respond_to?(:call)
89100
# proc, block, lambda, etc
90101
method.call(*args, **kargs, &blk)

0 commit comments

Comments
 (0)