convenient_service 0.4.0 → 0.6.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (127) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +16 -0
  3. data/LICENSE.txt +1 -1
  4. data/README.md +1 -1
  5. data/ROADMAP.md +14 -3
  6. data/Taskfile.yml +30 -0
  7. data/convenient_service.gemspec +4 -2
  8. data/lib/convenient_service/aliases.rb +6 -0
  9. data/lib/convenient_service/common/plugins/assigns_attributes_in_constructor/using_active_model_attribute_assignment/concern.rb +0 -2
  10. data/lib/convenient_service/common/plugins/assigns_attributes_in_constructor/using_dry_initializer/concern.rb +0 -2
  11. data/lib/convenient_service/core/entities/config/entities/method_middlewares/entities/container/commands/resolve_methods_middlewares_callers.rb +1 -24
  12. data/lib/convenient_service/dependencies.rb +7 -0
  13. data/lib/convenient_service/examples/dry/gemfile/dry_service/config.rb +3 -1
  14. data/lib/convenient_service/examples/rails/gemfile/rails_service/config.rb +3 -1
  15. data/lib/convenient_service/examples/standard/request_params/constants.rb +15 -0
  16. data/lib/convenient_service/examples/standard/request_params/entities/description.rb +40 -0
  17. data/lib/convenient_service/examples/standard/request_params/entities/format.rb +40 -0
  18. data/lib/convenient_service/examples/standard/request_params/entities/id.rb +47 -0
  19. data/lib/convenient_service/examples/standard/request_params/entities/logger.rb +21 -0
  20. data/lib/convenient_service/examples/standard/request_params/entities/request.rb +23 -0
  21. data/lib/convenient_service/examples/standard/request_params/entities/source.rb +40 -0
  22. data/lib/convenient_service/examples/standard/request_params/entities/tag.rb +40 -0
  23. data/lib/convenient_service/examples/standard/request_params/entities/title.rb +40 -0
  24. data/lib/convenient_service/examples/standard/request_params/entities.rb +11 -0
  25. data/lib/convenient_service/examples/standard/request_params/services/apply_default_param_values.rb +26 -0
  26. data/lib/convenient_service/examples/standard/request_params/services/cast_params.rb +38 -0
  27. data/lib/convenient_service/examples/standard/request_params/services/extract_params_from_body.rb +70 -0
  28. data/lib/convenient_service/examples/standard/request_params/services/extract_params_from_path.rb +62 -0
  29. data/lib/convenient_service/examples/standard/request_params/services/filter_out_unpermitted_params.rb +26 -0
  30. data/lib/convenient_service/examples/standard/request_params/services/log_request_params.rb +54 -0
  31. data/lib/convenient_service/examples/standard/request_params/services/merge_params.rb +26 -0
  32. data/lib/convenient_service/examples/standard/request_params/services/prepare.rb +65 -0
  33. data/lib/convenient_service/examples/standard/request_params/services/validate_casted_params.rb +94 -0
  34. data/lib/convenient_service/examples/standard/request_params/services/validate_uncasted_params.rb +72 -0
  35. data/lib/convenient_service/examples/standard/request_params/services.rb +13 -0
  36. data/lib/convenient_service/examples/standard/request_params/utils/array/wrap.rb +46 -0
  37. data/lib/convenient_service/examples/standard/request_params/utils/array.rb +21 -0
  38. data/lib/convenient_service/examples/standard/request_params/utils/http/request/parse_body.rb +42 -0
  39. data/lib/convenient_service/examples/standard/request_params/utils/http/request/parse_path.rb +40 -0
  40. data/lib/convenient_service/examples/standard/request_params/utils/http/request.rb +28 -0
  41. data/lib/convenient_service/examples/standard/request_params/utils/http.rb +3 -0
  42. data/lib/convenient_service/examples/standard/request_params/utils/integer/safe_parse.rb +31 -0
  43. data/lib/convenient_service/examples/standard/request_params/utils/integer.rb +25 -0
  44. data/lib/convenient_service/examples/standard/request_params/utils/json/safe_parse.rb +40 -0
  45. data/lib/convenient_service/examples/standard/request_params/utils/json.rb +21 -0
  46. data/lib/convenient_service/examples/standard/request_params/utils/object/blank.rb +34 -0
  47. data/lib/convenient_service/examples/standard/request_params/utils/object/present.rb +31 -0
  48. data/lib/convenient_service/examples/standard/request_params/utils/object.rb +26 -0
  49. data/lib/convenient_service/examples/standard/request_params/utils.rb +7 -0
  50. data/lib/convenient_service/examples/standard/request_params.rb +48 -0
  51. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/commands/generate_printable_method.rb +50 -0
  52. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/commands.rb +3 -0
  53. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/arguments/commands/apply_stub_to_track_delegations.rb +78 -0
  54. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/arguments/commands/generate_printable_arguments.rb +100 -0
  55. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/arguments/commands.rb +4 -0
  56. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/arguments.rb +95 -0
  57. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/base.rb +87 -0
  58. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/return_its_value.rb +129 -0
  59. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/with_any_arguments.rb +37 -0
  60. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/with_concrete_arguments.rb +37 -0
  61. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/sub_matchers/without_arguments.rb +37 -0
  62. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/values/base.rb +41 -0
  63. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/values/with_calling_original.rb +30 -0
  64. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings/values/without_calling_original.rb +30 -0
  65. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings.rb +12 -0
  66. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings_collection/errors.rb +57 -0
  67. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/chainings_collection.rb +171 -0
  68. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities/delegation.rb +79 -0
  69. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher/entities.rb +5 -0
  70. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities/matcher.rb +276 -0
  71. data/lib/convenient_service/rspec/matchers/custom/delegate_to/entities.rb +3 -0
  72. data/lib/convenient_service/rspec/matchers/custom/delegate_to.rb +58 -234
  73. data/lib/convenient_service/rspec/matchers/custom/singleton_prepend_module.rb +79 -0
  74. data/lib/convenient_service/rspec/matchers/custom.rb +1 -0
  75. data/lib/convenient_service/rspec/matchers/singleton_prepend_module.rb +13 -0
  76. data/lib/convenient_service/rspec/matchers.rb +2 -0
  77. data/lib/convenient_service/service/plugins/has_result_steps/concern.rb +25 -0
  78. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/commands/cast_method.rb +3 -0
  79. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/commands/cast_method_caller.rb +8 -1
  80. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/commands/cast_method_direction.rb +3 -0
  81. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/commands/cast_method_key.rb +8 -1
  82. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/commands/cast_method_name.rb +9 -5
  83. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/concern/instance_methods.rb +4 -0
  84. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities/callers/base.rb +4 -0
  85. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities/callers/reassignment/commands/define_method_in_container.rb +74 -0
  86. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities/callers/reassignment/commands.rb +3 -0
  87. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities/callers/reassignment.rb +50 -0
  88. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities/callers.rb +1 -0
  89. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities/values/reassignment.rb +43 -0
  90. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities/values.rb +3 -0
  91. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/entities.rb +1 -0
  92. data/lib/convenient_service/service/plugins/has_result_steps/entities/method/errors.rb +22 -0
  93. data/lib/convenient_service/service/plugins/has_result_steps/entities/step/concern/instance_methods.rb +13 -2
  94. data/lib/convenient_service/support/arguments/null_arguments.rb +28 -0
  95. data/lib/convenient_service/support/arguments.rb +87 -0
  96. data/lib/convenient_service/support/dependency_container/commands/assert_valid_scope.rb +32 -0
  97. data/lib/convenient_service/support/dependency_container/commands/create_methods_module.rb +27 -0
  98. data/lib/convenient_service/support/dependency_container/commands/import_method.rb +101 -0
  99. data/lib/convenient_service/support/dependency_container/commands.rb +5 -0
  100. data/lib/convenient_service/support/dependency_container/constants.rb +17 -0
  101. data/lib/convenient_service/support/dependency_container/entities/method.rb +114 -0
  102. data/lib/convenient_service/support/dependency_container/entities/method_collection.rb +98 -0
  103. data/lib/convenient_service/support/dependency_container/entities/namespace.rb +69 -0
  104. data/lib/convenient_service/support/dependency_container/entities/namespace_collection.rb +94 -0
  105. data/lib/convenient_service/support/dependency_container/entities.rb +6 -0
  106. data/lib/convenient_service/support/dependency_container/errors.rb +74 -0
  107. data/lib/convenient_service/support/dependency_container/export.rb +38 -0
  108. data/lib/convenient_service/support/dependency_container/import.rb +32 -0
  109. data/lib/convenient_service/support/dependency_container.rb +9 -0
  110. data/lib/convenient_service/support/not_passed.rb +7 -0
  111. data/lib/convenient_service/support/version/null_version.rb +7 -0
  112. data/lib/convenient_service/support/version.rb +11 -1
  113. data/lib/convenient_service/support.rb +3 -0
  114. data/lib/convenient_service/utils/module/fetch_own_const.rb +88 -0
  115. data/lib/convenient_service/utils/module/include_module.rb +38 -0
  116. data/lib/convenient_service/utils/module.rb +10 -0
  117. data/lib/convenient_service/utils/object/instance_variable_delete.rb +41 -0
  118. data/lib/convenient_service/utils/object/instance_variable_fetch.rb +4 -0
  119. data/lib/convenient_service/utils/object.rb +9 -0
  120. data/lib/convenient_service/utils/proc/conjunct.rb +1 -1
  121. data/lib/convenient_service/utils/proc/display.rb +43 -0
  122. data/lib/convenient_service/utils/proc.rb +5 -0
  123. data/lib/convenient_service/utils/string/split.rb +41 -0
  124. data/lib/convenient_service/utils/string.rb +5 -0
  125. data/lib/convenient_service/version.rb +1 -1
  126. data/lib/convenient_service.rb +6 -0
  127. metadata +123 -8
