excadg 0.1.4 → 0.2.0

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: 1081c1531553e5ca9113aab9d0f8dfaa7982438438f646df74fded7a17ec0e99
4
- data.tar.gz: 88d4ecf74086eddcea280c44844c47097aeca252b6f9170f585f0516368928f0
3
+ metadata.gz: 49311f7d62494f4eea817789e886391472cb5a4b9fc971e6d179baf9cbb679b6
4
+ data.tar.gz: 7d36c94f2b7c13441bd36c4d44c9a1a0e77e1b96cb14fa2ace12cf8068ff3668
5
5
  SHA512:
6
- metadata.gz: a086d015c957d0b86ecf3ec8de789aa4f151ea1687c6ddde4f5454b28ea1f4737bafc1f81ccac9dd0a1fed45603c6fae8dfc71649dcbe383f8cb9baf3f887e70
7
- data.tar.gz: 394abc39d249922bd7eec046e748673e0ad7eb294f3c91db5a15bbea26a99486f159a55411f1baed09c80f6e3e919654bce87eb97a441461c9b97e464ee97b96
6
+ metadata.gz: 5586527ad35c02823db4f521cb74f34a819e9164d538e8611b3cb68d3fe4c0b8b8c983c5fd91e72ff5cc327fc3d0d0c98ff04d18a4c853e43e26f04d95f742bd
7
+ data.tar.gz: a5a1f2cd5b58e8319ceb64794848006898a76cb9ba0ffea79be6ad4bd4d7fd0bd5ff730dc3aaae7437ebf09f0fdcef64a5466c48d635881e0dabb3cd02f2263e
data/README.md CHANGED
@@ -21,7 +21,7 @@ Main usage pattern for this project is a framework. It implies that you:
21
21
  ### Payload implementation
22
22
 
23
23
  Payload is a piece of code to execute in a single vertex. This code should be self-contained and use specific ways to communicate with other vertices.
24
- There is a [{ExcADG::Payload}](lib/excadg/payload.rb) module to make a custom payload class. Read through its documentation to get familiar with its interface.
24
+ There is a [{ExcADG::Payload}](lib/excadg/payload.rb) module to make a custom payload class. Read through its documentation to get familiar with the interface.
25
25
 
26
26
  Here is a very basic example of a payload that echoes a message:
27
27
 
@@ -33,29 +33,31 @@ class MyPayload
33
33
  end
34
34
  end
35
35
  ```
36
- More payload examples could be found in [{ExcADG::Payload::Example}](lib/excadg/payload/example.rb).
36
+ More payload examples could be found in the [{ExcADG::Payload::Example}](lib/excadg/payload/example.rb) module.
37
37
 
38
38
  Any payload has 3 parts:
39
39
  - code
40
40
  - arguments
41
41
  - dependencies data
42
42
 
43
- **Code** is defined in {Payload#get} method. E.g. `system 'echo here I am'` in the above example.
44
- **Arguments** are defined during payload object contruction (see {Payload#initialize}) as `args:` parameter. E.g. `MyPayload.new args: 'my custom message'`.
43
+ **Code** is defined in {ExcADG::Payload#get} method. E.g. `system 'echo here I am'` in the above example.
44
+ **Arguments** could be supplied during payload object contruction (see {ExcADG::Payload#initialize}) as `args:` parameter. E.g. `MyPayload.new args: 'my custom message'`.
45
45
  **Dependencies data** is provided by the framework and has all [{ExcADG::VStateData}](lib/excadg/vstate_data.rb) returned by the dependencies.
46
46
 
47
47
  ### Vertices constructing
48
48
 
49
- Vertex is a execution graph's building block. It executes its payload when all its dependencies are finished.
50
- Vertex can be created this way: `Vertex.new name: :myvertex, payload: MyPayload.new`. This vertex doesn't have any dependencies, so it starts execution immediately upon construction. Vertices without dependencies are always aneeded in the graph to start it.
49
+ Vertex is an execution graph's building block. It executes its payload when all its dependencies are finished.
50
+ Vertex can be created this way: `Vertex.new name: :myvertex, payload: MyPayload.new`. This vertex doesn't have any dependencies, so it starts execution immediately upon construction. Vertices without dependencies are always needed in the graph to start it.
51
51
 
52
- Another "kind" of vertices are vertices with dependencies. Dependencies are other vertices that the vertex waits to finish successfully before running its own payload.
53
- > Example: `Vertex.new name: :other_vertex, payload: MyPayload.new, deps: [:myvertex]` has exactly 1 dependency - vertex called `:myvertex`
54
- > `:other_vertex` won't start until `:myvertex` finishes.
55
- > Moreover, as described in [payload](#payload), `:other_vertex`'s payload will have access to data returned by `:myvertex`'s payload.
56
- > In this case, it'd be what `system 'echo here I am'` returns - `true`.
52
+ Another "kind" of vertices are vertices with dependencies. Dependencies are an array other vertices that the vertex waits to finish successfully before running its own payload.
57
53
 
58
- Dependencies could be specified both - using {ExcADG::Vertex} objects and names. E.g.
54
+ Example:
55
+ `Vertex.new name: :other_vertex, payload: MyPayload.new, deps: [:myvertex]` has exactly 1 dependency - the vertex called `:myvertex`
56
+ `:other_vertex` won't start until `:myvertex` finishes.
57
+ Moreover, as described in [payload](#excadgpayload), `:other_vertex`'s payload will have access to data returned by `:myvertex`'s payload.
58
+ In this case, it'd be what `system 'echo here I am'` returns - `true`.
59
+
60
+ Dependencies could be specified with both - {ExcADG::Vertex} objects and names. E.g.
59
61
  ``` ruby
