excadg 0.1.4 → 0.2.1
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 +4 -4
- data/README.md +24 -26
- data/lib/excadg/payload/example.rb +10 -15
- data/lib/excadg/rtimeout.rb +1 -0
- data/lib/excadg/state_machine.rb +17 -11
- data/lib/excadg/tui.rb +22 -19
- data/lib/excadg/vertex.rb +35 -25
- data/lib/excadg/vtimeout.rb +21 -0
- metadata +4 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: a0c8bef612487823d7ce7076e03cbd92f390e96217e7bd08fc46a3abddeb33ed
|
4
|
+
data.tar.gz: bb114bb5eece2992ca879b05686d801d016b1da8e017bfbb2b7adb10bd75cb29
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: ee44f5e96707f3aaa926961450a00ce04b71cbcbf8c2a260aa61fef3778ce7f1f47c89de3b14ca9cdcc8bb23b1f0b5de66719a20715fea5e6bf5e907c53959fd
|
7
|
+
data.tar.gz: 53e9f60bf74e6277b5c6b732f89c8945815bfba61ef56556c129ffc9f55b94c07e8b79b7b83131a63b6e3d42910d1c7aee65cb94ccb10927ad9c796c4d4aa982
|
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
|
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**
|
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
|
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
|
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
|
-
|
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](#
|
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](#
|
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
|
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](#
|
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
|
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](#
|
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](#
|
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](#
|
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](#
|
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
|
-
#
|
59
|
-
#
|
60
|
-
|
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
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
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
|
|
data/lib/excadg/rtimeout.rb
CHANGED
data/lib/excadg/state_machine.rb
CHANGED
@@ -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
|
-
|
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
|
-
|
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
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
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
|
46
|
+
printf @line_template, line
|
56
47
|
end
|
57
48
|
}
|
58
|
-
if content.size <
|
59
|
-
(
|
49
|
+
if content.size < @content_size[:height]
|
50
|
+
(@content_size[:height] - content.size).times { printf @line_template, ' ' }
|
60
51
|
else
|
61
|
-
printf
|
52
|
+
printf @line_template, '<some content did not fit and was cropped>'[..@content_size[:width]]
|
62
53
|
end
|
63
|
-
print "+-#{'-' *
|
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
|
-
|
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
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
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
|
-
|
101
|
+
Log.debug "another step fades: #{state_machine.state_data}" while state_machine.step
|
94
102
|
|
95
|
-
|
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
|
+
version: 0.2.1
|
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-
|
11
|
+
date: 2024-07-10 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rgl
|
@@ -51,7 +51,8 @@ files:
|
|
51
51
|
- lib/excadg/tui.rb
|
52
52
|
- lib/excadg/vertex.rb
|
53
53
|
- lib/excadg/vstate_data.rb
|
54
|
-
|
54
|
+
- lib/excadg/vtimeout.rb
|
55
|
+
homepage: https://github.com/skorobogatydmitry/excadg
|
55
56
|
licenses:
|
56
57
|
- LGPL-3.0-only
|
57
58
|
metadata:
|