@@ -0,0 +1,74 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Service
5
+ module Plugins
6
+ module HasResultSteps
7
+ module Entities
8
+ class Method
9
+ module Entities
10
+ module Callers
11
+ class Reassignment < Callers::Base
12
+ module Commands
13
+ class DefineMethodInContainer < Support::Command
14
+ include Support::Delegate
15
+
16
+ attr_reader :method, :container, :index
17
+
18
+ delegate :key, :name, to: :method
19
+
20
+ def initialize(method:, container:, index:)
21
+ @method = method
22
+ @container = container
23
+ @index = index
24
+ end
25
+
26
+ ##
27
+ # @return [Boolean]
28
+ #
29
+ def call
30
+ ##
31
+ # NOTE: `prepend` is thread-safe.
32
+ #
33
+ container.klass.prepend reassigned_methods
34
+
35
+ return false if Utils::Module.instance_method_defined?(reassigned_methods, method, private: true)
36
+
37
+ <<~RUBY.tap { |code| reassigned_methods.module_eval(code, __FILE__, __LINE__ + 1) }
38
+ def #{name}
39
+ step =
40
+ steps
41
+ .select { |step| step.has_reassignment?(__method__) }
42
+ .select(&:completed?)
43
+ .last
44
+
45
+ return super unless step
46
+
47
+ key = step.reassignment(__method__).key.to_sym
48
+
49
+ step.result.data[key]
50
+ end
51
+ RUBY
52
+
53
+ true
54
+ end
55
+
56
+ private
57
+
58
+ ##
59
+ # @return [Module]
60
+ #
61
+ def reassigned_methods
62
+ @reassigned_methods ||= Utils::Module.fetch_own_const(container.klass, :ReassignedMethods) { ::Module.new }
63
+ end
64
+ end
65
+ end
66
+ end
67
+ end
68
+ end
69
+ end
70
+ end
71
+ end
72
+ end
73
+ end
74
+ end
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "commands/define_method_in_container"
@@ -0,0 +1,50 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "reassignment/commands"
4
+
5
+ module ConvenientService
6
+ module Service
7
+ module Plugins
8
+ module HasResultSteps
9
+ module Entities
10
+ class Method
11
+ module Entities
12
+ module Callers
13
+ class Reassignment < Callers::Base
14
+ def reassignment?(name)
15
+ ##
16
+ # TODO: A better name for `object`. Wrapped object, `target`?
17
+ #
18
+ object.to_sym == name.to_sym
19
+ end
20
+
21
+ ##
22
+ # TODO: Separate `in` and `out` methods?
23
+ #
24
+ def calculate_value(method)
25
+ raise Errors::CallerCanNotCalculateReassignment.new(method: method)
26
+ end
27
+
28
+ def validate_as_input_for_container!(container, method:)
29
+ raise Errors::InputMethodReassignment.new(method: method, container: container)
30
+ end
31
+
32
+ def validate_as_output_for_container!(container, method:)
33
+ ##
34
+ # TODO: Raise when container has two reassignments with same name.
35
+ #
36
+ true
37
+ end
38
+
39
+ def define_output_in_container!(container, index:, method:)
40
+ Commands::DefineMethodInContainer.call(method: method, container: container, index: index)
41
+ end
42
+ end
43
+ end
44
+ end
45
+ end
46
+ end
47
+ end
48
+ end
49
+ end
50
+ end
@@ -4,4 +4,5 @@ require_relative "callers/base"
4
4
  require_relative "callers/alias"
