opera 0.3.3 → 0.3.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 66df118360280b866b0d104645f8f837d6046536167999f14fb4fdd97bb082a8
4
- data.tar.gz: cfe3b8d4d05f8e28900bf0339a70a5bb648ad26ee92c2c449eccb0ecee492887
3
+ metadata.gz: bc70b6a1f69200a0b37229ed04034c3bcc12c66c94dc9a5d0c066ceaf9971cdb
4
+ data.tar.gz: 7b369d14c001a62b07d91c43efd52e23ddb65749aae9a1a19e3cf49105e190a7
5
5
  SHA512:
6
- metadata.gz: b5bae55e20de7a7759b600050874bba4b14612635bb0616941317898ba0c537828131e24a145626b08c6d8d783ed38059dae8c55d59895dd705174aa385bfaae
7
- data.tar.gz: 2eb1c3540d2a855a9e71046499ef38f7fc9477d4acc76f92a7f0f3148daadb95cb8d4166a1e7d18a151f9a0107e42a4db50e965b232ef3228f0c61fb2dfa1962
6
+ metadata.gz: 8ae2fc0df4e8969215716157838de9ce26b4bfc66007c2c62b57e7a2ffd0b918be8974b7513f28d81c14593b41ecff1d573e4fca2ea4502761b9306abc273b4b
7
+ data.tar.gz: 6abf3cb52cd156ae7e458772842e62b1dbca0e2f93dde5af040e68c8c426c75088ae2ef2f11a164372a6d7535e4a0f6516643a78955a67f5040bf53f81ed6485
data/.gitignore CHANGED
@@ -7,5 +7,8 @@
7
7
  /spec/reports/
8
8
  /tmp/
9
9
 
10
+ #macos
11
+ .DS_Store
12
+
10
13
  # rspec failure tracking
11
14
  .rspec_status
data/CHANGELOG.md CHANGED
@@ -1,5 +1,15 @@
1
1
  # Opera Changelog
2
2
 
3
+ ### 0.3.5 - February 24, 2025
4
+
5
+ - Simplify instrumentation configuration
6
+ - Allow instrumentation logic to be defined at the source
7
+
8
+ ### 0.3.4 - February 19, 2025
9
+
10
+ - Add support for `benchmark` label
11
+ - Add support for instrumentation
12
+
3
13
  ### 0.3.3 - January 15, 2025
4
14
 
5
15
  - Fix issue with handling exceptions in nested operations
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- opera (0.3.3)
4
+ opera (0.3.5)
5
5
 
6
6
  GEM
7
7
  remote: https://rubygems.org/
data/README.md CHANGED
@@ -37,7 +37,10 @@ Simply initialise the configuration and chose what method you want to use to rep
37
37
  Opera::Operation::Config.configure do |config|
38
38
  config.transaction_class = ActiveRecord::Base
39
39
  config.transaction_method = :transaction
40
- config.transaction_options = { requires_new: true }
40
+ config.transaction_options = { requires_new: true, level: :step } # or level: :operation - default
41
+ config.instrumentation_class = Datadog::Tracing
42
+ config.instrumentation_method = :trace
43
+ config.instrumentation_options = { service: :operation }
41
44
  config.mode = :development # Can be set to production too
42
45
  config.reporter = defined?(Rollbar) ? Rollbar : Rails.logger
43
46
  end
@@ -52,7 +55,6 @@ Once opera gem is in your project you can start to build Operations
52
55
 
53
56
  ```ruby
54
57
  class A < Opera::Operation::Base
55
-
56
58
  configure do |config|
57
59
  config.transaction_class = Profile
58
60
  config.reporter = Rails.logger
@@ -96,18 +98,44 @@ Start developing your business logic, services and interactions as Opera::Operat
96
98
  When using Opera::Operation inside an engine add the following
97
99
  configuration to your spec_helper.rb or rails_helper.rb:
98
100
 
99
- ```
101
+ ```ruby
100
102
  Opera::Operation::Config.configure do |config|
101
103
  config.transaction_class = ActiveRecord::Base
102
104
  end
103
105
  ```
104
106
 
105
107
  Without this extra configuration you will receive:
106
- ```
108
+ ```ruby
107
109
  NoMethodError:
108
110
  undefined method `transaction' for nil:NilClass
109
111
  ```
110
112
 
