# frozen_string_literal: true require 'luna_park/errors' require 'luna_park/tools' require 'luna_park/notifiers/log' LunaPark::Tools.if_gem_installed('bugsnag') { require 'luna_park/notifiers/bugsnag' } require 'luna_park/extensions/attributable' require 'luna_park/extensions/callable' require 'luna_park/extensions/has_errors' module LunaPark module UseCases # The main goal of the use case is a high-level description # of the business process. This specific implementation # is based on the ideas of Ivar Jacobson from his article # Ivar Jacobson: Use Case 2.0. # # @example Create new user # module Errors # # To catch the errors, it's should be the error must # # be inherited from the class LunaPark::Errors::Business # class UserAlreadyExists < LunaPark::Errors::Business # message 'Sorry user with this email already created' # notify: :info # end # end # # class CreateUser < Scenario # attr_accessor :email, :password # # def perform # user = Entities::User.new # user.email = email # user.password = Service::Encode.call(password) # # DB.transaction do # raise Errors::UserAlreadyExists if Repo::Users.exists?(user) # Repo::Users.create(user) # end # end # end class Scenario include Extensions::Attributable extend Extensions::Callable include Extensions::HasErrors DEFAULT_NOTIFIER = Notifiers::Log.new private_constant :DEFAULT_NOTIFIER INIT = :initialized SUCCESS = :success FAIL = :fail private_constant :INIT, :SUCCESS, :FAIL # What status is the process of doing the work under the scenario. # It can be :initialized, :success, :failure # # @example when work just started # scenario = Scenario.new # scenario.state # => :initialized # # @example on fail # scenario.call # Something went wrong # scenario.state # => :failure # # @example on success # scenario.call # scenario.state # => :success attr_reader :state # If a failure occurs during the scenario, then this attribute will contain this error # Else it's nil. # # @example when work just started # scenario = Scenario.new # scenario.fail # => nil # # @example on fail # class Fail < Errors::Business; end # class FailScenario < Scenario # def perform # raise Fail # :result # end # end # # scenario = FailScenario.new # scenario.call # Something went wrong # scenario.fail # => # # # @example on success # scenario.call # scenario.fail # => nil attr_reader :failure # The result obtained during the execution of the scenario. # It's nil on failure scenario. # # @example when work just started # scenario = Scenario.new # scenario.data # => nil # # @example on fail # scenario.call # Something went wrong # scenario.data # => nil # # @example on success # class SuccessScenario < Scenario # def perform # :result # end # end # # scenario = SuccessScenario.new # scenario.call # scenario.data # => :result attr_reader :data # Current locale attr_reader :locale # Initialize new scenario # # @param notifier - custom notifier for the current instance of scenario # @param locale - custom locale for the current instance of scenario # @param attrs - the parameters that are needed to implement the scenario, usually the request model # # @example without parameters # class SayHello < Scenario # attr_accessor :first_name, :last_name # # def perform # t('hello_my_nme_is', first_name: first_name, last_name: last_name) # end # end # # hello = Scenario.new first_name: 'John', last_name: 'Doe' # hello.notifier # => Notifiers::Log # hello.locale # => nil # hello.first_name # => 'John' # hello.last_name # => 'Doe' # hello.call! # => 'Hello my name is John Doe' # # @example with custom parameters # hello = Scenario.new first_name: 'John', last_name: 'Doe', notifier: Notifier::Bugsnag, locale: :ru # hello.notifier # => Notifiers::Bugsnag # hello.locale # => :ru # hello.first_name # => 'John' # hello.last_name # => 'Doe' # hello.call! # => 'Добрый день, меня зовут John Doe' def initialize(notifier: nil, locale: nil, **attrs) set_attributes attrs @data = nil @failure = nil @locale = locale @notifier = notifier @state = INIT end def call! perform end # You must define this action and describe all business logic here. # When you run this method - it run as is, and does not change scenario instance. # # @abstract # # @example Fail way # class Shot < Scenario # attr_accessor :lucky_mode # # def perform # raise YouDied, 'Always something went wrong' unless lucky_mode # 'All good' # end # end # # bad_day = Shot.new lucky_mode: false # bad_day.call! # it raise - SomethingWentWrong: Always something went wrong # bad_day.state # => :initialized # # @example Main way # good_day = Shot.new lucky_mode: true # good_day.call! # => 'All good' # good_day.state # => :initialized # # @example Russian roulette # # `.call!` usually use for "scenario in scenario" # class RussianRoulette < Scenario # def perform # [true, true, true, true, true, false].shuffle do |bullet| # Shot.call! lucky_mode: bullet # end # end # end def perform raise Errors::AbstractMethod end # You must define this action and describe all business logic here. # When you run this method - it run as is, and does not change scenario instance. # # @abstract # # @example fail way # class YouDied < Errors::Business; end # # class Shot < Scenario # attr_accessor :lucky_mode # # def perform # raise YouDied, 'Always something went wrong' unless lucky_mode # 'All good' # end # end # # bad_day = Shot.new lucky_mode: false # bad_day.call # => # # bad_day.success? # => false # bad_day.fail? # => true # bad_day.data # => nil # bad_day.state # => :failure # bad_day.fail # => # # bad_day.fail_message # => '' # # @example main way # # good_day = Shot.new lucky_mode: true # good_day.call! # => 'All good' # good_day.state # => :initialized # # @example Russian roulette # class RussianRoulette < Scenario # def call! # [true, true, true, true, true, false].shuffle do |bullet| # Shot.call! lucky_mode: bullet # end # end # end def call rescue_exception { @data = call! } self end # Return notifier def notifier @notifier ||= self.class.default_notifier end # @return [Boolean] true if the scenario runs unsuccessfully def fail? state == FAIL end alias failure? fail? alias failed? fail? # @return [Boolean] true if the scenario runs successfully def success? state == SUCCESS end alias succeed? success? # @return [String] fail message def failure_message(locale: nil) failure&.message(locale: locale || self.locale) end class << self # @return Default notifier def default_notifier @default_notifier ||= DEFAULT_NOTIFIER end # Set notifier for this class # # @example set notifier # class Foobar < Scenario # notify_with Notifier::Bugsnag # # def perform # true # end # end # # Foobar.default_notifier # => Notifier::Bugsnag # Foobar.new.notifier # => Notifier::Bugsnag def notify_with(notifier) @default_notifier = notifier end end private def rescue_exception yield rescue Errors::Base => e @state = FAIL notify_error e if e.notify? handle_error e else @state = SUCCESS end def notify_error(error) notifier.post error, lvl: error.notify_lvl end def handle_error(error) case error when Errors::Business then on_catch(error) when Errors::System then on_raise(error) else raise ArgumentError, "Unknown error action #{error.class}" end end def on_catch(error) @failure = error end def on_raise(error) raise error end end end end