5
5
  require_relative "callers/proc"
6
6
  require_relative "callers/raw"
7
+ require_relative "callers/reassignment"
7
8
  require_relative "callers/usual"
@@ -0,0 +1,43 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Service
5
+ module Plugins
6
+ module HasResultSteps
7
+ module Entities
8
+ class Method
9
+ module Entities
10
+ module Values
11
+ ##
12
+ # TODO: Specs.
13
+ #
14
+ class Reassignment
15
+ include Support::Delegate
16
+
17
+ attr_reader :value
18
+
19
+ delegate :to_s, :to_sym, to: :value
20
+
21
+ ##
22
+ # @param value [String, Symbol] Method name to reassign.
23
+ #
24
+ def initialize(value)
25
+ @value = value
26
+ end
27
+
28
+ def ==(other)
29
+ return unless other.instance_of?(self.class)
30
+
31
+ return false if value != other.value
32
+
33
+ true
34
+ end
35
+ end
36
+ end
37
+ end
38
+ end
39
+ end
40
+ end
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "values/reassignment"
@@ -4,3 +4,4 @@ require_relative "entities/directions"
4
4
  require_relative "entities/callers"
5
5
  require_relative "entities/key"
6
6
  require_relative "entities/name"
7
+ require_relative "entities/values"
@@ -87,6 +87,28 @@ module ConvenientService
87
87
  end