60
62
  Broker.run
61
63
 
@@ -67,27 +69,27 @@ Vertex.new name: :final, payload: MyPayload.new, deps: [v1, :v2]
67
69
  Broker.wait_all
68
70
  ```
69
71
 
70
- *See [Broker section](#broker) for `Broker.run` and `Broker.wait_all` usage.*
72
+ *See [Broker section](#excadgbroker) for `Broker.run` and `Broker.wait_all` usage.*
71
73
 
72
74
  Using actual objects looks simpler, but it's less convenient, as it requires you to construct all vertices as they appear in the execution graph.
73
- However, names allows you to spawn vertices in arbitrary order and expect framework to figure execution order on the fly. See [run tool](#tool) as an example of using names.
75
+ However, names allows you to spawn vertices in arbitrary order and expect framework to figure execution order on the fly. See [run tool](#tool)'s code as an example of using names.
74
76
 
75
- *There is no need to store {ExcADG::Vertex} objects, as it and its data are available through [Broker](#broker)'s [DataStore](#data-store) and there is no interface to communicate with a {ExcADG::Vertex} directly.*
77
+ *There is no need to store {ExcADG::Vertex} objects, as it and its data are available through [Broker](#excadgbroker)'s [DataStore](#excadgdatastore) and there is no interface to communicate with an {ExcADG::Vertex} directly.*
76
78
 
77
79
  # Internals
78
80
 
79
81
  ## Overview
80
82
 
81
83
  This framework allows to spawn vertices. Once created, a vertex with payload starts immediately.
82
- This means that there is no central place where execution seqence is controlled besides vertex's own mechanism to wait for dependencies.
84
+ This means that there is no central place that controls execution seqence besides vertex's own mechanism to wait for dependencies.
83
85
 
84
- As Ractors doesn't allow to reliably exchange data between each other at the moment, the main Ractor (thread) has to spawn a broker to synchronize data exchange. See [broker](#broker) sections to learn more.
86
+ As Ractors doesn't allow to reliably exchange data between each other at the moment, the main Ractor (thread) has to spawn a broker to synchronize data exchange. See [broker](#excadgbroker) sections to learn more.
85
87
 
86
88
  This framework is based on [Ractors](https://docs.ruby-lang.org/en/master/ractor_md.html). It could be useful to get familiar with ractors before reading further.
87
89
 
88
90
  ## Vertice processing states
89
91
 
90
- Internally, each vertice goes through a sequence of states. Now it's **new**, **ready**, **done** and **failed**. Stages are controlled by the [{ExcADG::StateMachine}](#statemachine).
92
+ Internally, each vertice go through a sequence of states. Now it's **new**, **ready**, **done** and **failed**. Stages are controlled by the [{ExcADG::StateMachine}](#excadgstatemachine).
91
93
 
92
94
  {ExcADG::Vertex} starts as a **new** vertex and waits for its dependencies.
93
95
  When vertex received all dependencies states and made sure they're **done**, it becomes **ready** and starts its payload.
@@ -103,13 +105,13 @@ When a vertex changes its state, it (actually, state machine does that) notifies
103
105
  Broker is desired to be as thin as possible to keep most of the work for vertices.
104
106
 
105
107
  Each application should invoke {Broker.run} to enable messages processing.
106
- Its counterpart is {Broker.wait_all} which waits for all known vertices to reach one of the terminal states (**done** or **failed**) within specified timeout. Same as `Broker.run`, it spawns a thread and returns it. The main application could keep it in background and query or `.join` on it once all vertices are spawned. Once the thread finishes, the main app could lookup vertices execution results in {Broker.data_store} (see [DataStore](#data-store)).
108
+ Its counterpart is {Broker.wait_all} which waits for all known vertices to reach one of the terminal states (**done** or **failed**) within specified timeout. Same as `Broker.run`, it spawns a thread and returns it. The main application could keep it in background and query or `.join` on it once all vertices are spawned. Once the thread finishes, the main app could lookup vertices execution results in {Broker.data_store} (see [DataStore](#excadgdatastore)).
107
109
 
108
110
  > Beware that broker constantly uses main Ractor's ports - incoming and outgoing. Hence, `Ractor#take`, `Ractor.yield` or any other messaging in the main ractor conflict with broker.