113
+ ### Instrumentation
114
+
115
+ When you want to easily instrument your operations you can add this to the opera config:
116
+
117
+ ```ruby
118
+ Rails.application.configure do
119
+ config.x.instrumentation_class = Datadog::Tracing
120
+ config.x.instrumentation_method = :trace
121
+ config.x.instrumentation_options = { service: :opera }
122
+ end
123
+ ```
124
+
125
+ You can also instrument individual operations by adding this to the operation config:
126
+
127
+ ```ruby
128
+ class A < Opera::Operation::Base
129
+ configure do |config|
130
+ config.instrumentation_class = Datadog::Tracing
131
+ config.instrumentation_method = :trace
132
+ config.instrumentation_options = { service: :opera, level: :step }
133
+ end
134
+
135
+ # steps
136
+ end
137
+ ```
138
+
111
139
  ### Debugging
112
140
 
113
141
  When you want to easily debug exceptions you can add this
@@ -674,10 +702,12 @@ class Profile::Create < Opera::Operation::Base
674
702
 
675
703
  validate :profile_schema
676
704
 
677
- step :create
678
- step :update
705
+ benchmark :fast_section do
706
+ step :create
707
+ step :update
708
+ end
679
709
 
680
- benchmark do
710
+ benchmark :slow_section do
681
711
  step :send_email
682
712
  step :output
683
713
  end
@@ -717,7 +747,7 @@ Profile::Create.call(params: {
717
747
  }, dependencies: {
718
748
  current_account: Account.find(1)
719
749
  })