88
88
  end
89
89
 
90
+ class CallerCanNotCalculateReassignment < ::ConvenientService::Error
91
+ def initialize(method:)
92
+ message = <<~TEXT
93
+ Method caller failed to calculate reassignment for `#{method.name}`.
94
+
95
+ Method callers can calculate only `in` methods, while reassignments are always `out` methods.
96
+ TEXT
97
+
98
+ super(message)
99
+ end
100
+ end
101
+
102
+ class InputMethodReassignment < ConvenientService::Error
103
+ def initialize(method:, container:)
104
+ message = <<~TEXT
105
+ Reassignments are not allowed for `in` methods.
106
+ TEXT
107
+
108
+ super(message)
109
+ end
110
+ end
111
+
90
112
  class MethodIsNotInputMethod < ConvenientService::Error
91
113
  def initialize(method:, container:)
92
114
  message = <<~TEXT
@@ -50,10 +50,18 @@ module ConvenientService
50
50
  Utils::Bool.to_bool(organizer)
51
51
  end
52
52
 
53
+ def has_reassignment?(name)
54
+ outputs.any? { |output| output.reassignment?(name) }
55
+ end
56
+
53
57
  def completed?
54
58
  Utils::Bool.to_bool(@completed)
55
59
  end
56
60
 
61
+ def reassignment(name)
62
+ outputs.find { |output| output.reassignment?(name) }
63
+ end
64
+
57
65
  def params
