-
Notifications
You must be signed in to change notification settings - Fork 10
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
Support Ruby 3.0 kwarg-handling within .delay
API
#7
Changes from 6 commits
b5bb9b7
f0ab855
77c5061
459cea1
0e0f9c4
fddec5e
7c7769c
ce9b1cc
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,8 +1,8 @@ | ||
module Delayed | ||
class PerformableMethod | ||
attr_accessor :object, :method_name, :args | ||
attr_accessor :object, :method_name, :args, :kwargs | ||
|
||
def initialize(object, method_name, args) | ||
def initialize(object, method_name, args, kwargs) | ||
raise NoMethodError, "undefined method `#{method_name}' for #{object.inspect}" unless object.respond_to?(method_name, true) | ||
|
||
if !her_model?(object) && object.respond_to?(:persisted?) && !object.persisted? | ||
|
@@ -11,6 +11,7 @@ def initialize(object, method_name, args) | |
|
||
self.object = object | ||
self.args = args | ||
self.kwargs = kwargs | ||
self.method_name = method_name.to_sym | ||
end | ||
|
||
|
@@ -23,7 +24,13 @@ def display_name | |
end | ||
|
||
def perform | ||
object.send(method_name, *args) if object | ||
return unless object | ||
|
||
if kwargs.nil? || (RUBY_VERSION < '2.7' && kwargs.empty?) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There's a proposed PR in the one of the ancestor repos, but for this PR I took a very different strategy, because we want to both maintain Ruby 2.6 support and support the Ruby 3 case where both positional and keyword arguments are present. Holding onto There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. But I think that this should resolve collectiveidea/delayed_job#1134 and might be worth upstreaming, if someone gets a chance. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You can check if the callee uses kwargs. First get the method object with There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. In this case, I don't think we want it to actually care about the arity of the method it's about to call -- only the arity that was provided to it by the enqueuing call. (For example, if I do So the fix here is really just to prevent |
||
object.send(method_name, *args) | ||
else | ||
object.send(method_name, *args, **kwargs) | ||
end | ||
end | ||
|
||
def method(sym) | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,68 +1,78 @@ | ||
require 'helper' | ||
|
||
class MyMailer < ActionMailer::Base | ||
def signup(email) | ||
mail to: email, subject: 'Delaying Emails', from: 'delayedjob@example.com', body: 'Delaying Emails Body' | ||
end | ||
end | ||
describe Delayed::PerformableMailer do | ||
let(:mailer_class) do | ||
Class.new(ActionMailer::Base) do | ||
cattr_accessor(:emails) { [] } | ||
|
||
describe ActionMailer::Base do | ||
describe 'delay' do | ||
it 'enqueues a PerformableEmail job' do | ||
expect { | ||
job = MyMailer.delay.signup('john@example.com') | ||
expect(job.payload_object.class).to eq(Delayed::PerformableMailer) | ||
expect(job.payload_object.method_name).to eq(:signup) | ||
expect(job.payload_object.args).to eq(['john@example.com']) | ||
}.to change { Delayed::Job.count }.by(1) | ||
def signup(email, beta_tester: false) | ||
mail to: email, subject: "Delaying Emails (beta: #{beta_tester})", from: 'delayedjob@example.com', body: 'Delaying Emails Body' | ||
end | ||
end | ||
end | ||
|
||
describe 'delay on a mail object' do | ||
it 'raises an exception' do | ||
expect { | ||
MyMailer.signup('john@example.com').delay | ||
}.to raise_error(RuntimeError) | ||
end | ||
before do | ||
stub_const('MyMailer', mailer_class) | ||
end | ||
|
||
describe Delayed::PerformableMailer do | ||
describe 'perform' do | ||
it 'calls the method and #deliver on the mailer' do | ||
email = double('email', deliver: true) | ||
mailer_class = double('MailerClass', signup: email) | ||
mailer = described_class.new(mailer_class, :signup, ['john@example.com']) | ||
describe 'perform' do | ||
it 'calls the method and #deliver on the mailer' do | ||
mailer = MyMailer.new | ||
email = double('email', deliver: true) | ||
allow(mailer).to receive(:mail).and_return(email) | ||
mailer_job = described_class.new(mailer, :signup, ['john@example.com'], {}) | ||
|
||
expect(mailer_class).to receive(:signup).with('john@example.com') | ||
expect(email).to receive(:deliver) | ||
mailer.perform | ||
end | ||
expect(email).to receive(:deliver) | ||
mailer_job.perform | ||
end | ||
end | ||
end | ||
|
||
if defined?(ActionMailer::Parameterized::Mailer) | ||
describe ActionMailer::Parameterized::Mailer do | ||
describe ActionMailer::Base do | ||
describe 'delay' do | ||
it 'enqueues a PerformableEmail job' do | ||
expect { | ||
job = MyMailer.with(foo: 1, bar: 2).delay.signup('john@example.com') | ||
job = MyMailer.delay.signup('john@example.com', beta_tester: true) | ||
expect(job.payload_object.class).to eq(Delayed::PerformableMailer) | ||
expect(job.payload_object.object.class).to eq(described_class) | ||
expect(job.payload_object.object.instance_variable_get('@mailer')).to eq(MyMailer) | ||
expect(job.payload_object.object.instance_variable_get('@params')).to eq(foo: 1, bar: 2) | ||
expect(job.payload_object.method_name).to eq(:signup) | ||
expect(job.payload_object.args).to eq(['john@example.com']) | ||
expect(job.payload_object.kwargs).to eq(beta_tester: true) | ||
}.to change { Delayed::Job.count }.by(1) | ||
end | ||
end | ||
|
||
describe 'delay on a mail object' do | ||
it 'raises an exception' do | ||
expect { | ||
MyMailer.with(foo: 1, bar: 2).signup('john@example.com').delay | ||
MyMailer.signup('john@example.com').delay | ||
}.to raise_error(RuntimeError) | ||
end | ||
end | ||
end | ||
|
||
if defined?(ActionMailer::Parameterized::Mailer) | ||
describe ActionMailer::Parameterized::Mailer do | ||
describe 'delay' do | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I didn't actually add this section -- I just reorganized the file a little. |
||
it 'enqueues a PerformableEmail job' do | ||
expect { | ||
job = MyMailer.with(foo: 1, bar: 2).delay.signup('john@example.com', beta_tester: false) | ||
expect(job.payload_object.class).to eq(Delayed::PerformableMailer) | ||
expect(job.payload_object.object.class).to eq(described_class) | ||
expect(job.payload_object.object.instance_variable_get('@mailer')).to eq(MyMailer) | ||
expect(job.payload_object.object.instance_variable_get('@params')).to eq(foo: 1, bar: 2) | ||
expect(job.payload_object.method_name).to eq(:signup) | ||
expect(job.payload_object.args).to eq(['john@example.com']) | ||
expect(job.payload_object.kwargs).to eq(beta_tester: false) | ||
}.to change { Delayed::Job.count }.by(1) | ||
end | ||
end | ||
|
||
describe 'delay on a mail object' do | ||
it 'raises an exception' do | ||
expect { | ||
MyMailer.with(foo: 1, bar: 2).signup('john@example.com').delay | ||
}.to raise_error(RuntimeError) | ||
end | ||
end | ||
end | ||
end | ||
end |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Removing these -- they were already deprecated prior to our fork.