720
- #<Opera::Operation::Result:0x007ff414a01238 @errors={}, @exceptions={}, @information={:real=>1.800013706088066e-05, :total=>0.0}, @executions=[:profile_schema, :create, :update, :send_email, :output], @output={:model=>#<Profile id: 30, user_id: nil, linkedin_uid: nil, picture: nil, headline: nil, summary: nil, first_name: "foo", last_name: "bar", created_at: "2020-08-19 10:46:00", updated_at: "2020-08-18 10:46:00", agree_to_terms_and_conditions: nil, registration_status: "", account_id: 1, start_date: nil, supervisor_id: nil, picture_processing: false, statistics: {}, data: {}, notification_timestamps: {}, suggestions: {}, notification_settings: {}, contact_information: []>}>
750
+ #<Opera::Operation::Result:0x007ff414a01238 @errors={}, @exceptions={}, @information={fast_section: {:real=>0.300013706088066e-05, :total=>0.0}, slow_section: {:real=>1.800013706088066e-05, :total=>0.0}}, @executions=[:profile_schema, :create, :update, :send_email, :output], @output={:model=>#<Profile id: 30, user_id: nil, linkedin_uid: nil, picture: nil, headline: nil, summary: nil, first_name: "foo", last_name: "bar", created_at: "2020-08-19 10:46:00", updated_at: "2020-08-18 10:46:00", agree_to_terms_and_conditions: nil, registration_status: "", account_id: 1, start_date: nil, supervisor_id: nil, picture_processing: false, statistics: {}, data: {}, notification_timestamps: {}, suggestions: {}, notification_settings: {}, contact_information: []>}>
721
751
  ```
722
752
 
723
753
  ### Success
@@ -33,7 +33,9 @@ module Opera
33
33
  def call(args = {})
34
34
  operation = new(params: args.fetch(:params, {}), dependencies: args.fetch(:dependencies, {}))
35
35
  executor = Executor.new(operation)
36
- executor.evaluate_instructions(instructions)
36
+ Instrumentation.new(operation).instrument(name: self.name, level: :operation) do
37
+ executor.evaluate_instructions(instructions)
38
+ end
37
39
  executor.result
38
40
  end
39
41
 
@@ -35,6 +35,7 @@ module Opera
35
35
  instructions << if !blk.nil?
36
36
  {
37
37
  kind: instruction,
38
+ label: method,
38
39
  instructions: InnerBuilder.new(&blk).instructions
39
40
  }
40
41
  else
@@ -6,12 +6,16 @@ module Opera
6
6
  DEVELOPMENT_MODE = :development
7
7
  PRODUCTION_MODE = :production
8
8
 
9
- attr_accessor :transaction_class, :transaction_method, :transaction_options, :reporter, :mode
9
+ attr_accessor :transaction_class, :transaction_method, :transaction_options, :instrumentation_class,
10
+ :mode, :reporter
10
11
 
11
12
  def initialize
12
13
  @transaction_class = self.class.transaction_class
13
14
  @transaction_method = self.class.transaction_method || :transaction
14
15
  @transaction_options = self.class.transaction_options
16
+
17
+ @instrumentation_class = self.class.instrumentation_class
18
+
15
19
  @mode = self.class.mode || DEVELOPMENT_MODE
16
20
  @reporter = custom_reporter || self.class.reporter
17
21
 
@@ -35,7 +39,8 @@ module Opera
35
39
  end
36
40
 
37
41
  class << self
38
- attr_accessor :transaction_class, :transaction_method, :transaction_options, :reporter, :mode
42
+ attr_accessor :transaction_class, :transaction_method, :transaction_options, :instrumentation_class,
43
+ :mode, :reporter
39
44
 
40
45
  def configure
41
46
  yield self
@@ -7,10 +7,17 @@ module Opera
7
7
  class Benchmark < Executor
8
8
  def call(instruction)
9
9
  benchmark = ::Benchmark.measure do
10
+ instruction[:kind] = :step
10
11
  super
11
12
  end
12
13
 
13
- result.add_information(real: benchmark.real, total: benchmark.total)
14
+ result.add_information(benchmark_key(instruction) => { real: benchmark.real, total: benchmark.total })
15
+ end
16
+
17
+ private
18
+
19
+ def benchmark_key(instruction)
20
+ instruction[:method] || instruction[:label] || instruction[:instructions].map { |e| e[:method] }.join('-').to_sym
14
21
  end
15
22
  end
16
23
  end
@@ -8,8 +8,10 @@ module Opera
8
8
  def call(instruction)
9
9
  method = instruction[:method]
10
10
 
11
- operation.result.add_execution(method) unless production_mode?
12
- operation.send(method)
11
+ Instrumentation.new(operation).instrument(name: "##{method}", level: :step) do
12
+ operation.result.add_execution(method) unless production_mode?
13
+ operation.send(method)
14
+ end
13
15
  rescue StandardError => exception
14
16
  reporter&.error(exception)
15
17
  operation.result.add_exception(method, "#{exception.message}, for #{operation.inspect}", classname: operation.class.name)
@@ -0,0 +1,46 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Opera
4
+ module Operation
5
+ class Instrumentation
6
+ class Base
7
+ def self.instrument(operation, name:, level: :operation)
8
+ raise NotImplementedError, "#{self.class} must implement #instrument"
9
+ end
10
+ end
11
+
12
+ attr_reader :operation
13
+
14
+ def initialize(operation)
15
+ @operation = operation
16
+ end
17
+
18
+ def instrument(name:, level: :operation)
19
+ return yield if !instrumentation_enabled?
20
+ return yield if !instrumentation_compatible?
21
+
22
+ instrumentation_class.instrument(operation, name: name, level: level) do
23
+ yield
24
+ end
25
+ end
26
+
27
+ private
28
+
29
+ def config
30
+ operation.config
31
+ end
32
+
33
+ def instrumentation_class
34
+ config.instrumentation_class
35
+ end
36
+
37
+ def instrumentation_enabled?
38
+ !!config.instrumentation_class
39
+ end
40
+
41
+ def instrumentation_compatible?
42
+ config.instrumentation_class.ancestors.include?(Opera::Operation::Instrumentation::Base)
43
+ end
44
+ end
45
+ end
46
+ end
@@ -4,6 +4,7 @@ require 'opera/operation/attributes_dsl'
4
4
  require 'opera/operation/builder'
5
5
  require 'opera/operation/base'
6
6
  require 'opera/operation/executor'
7
+ require 'opera/operation/instrumentation'
7
8
  require 'opera/operation/result'
8
9
  require 'opera/operation/config'
9
10
  require 'opera/operation/instructions/executors/success'
data/lib/opera/version.rb CHANGED
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Opera
4
- VERSION = '0.3.3'
4
+ VERSION = '0.3.5'
5
5
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: opera
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.3.3
4
+ version: 0.3.5
5
5
  platform: ruby
6
6
  authors:
7
7
  - ProFinda Development Team
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2025-01-15 00:00:00.000000000 Z
11
+ date: 2025-02-24 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: dry-validation
@@ -91,6 +91,7 @@ files:
91
91
  - lib/opera/operation/instructions/executors/success.rb
92
92
  - lib/opera/operation/instructions/executors/transaction.rb
93
93
  - lib/opera/operation/instructions/executors/validate.rb
94
+ - lib/opera/operation/instrumentation.rb
94
95
  - lib/opera/operation/result.rb
95
96
  - lib/opera/version.rb
96
97
  - opera.gemspec