58
66
  @params ||= resolve_params
59
67
  end
@@ -113,14 +121,17 @@ module ConvenientService
113
121
  end
114
122
 
115
123
  ##
116
- # IMPORTANT: `service.result(**input_values)` is the reason, why services should have only kwargs as arguments.
124
+ # @internal
125
+ # IMPORTANT: `service.result(**input_values)` is the reason, why services should have only kwargs as arguments.
117
126
  #
118
127
  def calculate_result
119
128
  assert_has_organizer!
120
129
 
130
+ result = service.result(**input_values)
131
+
121
132
  mark_as_completed!
122
133
 
123
- service.result(**input_values)
134
+ result
124
135
  end
125
136
 
126
137
  def resolve_params
@@ -0,0 +1,28 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Support
5
+ class Arguments
6
+ ##
7
+ # @api private
8
+ #
9
+ class NullArguments < Support::Arguments
10
+ ##
11
+ # @return [void]
12
+ #
13
+ def initialize
14
+ @args = []
15
+ @kwargs = {}
16
+ @block = nil
17
+ end
18
+
19
+ ##
20
+ # @return [Boolean]
21
+ #
22
+ def null_arguments?
23
+ true
24
+ end
25
+ end
26
+ end
27
+ end
28
+ end
@@ -0,0 +1,87 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "arguments/null_arguments"
4
+
5
+ module ConvenientService
6
+ module Support
7
+ class Arguments
8
+ ##
9
+ # @!attribute [r] args
10
+ # @return [Array]
11
+ #
12
+ attr_reader :args
13
+
14
+ ##
15
+ # @!attribute [r] kwargs
16
+ # @return [Hash]
17
+ #
18
+ attr_reader :kwargs
19
+
20
+ ##
21
+ # @!attribute [r] block
22
+ # @return [Proc]
23
+ #
24
+ attr_reader :block
25
+
26
+ ##
27
+ # @param args [Array]
28
+ # @param kwargs [Hash]
29
+ # @param block [Proc]
30
+ # @return [void]
31
+ #
32
+ def initialize(*args, **kwargs, &block)
33
+ @args = args
34
+ @kwargs = kwargs
35
+ @block = block
36
+ end
37
+
38
+ class << self
39
+ ##
40
+ # @return [ConvenientService::Support::Arguments::NullArguments]
41
+ #
42
+ def null_arguments
43
+ @null_arguments ||= Support::Arguments::NullArguments.new
44
+ end
45
+ end
46
+
47
+ ##
48
+ # @return [Boolean]
49
+ #
50
+ def null_arguments?
51
+ false
52
+ end
53
+
54
+ ##
55
+ # @return [Booleam]
56
+ #
57
+ def any?
58
+ return true if args.any?
59
+ return true if kwargs.any?
60
+ return true if block
61
+
62
+ false
63
+ end
64
+
65
+ ##
66
+ # @return [Booleam]
67
+ #
68
+ def none?
69
+ !any?
70
+ end
71
+
72
+ ##
73
+ # @param other [Object]
74
+ # @return [Boolean]
75
+ #
76
+ def ==(other)
77
+ return unless other.instance_of?(self.class)
78
+
79
+ return false if args != other.args
80
+ return false if kwargs != other.kwargs
81
+ return false if block != other.block
82
+
83
+ true
84
+ end
85
+ end
86
+ end
87
+ end
@@ -0,0 +1,32 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Support
5
+ module DependencyContainer
6
+ module Commands
7
+ class AssertValidScope < Support::Command
8
+ ##
9
+ # @!attribute [r] scope
10
+ # @return [Object]
11
+ #
12
+ attr_reader :scope
13
+
14
+ ##
15
+ # @param scope [Object]
16
+ # @return [void]
17
+ #
18
+ def initialize(scope:)
19
+ @scope = scope
20
+ end
21
+
22
+ ##
23
+ # @return [Module]
24
+ #
25
+ def call
26
+ raise Errors::InvalidScope.new(scope: scope) unless Constants::SCOPES.include?(scope)
27
+ end
28
+ end
29
+ end
30
+ end
31
+ end
32
+ end
@@ -0,0 +1,27 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Support
5
+ module DependencyContainer
6
+ module Commands
7
+ class CreateMethodsModule < Support::Command
8
+ ##
9
+ # @return [Module]
10
+ #
11
+ def call
12
+ ::Module.new do
13
+ class << self
14
+ ##
15
+ # @return namespaces [ConvenientService::Support::DependencyContainer::Entities::NamespaceCollection]
16
+ #
17
+ def namespaces
18
+ @namespaces ||= Entities::NamespaceCollection.new
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
24
+ end
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,101 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Support
5
+ module DependencyContainer
6
+ module Commands
7
+ class ImportMethod < Support::Command
8
+ include Support::Delegate
9
+
10
+ ##
11
+ # @!attribute [r] importing_module
12
+ # @return [Module]
13
+ #
14
+ attr_reader :importing_module
15
+
16
+ ##
17
+ # @!attribute [r] exported_method
18
+ # @return [ConvenientService::Support::DependencyContainer::Method]
19
+ #
20
+ attr_reader :exported_method
21
+
22
+ ##
23
+ # @!attribute [r] prepend
24
+ # @return [Boolean]
25
+ #
26
+ attr_reader :prepend
27
+
28
+ ##
29
+ # @!attribute [r] scope
30
+ # @return [Symbol]
31
+ #
32
+ delegate :scope, to: :exported_method
33
+
34
+ ##
35
+ # @param importing_module [Module]
36
+ # @param exported_method [ConvenientService::Support::DependencyContainer::Method]
37
+ # @param prepend [Boolean]
38
+ #
39
+ def initialize(importing_module:, exported_method:, prepend:)
40
+ @importing_module = importing_module
41
+ @exported_method = exported_method
42
+ @prepend = prepend
43
+ end
44
+
45
+ ##
46
+ # @return [ConvenientService::Support::DependencyContainer::Method]
47
+ #
48
+ def call
49
+ import imported_scoped_methods
50
+
51
+ exported_method.define_in_module!(imported_scoped_methods)
52
+ end
53
+
54
+ private
55
+
56
+ ##
57
+ # @return [Module]
58
+ #
59
+ def imported_scoped_methods
60
+ @imported_scoped_methods ||= Utils::Module.fetch_own_const(importing_module, :"Imported#{imported_prefix}#{scoped_prefix}Methods") { Commands::CreateMethodsModule.call }
61
+ end
62
+
63
+ ##
64
+ # @return [String]
65
+ #
66
+ def imported_prefix
67
+ prepend ? "Prepended" : "Included"
68
+ end
69
+
70
+ ##
71
+ # @return [String]
72
+ #
73
+ def scoped_prefix
74
+ case scope
75
+ when Constants::INSTANCE_SCOPE then "Instance"
76
+ when Constants::CLASS_SCOPE then "Class"
77
+ end
78
+ end
79
+
80
+ ##
81
+ # @return [Module, Class]
82
+ #
83
+ def importer
84
+ case scope
85
+ when Constants::INSTANCE_SCOPE then importing_module
86
+ when Constants::CLASS_SCOPE then importing_module.singleton_class
87
+ end
88
+ end
89
+
90
+ ##
91
+ # @param mod [Module]
92
+ # @return [Module]
93
+ #
94
+ def import(mod)
95
+ prepend ? importer.prepend(mod) : importer.include(mod)
96
+ end
97
+ end
98
+ end
99
+ end
100
+ end
101
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "commands/assert_valid_scope"
4
+ require_relative "commands/create_methods_module"
5
+ require_relative "commands/import_method"
@@ -0,0 +1,17 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Support
5
+ module DependencyContainer
6
+ module Constants
7
+ SCOPES = [
8
+ INSTANCE_SCOPE = :instance,
9
+ CLASS_SCOPE = :class
10
+ ]
11
+
12
+ DEFAULT_SCOPE = Constants::INSTANCE_SCOPE
13
+ DEFAULT_PREPEND = false
14
+ end
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,114 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ConvenientService
4
+ module Support
5
+ module DependencyContainer
6
+ module Entities
7
+ class Method
8
+ ##
9
+ # @!attribute [r] full_name
10
+ # @return [String, Symbol]
11
+ #
12
+ attr_reader :full_name
13
+
14
+ ##
15
+ # @!attribute [r] scope
16
+ # @return [:instance, :class]
17
+ #
18
+ attr_reader :scope
19
+
20
+ ##
21
+ # @!attribute [r] body
22
+ # @return [Proc]
23
+ #
24
+ attr_reader :body
25
+
26
+ ##
27
+ # @param full_name [String, Symbol]
28
+ # @param scope [:instance, :class]
29
+ # @param body [Proc]
30
+ # @return [void]
31
+ #
32
+ def initialize(full_name:, scope:, body:)
33
+ @full_name = full_name
34
+ @scope = scope
35
+ @body = body
36
+ end
37
+
38
+ ##
39
+ # @return [String]
40
+ #
41
+ def name
42
+ @name ||= full_name_parts.last
43
+ end
44
+
45
+ ##
46
+ # @return [Array<ConvenientService::Support::DependencyContainer::Entities::Namespace>]
47
+ #
48
+ def namespaces
49
+ @namespaces ||= full_name_parts.slice(0..-2).map { |part| Entities::Namespace.new(name: part) }
50
+ end
51
+
52
+ ##
53
+ # @param mod [Module]
54
+ # @return [ConvenientService::Support::DependencyContainer::Entities::Method]
55
+ #
56
+ def define_in_module!(mod)
57
+ ##
58
+ # NOTE: `innermost_namespace` is just `mod`, when `namespaces` are empty.
59
+ #
60
+ innermost_namespace =
61
+ namespaces.reduce(mod) do |namespace, sub_namespace|
62
+ already_defined_sub_namespace = namespace.namespaces.find_by(name: sub_namespace.name)
63
+
64
+ ##
65
+ # NOTE:
66
+ # - Reuses already defined namespace from previous "imports".
67
+ # - In contrast, same methods are always redefined.
68
+ #
69
+ next already_defined_sub_namespace if already_defined_sub_namespace
70
+
71
+ namespace.namespaces << sub_namespace
72
+
73
+ namespace.define_method(sub_namespace.name) { sub_namespace.body.call }
74
+
75
+ sub_namespace
76
+ end
77
+
78
+ ##
79
+ # NOTE:
80
+ # - Same methods are redefined.
81
+ # - In contrast, same namespaces are always reused.
82
+ #
83
+ innermost_namespace.define_method(name, &body)
84
+
85
+ self
86
+ end
87
+
88
+ ##
89
+ # @param other [Object] Can be any type.
90
+ # @return [Boolean]
91
+ #
92
+ def ==(other)
93
+ return unless other.instance_of?(self.class)
94
+
95
+ return false if full_name != other.full_name
96
+ return false if scope != other.scope
97
+ return false if body != other.body
98
+
99
+ true
100
+ end
101
+
102
+ private
103
+
104
+ ##
105
+ # @return [Array<String>]
106
+ #
107
+ def full_name_parts
108
+ @full_name_parts ||= Utils::String.split(full_name, ".", "::").map(&:to_sym)
109
+ end
110
+ end
111
+ end
112
+ end
113
+ end
114
+ end