rage-rb 1.23.0 → 1.24.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 +4 -4
- data/CHANGELOG.md +22 -0
- data/CONTRIBUTING.md +240 -0
- data/README.md +1 -0
- data/lib/rage/application.rb +1 -0
- data/lib/rage/cable/cable.rb +20 -15
- data/lib/rage/cable/channel.rb +2 -1
- data/lib/rage/configuration.rb +122 -27
- data/lib/rage/controller/api.rb +5 -31
- data/lib/rage/deferred/deferred.rb +7 -0
- data/lib/rage/deferred/metadata.rb +8 -0
- data/lib/rage/deferred/scheduler.rb +25 -0
- data/lib/rage/deferred/task.rb +19 -5
- data/lib/rage/errors.rb +83 -0
- data/lib/rage/events/subscriber.rb +6 -1
- data/lib/rage/internal.rb +15 -6
- data/lib/rage/middleware/fiber_wrapper.rb +1 -0
- data/lib/rage/openapi/builder.rb +1 -1
- data/lib/rage/openapi/converter.rb +5 -1
- data/lib/rage/openapi/nodes/method.rb +2 -1
- data/lib/rage/openapi/nodes/root.rb +2 -1
- data/lib/rage/openapi/openapi.rb +1 -1
- data/lib/rage/openapi/parser.rb +73 -2
- data/lib/rage/openapi/parsers/ext/alba.rb +30 -2
- data/lib/rage/openapi/parsers/request.rb +2 -2
- data/lib/rage/openapi/parsers/response.rb +2 -2
- data/lib/rage/openapi/parsers/yaml.rb +27 -5
- data/lib/rage/params_parser.rb +2 -2
- data/lib/rage/pubsub/adapters/redis.rb +2 -1
- data/lib/rage/router/dsl.rb +7 -2
- data/lib/rage/sse/application.rb +1 -0
- data/lib/rage/telemetry/tracer.rb +1 -0
- data/lib/rage/version.rb +1 -1
- data/lib/rage-rb.rb +6 -0
- metadata +3 -3
- data/lib/rage/cable/adapters/base.rb +0 -16
- data/lib/rage/cable/adapters/redis.rb +0 -128
data/lib/rage/version.rb
CHANGED
data/lib/rage-rb.rb
CHANGED
|
@@ -40,6 +40,12 @@ module Rage
|
|
|
40
40
|
Rage::Events
|
|
41
41
|
end
|
|
42
42
|
|
|
43
|
+
# Shorthand to access {Rage::Errors Rage::Errors}.
|
|
44
|
+
# @return [Rage::Errors]
|
|
45
|
+
def self.errors
|
|
46
|
+
Rage::Errors
|
|
47
|
+
end
|
|
48
|
+
|
|
43
49
|
# Shorthand to access {Rage::SSE Rage::SSE}.
|
|
44
50
|
# @return [Rage::SSE]
|
|
45
51
|
def self.sse
|
metadata
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: rage-rb
|
|
3
3
|
version: !ruby/object:Gem::Version
|
|
4
|
-
version: 1.
|
|
4
|
+
version: 1.24.0
|
|
5
5
|
platform: ruby
|
|
6
6
|
authors:
|
|
7
7
|
- Roman Samoilov
|
|
@@ -134,6 +134,7 @@ files:
|
|
|
134
134
|
- Appraisals
|
|
135
135
|
- CHANGELOG.md
|
|
136
136
|
- CODE_OF_CONDUCT.md
|
|
137
|
+
- CONTRIBUTING.md
|
|
137
138
|
- Gemfile
|
|
138
139
|
- LICENSE.txt
|
|
139
140
|
- README.md
|
|
@@ -143,8 +144,6 @@ files:
|
|
|
143
144
|
- lib/rage.rb
|
|
144
145
|
- lib/rage/all.rb
|
|
145
146
|
- lib/rage/application.rb
|
|
146
|
-
- lib/rage/cable/adapters/base.rb
|
|
147
|
-
- lib/rage/cable/adapters/redis.rb
|
|
148
147
|
- lib/rage/cable/cable.rb
|
|
149
148
|
- lib/rage/cable/channel.rb
|
|
150
149
|
- lib/rage/cable/connection.rb
|
|
@@ -167,6 +166,7 @@ files:
|
|
|
167
166
|
- lib/rage/deferred/middleware_chain.rb
|
|
168
167
|
- lib/rage/deferred/proxy.rb
|
|
169
168
|
- lib/rage/deferred/queue.rb
|
|
169
|
+
- lib/rage/deferred/scheduler.rb
|
|
170
170
|
- lib/rage/deferred/task.rb
|
|
171
171
|
- lib/rage/env.rb
|
|
172
172
|
- lib/rage/errors.rb
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
class Rage::Cable::Adapters::Base
|
|
4
|
-
def pick_a_worker(&block)
|
|
5
|
-
_lock, lock_path = Tempfile.new.yield_self { |file| [file, file.path] }
|
|
6
|
-
|
|
7
|
-
Iodine.on_state(:on_start) do
|
|
8
|
-
if File.new(lock_path).flock(File::LOCK_EX | File::LOCK_NB)
|
|
9
|
-
if Rage.logger.debug?
|
|
10
|
-
puts "INFO: #{Process.pid} is managing #{self.class.name.split("::").last} subscriptions."
|
|
11
|
-
end
|
|
12
|
-
block.call
|
|
13
|
-
end
|
|
14
|
-
end
|
|
15
|
-
end
|
|
16
|
-
end
|
|
@@ -1,128 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
require "securerandom"
|
|
4
|
-
|
|
5
|
-
if !defined?(RedisClient)
|
|
6
|
-
fail <<~ERR
|
|
7
|
-
|
|
8
|
-
Redis adapter depends on the `redis-client` gem. Add the following line to your Gemfile:
|
|
9
|
-
gem "redis-client"
|
|
10
|
-
|
|
11
|
-
ERR
|
|
12
|
-
end
|
|
13
|
-
|
|
14
|
-
class Rage::Cable::Adapters::Redis < Rage::Cable::Adapters::Base
|
|
15
|
-
REDIS_STREAM_NAME = "rage:cable:messages"
|
|
16
|
-
DEFAULT_REDIS_OPTIONS = { reconnect_attempts: [0.05, 0.1, 0.5] }
|
|
17
|
-
REDIS_MIN_VERSION_SUPPORTED = Gem::Version.create(6)
|
|
18
|
-
|
|
19
|
-
def initialize(config)
|
|
20
|
-
@redis_stream = if (prefix = config.delete(:channel_prefix))
|
|
21
|
-
"#{prefix}:#{REDIS_STREAM_NAME}"
|
|
22
|
-
else
|
|
23
|
-
REDIS_STREAM_NAME
|
|
24
|
-
end
|
|
25
|
-
|
|
26
|
-
@redis_config = RedisClient.config(**DEFAULT_REDIS_OPTIONS.merge(config))
|
|
27
|
-
@server_uuid = SecureRandom.uuid
|
|
28
|
-
|
|
29
|
-
redis_version = get_redis_version
|
|
30
|
-
if redis_version < REDIS_MIN_VERSION_SUPPORTED
|
|
31
|
-
raise "Redis adapter only supports Redis 6+. Detected Redis version: #{redis_version}."
|
|
32
|
-
end
|
|
33
|
-
|
|
34
|
-
@trimming_strategy = redis_version < Gem::Version.create("6.2.0") ? :maxlen : :minid
|
|
35
|
-
|
|
36
|
-
pick_a_worker { poll }
|
|
37
|
-
end
|
|
38
|
-
|
|
39
|
-
def publish(stream_name, data)
|
|
40
|
-
message_uuid = SecureRandom.uuid
|
|
41
|
-
|
|
42
|
-
publish_redis.call(
|
|
43
|
-
"XADD",
|
|
44
|
-
@redis_stream,
|
|
45
|
-
trimming_method, "~", trimming_value,
|
|
46
|
-
"*",
|
|
47
|
-
"1", stream_name,
|
|
48
|
-
"2", data.to_json,
|
|
49
|
-
"3", @server_uuid,
|
|
50
|
-
"4", message_uuid
|
|
51
|
-
)
|
|
52
|
-
end
|
|
53
|
-
|
|
54
|
-
private
|
|
55
|
-
|
|
56
|
-
def publish_redis
|
|
57
|
-
@publish_redis ||= @redis_config.new_client
|
|
58
|
-
end
|
|
59
|
-
|
|
60
|
-
def trimming_method
|
|
61
|
-
@trimming_strategy == :maxlen ? "MAXLEN" : "MINID"
|
|
62
|
-
end
|
|
63
|
-
|
|
64
|
-
def trimming_value
|
|
65
|
-
@trimming_strategy == :maxlen ? "10000" : ((Time.now.to_f - 5 * 60) * 1000).to_i
|
|
66
|
-
end
|
|
67
|
-
|
|
68
|
-
def get_redis_version
|
|
69
|
-
service_redis = @redis_config.new_client
|
|
70
|
-
version = service_redis.call("INFO").match(/redis_version:([[:graph:]]+)/)[1]
|
|
71
|
-
|
|
72
|
-
Gem::Version.create(version)
|
|
73
|
-
|
|
74
|
-
rescue RedisClient::Error => e
|
|
75
|
-
puts "FATAL: Couldn't connect to Redis - all broadcasts will be limited to the current server."
|
|
76
|
-
puts e.backtrace.join("\n")
|
|
77
|
-
REDIS_MIN_VERSION_SUPPORTED
|
|
78
|
-
|
|
79
|
-
ensure
|
|
80
|
-
service_redis.close
|
|
81
|
-
end
|
|
82
|
-
|
|
83
|
-
def error_backoff_intervals
|
|
84
|
-
@error_backoff_intervals ||= Enumerator.new do |y|
|
|
85
|
-
y << 0.2 << 0.5 << 1 << 2 << 5
|
|
86
|
-
loop { y << 10 }
|
|
87
|
-
end
|
|
88
|
-
end
|
|
89
|
-
|
|
90
|
-
def poll
|
|
91
|
-
unless Fiber.scheduler
|
|
92
|
-
Fiber.set_scheduler(Rage::FiberScheduler.new)
|
|
93
|
-
end
|
|
94
|
-
|
|
95
|
-
Iodine.on_state(:start_shutdown) do
|
|
96
|
-
@stopping = true
|
|
97
|
-
end
|
|
98
|
-
|
|
99
|
-
Fiber.schedule do
|
|
100
|
-
read_redis = @redis_config.new_client
|
|
101
|
-
last_id = (Time.now.to_f * 1000).to_i
|
|
102
|
-
last_message_uuid = nil
|
|
103
|
-
|
|
104
|
-
loop do
|
|
105
|
-
data = read_redis.blocking_call(5, "XREAD", "COUNT", "100", "BLOCK", "5000", "STREAMS", @redis_stream, last_id)
|
|
106
|
-
|
|
107
|
-
if data
|
|
108
|
-
data[@redis_stream].each do |id, (_, stream_name, _, serialized_data, _, broadcaster_uuid, _, message_uuid)|
|
|
109
|
-
if broadcaster_uuid != @server_uuid && message_uuid != last_message_uuid
|
|
110
|
-
Rage.cable.__protocol.broadcast(stream_name, JSON.parse(serialized_data))
|
|
111
|
-
end
|
|
112
|
-
|
|
113
|
-
last_id = id
|
|
114
|
-
last_message_uuid = message_uuid
|
|
115
|
-
end
|
|
116
|
-
end
|
|
117
|
-
|
|
118
|
-
rescue RedisClient::Error => e
|
|
119
|
-
Rage.logger.error("Subscriber error: #{e.message} (#{e.class})")
|
|
120
|
-
sleep error_backoff_intervals.next
|
|
121
|
-
rescue => e
|
|
122
|
-
@stopping ? break : raise(e)
|
|
123
|
-
else
|
|
124
|
-
error_backoff_intervals.rewind
|
|
125
|
-
end
|
|
126
|
-
end
|
|
127
|
-
end
|
|
128
|
-
end
|