diff --git a/Gemfile b/Gemfile index 7a94f15..38fc49f 100644 --- a/Gemfile +++ b/Gemfile @@ -28,5 +28,6 @@ end group :development, :test do gem "activerecord" gem "rom-sql" + gem "sequel" gem "sqlite3", "~> 1.4" end diff --git a/lib/dry/operation/extensions/sequel.rb b/lib/dry/operation/extensions/sequel.rb new file mode 100644 index 0000000..850c3f8 --- /dev/null +++ b/lib/dry/operation/extensions/sequel.rb @@ -0,0 +1,111 @@ +# frozen_string_literal: true + +begin + require "sequel" +rescue LoadError + raise Dry::Operation::MissingDependencyError.new(gem: "sequel", extension: "Sequel") +end + +module Dry + class Operation + module Extensions + # Add Sequel transaction support to operations + # + # When this extension is included, you can use a `#transaction` method + # to wrap the desired steps in a Sequel transaction. If any of the steps + # returns a `Dry::Monads::Result::Failure`, the transaction will be rolled + # back and, as usual, the rest of the flow will be skipped. + # + # The extension expects the including class to give access to the Sequel + # database object via a `#db` method. + # + # ```ruby + # class MyOperation < Dry::Operation + # include Dry::Operation::Extensions::Sequel + # + # attr_reader :db + # + # def initialize(db:) + # @db = db + # end + # + # def call(input) + # attrs = step validate(input) + # user = transaction do + # new_user = step persist(attrs) + # step assign_initial_role(new_user) + # new_user + # end + # step notify(user) + # user + # end + # + # # ... + # end + # ``` + # + # By default, no options are passed to the Sequel transaction. You can + # change this when including the extension: + # + # ```ruby + # include Dry::Operation::Extensions::Sequel[isolation: :serializable] + # ``` + # + # Or you can change it at runtime: + # + # ```ruby + # transaction(isolation: :serializable) do + # # ... + # end + # ``` + # + # WARNING: Be aware that the `:requires_new` option is not yet supported. + # + # @see http://sequel.jeremyevans.net/rdoc/files/doc/transactions_rdoc.html + module Sequel + def self.included(klass) + klass.include(self[]) + end + + # Include the extension providing default options for the transaction. + # + # @param options [Hash] additional options for the Sequel transaction + def self.[](options = {}) + Builder.new(**options) + end + + # @api private + class Builder < Module + def initialize(**options) + super() + @options = options + end + + def included(klass) + class_exec(@options) do |default_options| + klass.define_method(:transaction) do |**opts, &steps| + raise Dry::Operation::ExtensionError, <<~MSG unless respond_to?(:db) + When using the Sequel extension, you need to define a #db method \ + that returns the Sequel database object + MSG + + intercepting_failure do + result = nil + db.transaction(**default_options.merge(opts)) do + intercepting_failure(->(failure) { + result = failure + raise ::Sequel::Rollback + }) do + result = steps.() + end + end + result + end + end + end + end + end + end + end + end +end diff --git a/spec/integration/extensions/active_record_spec.rb b/spec/integration/extensions/active_record_spec.rb index b32e566..3f004f8 100644 --- a/spec/integration/extensions/active_record_spec.rb +++ b/spec/integration/extensions/active_record_spec.rb @@ -156,7 +156,7 @@ def initialize(model) def call transaction do step create_record - transaction(requires_new: true) do + transaction(savepoint: true) do step failure end end diff --git a/spec/integration/extensions/sequel_spec.rb b/spec/integration/extensions/sequel_spec.rb new file mode 100644 index 0000000..9fc6c2d --- /dev/null +++ b/spec/integration/extensions/sequel_spec.rb @@ -0,0 +1,142 @@ +# frozen_string_literal: true + +require "spec_helper" + +RSpec.describe Dry::Operation::Extensions::Sequel do + include Dry::Monads[:result] + + let(:db) do + Sequel.sqlite + end + + before do + db.create_table(:users) do + primary_key :id + String :name + end + end + + after do + db.drop_table(:users) + end + + let(:base) do + Class.new(Dry::Operation) do + include Dry::Operation::Extensions::Sequel + + attr_reader :db + + def initialize(db:) + @db = db + super() + end + end + end + + it "rolls transaction back on failure" do + instance = Class.new(base) do + def call + transaction do + step create_user + step failure + end + end + + def create_user + Success(db[:users].insert(name: "John")) + end + + def failure + Failure(:failure) + end + end.new(db: db) + + instance.() + expect(db[:users].count).to be(0) + end + + it "acts transparently for the regular flow for a success" do + instance = Class.new(base) do + def call + transaction do + step create_user + step count_users + end + end + + def create_user + Success(db[:users].insert(name: "John")) + end + + def count_users + Success(db[:users].count) + end + end.new(db: db) + + expect(instance.()).to eql(Success(1)) + end + + it "acts transparently for the regular flow for a failure" do + instance = Class.new(base) do + def call + transaction do + step create_user + step failure + end + end + + def create_user + Success(db[:users].insert(name: "John")) + end + + def failure + Failure(:failure) + end + end.new(db: db) + + expect(instance.()).to eql(Failure(:failure)) + end + + it "accepts options for Sequel transaction method" do + instance = Class.new(base) do + def call + transaction(isolation: :serializable) do + step create_user + end + end + + def create_user + Success(db[:users].insert(name: "John")) + end + end.new(db: db) + + expect(db).to receive(:transaction).with(isolation: :serializable) + + instance.() + end + + xit "works with `requires_new` for nested transactions" do + instance = Class.new(base) do + def call + transaction do + step create_user + transaction(requires_new: true) do + step failure + end + end + end + + def create_user + Success(db[:users].insert(name: "John")) + end + + def failure + Failure(:failure) + end + end.new(db: db) + + instance.() + + expect(db[:users].count).to be(1) + end +end diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index cac3aca..91d54b8 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -15,6 +15,7 @@ require "dry/operation" require "dry/operation/extensions/active_record" require "dry/operation/extensions/rom" +require "dry/operation/extensions/sequel" SPEC_ROOT = Pathname(__dir__).realpath.freeze diff --git a/spec/unit/extensions/sequel_spec.rb b/spec/unit/extensions/sequel_spec.rb new file mode 100644 index 0000000..8f8dd04 --- /dev/null +++ b/spec/unit/extensions/sequel_spec.rb @@ -0,0 +1,48 @@ +# frozen_string_literal: true + +require "spec_helper" + +RSpec.describe Dry::Operation::Extensions::Sequel do + describe "#transaction" do + it "raises a meaningful error when #db method is not implemented" do + instance = Class.new(Dry::Operation).include(Dry::Operation::Extensions::Sequel).new + + expect { instance.transaction {} }.to raise_error( + Dry::Operation::ExtensionError, + /you need to define a #db method/ + ) + end + + it "forwards options to Sequel transaction call" do + db = double(:db) + instance = Class.new(Dry::Operation) do + include Dry::Operation::Extensions::Sequel + + attr_reader :db + + def initialize(db) + @db = db + end + end.new(db) + + expect(db).to receive(:transaction).with(isolation: :serializable) + instance.transaction(isolation: :serializable) {} + end + + it "merges options with default options" do + db = double(:db) + instance = Class.new(Dry::Operation) do + include Dry::Operation::Extensions::Sequel[savepoint: true] + + attr_reader :db + + def initialize(db) + @db = db + end + end.new(db) + + expect(db).to receive(:transaction).with(savepoint: true, isolation: :serializable) + instance.transaction(isolation: :serializable) {} + end + end +end