109
111
 
110
112
  ## {ExcADG::DataStore}
111
113
 
112
- It's a mostly internal [Broker's](#broker) object that holds all vertice's [{ExcADG::VStateData}](lib/excadg/vstate_data.rb).
114
+ It's a mostly internal [Broker's](#excadgbroker) object that holds all vertice's [{ExcADG::VStateData}](lib/excadg/vstate_data.rb).
113
115
 
114
116
  ## {ExcADG::StateMachine}
115
117
 
@@ -117,7 +119,7 @@ State machine is a small internal mechanism that helps vertices to
117
119
  - do state transitions
118
120
  - preserve state transition results locally
119
121
 
120
- State machine has transitions graph that is common for all vertices. Vertices bind certain logic to transitions. E.g. "wait for dependencies" or "run the payload". State transition mechanism ensures to run workload associated to the currently possible transition, process errors (if any) and send results to [Broker](#broker).
122
+ State machine has transitions graph that is common for all vertices. Vertices bind certain logic to transitions. E.g. "wait for dependencies" or "run the payload". State transition mechanism ensures to run workload associated to the currently possible transition, process errors (if any) and send results to [Broker](#excadgbroker).
121
123
 
122
124
  ## {ExcADG::Payload}
123
125
 
@@ -130,7 +132,7 @@ First way is `args` parameter of the {Payload#initialize}. It can be used on pay
130
132
  Second way is built-in. Vertex invokes payload with {ExcADG::Array} of dependencies {ExcADG::VStateData}, if payload's `Proc` is parametrized (has arity 1).
131
133
 
132
134
  > Be mindful about data your payload receives (args) and returns (state data). It could appear incompatible with ractors and cause vertice to fail.
133
- > Although these failures are hard to tell beforehand, [state machine](#statemachine) processes them gracefully.
135
+ > Although these failures are hard to tell beforehand, [state machine](#excadgstatemachine) processes them gracefully.
134
136
 
135
137
  ## {ExcADG::Log}
136
138
 
@@ -175,10 +177,6 @@ Logging is disabled by default, but it could be useful to debug tests. Add `ExcA
175
177
 
176
178
  > here is a list of improvemets could be implementex next
177
179
 
178
- - add timeouts
179
- - to payload
180
- - to whole vertex
181
- - to request processing
182
180
  - implement throttling (:suspended state)
183
181
  - limit # of running vertices
184
182
  - problem: can't find what to suspend
@@ -1,5 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ require 'benchmark'
4
+
3
5
  require_relative '../broker'
4
6
  require_relative '../request'
5
7
 
@@ -55,23 +57,16 @@ module ExcADG
55
57
  end
56
58
  end
57
59
 
58
- # calculate Nth member of the function
59
- # y = x(n) * (x(n-1)) where x(0) = 0 and x(1) = 1
60
- # it illustrates how payload can be customized with @args
61
- # and extended by adding methods
62
- class Multiplier
60
+ # payload that occupies a CPU core for several seconds,
61
+ # is suitable for perfomance tests
62
+ class Benchmark
63
63
  include Payload
64
64
  def get
65
- -> { get_el x: @args }
66
- end
67
-
68
- def get_el x:
69
- case x
70
- when 0 then 0
71
- when 1 then 1
72
- else
73
- x * get_el(x: x - 1)
74
- end
65
+ lambda {
66
+ Log.info(::Benchmark.measure {
67
+ 10_000.downto(1) { |i| 10_000.downto(1) { |j| i * j } }
68
+ })
69
+ }
75
70
  end
76
71
  end
77
72
 
@@ -9,6 +9,7 @@ module ExcADG
9
9
  return block.call if timeout.nil? || timeout.zero?
10
10
 
11
11
  timed_out = false
12
+ Thread.report_on_exception = false
12
13
  payload = Thread.new { Thread.current[:result] = block.call }
13
14
  Thread.new {
14
15
  sleep timeout
@@ -53,24 +53,30 @@ module ExcADG
53
53
  target = target_candidates.first
54
54
  Log.debug "found a candidate: #{target}"
55
55
  edge = GRAPH.edges.find { |e| e.source == @state && e.target = target }
56
- begin
56
+
57
+ with_fault_processing {
57
58
  @state_transition_data[target] = @state_edge_bindings[edge].call
58
59
  @state = target
59
60
  Log.debug "moved to #{@state}"
61
+ }
62
+ @state_transition_data[@state]
63
+ end
64
+
65
+ # invoke the block provided with state update on failures
66
+ def with_fault_processing
67
+ yield if block_given?
68
+ rescue StandardError => e
69
+ Log.error "step failed with #{e} / #{e.backtrace}"
70
+ @state_transition_data[:failed] = e
71
+ @state = :failed
72
+ ensure
73
+ begin
74
+ Broker.ask Request::Update.new data: state_data
60
75
  rescue StandardError => e
61
- Log.error "step failed with #{e} / #{e.backtrace}"
62
76
  @state_transition_data[:failed] = e
63
77
  @state = :failed
64
- ensure
65
- begin
66
- Broker.ask Request::Update.new data: state_data
67
- rescue StandardError => e
68
- @state_transition_data[:failed] = e
69
- @state = :failed
70
- Broker.ask Request::Update.new data: state_data
71
- end
78
+ Broker.ask Request::Update.new data: state_data
72
79
  end
73
- @state_transition_data[@state]
74
80
  end
75
81
 
76
82
  def assert_state_transition_bounds
data/lib/excadg/tui.rb CHANGED
@@ -12,16 +12,6 @@ module ExcADG
12
12
  MAX_VERTEX_TO_SHOW = 10
13
13
  DELAY = 0.2
14
14
  DEFAULT_BOX_SIZE = { height: 50, width: 150 }.freeze
15
- # TODO: do runtime calc
16
- BOX_SIZE = {
17
- height: DEFAULT_BOX_SIZE[:height] > (IO.console&.winsize&.first || 1000) ? IO.console.winsize.first : DEFAULT_BOX_SIZE[:height],
18
- width: DEFAULT_BOX_SIZE[:width] > (IO.console&.winsize&.last || 1000) ? IO.console.winsize.last : DEFAULT_BOX_SIZE[:width]
19
- }.freeze
20
- CONTENT_SIZE = {
21
- height: BOX_SIZE[:height] - 4, # 2 for borders, 1 for \n, 1 for remark
22
- width: BOX_SIZE[:width] - 5 # 2 for borders, 2 to indent
23
- }.freeze
24
- LINE_TEMPLATE = "| %-#{CONTENT_SIZE[:width]}s |\n".freeze
25
15
 
26
16
  @started_at = DateTime.now.strftime('%Q').to_i
27
17
 
@@ -47,20 +37,21 @@ module ExcADG
47
37
  # @param content is a list of lines to print
48
38
  def print_in_box content
49
39
  clear
50
- print "+-#{'-' * CONTENT_SIZE[:width]}-+\n"
51
- content[..CONTENT_SIZE[:height]].each { |line|
52
- if line.size > CONTENT_SIZE[:width]
53
- printf LINE_TEMPLATE, "#{line[...(CONTENT_SIZE[:width] - 3)]}..."
40
+ refresh_sizes
41
+ print "+-#{'-' * @content_size[:width]}-+\n"
42
+ content[..@content_size[:height]].each { |line|
43
+ if line.size > @content_size[:width]
44
+ printf @line_template, "#{line[...(@content_size[:width] - 3)]}..."
54
45
  else
55
- printf LINE_TEMPLATE, line
46
+ printf @line_template, line
56
47
  end
57
48
  }
58
- if content.size < CONTENT_SIZE[:height]
59
- (CONTENT_SIZE[:height] - content.size).times { printf LINE_TEMPLATE, ' ' }
49
+ if content.size < @content_size[:height]
50
+ (@content_size[:height] - content.size).times { printf @line_template, ' ' }
60
51
  else
61
- printf LINE_TEMPLATE, '<some content did not fit and was cropped>'[..CONTENT_SIZE[:width]]
52
+ printf @line_template, '<some content did not fit and was cropped>'[..@content_size[:width]]
62
53
  end
63
- print "+-#{'-' * CONTENT_SIZE[:width]}-+\n"
54
+ print "+-#{'-' * @content_size[:width]}-+\n"
64
55
  end
65
56
 
66
57
  def print_summary has_failed, timed_out
@@ -81,6 +72,18 @@ module ExcADG
81
72
  print "\e[2J\e[f"
82
73
  end
83
74
 
75
+ def refresh_sizes
76
+ box_size = {
77
+ height: IO.console&.winsize&.first.nil? || DEFAULT_BOX_SIZE[:height] < IO.console.winsize.first ? DEFAULT_BOX_SIZE[:height] : IO.console.winsize.first,
78
+ width: IO.console&.winsize&.last.nil? || DEFAULT_BOX_SIZE[:width] < IO.console&.winsize&.last ? DEFAULT_BOX_SIZE[:width] : IO.console.winsize.last
79
+ }.freeze
80
+ @content_size = {
81
+ height: box_size[:height] - 4, # 2 for borders, 1 for \n, 1 for remark
82
+ width: box_size[:width] - 5 # 2 for borders, 2 to indent
83
+ }.freeze
84
+ @line_template = "| %-#{@content_size[:width]}s |\n"
85
+ end
86
+
84
87
  # make states summary, one for a line with consistent placing
85
88
  def state_stats
86
89
  skeleton = StateMachine::GRAPH.vertices.collect { |v| [v, []] }.to_h
data/lib/excadg/vertex.rb CHANGED
@@ -7,6 +7,7 @@ require_relative 'payload'
7
7
  require_relative 'request'
8
8
  require_relative 'rtimeout'
9
9
  require_relative 'state_machine'
10
+ require_relative 'vtimeout'
10
11
 
11
12
  module ExcADG
12
13
  # Individual vertex of the execution graph to run in a separated Ractor
@@ -58,7 +59,7 @@ module ExcADG
58
59
  # @param payload Payload object to run in this Vertex
59
60
  # @param name optional vertex name to identify vertex
60
61
  # @param deps list of other Vertices or names to wait for
61
- # @param timeout total time in seconds for the payload to run
62
+ # @param timeout {ExcADG::VTimeouts} or total time in seconds for the payload to run
62
63
  # @raise Payload::IncorrectPayloadArity in case payload returns function with arity > 1
63
64
  # @raise Payload::NoPayloadSet in case payload provided has incorrect type
64
65
  def new payload:, name: nil, deps: [], timeout: nil
@@ -66,33 +67,42 @@ module ExcADG
66
67
 
67
68
  raise Payload::IncorrectPayloadArity, "arity is #{payload.get.arity}, supported only 0 and 1" unless [0, 1].include? payload.get.arity
68
69
 
69
- super(payload, name, timeout, DependencyManager.new(deps:)) { |payload, name, timeout, deps_manager|
70
+ dm = DependencyManager.new(deps: deps)
71
+ vtimeout = timeout.is_a?(VTimeout) ? timeout : VTimeout.new(payload: timeout)
72
+
73
+ super(payload, name, vtimeout, dm) { |payload, name, vtimeout, deps_manager|
70
74
  state_machine = StateMachine.new(name: name || "v#{number}".to_sym)
71
- Broker.ask Request::Update.new data: state_machine.state_data
72
- Log.info 'building vertex lifecycle'
73
- state_machine.bind_action(:new, :ready) {
74
- until deps_manager.deps.empty?
75
- deps_data = Broker.ask Request::GetStateData.new(deps: deps_manager.deps)
76
- deps_manager.deduct_deps deps_data
77
- sleep 0.2
78
- end
79
- deps_manager.data
80
- }
81
- state_machine.bind_action(:ready, :done) {
82
- function = payload.get
83
- await(timeout:) {
84
- case function.arity
85
- when 0 then function.call
86
- when 1 then function.call state_machine.state_data.data
87
- else
88
- raise Payload::IncorrectPayloadArity, "unexpected payload arity: #{function.arity}, supported only 0 and 1"
89
- end
90
- }
91
- }
75
+ state_machine.with_fault_processing {
76
+ await(timeout: vtimeout.global) {
77
+ Broker.ask Request::Update.new data: state_machine.state_data
78
+ Log.info 'building vertex lifecycle'
79
+ state_machine.bind_action(:new, :ready) {
80
+ await(timeout: vtimeout.deps) {
81
+ until deps_manager.deps.empty?
82
+ deps_data = Broker.ask Request::GetStateData.new(deps: deps_manager.deps)
83
+ deps_manager.deduct_deps deps_data
84
+ sleep 0.2
85
+ end
86
+ deps_manager.data
87
+ }
88
+ }
89
+ state_machine.bind_action(:ready, :done) {
90
+ function = payload.get
91
+ await(timeout: vtimeout.payload) {
92
+ case function.arity
93
+ when 0 then function.call
94
+ when 1 then function.call state_machine.state_data.data
95
+ else
96
+ raise Payload::IncorrectPayloadArity, "unexpected payload arity: #{function.arity}, supported only 0 and 1"
97
+ end
98
+ }
99
+ }
92
100
 
93
- Log.debug "another step fades: #{state_machine.state_data}" while state_machine.step
101
+ Log.debug "another step fades: #{state_machine.state_data}" while state_machine.step
94
102
 
95
- Log.debug 'shut down'
103
+ Log.debug 'shut down'
104
+ }
105
+ }
96
106
  }
97
107
  end
98
108
  end
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ExcADG
4
+ # complex vertex timeout to carry fine-grained timeouts
5
+ class VTimeout
6
+ attr_reader :global, :payload, :deps
7
+
8
+ # @param global timeout for the whole vertex in seconds
9
+ # @param deps for how long to wait for deps to complete in seconds
10
+ # @param payload how long to wait for payload in seconds
11
+ def initialize payload: nil, deps: nil, global: nil
12
+ if !global.nil? && global < (deps || 0) + (payload || 0) && (global < deps + payload)
13
+ raise "global timeout (#{global}) is less than sum of deps (#{deps}) and payload (#{payload}) timeouts"
14
+ end
15
+
16
+ @global = global
17
+ @deps = deps
18
+ @payload = payload
19
+ end
20
+ end
21
+ end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: excadg
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.4
4
+ version: 0.2.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - skorobogatydmitry
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2024-07-08 00:00:00.000000000 Z
11
+ date: 2024-07-10 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: rgl
@@ -51,6 +51,7 @@ files:
51
51
  - lib/excadg/tui.rb
52
52
  - lib/excadg/vertex.rb
53
53
  - lib/excadg/vstate_data.rb
54
+ - lib/excadg/vtimeout.rb
54
55
  homepage: https://rubygems.org/gems/excadg
55
56
  licenses:
56
57
  - LGPL-3.0-only