Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow developers to define #call with arguments for convenience #135

Closed
wants to merge 5 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 3 additions & 1 deletion .rubocop.yml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# This should always correspond to the required Ruby version specified in the
# gemspec.
AllCops:
TargetRubyVersion: 2.0
TargetRubyVersion: 2.1

# TODO: What should we do here?
Style/FrozenStringLiteralComment:
Expand Down Expand Up @@ -43,3 +43,5 @@ Style/StringLiterals:
EnforcedStyle: double_quotes
Style/SymbolArray:
Enabled: false
Style/WordArray:
Enabled: false
2 changes: 0 additions & 2 deletions .travis.yml
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,13 @@ cache: bundler
language: ruby
matrix:
allow_failures:
- rvm: "2.0"
- rvm: ruby-head
notifications:
webhooks:
on_start: always
urls:
- http://buildlight.collectiveidea.com/
rvm:
- "2.0"
- "2.1"
- "2.2"
- "2.3.4"
Expand Down
2 changes: 1 addition & 1 deletion interactor.gemspec
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ Gem::Specification.new do |spec|
spec.files = `git ls-files`.split($INPUT_RECORD_SEPARATOR)
spec.test_files = spec.files.grep(/^spec/)

spec.required_ruby_version = ">= 2.0"
spec.required_ruby_version = ">= 2.1"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A big thank-you for maintaining this value.


spec.add_development_dependency "bundler", "~> 1.9"
spec.add_development_dependency "rake", "~> 10.4"
Expand Down
26 changes: 25 additions & 1 deletion lib/interactor.rb
Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,7 @@ def run
# Raises Interactor::Failure if the context is failed.
def run!
with_hooks do
call
call(*arguments_for_call)
context.called!(self)
end
rescue
Expand All @@ -163,4 +163,28 @@ def call
# Returns nothing.
def rollback
end

private

# Internal: Determine what keyword arguments (if any) should be passed to the
# "call" instance method when invoking an Interactor. The "call" instance
# method may accept any number of keyword arguments. This method will extract
# values from the context in order to populate those arguments based on their
# names.
#
# Returns an Array of arguments to be applied as an argument list.
def arguments_for_call
positional_arguments = []
keyword_arguments = {}

method(:call).parameters.each do |(type, name)|

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i think should be here: self.class.instance_method(:call) instead of method(:call), because for some reason method(:call) returns the parameters from the class method "call", and not from the instance method "call" . I know this because i actually tried this and the only way i could make it work was to use self.class.instance_method .
Or maybe i am missing something.

next unless type == :keyreq || type == :key
next unless context.include?(name)

keyword_arguments[name] = context[name]
end

positional_arguments << keyword_arguments if keyword_arguments.any?
positional_arguments
end
end
11 changes: 10 additions & 1 deletion lib/interactor/context.rb
Original file line number Diff line number Diff line change
Expand Up @@ -121,7 +121,7 @@ def failure?
#
# Raises Interactor::Failure initialized with the Interactor::Context.
def fail!(context = {})
context.each { |key, value| modifiable[key.to_sym] = value }
context.each { |key, value| self[key] = value }
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

@failure = true
raise Failure, self
end
Expand Down Expand Up @@ -158,6 +158,15 @@ def rollback!
@rolled_back = true
end

# Public: Check for the presence of a given key in the context. This does
# not check whether the value is truthy, just whether the key is set to any
# value at all.
#
# Returns true if the key is found or false otherwise.
def include?(key)
table.include?(key.to_sym)
end

# Internal: An Array of successfully called Interactor instances invoked
# against this Interactor::Context instance.
#
Expand Down
20 changes: 20 additions & 0 deletions spec/interactor/context_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -164,6 +164,26 @@ module Interactor
end
end

describe "#include?" do
it "returns true if the key is found" do
context = Context.build(foo: "bar")

expect(context.include?(:foo)).to eq(true)
end

it "returns true if the symbolized key is found" do
context = Context.build(foo: "bar")

expect(context.include?("foo")).to eq(true)
end

it "returns false if the key is not found" do
context = Context.build(foo: "bar")

expect(context.include?(:hello)).to eq(false)
end
end

describe "#_called" do
let(:context) { Context.build }

Expand Down
62 changes: 62 additions & 0 deletions spec/interactor_spec.rb
Original file line number Diff line number Diff line change
@@ -1,3 +1,65 @@
describe Interactor do
include_examples :lint

describe "#call" do
let(:interactor) { Class.new.send(:include, described_class) }

context "keyword arguments" do
it "accepts required keyword arguments" do
interactor.class_eval do
def call(foo:)
context.output = foo
end
end

result = interactor.call(foo: "bar", hello: "world")

expect(result.output).to eq("bar")
end

it "accepts optional keyword arguments" do
interactor.class_eval do
def call(foo: "bar")
context.output = foo
end
end

result = interactor.call(foo: "baz", hello: "world")

expect(result.output).to eq("baz")
end

it "assigns absent keyword arguments" do
interactor.class_eval do
def call(foo: "bar")
context.output = foo
end
end

result = interactor.call(hello: "world")

expect(result.output).to eq("bar")
end

it "raises an error for missing keyword arguments" do
interactor.class_eval do
def call(foo:)
context.output = foo
end
end

expect { interactor.call(hello: "world") }.to raise_error(ArgumentError)
end

it "raises an error for call definitions with non-keyword arguments" do
interactor.class_eval do
def call(foo)
context.output = foo
end
end

expect { interactor.call(foo: "bar") }.to raise_error(ArgumentError)
end
end
end
end